我正在编译用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
还是使用
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代码,而不会抛出错误?
答案 0 :(得分:1)
正如评论中指出的那样,原因很可能是未定义的行为。事实证明,原因是在从未初始化的静态变量上取消引用的NULL指针。