1
0
mirror of https://github.com/alliedmodders/metamod-source.git synced 2025-01-19 08:52:34 +01:00
HLMetaModOfficial/loader/serverplugin.cpp
2008-11-16 03:47:58 -06:00

256 lines
5.7 KiB
C++

#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stddef.h>
#include "loader.h"
#include <sh_memfuncinfo.h>
#include <sh_memory.h>
#include "serverplugin.h"
typedef enum
{
PLUGIN_CONTINUE = 0,
PLUGIN_OVERRIDE,
PLUGIN_STOP,
} PLUGIN_RESULT;
typedef enum
{
eQueryCvarValueStatus_ValueIntact=0,
eQueryCvarValueStatus_CvarNotFound=1,
eQueryCvarValueStatus_NotACvar=2,
eQueryCvarValueStatus_CvarProtected=3
} EQueryCvarValueStatus;
typedef int QueryCvarCookie_t;
class CCommand;
class IServerPluginCallbacks;
struct edict_t;
#if defined WIN32
#define LIBRARY_EXT ".dll"
#else
#define LIBRARY_EXT "_i486.so"
#endif
class IRandomThings
{
public:
virtual PLUGIN_RESULT ClientCommand(edict_t *pEntity, const CCommand& args)
{
return PLUGIN_CONTINUE;
}
};
/**
* The vtable must match the general layout for ISPC. We modify the vtable
* based on what we get back.
*/
class ServerPlugin
{
IVspBridge* bridge;
unsigned int vsp_version;
bool load_allowed;
public:
ServerPlugin()
{
load_allowed = false;
}
virtual bool Load(QueryValveInterface engineFactory, QueryValveInterface gsFactory)
{
MetamodBackend backend = MMBackend_UNKNOWN;
if (!load_allowed)
return false;
load_allowed = false;
/* Check for L4D */
if (engineFactory("VEngineServer022", NULL) != NULL &&
engineFactory("VEngineCvar007", NULL) != NULL)
{
backend = MMBackend_Left4Dead;
}
else if (engineFactory("VEngineServer021", NULL) != NULL)
{
/* Check for OB */
if (engineFactory("VEngineCvar004", NULL) != NULL &&
engineFactory("VModelInfoServer002", NULL) != NULL)
{
backend = MMBackend_Episode2;
}
/* Check for EP1 */
else if (engineFactory("VModelInfoServer001", NULL) != NULL &&
(engineFactory("VEngineCvar003", NULL) != NULL ||
engineFactory("VEngineCvar002", NULL) != NULL))
{
backend = MMBackend_Episode1;
}
}
if (backend == MMBackend_UNKNOWN)
{
mm_LogFatal("Could not detect engine version, spewing stats:");
return false;
}
else if (backend >= MMBackend_Episode2)
{
/* We need to insert the right type of call into this vtable */
void **vtable_src;
void **vtable_dest;
IRandomThings sample;
SourceHook::MemFuncInfo mfp_dest, mfp_src;
mfp_dest.isVirtual = false;
mfp_src.isVirtual = false;
SourceHook::GetFuncInfo(&ServerPlugin::ClientCommand, mfp_dest);
SourceHook::GetFuncInfo(&IRandomThings::ClientCommand, mfp_src);
assert(mfp_dest.isVirtual);
assert(mfp_dest.thisptroffs == 0);
assert(mfp_dest.vtbloffs == 0);
assert(mfp_src.isVirtual);
assert(mfp_src.thisptroffs == 0);
assert(mfp_src.vtbloffs == 0);
vtable_src = (void **)*(void **)&sample;
vtable_dest = (void **)*(void **)this;
SourceHook::SetMemAccess(&vtable_dest[mfp_dest.vtblindex],
sizeof(void*),
SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC);
vtable_dest[mfp_dest.vtblindex] = vtable_src[mfp_src.vtblindex];
}
char error[255];
if (!mm_LoadMetamodLibrary(backend, error, sizeof(error)))
{
mm_LogFatal("Detected engine %d but could not load: %s", backend, error);
return false;
}
typedef IVspBridge *(*GetVspBridge)();
GetVspBridge get_bridge = (GetVspBridge)mm_GetProcAddress("GetVspBridge");
if (get_bridge == NULL)
{
mm_UnloadMetamodLibrary();
mm_LogFatal("Detected engine %d but could not find GetVspBridge callback.", backend);
return false;
}
bridge = get_bridge();
vsp_bridge_info info;
info.engineFactory = engineFactory;
info.gsFactory = gsFactory;
info.vsp_callbacks = (IServerPluginCallbacks*)this;
info.vsp_version = vsp_version;
strcpy(error, "Unknown error");
if (!bridge->Load(&info, error, sizeof(error)))
{
mm_UnloadMetamodLibrary();
mm_LogFatal("Unknown error loading Metamod for engine %d: %s", backend, error);
return false;
}
return true;
}
virtual void Unload()
{
if (bridge == NULL)
return;
bridge->Unload();
}
virtual void Pause()
{
}
virtual void UnPause()
{
}
virtual const char *GetPluginDescription()
{
if (bridge == NULL)
return "Metamod:Source Loader Shim";
return bridge->GetDescription();
}
virtual void LevelInit(char const *pMapName)
{
}
virtual void ServerActivate(edict_t *pEdictList, int edictCount, int clientMax)
{
}
virtual void GameFrame(bool simulating)
{
}
virtual void LevelShutdown()
{
}
virtual void ClientActive(edict_t *pEntity)
{
}
virtual void ClientDisconnect(edict_t *pEntity)
{
}
virtual void ClientPutInServer(edict_t *pEntity, char const *playername)
{
}
virtual void SetCommandClient(int index)
{
}
virtual void ClientSettingsChanged(edict_t *pEdict)
{
}
virtual PLUGIN_RESULT ClientConnect(bool *bAllowConnect,
edict_t *pEntity,
const char *pszName,
const char *pszAddress,
char *reject,
int maxrejectlen)
{
return PLUGIN_CONTINUE;
}
virtual PLUGIN_RESULT ClientCommand(edict_t *pEntity)
{
return PLUGIN_CONTINUE;
}
virtual PLUGIN_RESULT NetworkIDValidated(const char *pszUserName, const char *pszNetworkID)
{
return PLUGIN_CONTINUE;
}
virtual void OnQueryCvarValueFinished(QueryCvarCookie_t iCookie,
edict_t *pPlayerEntity,
EQueryCvarValueStatus eStatus,
const char *pCvarName,
const char *pCvarValue)
{
}
void PrepForLoad(unsigned int version)
{
vsp_version = version;
load_allowed = true;
}
bool IsLoaded()
{
return bridge != NULL;
}
};
ServerPlugin mm_vsp_callbacks;
void *mm_GetVspCallbacks(unsigned int version)
{
if (mm_vsp_callbacks.IsLoaded())
return NULL;
/* Only support versions 1 or 2 right now */
if (version > 2)
return NULL;
mm_vsp_callbacks.PrepForLoad(version);
return &mm_vsp_callbacks;
}