mirror of
https://github.com/rehlds/resemiclip.git
synced 2025-01-12 22:58:18 +03:00
Improved initialization of rehlds api
Prefer Sys_GetModuleHandle over Sys_LoadModule to avoid unnecessary reference count increase
This commit is contained in:
parent
33c4062e31
commit
8cae39a981
@ -64,7 +64,7 @@ EXPORT_FUNCTION IBaseInterface *CreateInterface( const char *pName, int *pReturn
|
|||||||
return pCur->m_CreateFn();
|
return pCur->m_CreateFn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( pReturnCode )
|
if ( pReturnCode )
|
||||||
{
|
{
|
||||||
*pReturnCode = IFACE_FAILED;
|
*pReturnCode = IFACE_FAILED;
|
||||||
@ -76,7 +76,7 @@ EXPORT_FUNCTION IBaseInterface *CreateInterface( const char *pName, int *pReturn
|
|||||||
static IBaseInterface *CreateInterfaceLocal( const char *pName, int *pReturnCode )
|
static IBaseInterface *CreateInterfaceLocal( const char *pName, int *pReturnCode )
|
||||||
{
|
{
|
||||||
InterfaceReg *pCur;
|
InterfaceReg *pCur;
|
||||||
|
|
||||||
for(pCur=InterfaceReg::s_pInterfaceRegs; pCur; pCur=pCur->m_pNext)
|
for(pCur=InterfaceReg::s_pInterfaceRegs; pCur; pCur=pCur->m_pNext)
|
||||||
{
|
{
|
||||||
if(strcmp(pCur->m_pName, pName) == 0)
|
if(strcmp(pCur->m_pName, pName) == 0)
|
||||||
@ -88,12 +88,12 @@ static IBaseInterface *CreateInterfaceLocal( const char *pName, int *pReturnCode
|
|||||||
return pCur->m_CreateFn();
|
return pCur->m_CreateFn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( pReturnCode )
|
if ( pReturnCode )
|
||||||
{
|
{
|
||||||
*pReturnCode = IFACE_FAILED;
|
*pReturnCode = IFACE_FAILED;
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -107,7 +107,7 @@ static IBaseInterface *CreateInterfaceLocal( const char *pName, int *pReturnCode
|
|||||||
// Input : pModuleName - module name
|
// Input : pModuleName - module name
|
||||||
// *pName - proc name
|
// *pName - proc name
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
//static hlds_run wants to use this function
|
//static hlds_run wants to use this function
|
||||||
void *Sys_GetProcAddress( const char *pModuleName, const char *pName )
|
void *Sys_GetProcAddress( const char *pModuleName, const char *pName )
|
||||||
{
|
{
|
||||||
return GetProcAddress( GetModuleHandle(pModuleName), pName );
|
return GetProcAddress( GetModuleHandle(pModuleName), pName );
|
||||||
@ -118,7 +118,7 @@ void *Sys_GetProcAddress( const char *pModuleName, const char *pName )
|
|||||||
// Input : pModuleName - module name
|
// Input : pModuleName - module name
|
||||||
// *pName - proc name
|
// *pName - proc name
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// hlds_run wants to use this function
|
// hlds_run wants to use this function
|
||||||
void *Sys_GetProcAddress( void *pModuleHandle, const char *pName )
|
void *Sys_GetProcAddress( void *pModuleHandle, const char *pName )
|
||||||
{
|
{
|
||||||
#if defined ( _WIN32 )
|
#if defined ( _WIN32 )
|
||||||
@ -128,6 +128,14 @@ void *Sys_GetProcAddress( void *pModuleHandle, const char *pName )
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Purpose: Returns a module handle by its name.
|
||||||
|
// Input : pModuleName - module name
|
||||||
|
// Output : the module handle or NULL in case of an error
|
||||||
|
CSysModule *Sys_GetModuleHandle(const char *pModuleName)
|
||||||
|
{
|
||||||
|
return reinterpret_cast<CSysModule *>(GetModuleHandle(pModuleName));
|
||||||
|
}
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// Purpose: Loads a DLL/component from disk and returns a handle to it
|
// Purpose: Loads a DLL/component from disk and returns a handle to it
|
||||||
// Input : *pModuleName - filename of the component
|
// Input : *pModuleName - filename of the component
|
||||||
@ -170,7 +178,7 @@ CSysModule *Sys_LoadModule( const char *pModuleName )
|
|||||||
#elif defined(OSX)
|
#elif defined(OSX)
|
||||||
printf("Error:%s\n",dlerror());
|
printf("Error:%s\n",dlerror());
|
||||||
_snprintf( str, sizeof(str), "%s.dylib", szAbsoluteModuleName );
|
_snprintf( str, sizeof(str), "%s.dylib", szAbsoluteModuleName );
|
||||||
hDLL = dlopen(str, RTLD_NOW);
|
hDLL = dlopen(str, RTLD_NOW);
|
||||||
#else
|
#else
|
||||||
printf("Error:%s\n",dlerror());
|
printf("Error:%s\n",dlerror());
|
||||||
_snprintf( str, sizeof(str), "%s.so", szAbsoluteModuleName );
|
_snprintf( str, sizeof(str), "%s.so", szAbsoluteModuleName );
|
||||||
@ -202,7 +210,7 @@ void Sys_UnloadModule( CSysModule *pModule )
|
|||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// Purpose: returns a pointer to a function, given a module
|
// Purpose: returns a pointer to a function, given a module
|
||||||
// Input : module - windows HMODULE from Sys_LoadModule()
|
// Input : module - windows HMODULE from Sys_LoadModule()
|
||||||
// *pName - proc name
|
// *pName - proc name
|
||||||
// Output : factory for this module
|
// Output : factory for this module
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
@ -216,9 +224,9 @@ CreateInterfaceFn Sys_GetFactory( CSysModule *pModule )
|
|||||||
return reinterpret_cast<CreateInterfaceFn>(GetProcAddress( hDLL, CREATEINTERFACE_PROCNAME ));
|
return reinterpret_cast<CreateInterfaceFn>(GetProcAddress( hDLL, CREATEINTERFACE_PROCNAME ));
|
||||||
#else
|
#else
|
||||||
// Linux gives this error:
|
// Linux gives this error:
|
||||||
//../public/interface.cpp: In function `IBaseInterface *(*Sys_GetFactory
|
//../public/interface.cpp: In function `IBaseInterface *(*Sys_GetFactory
|
||||||
//(CSysModule *)) (const char *, int *)':
|
//(CSysModule *)) (const char *, int *)':
|
||||||
//../public/interface.cpp:154: ISO C++ forbids casting between
|
//../public/interface.cpp:154: ISO C++ forbids casting between
|
||||||
//pointer-to-function and pointer-to-object
|
//pointer-to-function and pointer-to-object
|
||||||
//
|
//
|
||||||
// so lets get around it :)
|
// so lets get around it :)
|
||||||
@ -252,9 +260,9 @@ CreateInterfaceFn Sys_GetFactory( const char *pModuleName )
|
|||||||
return static_cast<CreateInterfaceFn>( Sys_GetProcAddress( pModuleName, CREATEINTERFACE_PROCNAME ) );
|
return static_cast<CreateInterfaceFn>( Sys_GetProcAddress( pModuleName, CREATEINTERFACE_PROCNAME ) );
|
||||||
#else
|
#else
|
||||||
// Linux gives this error:
|
// Linux gives this error:
|
||||||
//../public/interface.cpp: In function `IBaseInterface *(*Sys_GetFactory
|
//../public/interface.cpp: In function `IBaseInterface *(*Sys_GetFactory
|
||||||
//(const char *)) (const char *, int *)':
|
//(const char *)) (const char *, int *)':
|
||||||
//../public/interface.cpp:186: invalid static_cast from type `void *' to
|
//../public/interface.cpp:186: invalid static_cast from type `void *' to
|
||||||
//type `IBaseInterface *(*) (const char *, int *)'
|
//type `IBaseInterface *(*) (const char *, int *)'
|
||||||
//
|
//
|
||||||
// so lets use the old style cast.
|
// so lets use the old style cast.
|
||||||
|
@ -8,10 +8,10 @@
|
|||||||
|
|
||||||
// Versioning
|
// Versioning
|
||||||
// There are two versioning cases that are handled by this:
|
// There are two versioning cases that are handled by this:
|
||||||
// 1. You add functions to the end of an interface, so it is binary compatible with the previous interface. In this case,
|
// 1. You add functions to the end of an interface, so it is binary compatible with the previous interface. In this case,
|
||||||
// you need two EXPOSE_INTERFACEs: one to expose your class as the old interface and one to expose it as the new interface.
|
// you need two EXPOSE_INTERFACEs: one to expose your class as the old interface and one to expose it as the new interface.
|
||||||
// 2. You update an interface so it's not compatible anymore (but you still want to be able to expose the old interface
|
// 2. You update an interface so it's not compatible anymore (but you still want to be able to expose the old interface
|
||||||
// for legacy code). In this case, you need to make a new version name for your new interface, and make a wrapper interface and
|
// for legacy code). In this case, you need to make a new version name for your new interface, and make a wrapper interface and
|
||||||
// expose it for the old interface.
|
// expose it for the old interface.
|
||||||
|
|
||||||
//#if _MSC_VER >= 1300 // VC7
|
//#if _MSC_VER >= 1300 // VC7
|
||||||
@ -108,7 +108,7 @@ public:
|
|||||||
// This function is automatically exported and allows you to access any interfaces exposed with the above macros.
|
// This function is automatically exported and allows you to access any interfaces exposed with the above macros.
|
||||||
// if pReturnCode is set, it will return one of the following values
|
// if pReturnCode is set, it will return one of the following values
|
||||||
// extend this for other error conditions/code
|
// extend this for other error conditions/code
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
IFACE_OK = 0,
|
IFACE_OK = 0,
|
||||||
IFACE_FAILED
|
IFACE_FAILED
|
||||||
@ -133,6 +133,8 @@ extern CreateInterfaceFn Sys_GetFactory( const char *pModuleName );
|
|||||||
// load/unload components
|
// load/unload components
|
||||||
class CSysModule;
|
class CSysModule;
|
||||||
|
|
||||||
|
extern CSysModule *Sys_GetModuleHandle(const char *pModuleName);
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// Load & Unload should be called in exactly one place for each module
|
// Load & Unload should be called in exactly one place for each module
|
||||||
// The factory for that module should be passed on to dependent components for
|
// The factory for that module should be passed on to dependent components for
|
||||||
|
@ -3,8 +3,11 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
extern IRehldsApi* g_RehldsApi;
|
extern IRehldsApi *g_RehldsApi;
|
||||||
extern IRehldsHookchains* g_RehldsHookchains;
|
extern const RehldsFuncs_t *g_RehldsFuncs;
|
||||||
|
extern IRehldsServerData *g_RehldsData;
|
||||||
|
extern IRehldsHookchains *g_RehldsHookchains;
|
||||||
|
extern IRehldsServerStatic *g_RehldsSvs;
|
||||||
extern bool RehldsApi_Init();
|
extern bool RehldsApi_Init();
|
||||||
|
|
||||||
typedef struct packet_entities_s {
|
typedef struct packet_entities_s {
|
||||||
|
@ -1,48 +1,63 @@
|
|||||||
#include "precompiled.h"
|
#include "precompiled.h"
|
||||||
|
|
||||||
IRehldsApi* g_RehldsApi;
|
IRehldsApi *g_RehldsApi;
|
||||||
IRehldsHookchains* g_RehldsHookchains;
|
const RehldsFuncs_t *g_RehldsFuncs;
|
||||||
|
IRehldsServerData *g_RehldsData;
|
||||||
|
IRehldsHookchains *g_RehldsHookchains;
|
||||||
|
IRehldsServerStatic *g_RehldsSvs;
|
||||||
|
|
||||||
bool RehldsApi_TryInit(CSysModule* engineModule, char* failureReason)
|
bool RehldsApi_Init()
|
||||||
{
|
{
|
||||||
if (!engineModule) {
|
#ifdef WIN32
|
||||||
SEM_PRINT("[%s]: ReHLDS Failed to locate engine module.", Plugin_info.logtag);
|
// Find the most appropriate module handle from a list of DLL candidates
|
||||||
return false;
|
// Notes:
|
||||||
|
// - "swds.dll" is the library Dedicated Server
|
||||||
|
//
|
||||||
|
// Let's also attempt to locate the ReHLDS API in the client's library
|
||||||
|
// - "sw.dll" is the client library for Software render, with a built-in listenserver
|
||||||
|
// - "hw.dll" is the client library for Hardware render, with a built-in listenserver
|
||||||
|
const char *dllNames[] = { "swds.dll", "sw.dll", "hw.dll" }; // List of DLL candidates to lookup for the ReHLDS API
|
||||||
|
CSysModule *engineModule = NULL; // The module handle of the selected DLL
|
||||||
|
for (const char *dllName : dllNames)
|
||||||
|
{
|
||||||
|
if (engineModule = Sys_GetModuleHandle(dllName))
|
||||||
|
break; // gotcha
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
CSysModule *engineModule = Sys_GetModuleHandle("engine_i486.so");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (!engineModule)
|
||||||
|
return false;
|
||||||
|
|
||||||
CreateInterfaceFn ifaceFactory = Sys_GetFactory(engineModule);
|
CreateInterfaceFn ifaceFactory = Sys_GetFactory(engineModule);
|
||||||
|
if (!ifaceFactory)
|
||||||
if (!ifaceFactory) {
|
|
||||||
SEM_PRINT("[%s]: ReHLDS Failed to locate interface factory in engine module.", Plugin_info.logtag);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
int retCode = 0;
|
int retCode = 0;
|
||||||
|
|
||||||
g_RehldsApi = (IRehldsApi*)ifaceFactory(VREHLDS_HLDS_API_VERSION, &retCode);
|
g_RehldsApi = (IRehldsApi*)ifaceFactory(VREHLDS_HLDS_API_VERSION, &retCode);
|
||||||
|
|
||||||
if (!g_RehldsApi) {
|
if (!g_RehldsApi)
|
||||||
SEM_PRINT("[%s]: ReHLDS Failed to locate retrieve rehlds api interface from engine module, return code is %d.", Plugin_info.logtag, retCode);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
int majorVersion = g_RehldsApi->GetMajorVersion();
|
int majorVersion = g_RehldsApi->GetMajorVersion();
|
||||||
int minorVersion = g_RehldsApi->GetMinorVersion();
|
int minorVersion = g_RehldsApi->GetMinorVersion();
|
||||||
|
|
||||||
if (majorVersion != REHLDS_API_VERSION_MAJOR)
|
if (majorVersion != REHLDS_API_VERSION_MAJOR)
|
||||||
{
|
{
|
||||||
SEM_PRINT("[%s]: ReHLDS API major version mismatch; expected %d, real %d.", Plugin_info.logtag, REHLDS_API_VERSION_MAJOR, majorVersion);
|
SEM_PRINT("[%s]: ReHLDS API major version mismatch; expected %d, real %d\n", Plugin_info.logtag, REHLDS_API_VERSION_MAJOR, majorVersion);
|
||||||
|
|
||||||
// need to notify that it is necessary to update the ReHLDS.
|
// need to notify that it is necessary to update the ReHLDS.
|
||||||
if (majorVersion < REHLDS_API_VERSION_MAJOR)
|
if (majorVersion < REHLDS_API_VERSION_MAJOR)
|
||||||
{
|
{
|
||||||
SEM_PRINT("[%s]: Please update the ReHLDS up to a major version API >= %d.", Plugin_info.logtag, REHLDS_API_VERSION_MAJOR);
|
SEM_PRINT("[%s]: Please update the ReHLDS up to a major version API >= %d\n", Plugin_info.logtag, REHLDS_API_VERSION_MAJOR);
|
||||||
}
|
}
|
||||||
|
|
||||||
// need to notify that it is necessary to update the module.
|
// need to notify that it is necessary to update the module.
|
||||||
else if (majorVersion > REHLDS_API_VERSION_MAJOR)
|
else if (majorVersion > REHLDS_API_VERSION_MAJOR)
|
||||||
{
|
{
|
||||||
SEM_PRINT("[%s]: Please update the %s up to a major version API >= %d.", Plugin_info.logtag, Plugin_info.name, majorVersion);
|
SEM_PRINT("[%s]: Please update the %s up to a major version API >= %d\n", Plugin_info.logtag, Plugin_info.logtag, majorVersion);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
@ -50,39 +65,15 @@ bool RehldsApi_TryInit(CSysModule* engineModule, char* failureReason)
|
|||||||
|
|
||||||
if (minorVersion < REHLDS_API_VERSION_MINOR)
|
if (minorVersion < REHLDS_API_VERSION_MINOR)
|
||||||
{
|
{
|
||||||
SEM_PRINT("[%s]: ReHLDS API minor version mismatch; expected at least %d, real %d.", Plugin_info.logtag, REHLDS_API_VERSION_MINOR, minorVersion);
|
SEM_PRINT("[%s]: ReHLDS API minor version mismatch; expected at least %d, real %d\n", Plugin_info.logtag, REHLDS_API_VERSION_MINOR, minorVersion);
|
||||||
SEM_PRINT("[%s]: Please update the ReHLDS up to a minor version API >= %d.", Plugin_info.logtag, REHLDS_API_VERSION_MINOR);
|
SEM_PRINT("[%s]: Please update the ReHLDS up to a minor version API >= %d\n", Plugin_info.logtag, REHLDS_API_VERSION_MINOR);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
g_RehldsFuncs = g_RehldsApi->GetFuncs();
|
||||||
|
g_RehldsData = g_RehldsApi->GetServerData();
|
||||||
g_RehldsHookchains = g_RehldsApi->GetHookchains();
|
g_RehldsHookchains = g_RehldsApi->GetHookchains();
|
||||||
|
g_RehldsSvs = g_RehldsApi->GetServerStatic();
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool RehldsApi_Init()
|
|
||||||
{
|
|
||||||
char failReason[2048];
|
|
||||||
|
|
||||||
#ifdef WIN32
|
|
||||||
CSysModule* engineModule = Sys_LoadModule("swds.dll");
|
|
||||||
if (!RehldsApi_TryInit(engineModule, failReason))
|
|
||||||
{
|
|
||||||
engineModule = Sys_LoadModule("filesystem_stdio.dll");
|
|
||||||
if (!RehldsApi_TryInit(engineModule, failReason))
|
|
||||||
{
|
|
||||||
UTIL_LogPrintf("%s", failReason);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
CSysModule* engineModule = Sys_LoadModule("engine_i486.so");
|
|
||||||
if (!RehldsApi_TryInit(engineModule, failReason))
|
|
||||||
{
|
|
||||||
UTIL_LogPrintf("%s", failReason);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user