148 lines
3.3 KiB
C++
148 lines
3.3 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Shared library loading and symbol lookup.
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
|
|
#include "pch_tier0.h"
|
|
#include "tier0/dynfunction.h"
|
|
|
|
#if defined(WIN32)
|
|
typedef HMODULE LibraryHandle;
|
|
#define LoadLibraryHandle(libname) LoadLibrary(libname)
|
|
#define CloseLibraryHandle(handle) FreeLibrary(handle)
|
|
#define LookupInLibraryHandle(handle, fn) GetProcAddress(handle, fn)
|
|
#elif defined(POSIX)
|
|
#include <dlfcn.h>
|
|
typedef void *LibraryHandle;
|
|
#define LoadLibraryHandle(libname) dlopen(libname, RTLD_NOW)
|
|
#define CloseLibraryHandle(handle) dlclose(handle)
|
|
#define LookupInLibraryHandle(handle, fn) dlsym(handle, fn)
|
|
#else
|
|
#error Please define your platform.
|
|
#endif
|
|
|
|
#if 1
|
|
static inline void dbgdynfn(const char *fmt, ...) {}
|
|
#else
|
|
#define dbgdynfn printf
|
|
#endif
|
|
|
|
// NOTE: This has to be the last file included!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
class CSharedLibraryCache
|
|
{
|
|
public:
|
|
static CSharedLibraryCache &GetCache()
|
|
{
|
|
static CSharedLibraryCache Singleton;
|
|
return Singleton;
|
|
}
|
|
|
|
struct CSharedLibraryItem
|
|
{
|
|
CSharedLibraryItem(LibraryHandle handle, const char *name)
|
|
{
|
|
m_handle = handle;
|
|
m_name = new char[strlen(name) + 1];
|
|
m_next = NULL;
|
|
strcpy(m_name, name);
|
|
}
|
|
|
|
~CSharedLibraryItem()
|
|
{
|
|
dbgdynfn("CDynamicFunction: Closing library '%s' (%p)\n", m_name, (void *) m_handle);
|
|
CloseLibraryHandle(m_handle);
|
|
delete[] m_name;
|
|
delete m_next;
|
|
}
|
|
|
|
char *m_name;
|
|
CSharedLibraryItem *m_next;
|
|
LibraryHandle m_handle;
|
|
};
|
|
|
|
CSharedLibraryCache() : m_pList(NULL) {}
|
|
~CSharedLibraryCache() { CloseAllLibraries(); }
|
|
|
|
LibraryHandle GetHandle(const char *name)
|
|
{
|
|
CSharedLibraryItem *item = GetCacheItem(name);
|
|
if (item == NULL)
|
|
{
|
|
LibraryHandle lib = LoadLibraryHandle(name);
|
|
dbgdynfn("CDynamicFunction: Loading library '%s' (%p)\n", name, (void *) lib);
|
|
if (lib == NULL)
|
|
return NULL;
|
|
|
|
item = new CSharedLibraryItem(lib, name);
|
|
item->m_next = m_pList;
|
|
m_pList = item;
|
|
}
|
|
return item->m_handle;
|
|
}
|
|
|
|
void CloseLibrary(const char *name)
|
|
{
|
|
CSharedLibraryItem *item = GetCacheItem(name);
|
|
if (item)
|
|
{
|
|
assert(item == m_pList);
|
|
m_pList = item->m_next;
|
|
item->m_next = NULL;
|
|
delete item;
|
|
}
|
|
}
|
|
|
|
void CloseAllLibraries()
|
|
{
|
|
delete m_pList;
|
|
}
|
|
|
|
private:
|
|
CSharedLibraryItem *GetCacheItem(const char *name)
|
|
{
|
|
CSharedLibraryItem *prev = NULL;
|
|
CSharedLibraryItem *item = m_pList;
|
|
while (item)
|
|
{
|
|
if (strcmp(item->m_name, name) == 0)
|
|
{
|
|
// move this item to the front of the list, since there will
|
|
// probably be a big pile of these lookups in a row
|
|
// and then none ever again.
|
|
if (prev != NULL)
|
|
{
|
|
prev->m_next = item->m_next;
|
|
item->m_next = m_pList;
|
|
m_pList = item;
|
|
}
|
|
return item;
|
|
}
|
|
|
|
prev = item;
|
|
item = item->m_next;
|
|
}
|
|
return NULL; // not found.
|
|
}
|
|
|
|
CSharedLibraryItem *m_pList;
|
|
};
|
|
|
|
void *VoidFnPtrLookup_Tier0(const char *libname, const char *fn, void *fallback)
|
|
{
|
|
LibraryHandle lib = CSharedLibraryCache::GetCache().GetHandle(libname);
|
|
void *retval = NULL;
|
|
if (lib != NULL)
|
|
{
|
|
retval = LookupInLibraryHandle(lib, fn);
|
|
dbgdynfn("CDynamicFunction: Lookup of '%s' in '%s': %p\n", fn, libname, retval);
|
|
}
|
|
|
|
if (retval == NULL)
|
|
retval = fallback;
|
|
return retval;
|
|
}
|
|
|