未调用 RISCV 全局对象构造函数

RISCV Global object constructors are not being called

我 运行 遇到了与此类似的问题 post: Global constructor call not in .init_array section

我将重新使用他的示例,但我认为这是 RISCV 环境特有的问题,这就是我创建新的 post.

的原因
class foobar
{
    int i;

public:
    foobar()
    {
        i = 100;
    }

    void inc()
    {
        i++;
    }
};

foobar foo;

int main()
{
    foo.inc();
    for (;;);
}

问题是从未调用 foo 的构造函数。我在第一个增量上设置了断点,i 的值为 0,而不是预期的 100。

这是输出:riscv64-unknown-elf-objdump.exe -j .init_array -x objs/main.o

objs/main.o: file format elf32-littleriscv objs/main.o architecture: riscv:rv32, flags 0x00000011: HAS_RELOC, HAS_SYMS start address 0x00000000

Sections: Idx Name Size VMA LMA File off Algn 8 .init_array 00000004 00000000 00000000 00000154 2**2 CONTENTS, ALLOC, LOAD, RELOC, DATA SYMBOL TABLE: 00000000 l d .init_array 00000000 .init_array

RELOCATION RECORDS FOR [.init_array]: OFFSET TYPE VALUE 00000000 R_RISCV_32 _GLOBAL__sub_I_foo

哪个让我相信它应该被称为?我还测试了全局声明的标准 C 变量,这些变量确实被分配并显示在 .init_array.

我的 _init() 看起来像这样:

void _init(void)
{
    copy_section(&__sdata_load, &__sdata_start, &__sdata_end);
    copy_section(&__data_load, &__data_start, &__data_end);
    zero_section(&__sbss_start, &__sbss_end);
    zero_section(&__bss_start, &__bss_end);

    exit(main());
}

看起来其他基于 RISCV 的项目也不利用 libc 初始化原语(如 __libc_init_array()),作为一个额外的问题,我想知道为什么会这样?

我终于找到了一个正确的解决方案来解决我几个月前遇到的这个问题。这是我最终放入初始化过程的内容。这是在 main() 之前调用的。这在很大程度上受到我在原始问题中链接的 post 中提出的解决方案的影响。

extern void (**__preinit_array_start)();
extern void (**__preinit_array_end)();
extern void (**__init_array_start)();
extern void (**__init_array_end)();
extern void (**__fini_array_start)();
extern void (**__fini_array_end)();

static void execute_array(void (** _start)(), void (** _end)())
{
    size_t count;
    size_t i;

    count = _end - _start;
    for (i = 0; i < count; i++)
    {
        (*_start[i])();
    }
}

void _init(void)
{
    execute_array(__preinit_array_start, __preinit_array_end);
    execute_array(__init_array_start, __init_array_end);

    main();
}

void _fini()
{
    execute_array(__fini_array_start, __fini_array_end);
}