在我重新编译代码后,gdb 似乎无法找到断言失败的代码位置。更准确地说,我希望相对于断言失败的信号提升的位置是
0x00007ffff7a5ff00 in raise () from /lib64/libc.so.`6
相反,我得到
0x00007ffff7a5ff00 in ?? ()
例如,考虑以下代码
#include <assert.h>
int main()
{
assert(0);
return 0;
}
用调试符号编译并用 gdb 调试。
> gcc -g main.c
> gdb a.out
第一次运行gdb,就找到了位置,回溯报错:
GNU gdb (Gentoo 8.0.1 p1) 8.0.1
...
(gdb) r
Starting program: /home/myself/a.out
a.out: main.c:5: main: Assertion `0' failed.
Program received signal SIGABRT, Aborted.
0x00007ffff7a5ff00 in raise () from /lib64/libc.so.6
(gdb) bt
#0 0x00007ffff7a5ff00 in raise () from /lib64/libc.so.6
#1 0x00007ffff7a61baa in abort () from /lib64/libc.so.6
#2 0x00007ffff7a57cb7 in ?? () from /lib64/libc.so.6
#3 0x00007ffff7a57d72 in __assert_fail () from /lib64/libc.so.6
#4 0x00005555555546b3 in main () at main.c:5
(gdb)
当我重新编译代码时,问题就来了。重新编译后,我在同一个 gdb 实例中发出运行命令。gdb 重新读取符号,从头开始启动程序,但没有找到正确的位置:
(gdb) r
The program being debugged has been started already.
Start it from the beginning? (y or n) y
`/home/myself/a.out' has changed; re-reading symbols.
Starting program: /home/myself/a.out
a.out: main.c:5: main: Assertion `0' failed.
Program received signal SIGABRT, Aborted.
0x00007ffff7a5ff00 in ?? ()
(gdb) bt
#0 0x00007ffff7a5ff00 in ?? ()
#1 0x0000000000000000 in ?? ()
(gdb) up
Initial frame selected; you cannot go up.
(gdb) n
Cannot find bounds of current function
此时调试器无法使用。上不去,上前一步。作为一种解决方法,我可以手动重新加载文件,然后再次找到位置。
(gdb) file a.out
Load new symbol table from "a.out"? (y or n) y
Reading symbols from a.out...done.
(gdb) r
Starting program: /home/myself/a.out
a.out: main.c:5: main: Assertion `0' failed.
Program received signal SIGABRT, Aborted.
0x00007ffff7a5ff00 in raise () from /lib64/libc.so.6
(gdb)
不幸的是,以这种方式重新加载文件后,gdb 无法重置断点。
ERRATA CORRIGE:我在使用 gdb 7.12.1 重置断点时遇到了失败。升级到 8.0.1 后问题消失了。据说,这与错误修复https://sourceware.org/bugzilla/show_bug.cgi?id=21555有关。但是,仍然无法正确找到断言失败的代码位置。
有人知道这里发生了什么吗?
这在系统更新后开始发生。系统更新重新编译了所有系统库,包括glibc,作为位置无关代码,即使用-fPIC 编译。
另外,我使用的 gcc 版本是 6.4.0