我有一个程序,它与几个导出一些函数的 c++ 库静态链接:
extern "C"
{
KSrvRequestHandler* CreateRequestHandler( const char* name );
bool DestroyRequestHandler( KSrvRequestHandler* handler );
const char** ListRequestHandlerTypes();
}
然后主程序使用 GetProcAddress/dlsym 调用这些函数:
#ifdef WIN32
HINSTANCE hDll = GetModuleHandle( NULL );
mCreateHandler = GetProcAddress( hDll, createFuncName );
mDestroyHandler = GetProcAddress( hDll, destroyFuncName );
mGetHandlerTypes = GetProcAddress( hDll, listFuncName );
#else // POSIX
void* handle = dlopen( NULL, 0 );
mCreateHandler = dlsym( handle, createFuncName );
mDestroyHandler = dlsym( handle, destroyFuncName );
mGetHandlerTypes = dlsym( handle, listFuncName );
dlclose( handle );
#endif // !POSIX
所以这里的关键是我正在使用动态链接在我自己的主程序中调用一个函数。
(为什么我这样做超出了问题的范围,但简短的回答:这是一个插件架构,但我有一些直接链接到主二进制文件的标准插件 - 但我仍然想通过相同的插件加载来加载它们接口。例如,对于内置插件,我通过传入当前可执行文件作为插件接口的源来加载它们。)
问题是:链接器不知道我将需要这些函数,也没有将它们链接进去。
如何强制链接这些功能?对于动态库,导出它们就足够了。但是对于一个 exe,即使是 dll 导出的函数也会被链接器删除。
我知道我可以通过使主二进制文件将这些函数地址分配给某些东西或其他类似的黑客来强制链接。有正确的方法吗?
@UPDATE:所以我有一个可行的解决方案 - 但它的内部确实很难看。还在寻找更好的方法。
所以我必须以某种方式在加载内置接口的对象中定义我需要的符号。我认为没有办法强制链接器以其他方式链接符号。例如,我不知道如何构建一个函数库,该函数库总是链接它看起来是否需要。这完全由可执行文件的链接步骤决定。
所以在可执行文件中我有一个宏来定义我需要的内置接口。每个内置插件都有其所有接口函数的前缀,因此,在文件的顶部我这样做:
DEFINE_BUILT_IN_PLUGIN( PluginOne )
DEFINE_BUILT_IN_PLUGIN( PluginTwo )
这将强制定义我需要的函数。但是这样做的宏是如此丑陋,以至于我充满了愤怒和自我怀疑的感觉(为了便于阅读,我从宏中删除了尾部斜杠):
#define FORCE_UNDEFINED_SYMBOL(x)
void* _fp_ ## x ## _fp =(void*)&x;
if (((ptrv) _fp_ ## x ##_fp * ( rand() | 1 )) < 1 )
exit(0);
#define DEFINE_BUILT_IN_PLUGIN( PREFIX )
extern "C"
{
KSrvRequestHandler* PREFIX ## CreateRequestHandler( const char* name );
bool PREFIX ## DestroyRequestHandler( KSrvRequestHandler* handler );
const char** PREFIX ## ListRequestHandlerTypes();
}
class PREFIX ## HandlerInterfaceMagic
{
public:
PREFIX ## HandlerInterfaceMagic()
{
FORCE_UNDEFINED_SYMBOL( PREFIX ## CreateRequestHandler );
FORCE_UNDEFINED_SYMBOL( PREFIX ## DestroyRequestHandler );
FORCE_UNDEFINED_SYMBOL( PREFIX ## ListRequestHandlerTypes );
}
};
PREFIX ## HandlerInterfaceMagic PREFIX ## HandlerInterfaceMagicInstance;
由于编译器是一个优化天才,在 FORCE_UNDEFINED_SYMBOLS 中,我将竭尽全力欺骗编译器链接一个未引用的函数。该宏仅在函数内部有效。所以我必须创建这个虚假的魔法类。一定会有更好的办法。
无论如何 - 它确实有效。