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

什么会导致gcc部分编译C代码?,c,gcc,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 % (

我正在编译一个用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>

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

正如评论中指出的,原因很可能是未定义的行为。原来原因是对从未初始化过的静态变量执行了空指针解引用。

生成“乱码”的编译器通常表示代码中某个地方存在未定义的行为。问题仅在我导入“pycairo.h”调用
import\u cairo
宏时出现。如果我改为链接库的编译版本,代码将按预期工作。我倾向于得出结论,如果有一些UB,它可能是由pycairo.h引入我的代码中的。请尝试使用
-fsanitize=undefined
标志进行编译。我还怀疑您的timespec定义,尤其是对unsigned的强制转换。timespec的内容不是未签名的。那么为什么要强制转换呢?
-fsanizize
标志仅用于调试。它们将使程序非常慢,但应报告未定义的行为。
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