七、八年前写过一篇《用 GDB 调试程序》,于是,从那以后,很多朋友在 MSN 上以及给我发邮件询问我关于 GDB 的问题,一直到今天,还有人在问 GDB 的相关问题。这么多年来,有一些问题是大家反复在问的,一方面,我觉得我以前的文章可能没有说清楚,另一方面,我觉得大家常问的问题正是最有用的,所以,在这里罗列出来。希望大家补充。

一、多线程调试

多线程调试可能是问得最多的。其实,重要就是下面几个命令:

  • info thread 查看当前进程的线程。
  • thread 切换调试的线程为指定 ID 的线程。
  • break file.c:100 thread all 在 file.c 文件第 100 行处为所有经过这里的线程设置断点。
  • set scheduler-locking off|on|step,这个是问得最多的。在使用 step 或者 continue 命令调试当前被调试线程的时候,其他线程也是同时执行的,怎么只让被调试程序执行呢?通过这个命令就可以实现这个需求。

    • off 不锁定任何线程,也就是所有线程都执行,这是默认值。
    • on 只有当前被调试程序会执行。
    • step 在单步的时候,除了 next 过一个函数的情况 (熟悉情况的人可能知道,这其实是一个设置断点然后 continue 的行为) 以外,只有当前线程会执行。

二、调试宏

这个问题超多。在 GDB 下,我们无法 print 宏定义,因为宏是预编译的。但是我们还是有办法来调试宏,这个需要 GCC 的配合。

在 GCC 编译程序的时候,加上-ggdb3参数,这样,你就可以调试宏了。

另外,你可以使用下述的 GDB 的宏调试命令 来查看相关的宏。

  • info macro – 你可以查看这个宏在哪些文件里被引用了,以及宏定义是什么样的。
  • macro – 你可以查看宏展开的样子。

三、源文件

这个问题问的也是很多的,太多的朋友都说找不到源文件。在这里我想提醒大家做下面的检查:

  1. 编译程序员是否加上了 - g 参数以包含 debug 信息。
  2. 路径是否设置正确了。使用 GDB 的 directory 命令来设置源文件的目录。

下面给一个调试 / bin/ls 的示例(ubuntu 下)

  1. $ apt-get source coreutils
  2. $ sudo apt-get install coreutils-dbgsym
  3. $ gdb /bin/ls
  4. GNU gdb (GDB) 7.1-ubuntu
  5. (gdb) list main
  6. 1192 ls.c: No such file or directory.
  7. in ls.c
  8. (gdb) directory ~/src/coreutils-7.4/src/
  9. Source directories searched: /home/hchen/src/coreutils-7.4:$cdir:$cwd
  10. (gdb) list main
  11. 1192 }
  12. 1193 }
  13. 1194
  14. 1195 int
  15. 1196 main (int argc, char \*\*argv)
  16. 1197 {
  17. 1198 int i;
  18. 1199 struct pending \*thispend;
  19. 1200 int n\_files;
  20. 1201

四、条件断点

条件断点是语法是:break [where] if [condition],这种断点真是非常管用。尤其是在一个循环或递归中,或是要监视某个变量。注意,这个设置是在 GDB 中的,只不过每经过那个断点时 GDB 会帮你检查一下条件是否满足。

五、命令行参数

有时候,我们需要调试的程序需要有命令行参数,很多朋友都不知道怎么设置调试的程序的命令行参数。其实,有两种方法:

  1. gdb 命令行的 –args 参数
  2. gdb 环境中 set args 命令。

六、gdb 的变量

有时候,在调试程序时,我们不单单只是查看运行时的变量,我们还可以直接设置程序中的变量,以模拟一些很难在测试中出现的情况,比较一些出错,或是 switch 的分支语句。使用 set 命令可以修改程序中的变量。

另外,你知道 gdb 中也可以有变量吗?就像 shell 一样,gdb 中的变量以 $ 开头,比如你想打印一个数组中的个个元素,你可以这样:

  1. (gdb) set $i = 0
  2. (gdb) p a\[$i++\]
  3. ... #然后就一路回车下去了

当然,这里只是给一个示例,表示程序的变量和 gdb 的变量是可以交互的。

七、x 命令

也许,你很喜欢用 p 命令。所以,当你不知道变量名的时候,你可能会手足无措,因为 p 命令总是需要一个变量名的。x 命令是用来查看内存的,在 gdb 中 “help x” 你可以查看其帮助。

  • x/x 以十六进制输出
  • x/d 以十进制输出
  • x/c 以单字符输出
  • x/i 反汇编 – 通常,我们会使用 x/10i $ip-20 来查看当前的汇编($ip 是指令寄存器)
  • x/s 以字符串输出

八、command 命令

有一些朋友问我如何自动化调试。这里向大家介绍 command 命令,简单的理解一下,其就是把一组 gdb 的命令打包,有点像字处理软件的 “宏”。下面是一个示例:

  1. (gdb) break func
  2. Breakpoint 1 at 0x3475678: file test.c, line 12.
  3. (gdb) command 1
  4. Type commands for when breakpoint 1 is hit, one per line.
  5. End with a line saying just "end".
  6. >print arg1
  7. >print arg2
  8. >print arg3
  9. >end
  10. (gdb)

当我们的断点到达时,自动执行 command 中的三个命令,把 func 的三个参数值打出来。

(全文完)

https://coolshell.cn/articles/3643.html