475 lines
13 KiB
C++
475 lines
13 KiB
C++
/**
|
|
* vim: set ts=4 :
|
|
* =============================================================================
|
|
* SourceMod SDKTools Extension
|
|
* Copyright (C) 2004-2010 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 <http://www.gnu.org/licenses/>.
|
|
*
|
|
* 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 <http://www.sourcemod.net/license.php>.
|
|
*
|
|
* Version: $Id$
|
|
*/
|
|
|
|
#include <sourcemod_version.h>
|
|
#include "extension.h"
|
|
#include <compat_wrappers.h>
|
|
#include "vcallbuilder.h"
|
|
#include "vnatives.h"
|
|
#include "vhelpers.h"
|
|
#include "vglobals.h"
|
|
#include "tempents.h"
|
|
#include "vsound.h"
|
|
#include "output.h"
|
|
#include "hooks.h"
|
|
#include "gamerulesnatives.h"
|
|
#include <ISDKTools.h>
|
|
#include "clientnatives.h"
|
|
/**
|
|
* @file extension.cpp
|
|
* @brief Implements SDK Tools extension code.
|
|
*/
|
|
|
|
SH_DECL_HOOK6(IServerGameDLL, LevelInit, SH_NOATTRIB, false, bool, const char *, const char *, const char *, const char *, bool, bool);
|
|
|
|
SDKTools g_SdkTools; /**< Global singleton for extension's main interface */
|
|
IServerGameEnts *gameents = NULL;
|
|
IEngineTrace *enginetrace = NULL;
|
|
IEngineSound *engsound = NULL;
|
|
INetworkStringTableContainer *netstringtables = NULL;
|
|
IServerPluginHelpers *pluginhelpers = NULL;
|
|
IBinTools *g_pBinTools = NULL;
|
|
IGameConfig *g_pGameConf = NULL;
|
|
IGameHelpers *g_pGameHelpers = NULL;
|
|
IServerGameClients *serverClients = NULL;
|
|
IVoiceServer *voiceserver = NULL;
|
|
IPlayerInfoManager *playerinfomngr = NULL;
|
|
ICvar *icvar = NULL;
|
|
IServer *iserver = NULL;
|
|
CGlobalVars *gpGlobals;
|
|
|
|
#if SOURCE_ENGINE >= SE_ORANGEBOX
|
|
IServerTools *servertools = NULL;
|
|
#endif
|
|
|
|
SourceHook::CallClass<IVEngineServer> *enginePatch = NULL;
|
|
SourceHook::CallClass<IEngineSound> *enginesoundPatch = NULL;
|
|
HandleType_t g_CallHandle = 0;
|
|
HandleType_t g_TraceHandle = 0;
|
|
ISDKTools *g_pSDKTools;
|
|
|
|
SMEXT_LINK(&g_SdkTools);
|
|
|
|
extern sp_nativeinfo_t g_CallNatives[];
|
|
extern sp_nativeinfo_t g_TENatives[];
|
|
extern sp_nativeinfo_t g_TRNatives[];
|
|
extern sp_nativeinfo_t g_StringTableNatives[];
|
|
extern sp_nativeinfo_t g_VoiceNatives[];
|
|
extern sp_nativeinfo_t g_EntInputNatives[];
|
|
extern sp_nativeinfo_t g_TeamNatives[];
|
|
extern sp_nativeinfo_t g_GameRulesNatives[];
|
|
extern sp_nativeinfo_t g_ClientNatives[];
|
|
|
|
static void InitSDKToolsAPI();
|
|
|
|
bool SDKTools::SDK_OnLoad(char *error, size_t maxlength, bool late)
|
|
{
|
|
HandleError err;
|
|
|
|
if (!gameconfs->LoadGameConfigFile("sdktools.games", &g_pGameConf, error, maxlength))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
sharesys->AddDependency(myself, "bintools.ext", true, true);
|
|
sharesys->AddNatives(myself, g_CallNatives);
|
|
sharesys->AddNatives(myself, g_Natives);
|
|
sharesys->AddNatives(myself, g_TENatives);
|
|
sharesys->AddNatives(myself, g_SoundNatives);
|
|
sharesys->AddNatives(myself, g_TRNatives);
|
|
sharesys->AddNatives(myself, g_StringTableNatives);
|
|
sharesys->AddNatives(myself, g_VoiceNatives);
|
|
sharesys->AddNatives(myself, g_EntInputNatives);
|
|
sharesys->AddNatives(myself, g_TeamNatives);
|
|
sharesys->AddNatives(myself, g_EntOutputNatives);
|
|
sharesys->AddNatives(myself, g_GameRulesNatives);
|
|
sharesys->AddNatives(myself, g_ClientNatives);
|
|
|
|
SM_GET_IFACE(GAMEHELPERS, g_pGameHelpers);
|
|
|
|
playerhelpers->AddClientListener(&g_SdkTools);
|
|
g_CallHandle = handlesys->CreateType("ValveCall", this, 0, NULL, NULL, myself->GetIdentity(), &err);
|
|
if (g_CallHandle == 0)
|
|
{
|
|
snprintf(error, maxlength, "Could not create call handle type (err: %d)", err);
|
|
return false;
|
|
}
|
|
|
|
TypeAccess TraceAccess;
|
|
handlesys->InitAccessDefaults(&TraceAccess, NULL);
|
|
TraceAccess.ident = myself->GetIdentity();
|
|
TraceAccess.access[HTypeAccess_Create] = true;
|
|
TraceAccess.access[HTypeAccess_Inherit] = true;
|
|
g_TraceHandle = handlesys->CreateType("TraceRay", this, 0, &TraceAccess, NULL, myself->GetIdentity(), &err);
|
|
if (g_TraceHandle == 0)
|
|
{
|
|
handlesys->RemoveType(g_CallHandle, myself->GetIdentity());
|
|
g_CallHandle = 0;
|
|
snprintf(error, maxlength, "Could not create traceray handle type (err: %d)", err);
|
|
return false;
|
|
}
|
|
|
|
#if SOURCE_ENGINE >= SE_ORANGEBOX
|
|
g_pCVar = icvar;
|
|
#endif
|
|
CONVAR_REGISTER(this);
|
|
|
|
SH_ADD_HOOK_MEMFUNC(IServerGameDLL, LevelInit, gamedll, this, &SDKTools::LevelInit, true);
|
|
|
|
playerhelpers->RegisterCommandTargetProcessor(this);
|
|
|
|
MathLib_Init(2.2f, 2.2f, 0.0f, 2);
|
|
|
|
spengine = g_pSM->GetScriptingEngine();
|
|
|
|
plsys->AddPluginsListener(&g_OutputManager);
|
|
|
|
CDetourManager::Init(g_pSM->GetScriptingEngine(), g_pGameConf);
|
|
g_OutputManager.Init();
|
|
|
|
VoiceInit();
|
|
|
|
GetIServer();
|
|
|
|
GameRulesNativesInit();
|
|
|
|
InitSDKToolsAPI();
|
|
|
|
return true;
|
|
}
|
|
|
|
void SDKTools::OnHandleDestroy(HandleType_t type, void *object)
|
|
{
|
|
if (type == g_CallHandle)
|
|
{
|
|
ValveCall *v = (ValveCall *)object;
|
|
delete v;
|
|
}
|
|
else if (type == g_TraceHandle)
|
|
{
|
|
trace_t *tr = (trace_t *)object;
|
|
delete tr;
|
|
}
|
|
}
|
|
|
|
void SDKTools::SDK_OnUnload()
|
|
{
|
|
SourceHook::List<ValveCall *>::iterator iter;
|
|
for (iter = g_RegCalls.begin();
|
|
iter != g_RegCalls.end();
|
|
iter++)
|
|
{
|
|
delete (*iter);
|
|
}
|
|
g_RegCalls.clear();
|
|
ShutdownHelpers();
|
|
|
|
if (g_pAcceptInput)
|
|
{
|
|
g_pAcceptInput->Destroy();
|
|
g_pAcceptInput = NULL;
|
|
}
|
|
|
|
g_TEManager.Shutdown();
|
|
s_TempEntHooks.Shutdown();
|
|
s_SoundHooks.Shutdown();
|
|
g_Hooks.Shutdown();
|
|
|
|
gameconfs->CloseGameConfigFile(g_pGameConf);
|
|
playerhelpers->RemoveClientListener(&g_SdkTools);
|
|
playerhelpers->UnregisterCommandTargetProcessor(this);
|
|
plsys->RemovePluginsListener(&g_OutputManager);
|
|
|
|
SH_REMOVE_HOOK_MEMFUNC(IServerGameDLL, LevelInit, gamedll, this, &SDKTools::LevelInit, true);
|
|
|
|
if (enginePatch)
|
|
{
|
|
SH_RELEASE_CALLCLASS(enginePatch);
|
|
enginePatch = NULL;
|
|
}
|
|
if (enginesoundPatch)
|
|
{
|
|
SH_RELEASE_CALLCLASS(enginesoundPatch);
|
|
enginesoundPatch = NULL;
|
|
}
|
|
|
|
bool err;
|
|
if (g_CallHandle != 0)
|
|
{
|
|
if ((err = handlesys->RemoveType(g_CallHandle, myself->GetIdentity())) != true)
|
|
{
|
|
g_pSM->LogError(myself, "Could not remove call handle (type=%x, err=%d)", g_CallHandle, err);
|
|
}
|
|
}
|
|
|
|
if (g_TraceHandle != 0)
|
|
{
|
|
if ((err = handlesys->RemoveType(g_TraceHandle, myself->GetIdentity())) != true)
|
|
{
|
|
g_pSM->LogError(myself, "Could not remove trace handle (type=%x, err=%d)", g_TraceHandle, err);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool SDKTools::SDK_OnMetamodLoad(ISmmAPI *ismm, char *error, size_t maxlen, bool late)
|
|
{
|
|
GET_V_IFACE_ANY(GetServerFactory, gameents, IServerGameEnts, INTERFACEVERSION_SERVERGAMEENTS);
|
|
GET_V_IFACE_ANY(GetEngineFactory, engsound, IEngineSound, IENGINESOUND_SERVER_INTERFACE_VERSION);
|
|
GET_V_IFACE_ANY(GetEngineFactory, enginetrace, IEngineTrace, INTERFACEVERSION_ENGINETRACE_SERVER);
|
|
GET_V_IFACE_ANY(GetEngineFactory, netstringtables, INetworkStringTableContainer, INTERFACENAME_NETWORKSTRINGTABLESERVER);
|
|
GET_V_IFACE_ANY(GetEngineFactory, pluginhelpers, IServerPluginHelpers, INTERFACEVERSION_ISERVERPLUGINHELPERS);
|
|
GET_V_IFACE_ANY(GetServerFactory, serverClients, IServerGameClients, INTERFACEVERSION_SERVERGAMECLIENTS);
|
|
GET_V_IFACE_ANY(GetEngineFactory, voiceserver, IVoiceServer, INTERFACEVERSION_VOICESERVER);
|
|
GET_V_IFACE_ANY(GetServerFactory, playerinfomngr, IPlayerInfoManager, INTERFACEVERSION_PLAYERINFOMANAGER);
|
|
GET_V_IFACE_CURRENT(GetEngineFactory, icvar, ICvar, CVAR_INTERFACE_VERSION);
|
|
|
|
#if SOURCE_ENGINE >= SE_ORANGEBOX
|
|
GET_V_IFACE_ANY(GetServerFactory, servertools, IServerTools, VSERVERTOOLS_INTERFACE_VERSION);
|
|
#endif
|
|
|
|
gpGlobals = ismm->GetCGlobals();
|
|
enginePatch = SH_GET_CALLCLASS(engine);
|
|
enginesoundPatch = SH_GET_CALLCLASS(engsound);
|
|
|
|
return true;
|
|
}
|
|
|
|
void SDKTools::SDK_OnAllLoaded()
|
|
{
|
|
SM_GET_LATE_IFACE(BINTOOLS, g_pBinTools);
|
|
|
|
if (!g_pBinTools)
|
|
{
|
|
return;
|
|
}
|
|
|
|
g_TEManager.Initialize();
|
|
s_TempEntHooks.Initialize();
|
|
s_SoundHooks.Initialize();
|
|
g_Hooks.Initialize();
|
|
InitializeValveGlobals();
|
|
}
|
|
|
|
bool SDKTools::QueryRunning(char *error, size_t maxlength)
|
|
{
|
|
SM_CHECK_IFACE(BINTOOLS, g_pBinTools);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SDKTools::QueryInterfaceDrop(SMInterface *pInterface)
|
|
{
|
|
if (pInterface == g_pBinTools)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return IExtensionInterface::QueryInterfaceDrop(pInterface);
|
|
}
|
|
|
|
void SDKTools::NotifyInterfaceDrop(SMInterface *pInterface)
|
|
{
|
|
SourceHook::List<ValveCall *>::iterator iter;
|
|
for (iter = g_RegCalls.begin();
|
|
iter != g_RegCalls.end();
|
|
iter++)
|
|
{
|
|
delete (*iter);
|
|
}
|
|
g_RegCalls.clear();
|
|
ShutdownHelpers();
|
|
|
|
g_TEManager.Shutdown();
|
|
s_TempEntHooks.Shutdown();
|
|
|
|
if (g_pAcceptInput)
|
|
{
|
|
g_pAcceptInput->Destroy();
|
|
g_pAcceptInput = NULL;
|
|
}
|
|
}
|
|
|
|
bool SDKTools::RegisterConCommandBase(ConCommandBase *pVar)
|
|
{
|
|
#if defined METAMOD_PLAPI_VERSION
|
|
return g_SMAPI->RegisterConCommandBase(g_PLAPI, pVar);
|
|
#else
|
|
return g_SMAPI->RegisterConCmdBase(g_PLAPI, pVar);
|
|
#endif
|
|
}
|
|
|
|
bool SDKTools::LevelInit(char const *pMapName, char const *pMapEntities, char const *pOldLevel, char const *pLandmarkName, bool loadGame, bool background)
|
|
{
|
|
const char *name;
|
|
char key[32];
|
|
int count, n = 1;
|
|
|
|
if (!(name=g_pGameConf->GetKeyValue("SlapSoundCount")))
|
|
{
|
|
RETURN_META_VALUE(MRES_IGNORED, true);
|
|
}
|
|
|
|
count = atoi(name);
|
|
|
|
while (n <= count)
|
|
{
|
|
snprintf(key, sizeof(key), "SlapSound%d", n);
|
|
if ((name=g_pGameConf->GetKeyValue(key)))
|
|
{
|
|
engsound->PrecacheSound(name, true);
|
|
}
|
|
n++;
|
|
}
|
|
|
|
RETURN_META_VALUE(MRES_IGNORED, true);
|
|
}
|
|
|
|
bool SDKTools::ProcessCommandTarget(cmd_target_info_t *info)
|
|
{
|
|
IGamePlayer *pAdmin = info->admin ? playerhelpers->GetGamePlayer(info->admin) : NULL;
|
|
|
|
if (strcmp(info->pattern, "@aim") == 0)
|
|
{
|
|
/* The server can't aim, of course. */
|
|
if (pAdmin == NULL)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
int player_index;
|
|
if ((player_index = GetClientAimTarget(pAdmin->GetEdict(), true)) < 1)
|
|
{
|
|
info->reason = COMMAND_TARGET_NONE;
|
|
info->num_targets = 0;
|
|
return true;
|
|
}
|
|
|
|
IGamePlayer *pTarget = playerhelpers->GetGamePlayer(player_index);
|
|
|
|
if (pTarget == NULL)
|
|
{
|
|
info->reason = COMMAND_TARGET_NONE;
|
|
info->num_targets = 0;
|
|
return true;
|
|
}
|
|
|
|
info->reason = playerhelpers->FilterCommandTarget(pAdmin, pTarget, info->flags);
|
|
if (info->reason != COMMAND_TARGET_VALID)
|
|
{
|
|
info->num_targets = 0;
|
|
return true;
|
|
}
|
|
|
|
info->targets[0] = player_index;
|
|
info->num_targets = 1;
|
|
info->reason = COMMAND_TARGET_VALID;
|
|
info->target_name_style = COMMAND_TARGETNAME_RAW;
|
|
snprintf(info->target_name, info->target_name_maxlength, "%s", pTarget->GetName());
|
|
return true;
|
|
}
|
|
else if (strcmp(info->pattern, "@spec") == 0)
|
|
{
|
|
const char *teamname = tools_GetTeamName(1);
|
|
if (strcasecmp(teamname, "spectator") != 0)
|
|
return false;
|
|
info->num_targets = 0;
|
|
for (int i = 1; i <= playerhelpers->GetMaxClients(); i++)
|
|
{
|
|
IGamePlayer *player = playerhelpers->GetGamePlayer(i);
|
|
if (player == NULL || !player->IsInGame())
|
|
continue;
|
|
IPlayerInfo *plinfo = player->GetPlayerInfo();
|
|
if (plinfo == NULL)
|
|
continue;
|
|
if (plinfo->GetTeamIndex() == 1 &&
|
|
playerhelpers->FilterCommandTarget(pAdmin, player, info->flags) ==
|
|
COMMAND_TARGET_VALID)
|
|
{
|
|
info->targets[info->num_targets++] = i;
|
|
}
|
|
}
|
|
info->reason = info->num_targets > 0 ? COMMAND_TARGET_VALID : COMMAND_TARGET_EMPTY_FILTER;
|
|
info->target_name_style = COMMAND_TARGETNAME_ML;
|
|
snprintf(info->target_name, info->target_name_maxlength, "all spectators");
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
const char *SDKTools::GetExtensionVerString()
|
|
{
|
|
return SM_FULL_VERSION;
|
|
}
|
|
|
|
const char *SDKTools::GetExtensionDateString()
|
|
{
|
|
return SM_BUILD_TIMESTAMP;
|
|
}
|
|
|
|
void SDKTools::OnClientPutInServer(int client)
|
|
{
|
|
g_Hooks.OnClientPutInServer(client);
|
|
}
|
|
|
|
class SDKTools_API : public ISDKTools
|
|
{
|
|
public:
|
|
virtual const char *GetInterfaceName()
|
|
{
|
|
return SMINTERFACE_SDKTOOLS_NAME;
|
|
}
|
|
|
|
virtual unsigned int GetInterfaceVersion()
|
|
{
|
|
return SMINTERFACE_SDKTOOLS_VERSION;
|
|
}
|
|
|
|
virtual IServer *GetIServer()
|
|
{
|
|
return iserver;
|
|
}
|
|
|
|
virtual void *GetGameRules()
|
|
{
|
|
if (!g_pGameRules)
|
|
return NULL;
|
|
|
|
return *g_pGameRules;
|
|
}
|
|
} g_SDKTools_API;
|
|
|
|
static void InitSDKToolsAPI()
|
|
{
|
|
g_pSDKTools = &g_SDKTools_API;
|
|
sharesys->AddInterface(myself, g_pSDKTools);
|
|
}
|