是什么导致我出现细分错误(核心已转储)

时间:2019-03-24 21:57:34

标签: c linux fault

我正在读一本有关编码操作系统的书,我正在写书中包含的C代码示例,并在终端中编译和测试代码,但是我遇到了与此代码有关的问题。

其中包含此代码的文件名为“ hello.c”。 我用“ gcc hello.c -o hello”编译了文件 然后使用“ ./hello”来运行它。

我收到消息分段错误(核心已转储), 而且我不确定自己在做什么错。

#include <stdio.h>

void preinit1() {
  printf("%s\n", __FUNCTION__);
}

void preinit2() {
  printf("%s\n", __FUNCTION__);
}

void init1() {
  printf("%s\n", __FUNCTION__);
}

void init2() {
  printf("%s\n", __FUNCTION__);
}

typedef void (*preinit)();
typedef void (*init)();

__attribute__((section(".init_array"))) preinit
  preinit_arr[2] = {preinit1, preinit2};

__attribute__((section(".init_array"))) init
  init_arr[2] = {init1, init2};

int main(int argc, char *argv[])
{
    printf("hello world!\n");

    return 0;
}

1 个答案:

答案 0 :(得分:2)

我不认为您应该向该部分添加数组(示例中出现错误,将.init_array初始化了两次)。

__attribute__((section(".preinit_array"))) preinit preinit_arr1 = preinit1;
__attribute__((section(".preinit_array"))) preinit preinit_arr2 = preinit2;

__attribute__((section(".init_array"))) init init_arr1 = init1;
__attribute__((section(".init_array"))) init init_arr2 = init2;

这是使用数组声明的部分

objdump -s -j .init_array hello.orig 

hello.orig:     file format elf64-x86-64

Contents of section .init_array:
 3dc0 30110000 00000000 00000000 00000000  0...............
 3dd0 35110000 00000000 48110000 00000000  5.......H.......
 3de0 5b110000 00000000 6e110000 00000000  [.......n.......

这是有效的部分

objdump -s -j .init_array hello

hello:     file format elf64-x86-64

Contents of section .init_array:
 3dc8 30110000 00000000 35110000 00000000  0.......5.......
 3dd8 48110000 00000000 5b110000 00000000  H.......[.......
 3de8 6e110000 00000000                    n.......

在前一种情况下,3dc8处有一个空指针可能会导致分割(请参见下文),我不知道为什么会有其他人可以解释。

(gdb) bt
#0  0x0000000000000000 in ?? ()
#1  0x00005555555551f5 in __libc_csu_init ()
#2  0x00007ffff7dec02a in __libc_start_main (main=0x555555555181 <main>, argc=1, argv=0x7fffffffe1f8, init=0x5555555551b0 <__libc_csu_init>, 
    fini=<optimized out>, rtld_fini=<optimized out>, stack_end=0x7fffffffe1e8) at ../csu/libc-start.c:264
#3  0x000055555555507a in _start ()

仅修复有关.preinit_array的错字,才能使preinit调用起作用,而不能进行init