什么可以导致gcc部分编译C代码?

时间:2018-04-11 22:06:43

标签: c gcc

我正在编译用C编写的Python扩展,它使用外部C扩展(pycairo)。当我编译并链接外部库时,gcc生成一个有效的.so对象。但是,如果我使用pycairo提供的PyCapsule形式的C API,gcc会在我的一个C源中部分组装一个函数。功能

static PyObject *
Canvas_show(Canvas* self) {
  struct timespec ts = {self->interval / 1e9, self->interval % (unsigned int) 1e9};
  int running;

  // Input events
  XSelectInput(self->display, self->win_id, ButtonPressMask | KeyPressMask);
  XMapWindow(self->display, self->win_id);

  PyObject *args = Py_BuildValue("(O)", PycairoContext_FromContext(self->context, &PycairoContext_Type, NULL));
  for (running = 1; running;)
  {
    Canvas_on_draw(self, args, NULL);

    nanosleep(&ts, NULL);
    cairo_surface_flush(self->surface);

    switch (cairo_check_event(self->surface, 0)) {
    case 0xff1b:   // Esc
    case -1:       // left mouse button
      running = 0;
      break;
    }
  }

  Py_DECREF(args);
  cairo_destroy(self->context);
  cairo_surface_destroy(self->surface);
  XCloseDisplay(self->display);

  return Py_True;
}

由gcc编译成以下几条指令

00000000000010f0 <Canvas_show>:
    10f0:       53                      push   rbx
    10f1:       48 89 fb                mov    rbx,rdi
    10f4:       48 8b 77 38             mov    rsi,QWORD PTR [rdi+0x38]
    10f8:       48 8b 7f 30             mov    rdi,QWORD PTR [rdi+0x30]
    10fc:       ba 05 00 00 00          mov    edx,0x5
    1101:       e8 8a fb ff ff          call   c90 <XSelectInput@plt>
    1106:       48 8b 73 38             mov    rsi,QWORD PTR [rbx+0x38]
    110a:       48 8b 7b 30             mov    rdi,QWORD PTR [rbx+0x30]
    110e:       e8 bd fb ff ff          call   cd0 <XMapWindow@plt>
    1113:       48 8b 04 25 08 00 00    mov    rax,QWORD PTR ds:0x8
    111a:       00 
    111b:       0f 0b                   ud2    

Disassembly of section .fini:

0000000000001120 <_fini>:
    1120:       48 83 ec 08             sub    rsp,0x8
    1124:       48 83 c4 08             add    rsp,0x8
    1128:       c3                      ret    
<EOF>

从1113开始的代码似乎已经是胡言乱语,这是我在运行测试时获得SIGSEGV的地方。

无论我使用distutils还是使用

手动编译源代码,都会生成相同的.so文件
gcc -shared x11/x11module.c x11/canvas.c -o x11/x11.so -g -O3 -Wall -Wstrict-prototypes -fPIC -I/usr/include/python3.6 -I/usr/include/cairo/ -lcairo -lX11

什么可以导致gcc部分编译有效的C代码,而不会抛出错误?

1 个答案:

答案 0 :(得分:1)

正如评论中指出的那样,原因很可能是未定义的行为。事实证明,原因是在从未初始化的静态变量上取消引用的NULL指针。