/** * vim: set ts=4 : * ============================================================================= * SourceMod SDKTools Extension * Copyright (C) 2004-2008 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, version 3.0, as published by the * Free Software Foundation. * * 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, see . * * As a special exception, AlliedModders LLC gives you permission to link the * code of this program (as well as its derivative works) to "Half-Life 2," the * "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software * by the Valve Corporation. You must obey the GNU General Public License in * all respects for all other code used. Additionally, AlliedModders LLC grants * this exception to all derivative works. AlliedModders LLC defines further * exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007), * or . * * Version: $Id$ */ #include "extension.h" #include "util.h" #include "vhelpers.h" CallHelper s_Teleport; CallHelper s_GetVelocity; CallHelper s_EyeAngles; class CTraceFilterSimple : public CTraceFilterEntitiesOnly { public: CTraceFilterSimple(const IHandleEntity *passentity): m_pPassEnt(passentity) { } virtual bool ShouldHitEntity(IHandleEntity *pServerEntity, int contentsMask) { if (pServerEntity == m_pPassEnt) { return false; } return true; } private: const IHandleEntity *m_pPassEnt; }; bool SetupTeleport() { if (s_Teleport.setup) { return s_Teleport.supported; } /* Setup Teleport */ int offset; if (g_pGameConf->GetOffset("Teleport", &offset)) { PassInfo info[3]; info[0].flags = info[1].flags = info[2].flags = PASSFLAG_BYVAL; info[0].size = info[1].size = info[2].size = sizeof(void *); info[0].type = info[1].type = info[2].type = PassType_Basic; s_Teleport.call = g_pBinTools->CreateVCall(offset, 0, 0, NULL, info, 3); if (s_Teleport.call != NULL) { s_Teleport.supported = true; } } s_Teleport.setup = true; return s_Teleport.supported; } void Teleport(CBaseEntity *pEntity, Vector *origin, QAngle *ang, Vector *velocity) { unsigned char params[sizeof(void *) * 4]; unsigned char *vptr = params; *(CBaseEntity **)vptr = pEntity; vptr += sizeof(CBaseEntity *); *(Vector **)vptr = origin; vptr += sizeof(Vector *); *(QAngle **)vptr = ang; vptr += sizeof(QAngle *); *(Vector **)vptr = velocity; s_Teleport.call->Execute(params, NULL); } bool IsTeleportSupported() { return SetupTeleport(); } bool SetupGetVelocity() { if (s_GetVelocity.setup) { return s_GetVelocity.supported; } int offset; if (g_pGameConf->GetOffset("GetVelocity", &offset)) { PassInfo info[2]; info[0].flags = info[1].flags = PASSFLAG_BYVAL; info[0].size = info[1].size = sizeof(void *); info[0].type = info[1].type = PassType_Basic; s_GetVelocity.call = g_pBinTools->CreateVCall(offset, 0, 0, NULL, info, 2); if (s_GetVelocity.call != NULL) { s_GetVelocity.supported = true; } } s_GetVelocity.setup = true; return s_GetVelocity.supported; } void GetVelocity(CBaseEntity *pEntity, Vector *velocity, AngularImpulse *angvelocity) { unsigned char params[sizeof(void *) * 3]; unsigned char *vptr = params; *(CBaseEntity **)vptr = pEntity; vptr += sizeof(CBaseEntity *); *(Vector **)vptr = velocity; vptr += sizeof(Vector *); *(AngularImpulse **)vptr = angvelocity; s_GetVelocity.call->Execute(params, NULL); } bool IsGetVelocitySupported() { return SetupGetVelocity(); } bool SetupGetEyeAngles() { if (s_EyeAngles.setup) { return s_EyeAngles.supported; } int offset; if (g_pGameConf->GetOffset("EyeAngles", &offset)) { PassInfo info[1]; info[0].flags = PASSFLAG_BYVAL; info[0].size = sizeof(void *); info[0].type = PassType_Basic; s_EyeAngles.call = g_pBinTools->CreateVCall(offset, 0, 0, info, NULL, 0); if (s_EyeAngles.call != NULL) { s_EyeAngles.supported = true; } } s_EyeAngles.setup = true; return s_EyeAngles.supported; } bool GetEyeAngles(CBaseEntity *pEntity, QAngle *pAngles) { if (!IsEyeAnglesSupported()) { return false; } QAngle *pRetAngle = NULL; unsigned char params[sizeof(void *)]; unsigned char *vptr = params; *(CBaseEntity **)vptr = pEntity; s_EyeAngles.call->Execute(params, &pRetAngle); if (pRetAngle == NULL) { return false; } *pAngles = *pRetAngle; return true; } int GetClientAimTarget(edict_t *pEdict, bool only_players) { CBaseEntity *pEntity = pEdict->GetUnknown() ? pEdict->GetUnknown()->GetBaseEntity() : NULL; if (pEntity == NULL) { return -1; } Vector eye_position; QAngle eye_angles; /* Get the private information we need */ serverClients->ClientEarPosition(pEdict, &eye_position); if (!GetEyeAngles(pEntity, &eye_angles)) { return -2; } Vector aim_dir; AngleVectors(eye_angles, &aim_dir); VectorNormalize(aim_dir); Vector vec_end = eye_position + aim_dir * 8000; Ray_t ray; ray.Init(eye_position, vec_end); trace_t tr; CTraceFilterSimple simple(pEdict->GetIServerEntity()); enginetrace->TraceRay(ray, MASK_SOLID|CONTENTS_DEBRIS|CONTENTS_HITBOX, &simple, &tr); if (tr.fraction == 1.0f || tr.m_pEnt == NULL) { return -1; } int ent_ref = gamehelpers->EntityToBCompatRef(tr.m_pEnt); int ent_index = gamehelpers->ReferenceToIndex(ent_ref); IGamePlayer *pTargetPlayer = playerhelpers->GetGamePlayer(ent_index); if (pTargetPlayer != NULL && !pTargetPlayer->IsInGame()) { return -1; } else if (only_players && pTargetPlayer == NULL) { return -1; } return ent_ref; } bool IsEyeAnglesSupported() { return SetupGetEyeAngles(); } bool GetPlayerInfo(int client, player_info_t *info) { #if SOURCE_ENGINE >= SE_ORANGEBOX return engine->GetPlayerInfo(client, info); #else return (iserver) ? iserver->GetPlayerInfo(client-1, info) : false; #endif } void ShutdownHelpers() { s_Teleport.Shutdown(); s_GetVelocity.Shutdown(); s_EyeAngles.Shutdown(); } const char *GetDTTypeName(int type) { switch (type) { case DPT_Int: { return "integer"; } case DPT_Float: { return "float"; } case DPT_Vector: { return "vector"; } case DPT_String: { return "string"; } case DPT_Array: { return "array"; } case DPT_DataTable: { return "datatable"; } default: { return NULL; } } return NULL; } void UTIL_DrawSendTable_XML(FILE *fp, SendTable *pTable, int space_count) { char spaces[255]; for (int i = 0; i < space_count; i++) { spaces[i] = ' '; } spaces[space_count] = '\0'; const char *type_name; SendTable *pOtherTable; SendProp *pProp; fprintf(fp, " %s\n", spaces, pTable->GetName()); for (int i = 0; i < pTable->GetNumProps(); i++) { pProp = pTable->GetProp(i); fprintf(fp, " %s\n", spaces, pProp->GetName()); if ((type_name = GetDTTypeName(pProp->GetType())) != NULL) { fprintf(fp, " %s%s\n", spaces, type_name); } else { fprintf(fp, " %s%d\n", spaces, pProp->GetType()); } fprintf(fp, " %s%d\n", spaces, pProp->GetOffset()); fprintf(fp, " %s%d\n", spaces, pProp->m_nBits); if ((pOtherTable = pTable->GetProp(i)->GetDataTable()) != NULL) { UTIL_DrawSendTable_XML(fp, pOtherTable, space_count + 3); } fprintf(fp, " %s\n", spaces); } fprintf(fp, " %s\n", spaces); } void UTIL_DrawServerClass_XML(FILE *fp, ServerClass *sc) { fprintf(fp, "\n", sc->GetName()); UTIL_DrawSendTable_XML(fp, sc->m_pTable, 0); fprintf(fp, "\n"); } void UTIL_DrawSendTable(FILE *fp, SendTable *pTable, int level) { char spaces[255]; for (int i=0; iGetName()); for (int i=0; iGetNumProps(); i++) { pProp = pTable->GetProp(i); name = pProp->GetName(); if (pProp->GetDataTable()) { UTIL_DrawSendTable(fp, pProp->GetDataTable(), level + 1); } else { type = GetDTTypeName(pProp->GetType()); if (type != NULL) { fprintf(fp, "%s-Member: %s (offset %d) (type %s) (bits %d)\n", spaces, pProp->GetName(), pProp->GetOffset(), type, pProp->m_nBits); } else { fprintf(fp, "%s-Member: %s (offset %d) (type %d) (bits %d)\n", spaces, pProp->GetName(), pProp->GetOffset(), pProp->GetType(), pProp->m_nBits); } } } } CON_COMMAND(sm_dump_netprops_xml, "Dumps the networkable property table as an XML file") { #if SOURCE_ENGINE <= SE_DARKMESSIAH CCommand args; #endif if (args.ArgC() < 2) { META_CONPRINT("Usage: sm_dump_netprops_xml \n"); return; } const char *file = args.Arg(1); if (!file || file[0] == '\0') { META_CONPRINT("Usage: sm_dump_netprops_xml \n"); return; } char path[PLATFORM_MAX_PATH]; g_pSM->BuildPath(Path_Game, path, sizeof(path), "%s", file); FILE *fp = NULL; if ((fp = fopen(path, "wt")) == NULL) { META_CONPRINTF("Could not open file \"%s\"\n", path); return; } fprintf(fp, "\n\n"); fprintf(fp, "\n\n", g_pSM->GetGameFolderName()); ServerClass *pBase = gamedll->GetAllServerClasses(); while (pBase != NULL) { UTIL_DrawServerClass_XML(fp, pBase); pBase = pBase->m_pNext; } fclose(fp); } CON_COMMAND(sm_dump_netprops, "Dumps the networkable property table as a text file") { #if SOURCE_ENGINE <= SE_DARKMESSIAH CCommand args; #endif if (args.ArgC() < 2) { META_CONPRINT("Usage: sm_dump_netprops \n"); return; } const char *file = args.Arg(1); if (!file || file[0] == '\0') { META_CONPRINT("Usage: sm_dump_netprops \n"); return; } char path[PLATFORM_MAX_PATH]; g_pSM->BuildPath(Path_Game, path, sizeof(path), "%s", file); FILE *fp = NULL; if ((fp = fopen(path, "wt")) == NULL) { META_CONPRINTF("Could not open file \"%s\"\n", path); return; } fprintf(fp, "// Dump of all network properties for \"%s\" follows\n//\n\n", g_pSM->GetGameFolderName()); ServerClass *pBase = gamedll->GetAllServerClasses(); while (pBase != NULL) { fprintf(fp, "%s:\n", pBase->GetName()); UTIL_DrawSendTable(fp, pBase->m_pTable, 1); pBase = pBase->m_pNext; } fclose(fp); } #if defined SUBPLATFORM_SECURECRT void _ignore_invalid_parameter( const wchar_t * expression, const wchar_t * function, const wchar_t * file, unsigned int line, uintptr_t pReserved ) { /* Wow we don't care, thanks Microsoft. */ } #endif CON_COMMAND(sm_dump_classes, "Dumps the class list as a text file") { #if SOURCE_ENGINE <= SE_DARKMESSIAH CCommand args; #endif if (args.ArgC() < 2) { META_CONPRINT("Usage: sm_dump_classes \n"); return; } const char *file = args.Arg(1); if (!file || file[0] == '\0') { META_CONPRINT("Usage: sm_dump_classes \n"); return; } ICallWrapper *pWrapper = NULL; if (!pWrapper) { PassInfo retData; retData.flags = PASSFLAG_BYVAL; retData.size = sizeof(void *); retData.type = PassType_Basic; void *addr; if (!g_pGameConf->GetMemSig("EntityFactory", &addr) || addr == NULL) { META_CONPRINT("Failed to locate function\n"); return; } pWrapper = g_pBinTools->CreateCall(addr, CallConv_Cdecl, &retData, NULL, 0); } void *returnData = NULL; pWrapper->Execute(NULL, &returnData); pWrapper->Destroy(); if (returnData == NULL) { return; } CEntityFactoryDictionary *dict = ( CEntityFactoryDictionary * )returnData; if ( !dict ) { return; } char path[PLATFORM_MAX_PATH]; g_pSM->BuildPath(Path_Game, path, sizeof(path), "%s", file); FILE *fp = NULL; if ((fp = fopen(path, "wt")) == NULL) { META_CONPRINTF("Could not open file \"%s\"\n", path); return; } char buffer[80]; buffer[0] = 0; #if defined SUBPLATFORM_SECURECRT _invalid_parameter_handler handler = _set_invalid_parameter_handler(_ignore_invalid_parameter); #endif time_t t = g_pSM->GetAdjustedTime(); size_t written = strftime(buffer, sizeof(buffer), "%d/%m/%Y", localtime(&t)); #if defined SUBPLATFORM_SECURECRT _set_invalid_parameter_handler(handler); #endif fprintf(fp, "// Dump of all classes for \"%s\" as at %s\n//\n\n", g_pSM->GetGameFolderName(), buffer); for ( int i = dict->m_Factories.First(); i != dict->m_Factories.InvalidIndex(); i = dict->m_Factories.Next( i ) ) { IServerNetworkable *entity = dict->Create(dict->m_Factories.GetElementName(i)); ServerClass *sclass = entity->GetServerClass(); fprintf(fp,"%s - %s\n",sclass->GetName(), dict->m_Factories.GetElementName(i)); typedescription_t *datamap = gamehelpers->FindInDataMap(gamehelpers->GetDataMap(entity->GetBaseEntity()), "m_iEFlags"); int *eflags = (int *)((char *)entity->GetBaseEntity() + datamap->fieldOffset[TD_OFFSET_NORMAL]); *eflags |= (1<<0); // EFL_KILLME } fclose(fp); } char *UTIL_FlagsToString(int flags) { static char str[1024]; str[0] = 0; if (flags & FTYPEDESC_GLOBAL) { strcat(str, "Global|"); } if (flags & FTYPEDESC_SAVE) { strcat(str, "Save|"); } if (flags & FTYPEDESC_KEY) { strcat(str, "Key|"); } if (flags & FTYPEDESC_INPUT) { strcat(str, "Input|"); } if (flags & FTYPEDESC_OUTPUT) { strcat(str, "Output|"); } if (flags & FTYPEDESC_FUNCTIONTABLE) { strcat(str, "FunctionTable|"); } if (flags & FTYPEDESC_PTR) { strcat(str, "Ptr|"); } if (flags & FTYPEDESC_OVERRIDE) { strcat(str, "Override|"); } int len = strlen(str) - 1; if (len > 0) { str[len] = 0; // Strip the final '|' } return str; } void UTIL_DrawDataTable(FILE *fp, datamap_t *pMap, int level) { char spaces[255]; for (int i=0; idataNumFields; i++) { if (pMap->dataDesc[i].fieldName == NULL) { continue; } if (pMap->dataDesc[i].td) { fprintf(fp, " %sSub-Class Table (%d Deep): %s - %s\n", spaces, level+1, pMap->dataDesc[i].fieldName, pMap->dataDesc[i].td->dataClassName); UTIL_DrawDataTable(fp, pMap->dataDesc[i].td, level+1); } else { externalname = pMap->dataDesc[i].externalName; flags = UTIL_FlagsToString(pMap->dataDesc[i].flags); if (externalname == NULL) { fprintf(fp,"%s- %s (%s)(%i Bytes)\n", spaces, pMap->dataDesc[i].fieldName, flags, pMap->dataDesc[i].fieldSizeInBytes); } else { fprintf(fp,"%s- %s (%s)(%i Bytes) - %s\n", spaces, pMap->dataDesc[i].fieldName, flags, pMap->dataDesc[i].fieldSizeInBytes, externalname); } } } pMap = pMap->baseMap; } } CON_COMMAND(sm_dump_datamaps, "Dumps the data map list as a text file") { #if SOURCE_ENGINE <= SE_DARKMESSIAH CCommand args; #endif if (args.ArgC() < 2) { META_CONPRINT("Usage: sm_dump_datamaps \n"); return; } const char *file = args.Arg(1); if (!file || file[0] == '\0') { META_CONPRINT("Usage: sm_dump_datamaps \n"); return; } ICallWrapper *pWrapper = NULL; if (!pWrapper) { PassInfo retData; retData.flags = PASSFLAG_BYVAL; retData.size = sizeof(void *); retData.type = PassType_Basic; void *addr; if (!g_pGameConf->GetMemSig("EntityFactory", &addr) || addr == NULL) { META_CONPRINT("Failed to locate function\n"); return; } pWrapper = g_pBinTools->CreateCall(addr, CallConv_Cdecl, &retData, NULL, 0); } void *returnData = NULL; pWrapper->Execute(NULL, &returnData); pWrapper->Destroy(); if (returnData == NULL) { return; } CEntityFactoryDictionary *dict = ( CEntityFactoryDictionary * )returnData; if ( !dict ) { return; } char path[PLATFORM_MAX_PATH]; g_pSM->BuildPath(Path_Game, path, sizeof(path), "%s", file); FILE *fp = NULL; if ((fp = fopen(path, "wt")) == NULL) { META_CONPRINTF("Could not open file \"%s\"\n", path); return; } char buffer[80]; buffer[0] = 0; #if defined SUBPLATFORM_SECURECRT _invalid_parameter_handler handler = _set_invalid_parameter_handler(_ignore_invalid_parameter); #endif time_t t = g_pSM->GetAdjustedTime(); size_t written = strftime(buffer, sizeof(buffer), "%d/%m/%Y", localtime(&t)); #if defined SUBPLATFORM_SECURECRT _set_invalid_parameter_handler(handler); #endif fprintf(fp, "// Dump of all datamaps for \"%s\" as at %s\n//\n//\n", g_pSM->GetGameFolderName(), buffer); fprintf(fp, "// Flag Details:\n//\n"); fprintf(fp, "// Global: This field is masked for global entity save/restore\n"); fprintf(fp, "// Save: This field is saved to disk\n"); fprintf(fp, "// Key: This field can be requested and written to by string name at load time\n"); fprintf(fp, "// Input: This field can be written to by string name at run time, and a function called\n"); fprintf(fp, "// Output: This field propogates it's value to all targets whenever it changes\n"); fprintf(fp, "// FunctionTable: This is a table entry for a member function pointer\n"); fprintf(fp, "// Ptr: This field is a pointer, not an embedded object\n"); fprintf(fp, "// Override: The field is an override for one in a base class (only used by prediction system for now)\n"); fprintf(fp, "//\n\n"); for ( int i = dict->m_Factories.First(); i != dict->m_Factories.InvalidIndex(); i = dict->m_Factories.Next( i ) ) { IServerNetworkable *entity = dict->Create(dict->m_Factories.GetElementName(i)); ServerClass *sclass = entity->GetServerClass(); datamap_t *pMap = gamehelpers->GetDataMap(entity->GetBaseEntity()); fprintf(fp,"%s - %s\n", sclass->GetName(), dict->m_Factories.GetElementName(i)); UTIL_DrawDataTable(fp, pMap, 0); typedescription_t *datamap = gamehelpers->FindInDataMap(pMap, "m_iEFlags"); int *eflags = (int *)((char *)entity->GetBaseEntity() + datamap->fieldOffset[TD_OFFSET_NORMAL]); *eflags |= (1<<0); // EFL_KILLME } fclose(fp); }