0

让这个 C 文件:

#include <stdio.h>
#include <signal.h>

void handle(){
    return;
}

int main() {
    struct sigaction action;
    action.sa_handler = &handle;
    printf("%d\n", action.sa_flags);
    return 0;
}

使用 gcc-7 (Ubuntu 7.4.0-1ubuntu1~18.04.1) 编译会引发警告:

src/flags.c:14:2: warning: ‘action.sa_flags’ is used uninitialized in this function [-Wuninitialized]
  printf("%d\n", action.sa_flags);

但是,使用 gcc-5 (Ubuntu 5.4.0-6ubuntu1~16.04.11),没有警告。

使用 gcc-7,该sa_flags值实际上没有被初始化:

(gdb) x action.sa_flags
0x555545a0:    Cannot access memory at address 0x555545a0

在 gcc-5 中,它得到一个奇怪的值:

(gdb) x action.sa_flags
0x4004a0 <_start>:    0x8949ed31

我不明白为什么:

  • 使用 gcc-7,sa_flags不会初始化为 0。
  • 使用 gcc-5,sa_flags并且_start函数具有相同的地址。
4

1 回答 1

0

struct sigaction action任一编译器中都没有初始化。您设置action.sa_handler但所有其他字段保持未初始化。从 gcc-7 开始,编译器足够聪明,可以注意到您只初始化了结构的某些字段,而其他字段未初始化。Gcc-5 根本就没有那么聪明,但这并不意味着该结构曾经被初始化过。您在 action.sa_flags 中看到的只是调用 main 之前 crt0 代码在堆栈中留下的任何垃圾。

尽管结构中的指针在技术上是错误的,但一般解决方案是将结构 memset() 设置为 0。

于 2019-10-14T14:11:22.867 回答