c++ - 如何在可执行共享库 (.so) 中触发全局变量的 c'tors?

标签 c++ linker shared-libraries factory elf

我有一个共享库,我想使其可执行,类似于 libc。当库执行时,我希望它转储在特定抽象工厂(这是 C++)中注册的类的名称列表。我使用通过全局变量的初始化/构造向工厂注册类的标准技术。

有几个教程介绍如何使共享库可执行(例如 herehere)。它相对简单。然而,当我尝试它时,我发现入口点是在调用任何全局构造函数之前执行的。在我的例子中,这意味着我的工厂没有注册任何类,所以我没有要打印的信息。

我想在调用构造函数后执行入口点,或者学习如何从我的入口点函数自己触发构造。这可能吗?有谁知道该怎么做?也许有一个内部 libc 函数我可以外部调用?

最佳答案

我相信我已经找到了可行的解决方案。它基于创建 -nostdlib 可执行文件(例如操作系统内核)的技术。但是,在这种情况下,我们的共享库仍然链接标准库。我找到了 this RaspberryPi 论坛帖子特别有用。

解决方案是手动执行存储在共享库的嵌入式init_array 中的函数指针。诀窍是使用链接描述文件来定义用于访问该数组的指针。然后我们在程序代码中extern这些指针。我们也可以重复执行析构函数的过程。

test.cpp 中,我们有以下内容:

#include <cstdio>
#include <unistd.h>

class Test
{
public:
    Test() { printf("Hello world!\n"); }
    ~Test() { printf("Goodbye world!\n"); }
};

Test myGlobal;  // a global class instance

extern "C"
{
// ensures linker generates executable .so (assuming x86_64)
extern const char test_interp[] __attribute__((section(".interp"))) =
    "/lib64/ld-linux-x86-64.so.2";

// function walks the init_array and executes constructors
void manual_init(void)
{
    typedef void (*constructor_t)(void);
    // _manual_init_array_start and _manual_init_array_end
    // are created by linker script.
    extern constructor_t _manual_init_array_start[];
    extern constructor_t _manual_init_array_end[];

    constructor_t *fn = _manual_init_array_start;
    while(fn < _manual_init_array_end)
    {
        (*fn++)();
    }
}

// function walks the fini_array and executes destructors
void manual_fini(void)
{
    typedef void (*destructor_t)(void);
    // _manual_init_array_start and _manual_init_array_end
    // are created by linker script.
    extern destructor_t _manual_fini_array_start[];
    extern destructor_t _manual_fini_array_end[];

    destructor_t *fn = _manual_fini_array_start;
    while(fn < _manual_fini_array_end)
    {
        (*fn++)();
    }
}

// entry point for libtest.so when it is executed
void lib_entry(void)
{
    manual_init();  // call ctors
    printf("Entry point of the library!\n");
    manual_fini();  // call dtors

    _exit(0);
}

我们需要通过链接描述文件手动定义_manual* 指针。我们必须使用 INSERT 关键字,这样才不会完全替换 ld 的默认链接描述文件。在文件 test.ld 中,我们有:

SECTIONS
{
    .init_array : ALIGN(4)
    {
        _manual_init_array_start = .;
        *(.init_array)
        *(SORT_BY_INIT_PRIORITY(.init_array.*))
        _manual_init_array_end = .;
    }
}
INSERT AFTER .init; /* use INSERT so we don't override default linker script */

SECTIONS
{
    .fini_array : ALIGN(4)
    {
        _manual_fini_array_start = .;
        *(.fini_array)
        *(SORT_BY_INIT_PRIORITY(.fini_array.*))
        _manual_fini_array_end = .;
    }
}
INSERT AFTER .fini; /* use INSERT so we don't override default linker script */

我们必须向链接器提供两个参数:(1) 我们的链接描述文件和 (2) 我们库的入口点。要编译,我们执行以下操作:

 g++ test.cpp -fPIC -Wl,-T,test.ld -Wl,-e,lib_entry -shared -o libtest.so

在命令行执行 libtest.so 时,我们得到以下输出:

$ ./libtest.so 
Hello world!
Entry point of the library!
Goodbye world!

除了 test.cpp 之外,我还尝试在 .cpp 文件中定义全局变量,这些文件也被编译到共享库中。这些全局变量的 ctor 调用包含在 init_array 中,因此它们由 manual_init() 调用。当库作为常规共享库加载时,它“像正常一样”​​工作。

关于c++ - 如何在可执行共享库 (.so) 中触发全局变量的 c'tors?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24316818/

相关文章:

c++, protected 抽象虚拟基纯虚拟私有(private)析构函数

c++ - 可变参数函数崩溃

c - 了解编译、加载器、链接器和进程结构的资源

c++ - 如何正确定义C++类析构函数并将其链接到主文件?

android - 已编译的 JNI 库 (libmupdf.so),适用于 android 2.2 和 Android 模拟器,但在 android 2.1 上运行失败

c - 如何使用 automake 创建 JNI/JNA 可以使用的共享库(dylib)?

c++ - C++中的直接构造函数调用

c++ - C++ 自动变量的大小可以变化吗?

ubuntu - 在 Ubuntu 上链接 libusb 不起作用

matlab - 无效的 mex 文件、libarmadillo、无法打开共享对象文件