mirror of
https://github.com/rehlds/metamod-r.git
synced 2025-01-19 10:08:21 +03:00
0118e05280
Prefer Sys_GetModuleHandle over Sys_LoadModule to avoid unnecessary reference count increase
245 lines
7.1 KiB
C++
245 lines
7.1 KiB
C++
/*
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation; either version 2 of the License, or (at
|
|
* your option) any later version.
|
|
*
|
|
* 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, write to the Free Software Foundation,
|
|
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
* In addition, as a special exception, the author gives permission to
|
|
* link the code of this program with the Half-Life Game Engine ("HL
|
|
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
|
|
* L.L.C ("Valve"). You must obey the GNU General Public License in all
|
|
* respects for all of the code used other than the HL Engine and MODs
|
|
* from Valve. If you modify this file, you may extend this exception
|
|
* to your version of the file, but you are not obligated to do so. If
|
|
* you do not wish to do so, delete this exception statement from your
|
|
* version.
|
|
*
|
|
*/
|
|
|
|
#include "interface.h"
|
|
|
|
#ifdef _WIN32
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#include "windows.h"
|
|
#endif // _WIN32
|
|
|
|
// InterfaceReg
|
|
InterfaceReg *InterfaceReg::s_pInterfaceRegs = nullptr;
|
|
|
|
InterfaceReg::InterfaceReg(InstantiateInterfaceFn fn, const char *pName) : m_pName(pName)
|
|
{
|
|
m_CreateFn = fn;
|
|
m_pNext = s_pInterfaceRegs;
|
|
s_pInterfaceRegs = this;
|
|
}
|
|
|
|
// This is the primary exported function by a dll, referenced by name via dynamic binding
|
|
// that exposes an opqaue function pointer to the interface.
|
|
//
|
|
// We have the Internal variant so Sys_GetFactoryThis() returns the correct internal
|
|
// symbol under GCC/Linux/Mac as CreateInterface is DLL_EXPORT so its global so the loaders
|
|
// on those OS's pick exactly 1 of the CreateInterface symbols to be the one that is process wide and
|
|
// all Sys_GetFactoryThis() calls find that one, which doesn't work. Using the internal walkthrough here
|
|
// makes sure Sys_GetFactoryThis() has the dll specific symbol and GetProcAddress() returns the module specific
|
|
// function for CreateInterface again getting the dll specific symbol we need.
|
|
EXPORT_FUNCTION IBaseInterface *CreateInterface(const char *pName, int *pReturnCode)
|
|
{
|
|
InterfaceReg *pCur;
|
|
for (pCur = InterfaceReg::s_pInterfaceRegs; pCur; pCur = pCur->m_pNext)
|
|
{
|
|
if (strcmp(pCur->m_pName, pName) == 0)
|
|
{
|
|
if (pReturnCode)
|
|
{
|
|
*pReturnCode = IFACE_OK;
|
|
}
|
|
|
|
return pCur->m_CreateFn();
|
|
}
|
|
}
|
|
|
|
if (pReturnCode)
|
|
{
|
|
*pReturnCode = IFACE_FAILED;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
// Linux doesn't have this function so this emulates its functionality
|
|
void *GetModuleHandle(const char *name)
|
|
{
|
|
void *handle;
|
|
if (name == nullptr)
|
|
{
|
|
// hmm, how can this be handled under linux....
|
|
// is it even needed?
|
|
return nullptr;
|
|
}
|
|
|
|
if ((handle = dlopen(name, RTLD_NOW)) == nullptr)
|
|
{
|
|
//printf("Error:%s\n",dlerror());
|
|
// couldn't open this file
|
|
return nullptr;
|
|
}
|
|
|
|
// read "man dlopen" for details
|
|
// in short dlopen() inc a ref count
|
|
// so dec the ref count by performing the close
|
|
dlclose(handle);
|
|
return handle;
|
|
}
|
|
#endif // _WIN32
|
|
|
|
// Purpose: returns a pointer to a function, given a module
|
|
// Input : pModuleName - module name
|
|
// *pName - proc name
|
|
//static hlds_run wants to use this function
|
|
void *Sys_GetProcAddress(const char *pModuleName, const char *pName)
|
|
{
|
|
return GetProcAddress(GetModuleHandle(pModuleName), pName);
|
|
}
|
|
|
|
// Purpose: returns a pointer to a function, given a module
|
|
// Input : pModuleName - module name
|
|
// *pName - proc name
|
|
// hlds_run wants to use this function
|
|
void *Sys_GetProcAddress(void *pModuleHandle, const char *pName)
|
|
{
|
|
return GetProcAddress((HMODULE)pModuleHandle, pName);
|
|
}
|
|
|
|
// 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
|
|
// Input : *pModuleName - filename of the component
|
|
// Output : opaque handle to the module (hides system dependency)
|
|
CSysModule *Sys_LoadModule(const char *pModuleName)
|
|
{
|
|
#ifdef _WIN32
|
|
HMODULE hDLL = LoadLibrary(pModuleName);
|
|
#else
|
|
HMODULE hDLL = nullptr;
|
|
char szAbsoluteModuleName[1024];
|
|
if (pModuleName[0] != '/')
|
|
{
|
|
char szCwd[1024];
|
|
getcwd(szCwd, sizeof(szCwd));
|
|
if (szCwd[strlen(szCwd) - 1] == '/')
|
|
szCwd[strlen(szCwd) - 1] = '\0';
|
|
|
|
_snprintf(szAbsoluteModuleName, sizeof(szAbsoluteModuleName), "%s/%s", szCwd, pModuleName);
|
|
hDLL = dlopen(szAbsoluteModuleName, RTLD_NOW);
|
|
}
|
|
else
|
|
{
|
|
_snprintf(szAbsoluteModuleName, sizeof(szAbsoluteModuleName), "%s", pModuleName);
|
|
hDLL = dlopen(pModuleName, RTLD_NOW);
|
|
}
|
|
#endif // _WIN32
|
|
|
|
if (!hDLL)
|
|
{
|
|
char str[512];
|
|
|
|
#if defined(_WIN32)
|
|
_snprintf(str, sizeof(str), "%s.dll", pModuleName);
|
|
hDLL = LoadLibrary(str);
|
|
#elif defined(OSX)
|
|
printf("Error: %s\n", dlerror());
|
|
_snprintf(str, sizeof(str), "%s.dylib", szAbsoluteModuleName);
|
|
hDLL = dlopen(str, RTLD_NOW);
|
|
#else
|
|
printf("Error: %s\n", dlerror());
|
|
_snprintf(str, sizeof(str), "%s.so", szAbsoluteModuleName);
|
|
hDLL = dlopen(str, RTLD_NOW);
|
|
#endif
|
|
}
|
|
|
|
return reinterpret_cast<CSysModule *>(hDLL);
|
|
}
|
|
|
|
// Purpose: Unloads a DLL/component from
|
|
// Input : *pModuleName - filename of the component
|
|
// Output : opaque handle to the module (hides system dependency)
|
|
void Sys_UnloadModule(CSysModule *pModule)
|
|
{
|
|
if (!pModule)
|
|
return;
|
|
|
|
HMODULE hDLL = reinterpret_cast<HMODULE>(pModule);
|
|
|
|
#ifdef _WIN32
|
|
FreeLibrary(hDLL);
|
|
#else
|
|
dlclose(hDLL);
|
|
#endif // _WIN32
|
|
}
|
|
|
|
// Purpose: returns a pointer to a function, given a module
|
|
// Input : module - windows HMODULE from Sys_LoadModule()
|
|
// *pName - proc name
|
|
// Output : factory for this module
|
|
CreateInterfaceFn Sys_GetFactory(CSysModule *pModule)
|
|
{
|
|
if (!pModule)
|
|
return nullptr;
|
|
|
|
return reinterpret_cast<CreateInterfaceFn>(Sys_GetProcAddress(pModule, CREATEINTERFACE_PROCNAME));
|
|
}
|
|
|
|
// Purpose: returns the instance of this module
|
|
// Output : CreateInterfaceFn
|
|
CreateInterfaceFn Sys_GetFactoryThis()
|
|
{
|
|
return CreateInterface;
|
|
}
|
|
|
|
// Purpose: returns the instance of the named module
|
|
// Input : *pModuleName - name of the module
|
|
// Output : CreateInterfaceFn - instance of that module
|
|
CreateInterfaceFn Sys_GetFactory(const char *pModuleName)
|
|
{
|
|
return reinterpret_cast<CreateInterfaceFn>(Sys_GetProcAddress(pModuleName, CREATEINTERFACE_PROCNAME));
|
|
}
|
|
|
|
// Purpose: finds a particular interface in the factory set
|
|
void *InitializeInterface(char const *interfaceName, CreateInterfaceFn *factoryList, int numFactories)
|
|
{
|
|
void *retval;
|
|
|
|
for (int i = 0; i < numFactories; i++)
|
|
{
|
|
CreateInterfaceFn factory = factoryList[ i ];
|
|
if (!factory)
|
|
continue;
|
|
|
|
retval = factory(interfaceName, nullptr);
|
|
if (retval)
|
|
return retval;
|
|
}
|
|
|
|
// No provider for requested interface!!!
|
|
// assert(!"No provider for requested interface!!!");
|
|
|
|
return nullptr;
|
|
}
|