- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一个程序可以静态链接到几个导出一些函数的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
所以这里的关键是我使用动态链接在我自己的主程序中调用一个函数。
(为什么我这样做超出了问题的范围,但简短的回答:这是一个插件架构,但我有一些标准插件直接链接到主二进制文件 - 但我仍然想通过加载它们相同的插件加载接口(interface)。例如,对于内置插件,我通过传入当前可执行文件作为插件接口(interface)的源来加载它们。)
问题是:链接器不知道我将需要这些函数,也没有链接它们。
如何强制链接这些函数?对于动态库,导出它们就足够了。但是对于exe,连dll导出的函数都会被链接器删除。
我知道我可以通过让主二进制文件将这些函数地址分配给某些东西或其他类似的 hack 来强制链接。有正确的方法吗?
@UPDATE:所以我有一个可行的解决方案 - 但它的内部确实很丑陋。仍在寻找更好的方法。
所以我必须以某种方式在加载内置接口(interface)的对象中定义我需要的符号。我认为没有办法强制链接器以其他方式链接到符号中。例如。据我所知,没有办法构建一个函数库,无论它看起来是否需要,它总是被链接起来。这完全取决于可执行文件的链接步骤。
所以在可执行文件中我有一个定义我需要的内置接口(interface)的宏。每个内置插件都有一个指向其所有接口(interface)函数的前缀,因此,在文件的顶部我做了:
DEFINE_BUILT_IN_PLUGIN( PluginOne )
DEFINE_BUILT_IN_PLUGIN( PluginTwo )
这将强制定义我需要的函数。但是这样做的宏是如此丑陋,以至于我充满了愤怒和 self 怀疑的感觉(为了便于阅读,我已经从宏中删除了尾部的斜线):
#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 中,我将竭尽全力诱使编译器链接一个未引用的函数。该宏仅在函数内部起作用。所以我必须创建这个伪造的 Magic 类。一定有更好的方法。
无论如何 - 它确实有效。
最佳答案
我至少见过两种不同的方法来解决类似的任务。
例如,在 Qt 中,您可以拥有需要通过调用特定宏“导入”到主可执行文件中的静态插件:
https://qt-project.org/doc/qt-4.8/qtplugin.html#Q_IMPORT_PLUGIN
它创建自定义类的静态实例,其构造函数调用从静态插件导出的初始化函数。
Poco 人员在 Linux 上使用 extern "C"
声明,在 Windows 上使用 pragma 强制从静态库中导出特定符号:
__pragma(comment (linker, "/export:CreateRequestHandler"))
在 Linux 上使用相同的 extern "C"
声明强制链接到静态库,在 Windows 上使用链接器编译指示:
__pragma(comment (linker, "/include:CreateRequestHandler"))
您可以在 blog post 中找到详细信息.
关于c++ - 防止链接器删除导出的函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17070536/
我之前让 dll 注入(inject)器变得简单,但我有 Windows 7,我用 C# 和 C++ 做了它,它工作得很好!但是现在当我在 Windows 8 中尝试相同的代码时,它似乎没有以正确的方
我正在尝试制作一个名为 core-splitter 的元素,该元素在 1.0 中已弃用,因为它在我们的项目中起着关键作用。 如果您不知道 core-splitter 的作用,我可以提供一个简短的描述。
我有几个不同的蜘蛛,想一次运行所有它们。基于 this和 this ,我可以在同一个进程中运行多个蜘蛛。但是,我不知道如何设计一个信号系统来在所有蜘蛛都完成后停止 react 器。 我试过了: cra
有没有办法在达到特定条件时停止扭曲 react 器。例如,如果一个变量被设置为某个值,那么 react 器应该停止吗? 最佳答案 理想情况下,您不会将变量设置为一个值并停止 react 器,而是调用
https://code.angularjs.org/1.0.0rc9/angular-1.0.0rc9.js 上面的链接定义了外部js文件,我不知道Angular-1.0.0rc9.js的注入(in
我正在尝试运行一个函数并将服务注入(inject)其中。我认为这可以使用 $injector 轻松完成.所以我尝试了以下(简化示例): angular.injector().invoke( [ "$q
在 google Guice 中,我可以使用函数 createInjector 创建基于多个模块的注入(inject)器。 因为我使用 GWT.create 在 GoogleGin 中实例化注入(in
我在 ASP.NET Core 1.1 解决方案中使用配置绑定(bind)。基本上,我在“ConfigureServices Startup”部分中有一些用于绑定(bind)的简单代码,如下所示: s
我在 Spring MVC 中设置 initBinder 时遇到一些问题。我有一个 ModelAttribute,它有一个有时会显示的字段。 public class Model { privat
我正在尝试通过jquery post发布knockoutjs View 模型 var $form = $('#barcodeTemplate form'); var data = ko.toJS(vm
如何为包含多态对象集合的复杂模型编写自定义模型绑定(bind)程序? 我有下一个模型结构: public class CustomAttributeValueViewModel { publi
您好,我正在尝试实现我在 this article 中找到的扩展方法对于简单的注入(inject)器,因为它不支持开箱即用的特定构造函数的注册。 根据这篇文章,我需要用一个假的委托(delegate)
你好,我想自动注册我的依赖项。 我现在拥有的是: public interface IRepository where T : class public interface IFolderReposi
我正在使用 Jasmine 测试一些 Angular.js 代码。为此,我需要一个 Angular 注入(inject)器: var injector = angular.injector(['ng'
我正在使用 Matlab 代码生成器。不可能包含代码风格指南。这就是为什么我正在寻找一个工具来“ reshape ”、重命名和重新格式化生成的代码,根据我的: 功能横幅约定 文件横幅约定 命名约定 等
这个问题在这里已经有了答案: Where and why do I have to put the "template" and "typename" keywords? (8 个答案) 关闭 8
我开发了一种工具,可以更改某些程序的外观。为此,我需要在某些进程中注入(inject)一个 dll。 现在我基本上使用这个 approach .问题通常是人们无法注入(inject) dll,因为他们
我想使用 swing、spring 和 hibernate 编写一个 java 应用程序。 我想使用数据绑定(bind)器用 bean 的值填充 gui,并且我还希望它反射(reflect) gui
我有这段代码,当两个蜘蛛完成后,程序仍在运行。 #!C:\Python27\python.exe from twisted.internet import reactor from scrapy.cr
要点是 Spring Batch (v2) 测试框架具有带有 @Autowired 注释的 JobLauncherTestUtils.setJob。我们的测试套件有多个 Job 类提供者。因为这个类不
我是一名优秀的程序员,十分优秀!