You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
261 lines
6.1 KiB
261 lines
6.1 KiB
/* |
|
lib_posix.c - dynamic library code for POSIX systems |
|
Copyright (C) 2018 Flying With Gauss |
|
|
|
This program is free software: you can redistribute it and/sor modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 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. |
|
*/ |
|
#define _GNU_SOURCE |
|
#include "platform/platform.h" |
|
#if XASH_LIB == LIB_POSIX |
|
|
|
|
|
#include "common.h" |
|
#include "library.h" |
|
#include "filesystem.h" |
|
#include "server.h" |
|
#include "platform/android/lib_android.h" |
|
#include "platform/emscripten/lib_em.h" |
|
#include "platform/apple/lib_ios.h" |
|
|
|
#ifdef XASH_DLL_LOADER // wine-based dll loader |
|
void * Loader_LoadLibrary (const char *name); |
|
void * Loader_GetProcAddress (void *hndl, const char *name); |
|
void Loader_FreeLibrary(void *hndl); |
|
void *Loader_GetDllHandle( void *hndl ); |
|
const char * Loader_GetFuncName( void *hndl, void *func); |
|
const char * Loader_GetFuncName_int( void *wm , void *func); |
|
#endif |
|
|
|
|
|
#ifdef XASH_NO_LIBDL |
|
#ifndef XASH_DLL_LOADER |
|
#error Enable at least one dll backend!!! |
|
#endif // XASH_DLL_LOADER |
|
|
|
void *dlsym(void *handle, const char *symbol ) |
|
{ |
|
Con_DPrintf( "dlsym( %p, \"%s\" ): stub\n", handle, symbol ); |
|
return NULL; |
|
} |
|
|
|
void *dlopen(const char *name, int flag ) |
|
{ |
|
Con_DPrintf( "dlopen( \"%s\", %d ): stub\n", name, flag ); |
|
return NULL; |
|
} |
|
|
|
int dlclose(void *handle) |
|
{ |
|
Con_DPrintf( "dlsym( %p ): stub\n", handle ); |
|
return 0; |
|
} |
|
|
|
char *dlerror( void ) |
|
{ |
|
return "Loading ELF libraries not supported in this build!\n"; |
|
} |
|
|
|
int dladdr( const void *addr, Dl_info *info ) |
|
{ |
|
return 0; |
|
} |
|
#endif // XASH_NO_LIBDL |
|
|
|
qboolean COM_CheckLibraryDirectDependency( const char *name, const char *depname, qboolean directpath ) |
|
{ |
|
// TODO: implement |
|
return true; |
|
} |
|
|
|
void *COM_LoadLibrary( const char *dllname, int build_ordinals_table, qboolean directpath ) |
|
{ |
|
dll_user_t *hInst = NULL; |
|
void *pHandle = NULL; |
|
#if XASH_HAIKU |
|
const char *libdir; |
|
#endif |
|
|
|
COM_ResetLibraryError(); |
|
|
|
// platforms where gameinfo mechanism is impossible |
|
#ifdef Platform_POSIX_LoadLibrary |
|
return Platform_POSIX_LoadLibrary( dllname ); |
|
#endif |
|
|
|
#if XASH_HAIKU |
|
// First look for libraries in the mirror directory |
|
libdir = getenv( "XASH3D_MIRRORDIR" ); |
|
if( libdir ) { |
|
char path[MAX_SYSPATH]; |
|
char game[MAX_SYSPATH] = { 0 }; |
|
if( GI && !Q_strstr( dllname, "menu" ) ) |
|
Q_snprintf( game, MAX_SYSPATH, "/%s", GI->gamefolder ); |
|
Q_snprintf( path, MAX_SYSPATH, "%s%s/%s", libdir, game, dllname ); |
|
pHandle = dlopen( path, RTLD_NOW ); |
|
if( pHandle ) |
|
return pHandle; |
|
|
|
COM_PushLibraryError( dlerror() ); |
|
} |
|
// Then through FS_FindLibrary() function in the gamebase directory |
|
#endif |
|
|
|
// platforms where gameinfo mechanism is working goes here |
|
// and use FS_FindLibrary |
|
hInst = FS_FindLibrary( dllname, directpath ); |
|
if( !hInst ) |
|
{ |
|
// HACKHACK: direct load dll |
|
#ifdef XASH_DLL_LOADER |
|
if( host.enabledll && ( pHandle = Loader_LoadLibrary(dllname)) ) |
|
{ |
|
return pHandle; |
|
} |
|
#endif |
|
|
|
// try to find by linker(LD_LIBRARY_PATH, DYLD_LIBRARY_PATH, LD_32_LIBRARY_PATH and so on...) |
|
if( !pHandle ) |
|
{ |
|
pHandle = dlopen( dllname, RTLD_LAZY ); |
|
if( pHandle ) |
|
return pHandle; |
|
|
|
COM_PushLibraryError( va( "Failed to find library %s", dllname )); |
|
COM_PushLibraryError( dlerror() ); |
|
return NULL; |
|
} |
|
} |
|
|
|
if( hInst->custom_loader ) |
|
{ |
|
COM_PushLibraryError( va( "Custom library loader is not available. Extract library %s and fix gameinfo.txt!", hInst->fullPath )); |
|
Mem_Free( hInst ); |
|
return NULL; |
|
} |
|
|
|
#ifdef XASH_DLL_LOADER |
|
if( host.enabledll && ( !Q_stricmp( COM_FileExtension( hInst->shortPath ), "dll" ) ) ) |
|
{ |
|
if( hInst->encrypted ) |
|
{ |
|
COM_PushLibraryError( va( "Library %s is encrypted. Cannot load", hInst->shortPath ) ); |
|
Mem_Free( hInst ); |
|
return NULL; |
|
} |
|
|
|
if( !( hInst->hInstance = Loader_LoadLibrary( hInst->fullPath ) ) ) |
|
{ |
|
COM_PushLibraryError( va( "Failed to load DLL with DLL loader: %s", hInst->shortPath ) ); |
|
Mem_Free( hInst ); |
|
return NULL; |
|
} |
|
} |
|
else |
|
#endif |
|
{ |
|
if( !( hInst->hInstance = dlopen( hInst->fullPath, RTLD_LAZY ) ) ) |
|
{ |
|
COM_PushLibraryError( dlerror() ); |
|
Mem_Free( hInst ); |
|
return NULL; |
|
} |
|
} |
|
|
|
pHandle = hInst->hInstance; |
|
|
|
Mem_Free( hInst ); |
|
|
|
return pHandle; |
|
} |
|
|
|
void COM_FreeLibrary( void *hInstance ) |
|
{ |
|
#ifdef XASH_DLL_LOADER |
|
void *wm; |
|
if( host.enabledll && (wm = Loader_GetDllHandle( hInstance )) ) |
|
return Loader_FreeLibrary( hInstance ); |
|
else |
|
#endif |
|
{ |
|
#ifdef Platform_POSIX_FreeLibrary |
|
Platform_POSIX_FreeLibrary( hInstance ); |
|
#else |
|
dlclose( hInstance ); |
|
#endif |
|
} |
|
} |
|
|
|
void *COM_GetProcAddress( void *hInstance, const char *name ) |
|
{ |
|
#ifdef XASH_DLL_LOADER |
|
void *wm; |
|
if( host.enabledll && (wm = Loader_GetDllHandle( hInstance )) ) |
|
return Loader_GetProcAddress(hInstance, name); |
|
else |
|
#endif |
|
#if Platform_POSIX_GetProcAddress |
|
return Platform_POSIX_GetProcAddress( hInstance, name ); |
|
#else |
|
return dlsym( hInstance, name ); |
|
#endif |
|
} |
|
|
|
void *COM_FunctionFromName( void *hInstance, const char *pName ) |
|
{ |
|
void *function; |
|
if( !( function = COM_GetProcAddress( hInstance, pName ) ) ) |
|
{ |
|
Con_Reportf( S_ERROR "FunctionFromName: Can't get symbol %s: %s\n", pName, dlerror()); |
|
} |
|
return function; |
|
} |
|
|
|
#ifdef XASH_DYNAMIC_DLADDR |
|
static int d_dladdr( void *sym, Dl_info *info ) |
|
{ |
|
static int (*dladdr_real) ( void *sym, Dl_info *info ); |
|
|
|
if( !dladdr_real ) |
|
dladdr_real = dlsym( (void*)(size_t)(-1), "dladdr" ); |
|
|
|
memset( info, 0, sizeof( *info ) ); |
|
|
|
if( !dladdr_real ) |
|
return -1; |
|
|
|
return dladdr_real( sym, info ); |
|
} |
|
#define dladdr d_dladdr |
|
#endif |
|
|
|
const char *COM_NameForFunction( void *hInstance, void *function ) |
|
{ |
|
#ifdef XASH_DLL_LOADER |
|
void *wm; |
|
if( host.enabledll && (wm = Loader_GetDllHandle( hInstance )) ) |
|
return Loader_GetFuncName_int(wm, function); |
|
else |
|
#endif |
|
// NOTE: dladdr() is a glibc extension |
|
{ |
|
Dl_info info = {0}; |
|
dladdr((void*)function, &info); |
|
if(info.dli_sname) |
|
return info.dli_sname; |
|
} |
|
#ifdef XASH_ALLOW_SAVERESTORE_OFFSETS |
|
return COM_OffsetNameForFunction( function ); |
|
#else |
|
return NULL; |
|
#endif |
|
} |
|
|
|
#endif // _WIN32
|
|
|