--HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%401081
		
			
				
	
	
		
			399 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			399 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
|  * vim: set ts=4 :
 | |
|  * ===============================================================
 | |
|  * SourceMod SDK Tools Extension
 | |
|  * Copyright (C) 2004-2007 AlliedModders LLC. All rights reserved.
 | |
|  * ===============================================================
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  * 
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program; if not, write to the Free Software
 | |
|  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 | |
|  *
 | |
|  * Version: $Id$
 | |
|  */
 | |
| 
 | |
| #include "tempents.h"
 | |
| 
 | |
| TempEntityManager g_TEManager;
 | |
| ICallWrapper *g_GetServerClass = NULL;
 | |
| 
 | |
| CON_COMMAND(sm_print_telist, "Prints the temp entity list")
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		META_CONPRINT("The tempent portion of SDKTools failed to load.\n");
 | |
| 		META_CONPRINT("Check that you have the latest sdktools.games.txt file!\n");
 | |
| 		return;
 | |
| 	}
 | |
| 	g_TEManager.DumpList();
 | |
| }
 | |
| 
 | |
| CON_COMMAND(sm_dump_teprops, "Dumps tempentity props to a file")
 | |
| {
 | |
| 	if (!g_TEManager.IsAvailable())
 | |
| 	{
 | |
| 		META_CONPRINT("The tempent portion of SDKTools failed to load.\n");
 | |
| 		META_CONPRINT("Check that you have the latest sdktools.games.txt file!\n");
 | |
| 		return;
 | |
| 	}
 | |
| 	int argc = engine->Cmd_Argc();
 | |
| 	if (argc < 2)
 | |
| 	{
 | |
| 		META_CONPRINT("Usage: sm_dump_teprops <file>\n");
 | |
| 		return;
 | |
| 	}
 | |
| 	const char *arg = engine->Cmd_Argv(1);
 | |
| 	FILE *fp = NULL;
 | |
| 	if (!arg || arg[0] == '\0' || ((fp = fopen(arg, "wt")) == NULL))
 | |
| 	{
 | |
| 		META_CONPRINTF("Could not open file \"%s\"\n", arg);
 | |
| 		return;
 | |
| 	}
 | |
| 	g_TEManager.DumpProps(fp);
 | |
| 	fclose(fp);
 | |
| }
 | |
| 
 | |
| /*************************
 | |
| *                        *
 | |
| * Temp Entities Wrappers *
 | |
| *                        *
 | |
| **************************/
 | |
| 
 | |
| TempEntityInfo::TempEntityInfo(const char *name, void *me)
 | |
| {
 | |
| 	m_Name.assign(name);
 | |
| 	m_Me = me;
 | |
| 	g_GetServerClass->Execute(&m_Me, &m_Sc);
 | |
| }
 | |
| 
 | |
| const char *TempEntityInfo::GetName()
 | |
| {
 | |
| 	return m_Name.c_str();
 | |
| }
 | |
| 
 | |
| ServerClass *TempEntityInfo::GetServerClass()
 | |
| {
 | |
| 	return m_Sc;
 | |
| }
 | |
| 
 | |
| bool TempEntityInfo::IsValidProp(const char *name)
 | |
| {
 | |
| 	return (g_pGameHelpers->FindInSendTable(m_Sc->GetName(), name)) ? true : false;
 | |
| }
 | |
| 
 | |
| int TempEntityInfo::_FindOffset(const char *name, int *size)
 | |
| {
 | |
| 	int offset;
 | |
| 
 | |
| 	SendProp *prop = g_pGameHelpers->FindInSendTable(m_Sc->GetName(), name);
 | |
| 	if (!prop)
 | |
| 	{
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	offset = prop->GetOffset();
 | |
| 	if (size)
 | |
| 	{
 | |
| 		*size = prop->m_nBits;
 | |
| 	}
 | |
| 
 | |
| 	return offset;
 | |
| }
 | |
| 
 | |
| bool TempEntityInfo::TE_SetEntData(const char *name, int value)
 | |
| {
 | |
| 	/* Search for our offset */
 | |
| 	int size;
 | |
| 	int offset = _FindOffset(name, &size);
 | |
| 
 | |
| 	if (offset < 0)
 | |
| 	{
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (size <= 8)
 | |
| 	{
 | |
| 		*((uint8_t *)m_Me + offset) = value;
 | |
| 	} else if (size <= 16) {
 | |
| 		*(short *)((uint8_t *)m_Me + offset) = value;
 | |
| 	} else if (size <= 32) {
 | |
| 		*(int *)((uint8_t *)m_Me + offset) = value;
 | |
| 	} else {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool TempEntityInfo::TE_SetEntDataFloat(const char *name, float value)
 | |
| {
 | |
| 	/* Search for our offset */
 | |
| 	int offset = _FindOffset(name);
 | |
| 
 | |
| 	if (offset < 0)
 | |
| 	{
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	*(float *)((uint8_t *)m_Me + offset) = value;
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool TempEntityInfo::TE_SetEntDataVector(const char *name, float vector[3])
 | |
| {
 | |
| 	/* Search for our offset */
 | |
| 	int offset = _FindOffset(name);
 | |
| 
 | |
| 	if (offset < 0)
 | |
| 	{
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	Vector *v = (Vector *)((uint8_t *)m_Me + offset);
 | |
| 	v->x = vector[0];
 | |
| 	v->y = vector[1];
 | |
| 	v->z = vector[2];
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool TempEntityInfo::TE_SetEntDataFloatArray(const char *name, cell_t *array, int size)
 | |
| {
 | |
| 	/* Search for our offset */
 | |
| 	int offset = _FindOffset(name);
 | |
| 
 | |
| 	if (offset < 0)
 | |
| 	{
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	float *base = (float *)((uint8_t *)m_Me + offset);
 | |
| 	for (int i=0; i<size; i++)
 | |
| 	{
 | |
| 		base[i] = sp_ctof(array[i]);
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void TempEntityInfo::Send(IRecipientFilter &filter, float delay)
 | |
| {
 | |
| 	engine->PlaybackTempEntity(filter, delay, m_Me, m_Sc->m_pTable, m_Sc->m_ClassID);
 | |
| }
 | |
| 
 | |
| /**********************
 | |
| *                     *
 | |
| * Temp Entity Manager *
 | |
| *                     *
 | |
| ***********************/
 | |
| 
 | |
| void TempEntityManager::Initialize()
 | |
| {
 | |
| 	void *addr;
 | |
| 	int offset;
 | |
| 	m_Loaded = false;
 | |
| 
 | |
| 	/* Read our sigs and offsets from the config file */
 | |
| 	if (!g_pGameConf->GetMemSig("CBaseTempEntity", &addr) || !addr)
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!g_pGameConf->GetOffset("s_pTempEntities", &offset))
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!g_pGameConf->GetOffset("GetTEName", &m_NameOffs))
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!g_pGameConf->GetOffset("GetTENext", &m_NextOffs))
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!g_pGameConf->GetOffset("TE_GetServerClass", &m_GetClassNameOffs))
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* Store the head of the TE linked list */
 | |
| 	m_ListHead = **(void ***)((unsigned char *)addr + offset);
 | |
| 
 | |
| 	/* Create our trie */
 | |
| 	m_TempEntInfo = adtfactory->CreateBasicTrie();
 | |
| 
 | |
| 	/* Create the GetServerClass call */
 | |
| 	PassInfo retinfo;
 | |
| 	retinfo.flags = PASSFLAG_BYVAL;
 | |
| 	retinfo.type = PassType_Basic;
 | |
| 	retinfo.size = sizeof(ServerClass *);
 | |
| 	g_GetServerClass = g_pBinTools->CreateVCall(m_GetClassNameOffs, 0, 0, &retinfo, NULL, 0);
 | |
| 
 | |
| 	/* We're done */
 | |
| 	m_Loaded = true;
 | |
| }
 | |
| 
 | |
| bool TempEntityManager::IsAvailable()
 | |
| {
 | |
| 	return m_Loaded;
 | |
| }
 | |
| 
 | |
| void TempEntityManager::Shutdown()
 | |
| {
 | |
| 	if (!IsAvailable())
 | |
| 	{
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	List<TempEntityInfo *>::iterator iter;
 | |
| 	for (iter=m_TEList.begin(); iter!=m_TEList.end(); iter++)
 | |
| 	{
 | |
| 		delete (*iter);
 | |
| 	}
 | |
| 	m_TEList.clear();
 | |
| 
 | |
| 	m_TempEntInfo->Destroy();
 | |
| 	g_GetServerClass->Destroy();
 | |
| 	g_GetServerClass = NULL;
 | |
| 	m_ListHead = NULL;
 | |
| 	m_NextOffs = m_NameOffs = m_GetClassNameOffs = 0;
 | |
| 	m_Loaded = false;
 | |
| }
 | |
| 
 | |
| TempEntityInfo *TempEntityManager::GetTempEntityInfo(const char *name)
 | |
| {
 | |
| 	/* If the system is unloaded skip the search */
 | |
| 	if (!IsAvailable())
 | |
| 	{
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	TempEntityInfo *te = NULL;
 | |
| 
 | |
| 	/* Start searching for the TE inside the engine */
 | |
| 	if (!m_TempEntInfo->Retrieve(name, reinterpret_cast<void **>(&te)))
 | |
| 	{
 | |
| 		void *iter = m_ListHead;
 | |
| 		while (iter)
 | |
| 		{
 | |
| 			const char *realname = *(const char **)((unsigned char *)iter + m_NameOffs);
 | |
| 			if (!realname)
 | |
| 			{
 | |
| 				continue;
 | |
| 			}
 | |
| 			if (strcmp(name, realname) == 0)
 | |
| 			{
 | |
| 				te = new TempEntityInfo(name, iter);
 | |
| 				m_TempEntInfo->Insert(name, (void *)te);
 | |
| 				m_TEList.push_back(te);
 | |
| 				return te;
 | |
| 			}
 | |
| 			iter = *(void **)((unsigned char *)iter + m_NextOffs);
 | |
| 		}
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return te;
 | |
| }
 | |
| 
 | |
| void TempEntityManager::DumpList()
 | |
| {
 | |
| 	unsigned int index = 0;
 | |
| 	META_CONPRINT("Listing temp entities:\n");
 | |
| 	void *iter = m_ListHead;
 | |
| 	while (iter)
 | |
| 	{
 | |
| 		const char *realname = *(const char **)((unsigned char *)iter + m_NameOffs);
 | |
| 		if (!realname)
 | |
| 		{
 | |
| 			break;
 | |
| 		}
 | |
| 		TempEntityInfo *info = GetTempEntityInfo(realname);
 | |
| 		if (!info)
 | |
| 		{
 | |
| 			continue;
 | |
| 		}
 | |
| 		ServerClass *sc = info->GetServerClass();
 | |
| 		META_CONPRINTF("[%02d] %s (%s)\n", index++, realname, sc->GetName());
 | |
| 		iter = *(void **)((unsigned char *)iter + m_NextOffs);
 | |
| 	}
 | |
| 	META_CONPRINTF("%d tempent%s found.\n", index, (index == 1) ? " was" : "s were");
 | |
| }
 | |
| 
 | |
| const char *SendPropTypeToString(SendPropType type)
 | |
| {
 | |
| 	if (type == DPT_Int)
 | |
| 	{
 | |
| 		return "int";
 | |
| 	} else if (type == DPT_Float) {
 | |
| 		return "float";
 | |
| 	} else if (type == DPT_Vector) {
 | |
| 		return "vector";
 | |
| 	} else if (type == DPT_String) {
 | |
| 		return "string";
 | |
| 	} else if (type == DPT_Array) {
 | |
| 		return "array";
 | |
| 	} else if (type == DPT_DataTable) {
 | |
| 		return "datatable";
 | |
| 	} else {
 | |
| 		return "unknown";
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void _DumpProps(FILE *fp, SendTable *pTable)
 | |
| {
 | |
| 	SendTable *pOther;
 | |
| 	for (int i=0; i<pTable->GetNumProps(); i++)
 | |
| 	{
 | |
| 		SendProp *prop = pTable->GetProp(i);
 | |
| 		if ((pOther = prop->GetDataTable()) != NULL)
 | |
| 		{
 | |
| 			_DumpProps(fp, pOther);
 | |
| 		} else {
 | |
| 			fprintf(fp, "\t\t\t\"%s\"\t\t\"%s\"\n", 
 | |
| 				prop->GetName() ? prop->GetName() : "unknown",
 | |
| 				SendPropTypeToString(prop->GetType()));
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TempEntityManager::DumpProps(FILE *fp)
 | |
| {
 | |
| 	unsigned int index = 0;
 | |
| 	void *iter = m_ListHead;
 | |
| 	fprintf(fp, "\"TempEnts\"\n{\n");
 | |
| 	while (iter)
 | |
| 	{
 | |
| 		const char *realname = *(const char **)((unsigned char *)iter + m_NameOffs);
 | |
| 		if (!realname)
 | |
| 		{
 | |
| 			break;
 | |
| 		}
 | |
| 		TempEntityInfo *info = GetTempEntityInfo(realname);
 | |
| 		if (!info)
 | |
| 		{
 | |
| 			continue;
 | |
| 		}
 | |
| 		ServerClass *sc = info->GetServerClass();
 | |
| 		fprintf(fp, "\t\"%s\"\n", sc->GetName());
 | |
| 		fprintf(fp, "\t{\n");
 | |
| 		fprintf(fp, "\t\t\"name\"\t\t\"%s\"\n", realname);
 | |
| 		fprintf(fp, "\t\t\"index\"\t\t\"%d\"\n", index++);
 | |
| 		fprintf(fp, "\t\t\"SendTable\"\n\t\t{\n");
 | |
| 		_DumpProps(fp, sc->m_pTable);
 | |
| 		fprintf(fp, "\t\t}\n\t}\n");
 | |
| 		iter = *(void **)((unsigned char *)iter + m_NextOffs);
 | |
| 	}
 | |
| 	fprintf(fp, "}\n");
 | |
| 	META_CONPRINTF("%d tempent%s written to file.\n", index, (index == 1) ? " was" : "s were");
 | |
| }
 |