/** * vim: set ts=4 : * ============================================================================= * SourceMod Counter-Strike:Source 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 "RegNatives.h" #include "forwards.h" #include "util_cstrike.h" #include int g_iPriceOffset = -1; #define REGISTER_NATIVE_ADDR(name, code) \ void *addr; \ if (!g_pGameConf->GetMemSig(name, &addr) || !addr) \ { \ return pContext->ThrowNativeError("Failed to locate function"); \ } \ code; \ g_RegNatives.Register(pWrapper); inline CBaseEntity *GetCBaseEntity(int num, bool isplayer) { edict_t *pEdict = gamehelpers->EdictOfIndex(num); if (!pEdict || pEdict->IsFree()) { return NULL; } if (num > 0 && num <= playerhelpers->GetMaxClients()) { IGamePlayer *pPlayer = playerhelpers->GetGamePlayer(pEdict); if (!pPlayer || !pPlayer->IsConnected()) { return NULL; } } else if (isplayer) { return NULL; } IServerUnknown *pUnk; if ((pUnk=pEdict->GetUnknown()) == NULL) { return NULL; } return pUnk->GetBaseEntity(); } bool UTIL_FindDataTable(SendTable *pTable, const char *name, sm_sendprop_info_t *info, unsigned int offset) { const char *pname; int props = pTable->GetNumProps(); SendProp *prop; SendTable *table; for (int i=0; iGetProp(i); if ((table = prop->GetDataTable()) != NULL) { pname = table->GetName(); if (pname && strcmp(name, pname) == 0) { info->prop = prop; info->actual_offset = offset + info->prop->GetOffset(); return true; } if (UTIL_FindDataTable(table, name, info, offset + prop->GetOffset()) ) { return true; } } } return false; } //Taken From Sourcemod Core unsigned int strncopy(char *dest, const char *src, size_t count) { if (!count) { return 0; } char *start = dest; while ((*src) && (--count)) { *dest++ = *src++; } *dest = '\0'; return (dest - start); } static cell_t CS_RespawnPlayer(IPluginContext *pContext, const cell_t *params) { static ICallWrapper *pWrapper = NULL; if (!pWrapper) { REGISTER_NATIVE_ADDR("RoundRespawn", pWrapper = g_pBinTools->CreateCall(addr, CallConv_ThisCall, NULL, NULL, 0)); } CBaseEntity *pEntity; if (!(pEntity=GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } pWrapper->Execute(&pEntity, NULL); return 1; } static cell_t CS_SwitchTeam(IPluginContext *pContext, const cell_t *params) { static ICallWrapper *pWrapper = NULL; if (!pWrapper) { REGISTER_NATIVE_ADDR("SwitchTeam", PassInfo pass[1]; \ pass[0].flags = PASSFLAG_BYVAL; \ pass[0].size = sizeof(int); \ pass[0].type = PassType_Basic; \ pWrapper = g_pBinTools->CreateCall(addr, CallConv_ThisCall, NULL, pass, 1)) } CBaseEntity *pEntity; if (!(pEntity=GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } unsigned char vstk[sizeof(CBaseEntity *) + sizeof(int)]; unsigned char *vptr = vstk; *(CBaseEntity **)vptr = pEntity; vptr += sizeof(CBaseEntity *); *(int *)vptr = params[2]; pWrapper->Execute(vstk, NULL); return 1; } static cell_t CS_DropWeapon(IPluginContext *pContext, const cell_t *params) { static ICallWrapper *pWrapper = NULL; if (!pWrapper) { REGISTER_NATIVE_ADDR("CSWeaponDrop", PassInfo pass[3]; \ pass[0].flags = PASSFLAG_BYVAL; \ pass[0].type = PassType_Basic; \ pass[0].size = sizeof(CBaseEntity *); \ pass[1].flags = PASSFLAG_BYVAL; \ pass[1].type = PassType_Basic; \ pass[1].size = sizeof(bool); \ pass[2].flags = PASSFLAG_BYVAL; \ pass[2].type = PassType_Basic; \ pass[2].size = sizeof(bool); \ pWrapper = g_pBinTools->CreateCall(addr, CallConv_ThisCall, NULL, pass, 3)) } CBaseEntity *pEntity; if (!(pEntity = GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } CBaseEntity *pWeapon; if (!(pWeapon = GetCBaseEntity(params[2], false))) { return pContext->ThrowNativeError("Weapon index %d is not valid", params[2]); } //Psychonic is awesome for this sm_sendprop_info_t spi; IServerUnknown *pUnk = (IServerUnknown *)pWeapon; IServerNetworkable *pNet = pUnk->GetNetworkable(); if (!UTIL_FindDataTable(pNet->GetServerClass()->m_pTable, "DT_WeaponCSBase", &spi, 0)) return pContext->ThrowNativeError("Entity index %d is not a weapon", params[2]); if (!gamehelpers->FindSendPropInfo("CBaseCombatWeapon", "m_hOwnerEntity", &spi)) return pContext->ThrowNativeError("Invalid entity index %d for weapon", params[2]); CBaseHandle &hndl = *(CBaseHandle *)((uint8_t *)pWeapon + spi.actual_offset); if (params[1] != hndl.GetEntryIndex()) return pContext->ThrowNativeError("Weapon %d is not owned by client %d", params[2], params[1]); if (params[4] == 1 && g_pCSWeaponDropDetoured) g_pIgnoreCSWeaponDropDetour = true; unsigned char vstk[sizeof(CBaseEntity *) * 2 + sizeof(bool) * 2]; unsigned char *vptr = vstk; // first one is always false. second is true to toss, false to just drop *(CBaseEntity **)vptr = pEntity; vptr += sizeof(CBaseEntity *); *(CBaseEntity **)vptr = pWeapon; vptr += sizeof(CBaseEntity *); *(bool *)vptr = false; vptr += sizeof(bool); *(bool *)vptr = (params[3]) ? true : false; pWrapper->Execute(vstk, NULL); return 1; } static cell_t CS_TerminateRound(IPluginContext *pContext, const cell_t *params) { if (g_pSDKTools == NULL) { smutils->LogError(myself, "SDKTools interface not found. TerminateRound native disabled."); } else if (g_pSDKTools->GetInterfaceVersion() < 2) { // THIS ISN'T DA LIMBO STICK. LOW IS BAD return pContext->ThrowNativeError("SDKTools interface is outdated. TerminateRound native disabled."); } static ICallWrapper *pWrapper = NULL; if (!pWrapper) { REGISTER_NATIVE_ADDR("TerminateRound", PassInfo pass[2]; \ pass[0].flags = PASSFLAG_BYVAL; \ pass[0].type = PassType_Basic; \ pass[0].size = sizeof(float); \ pass[1].flags = PASSFLAG_BYVAL; \ pass[1].type = PassType_Basic; \ pass[1].size = sizeof(int); \ pWrapper = g_pBinTools->CreateCall(addr, CallConv_ThisCall, NULL, pass, 2)) } void *gamerules = g_pSDKTools->GetGameRules(); if (gamerules == NULL) { return pContext->ThrowNativeError("GameRules not available. TerminateRound native disabled."); } if (params[3] == 1 && g_pTerminateRoundDetoured) g_pIgnoreTerminateDetour = true; unsigned char vstk[sizeof(void *) + sizeof(float)+ sizeof(int)]; unsigned char *vptr = vstk; *(void **)vptr = gamerules; vptr += sizeof(void *); *(float *)vptr = sp_ctof(params[1]); vptr += sizeof(float); *(int*)vptr = params[2]; pWrapper->Execute(vstk, NULL); return 1; } static cell_t CS_GetTranslatedWeaponAlias(IPluginContext *pContext, const cell_t *params) { char *weapon; char *dest; pContext->LocalToString(params[2], &dest); pContext->LocalToString(params[1], &weapon); const char *ret = GetTranslatedWeaponAlias(weapon); strncopy(dest, ret, params[3]); return 1; } static cell_t CS_GetWeaponPrice(IPluginContext *pContext, const cell_t *params) { //Hard code return values for weapons that dont call GetWeaponPrice and always use default value. int id = GetRealWeaponID(params[2]); #if SOURCE_ENGINE == SE_CSGO if (id == WEAPON_C4 || id == WEAPON_KNIFE || id == WEAPON_KNIFE_GG) #else if (id == WEAPON_C4 || id == WEAPON_KNIFE || id == WEAPON_SHIELD) #endif return 0; if (id == WEAPON_KEVLAR) return 650; if (id == WEAPON_ASSAULTSUIT) return 1000; if (id == WEAPON_NIGHTVISION) return 1250; #if SOURCE_ENGINE == SE_CSGO if (id == WEAPON_DEFUSER) return 200; #endif if (id == 0) return 0; if (g_iPriceOffset == -1) { if (!g_pGameConf->GetOffset("WeaponPrice", &g_iPriceOffset)) { g_iPriceOffset = -1; return pContext->ThrowNativeError("Failed to get WeaponPrice offset"); } } CBaseEntity *pEntity; if (!(pEntity = GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } void *info = GetWeaponInfo(id); if (!info) return pContext->ThrowNativeError("Failed to get weaponinfo"); int price = *(int *)((intptr_t)info + g_iPriceOffset); if (params[3] || weaponNameOffset == -1) return price; const char *weapon_name = (const char *)((intptr_t)info + weaponNameOffset); #if SOURCE_ENGINE == SE_CSGO return CallPriceForwardCSGO(params[1], weapon_name, price); #else return CallPriceForward(params[1], weapon_name, price); #endif } static cell_t CS_GetClientClanTag(IPluginContext *pContext, const cell_t *params) { static void *addr; if (!addr) { if (!g_pGameConf->GetMemSig("SetClanTag", &addr) || !addr) { return pContext->ThrowNativeError("Failed to locate function"); } } CBaseEntity *pEntity; if (!(pEntity = GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } static int tagOffsetOffset = -1; static int tagOffset; if (tagOffsetOffset == -1) { if (!g_pGameConf->GetOffset("ClanTagOffset", &tagOffsetOffset)) { tagOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find ClanTagOffset gamedata"); } tagOffset = *(int *)((intptr_t)addr + tagOffsetOffset); } size_t len; const char *src = (char *)((intptr_t)pEntity + tagOffset); pContext->StringToLocalUTF8(params[2], params[3], src, &len); return len; } static cell_t CS_SetClientClanTag(IPluginContext *pContext, const cell_t *params) { static ICallWrapper *pWrapper = NULL; if (!pWrapper) { REGISTER_NATIVE_ADDR("SetClanTag", PassInfo pass[2]; \ pass[0].flags = PASSFLAG_BYVAL; \ pass[0].type = PassType_Basic; \ pass[0].size = sizeof(CBaseEntity *); \ pass[1].flags = PASSFLAG_BYVAL; \ pass[1].type = PassType_Basic; \ pass[1].size = sizeof(char *); \ pWrapper = g_pBinTools->CreateCall(addr, CallConv_ThisCall, NULL, pass, 2)) } CBaseEntity *pEntity; if (!(pEntity = GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } char *szNewTag; pContext->LocalToString(params[2], &szNewTag); unsigned char vstk[sizeof(CBaseEntity *) + sizeof(char *)]; unsigned char *vptr = vstk; *(CBaseEntity **)vptr = pEntity; vptr += sizeof(CBaseEntity *); *(char **)vptr = szNewTag; pWrapper->Execute(vstk, NULL); return 1; } static cell_t CS_AliasToWeaponID(IPluginContext *pContext, const cell_t *params) { char *weapon; pContext->LocalToString(params[1], &weapon); return GetFakeWeaponID(AliasToWeaponID(weapon)); } static cell_t CS_SetTeamScore(IPluginContext *pContext, const cell_t *params) { if (g_pSDKTools == NULL) { smutils->LogError(myself, "SDKTools interface not found. CS_SetTeamScore native disabled."); } else if (g_pSDKTools->GetInterfaceVersion() < 2) { // THIS ISN'T DA LIMBO STICK. LOW IS BAD return pContext->ThrowNativeError("SDKTools interface is outdated. CS_SetTeamScore native disabled."); } static void *addr; if (!addr) { if (!g_pGameConf->GetMemSig("CheckWinLimit", &addr) || !addr) { return pContext->ThrowNativeError("Failed to locate CheckWinLimit function"); } } static int ctTeamOffsetOffset = -1; static int ctTeamOffset; static int tTeamOffsetOffset = -1; static int tTeamOffset; if (ctTeamOffsetOffset == -1) { if (!g_pGameConf->GetOffset("CTTeamScoreOffset", &ctTeamOffsetOffset)) { ctTeamOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find CTTeamOffset gamedata"); } ctTeamOffset = *(int *)((intptr_t)addr + ctTeamOffsetOffset); } if (tTeamOffsetOffset == -1) { if (!g_pGameConf->GetOffset("TTeamScoreOffset", &tTeamOffsetOffset)) { tTeamOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find CTTeamOffset gamedata"); } tTeamOffset = *(int *)((intptr_t)addr + tTeamOffsetOffset); } void *gamerules = g_pSDKTools->GetGameRules(); if (gamerules == NULL) { return pContext->ThrowNativeError("GameRules not available. CS_SetTeamScore native disabled."); } if (params[1] == 3) { *(int16_t *)((intptr_t)gamerules+ctTeamOffset) = params[2]; return 1; } else if (params[1] == 2) { *(int16_t *)((intptr_t)gamerules+tTeamOffset) = params[2]; return 1; } return pContext->ThrowNativeError("Invalid team index passed (%i).", params[1]); } static cell_t CS_GetTeamScore(IPluginContext *pContext, const cell_t *params) { if (g_pSDKTools == NULL) { smutils->LogError(myself, "SDKTools interface not found. CS_GetTeamScore native disabled."); } else if (g_pSDKTools->GetInterfaceVersion() < 2) { // THIS ISN'T DA LIMBO STICK. LOW IS BAD return pContext->ThrowNativeError("SDKTools interface is outdated. CS_GetTeamScore native disabled."); } static void *addr; if (!addr) { if (!g_pGameConf->GetMemSig("CheckWinLimit", &addr) || !addr) { return pContext->ThrowNativeError("Failed to locate CheckWinLimit function"); } } static int ctTeamOffsetOffset = -1; static int ctTeamOffset; static int tTeamOffsetOffset = -1; static int tTeamOffset; if (ctTeamOffsetOffset == -1) { if (!g_pGameConf->GetOffset("CTTeamScoreOffset", &ctTeamOffsetOffset)) { ctTeamOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find CTTeamOffset gamedata"); } ctTeamOffset = *(int *)((intptr_t)addr + ctTeamOffsetOffset); } if (tTeamOffsetOffset == -1) { if (!g_pGameConf->GetOffset("TTeamScoreOffset", &tTeamOffsetOffset)) { tTeamOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find CTTeamOffset gamedata"); } tTeamOffset = *(int *)((intptr_t)addr + tTeamOffsetOffset); } void *gamerules = g_pSDKTools->GetGameRules(); if (gamerules == NULL) { return pContext->ThrowNativeError("GameRules not available. CS_GetTeamScore native disabled."); } if (params[1] == 3) { return *(int16_t *)((intptr_t)gamerules+ctTeamOffset); } else if (params[1] == 2) { return *(int16_t *)((intptr_t)gamerules+tTeamOffset); } return pContext->ThrowNativeError("Invalid team index passed (%i).", params[1]); } static cell_t CS_SetMVPCount(IPluginContext *pContext, const cell_t *params) { static void *addr; if (!addr) { if (!g_pGameConf->GetMemSig("IncrementNumMVPs", &addr) || !addr) { return pContext->ThrowNativeError("Failed to locate IncrementNumMVPs function"); } } CBaseEntity *pEntity; if (!(pEntity = GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } static int mvpOffsetOffset = -1; static int mvpOffset; if (mvpOffsetOffset == -1) { if (!g_pGameConf->GetOffset("MVPCountOffset", &mvpOffsetOffset)) { mvpOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find MVPCountOffset gamedata"); } mvpOffset = *(int *)((intptr_t)addr + mvpOffsetOffset); } *(int *)((intptr_t)pEntity + mvpOffset) = params[2]; return 1; } static cell_t CS_GetMVPCount(IPluginContext *pContext, const cell_t *params) { static void *addr; if (!addr) { if (!g_pGameConf->GetMemSig("IncrementNumMVPs", &addr) || !addr) { return pContext->ThrowNativeError("Failed to locate IncrementNumMVPs function"); } } CBaseEntity *pEntity; if (!(pEntity = GetCBaseEntity(params[1], true))) { return pContext->ThrowNativeError("Client index %d is not valid", params[1]); } static int mvpOffsetOffset = -1; static int mvpOffset; if (mvpOffsetOffset == -1) { if (!g_pGameConf->GetOffset("MVPCountOffset", &mvpOffsetOffset)) { mvpOffsetOffset = -1; return pContext->ThrowNativeError("Unable to find MVPCountOffset gamedata"); } mvpOffset = *(int *)((intptr_t)addr + mvpOffsetOffset); } return *(int *)((intptr_t)pEntity + mvpOffset); } sp_nativeinfo_t g_CSNatives[] = { {"CS_RespawnPlayer", CS_RespawnPlayer}, {"CS_SwitchTeam", CS_SwitchTeam}, {"CS_DropWeapon", CS_DropWeapon}, {"CS_TerminateRound", CS_TerminateRound}, {"CS_GetTranslatedWeaponAlias", CS_GetTranslatedWeaponAlias}, {"CS_GetWeaponPrice", CS_GetWeaponPrice}, {"CS_GetClientClanTag", CS_GetClientClanTag}, {"CS_SetClientClanTag", CS_SetClientClanTag}, {"CS_AliasToWeaponID", CS_AliasToWeaponID}, {"CS_GetTeamScore", CS_GetTeamScore}, {"CS_SetTeamScore", CS_SetTeamScore}, {"CS_GetMVPCount", CS_GetMVPCount}, {"CS_SetMVPCount", CS_SetMVPCount}, {NULL, NULL} };