@comzyh
2019-04-04T10:30:08.000000Z
字数 2792
阅读 1496
GDB
GDB 是 GNU 提供的流行Debug 工具,GDB 可以对任何可执行文件进行Debug
最简单的方法
gdb ./a.out
如何需要使用命令行参数,则需要使用 --args 参数
gdb --args ./a.out soomeargs --arg1 value 1 --args2 value2
进入GDB后使用 r 开始执行程序
当程序运行遇到关键错误时,GDB会捕获错误并开始调试
当然,最重要的一点,gdb 使用 quit 命令或者 Ctrl + D 退出
Linux 支持在程序异常退出时使用 CoreDump 将现场内存保存在磁盘上(文件名默认为core). GDB 可以利用CoreDump文件恢复现场调试。
如果想使用CoreDump,首先要用ulimt 打开 CoreDump 功能
ulimit -c unlimited
这样程序异常退出时才会保存 core 文件。调试时使用
gdb ./a.out core
来启动调试
gdb 如果不能在可执行文件中找到符号(symbol)信息,调试时功能会受到极大限制,不能方便的使用断点, 不能查看代码,不能定位到崩溃的具体文件和行数。如果想要使用 GDB 调试,一般需要在编译时加上 -g 参数,加入符号信息。
gdb a.cpp -g
如果调试时发现有些变量不能输出,可能是 编译优化级别太高导致。仅在有必要的情况下,可以加入-O0 (注意是大写英文字母O 和阿拉伯数字0)选项
g++ a.cpp -g -O0
当触发断点,或者异常中断时,GDB会切换到相应的栈帧 (frame) 等待用户调试
常用的调试指令有
用法在下面介绍
下面给一个含有错误样例代码,演示 GDB 常用的调试命令
int divide(int x, int y) {return x / y;}int main() {for (int i = 10; i >= 0; i--) {int j = divide(10, i - 1);}}
g++ debug.cpp -g -O0
gdb ./a.out
显然这段代码会产生除0错误, 先执行(输入r)gdb 输出如下
(gdb) rStarting program: /home/comzyh/Projects/tmp/a.outProgram received signal SIGFPE, Arithmetic exception.0x0000555555554608 in divide (x=10, y=0) at debug.cpp:22 return x / y;(gdb)
我们能看到很多信息, 比如错误出现在 debug.cpp 文件的第3行,这行的代码是 return x / y;, 出现的错误是算数错误(Arithmetic exception)
现在可以使用××打印变量××(p 命令)功能,看看 x 和 y 到底是什么
(gdb) p x$1 = 10(gdb) p y$2 = 0
我们知道了,错误是因为 y = 0,出现了除 0 错误
那么为什么会执行这段代码呢?可以使用 list 命令查看附近的代码. 需要指出的是,list 默认向下浏览代码,你可以使用l - 向上浏览代码
(gdb) l1 int divide(int x, int y) {2 return x / y;3 }4 int main() {5 for (int i = 10; i >= 0; i--) {6 int j = divide(10, i - 1);7 }8 }
到这里我们知道了,y 是 0 的原因 divide 函数的输入参数 y 是 0,那么是谁调用了这个函数呢?我们可以使用 查看调用栈 功能 (bt命令)
(gdb) bt#0 0x0000555555554608 in divide (x=10, y=0) at debug.cpp:2#1 0x0000555555554631 in main () at debug.cpp:6
我们可以看出,是debug.cpp 的第 6 行 main 函数调用了 divide 函数,还能看到divide 函数的实参是 x = 10, y = 0.
此时我们应当使用切换栈帧 (frame命令) 回到 main 函数查看调用 divide 的代码. #1 代表main 函数的栈帧编号为 1
(gdb) f 1#1 0x0000555555554634 in main () at debug.cpp:66 int j = divide(10, i - 1);(gdb) l1 int divide(int x, int y) {2 return x / y;3 }4 int main() {5 for (int i = 10; i >= 0; i--) {6 int j = divide(10, i - 1);7 }8 }
我们通过切换栈帧和打印代码看到,传给 divide 函数的 y 参数是 循环变量 i. 我们可以打印此时 i 的值
(gdb) p i$3 = 1(gdb) p i - 1$4 = 0
print 指令不仅可以查看变量的值,还能进行简单的运算,例如对vector的元素的访问,可以直接 p vec[10] 这样操作。
最后我们尝试下断点 (breakpoint). 只要gdb等待你的输入,你就可以添加断点,比如我们给 main 函数的 第6行添加断点并重新运行
(gdb) b main.cpp:6No source file named main.cpp.Make breakpoint pending on future shared library load? (y or [n]) n(gdb) b debug.cpp:6Breakpoint 1 at 0x555555554622: file debug.cpp, line 6.(gdb) rThe program being debugged has been started already.Start it from the beginning? (y or n) yStarting program: /home/comzyh/Projects/tmp/a.outBreakpoint 1, main () at debug.cpp:66 int j = divide(10, i - 1);(gdb) p i$3 = 10
GDB 的确帮我们中断了程序并进入调试,此时我们可以继续(continue)
(gdb) cContinuing.Breakpoint 1, main () at debug.cpp:66 int j = divide(10, i - 1);
断点不是一次性的,再次执行代码会再次触发断点。此时我们可以删除断点(d),需要指出断点编号
(gdb) d 1(gdb) cContinuing.Breakpoint 2, main () at debug.cpp:66 int j = divide(10, i - 1);
最后,退出GDB (quit)
(gdb) quitA debugging session is active.Inferior 1 [process 15063] will be killed.Quit anyway? (y or n) y