[关闭]
@comzyh 2019-04-04T18:30:08.000000Z 字数 2792 阅读 1216

GDB 调试极简入门

GDB


GDB 是 GNU 提供的流行Debug 工具,GDB 可以对任何可执行文件进行Debug

使用GDB 启动程序

最简单的方法

gdb ./a.out

如何需要使用命令行参数,则需要使用 --args 参数

gdb --args ./a.out soomeargs --arg1 value 1 --args2 value2

进入GDB后使用 r 开始执行程序
当程序运行遇到关键错误时,GDB会捕获错误并开始调试

当然,最重要的一点,gdb 使用 quit 命令或者 Ctrl + D 退出

使用CoreDump调试程序

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 常用的调试命令

  1. 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. }

g++ debug.cpp -g -O0
gdb ./a.out

显然这段代码会产生除0错误, 先执行(输入r)gdb 输出如下

  1. (gdb) r
  2. Starting program: /home/comzyh/Projects/tmp/a.out
  3. Program received signal SIGFPE, Arithmetic exception.
  4. 0x0000555555554608 in divide (x=10, y=0) at debug.cpp:2
  5. 2 return x / y;
  6. (gdb)

我们能看到很多信息, 比如错误出现在 debug.cpp 文件的第3行,这行的代码是 return x / y;, 出现的错误是算数错误(Arithmetic exception)

现在可以使用××打印变量××(p 命令)功能,看看 x 和 y 到底是什么

  1. (gdb) p x
  2. $1 = 10
  3. (gdb) p y
  4. $2 = 0

我们知道了,错误是因为 y = 0,出现了除 0 错误

那么为什么会执行这段代码呢?可以使用 list 命令查看附近的代码. 需要指出的是,list 默认向下浏览代码,你可以使用l - 向上浏览代码

  1. (gdb) l
  2. 1 int divide(int x, int y) {
  3. 2 return x / y;
  4. 3 }
  5. 4 int main() {
  6. 5 for (int i = 10; i >= 0; i--) {
  7. 6 int j = divide(10, i - 1);
  8. 7 }
  9. 8 }

到这里我们知道了,y 是 0 的原因 divide 函数的输入参数 y 是 0,那么是谁调用了这个函数呢?我们可以使用 查看调用栈 功能 (bt命令)

  1. (gdb) bt
  2. #0 0x0000555555554608 in divide (x=10, y=0) at debug.cpp:2
  3. #1 0x0000555555554631 in main () at debug.cpp:6

我们可以看出,是debug.cpp 的第 6 行 main 函数调用了 divide 函数,还能看到divide 函数的实参是 x = 10, y = 0.
此时我们应当使用切换栈帧 (frame命令) 回到 main 函数查看调用 divide 的代码. #1 代表main 函数的栈帧编号为 1

  1. (gdb) f 1
  2. #1 0x0000555555554634 in main () at debug.cpp:6
  3. 6 int j = divide(10, i - 1);
  4. (gdb) l
  5. 1 int divide(int x, int y) {
  6. 2 return x / y;
  7. 3 }
  8. 4 int main() {
  9. 5 for (int i = 10; i >= 0; i--) {
  10. 6 int j = divide(10, i - 1);
  11. 7 }
  12. 8 }

我们通过切换栈帧和打印代码看到,传给 divide 函数的 y 参数是 循环变量 i. 我们可以打印此时 i 的值

  1. (gdb) p i
  2. $3 = 1
  3. (gdb) p i - 1
  4. $4 = 0

print 指令不仅可以查看变量的值,还能进行简单的运算,例如对vector的元素的访问,可以直接 p vec[10] 这样操作。

最后我们尝试下断点 (breakpoint). 只要gdb等待你的输入,你就可以添加断点,比如我们给 main 函数的 第6行添加断点并重新运行

  1. (gdb) b main.cpp:6
  2. No source file named main.cpp.
  3. Make breakpoint pending on future shared library load? (y or [n]) n
  4. (gdb) b debug.cpp:6
  5. Breakpoint 1 at 0x555555554622: file debug.cpp, line 6.
  6. (gdb) r
  7. The program being debugged has been started already.
  8. Start it from the beginning? (y or n) y
  9. Starting program: /home/comzyh/Projects/tmp/a.out
  10. Breakpoint 1, main () at debug.cpp:6
  11. 6 int j = divide(10, i - 1);
  12. (gdb) p i
  13. $3 = 10

GDB 的确帮我们中断了程序并进入调试,此时我们可以继续(continue)

  1. (gdb) c
  2. Continuing.
  3. Breakpoint 1, main () at debug.cpp:6
  4. 6 int j = divide(10, i - 1);

断点不是一次性的,再次执行代码会再次触发断点。此时我们可以删除断点(d),需要指出断点编号

  1. (gdb) d 1
  2. (gdb) c
  3. Continuing.
  4. Breakpoint 2, main () at debug.cpp:6
  5. 6 int j = divide(10, i - 1);

最后,退出GDB (quit)

  1. (gdb) quit
  2. A debugging session is active.
  3. Inferior 1 [process 15063] will be killed.
  4. Quit anyway? (y or n) y
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注