GDB常用调试命令以及多进程多线程调试

时间:2022-05-31 16:39:25


   今天把所有的GDB常用的调试命令都过了一遍,尤其是多线程调试。所以先做个总结,以免忘记。以后如果有新的有用的命令会继续追加。

    本文地址:(LYanger的博客:http://blog.csdn.net/freeelinux/article/details/53700266


一:普通命令


1.list命令

  • list  linenum      显示程序第linenum行周围的程序
  • list  function      显示函数名为function的函数的源程序
  • list                      显示当前行后面的源程序
  • list -                    显示当前行前面的源程序

2.run(r)

    运行命令。
  • run args         run命令可以直接接命令行参数值,也可以在执行run之前通过 set args + 参数值实现。 

3.break(b)

    打断点,使用方法:
  • b  linenum                        在某行打断点
  • b  +offset/-offset               在当前行号的前面或后面offset停住
  • b  filename:linenum         在某文件的某行打断点
  • b  filename:function         在某文件某个函数入口停住
  • b  *address                      在程序的运行地址处停住
  • b                                      没有参数在下一句停住
  • b where if condition         当某个条件满足时,在某一行停住(这个很有用,比如b 10 if ret == 5)
    对于break命令,我们要灵活使用。例如打多个断点。多线程程序中我们可以主函数中线程创建后立即打断点,执行线程函数入口打断点等。    关闭断点:delete(d) breakpoint-id

4.单步命令

    普通用法就不说了。
  • step count         一次性执行count步,如果有函数会进入函数
  • next count         一次执行count,不进入函数
  • finish                 运行程序,直到当前函数完成返回,并打印函数返回时的堆栈地址和返回值以及参数信息
  • until                   退出循环体(尤其是针对for循环这种,很烦的)

5.continue命令

    当程序被停住之后,可以使用continue(c)命令,恢复程序的运行直到程序结束,或到达下一个断点。这里要注意如果没有断点程序是会直接结束的。    

6.print(p)命令

    这个命令比较常用,用来查看我们想看的内容。比如有关数组可以看全部,也可以看从左到右某一部分: GDB常用调试命令以及多进程多线程调试
    print命令针对变量查看的输出格式有:
  • x 按十六进制格式显示变量
  • d 按十进制格式显示变量
  • u 按十六进制格式显示无符号整型
  • o 按八进制格式显示变量
  • t 按二进制格式显示变量t 按二进制格式显示变量
  • a 按十六进制格式显示变量
  • c 按字符格式显示变量
  • f 按浮点数格式显示变量

7.watch命令

    这个命令比较有用。watch一般用来观察某个表达式(变量也是一种表达式)的值是否有变化,如果有变化,马上停住程序。我们有一下几种方法设置观察点:
  • watch   expr                 为表达式expr设置一个观察点,一旦表达式值有变化,马上停住程序
  • rwatch  expr                  当表达式expr被读时,停住程序
  • awatch expr                  当表达式的值被读或被写时,停住程序。
  • info      watchpoints       列出所有观察点(info指令通常可以去套)
    举例如下,演示观测*i的值,一旦变化停下来:
GDB常用调试命令以及多进程多线程调试
         在循环中我们也可以使用watch,配合ignore,它是除了until命令之外又一个可以让我们跳出循环的方法,不过watch+ignore更强大,可以任意跳转到第i次循环。它们的意思就是观察一个变量,可以理解为断点,ignore这个断点多少次,然后用continue就可以直接跳过了。 GDB常用调试命令以及多进程多线程调试

8.examine命令

    使用该命令来查看内存地址中的值。语法是:x/u addr      addr表示一个内存地址。“x/”后的n、f、u都是可选的参数,n 是一个正整数,表示显示内存的长度,也就是说从当前地址向后显示几个地址的内容;f 表示显示的格式,如果地址所指的是字符串,那么格式可以是s,如果地址是指令地址,那么格式可以是i;u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4字节。u参数可以被一些字符代替:b表示单字节,h表示双字节,w表示四字节,g表示八字节。当我们指定了字节长度后,GDB会从指定的内存地址开始,读写指定字节,并把其当作一个值取出来。n、f、u这3个参数可以一起使用,例如命令“x/3uh 0x54320”表示从内存地址0x54320开始以双字节为1个单位(h)、16进制方式(u)显示3个单位(3)的内存。 GDB常用调试命令以及多进程多线程调试

9.jump命令

    jump命令不会改变程序栈的内容,一般只在同一函数内跳转。
  • jump linespec     指定下一条语句的运行点,linespec可以是linenum,filename+linenum,+offset这几种形式
  • jump address      跳到代码行的地址

10.signal命令

使用signal 信号名(如SIGINT)这种方式把信号发送给程序,如果程序注册了signal_handler函数,还可以进行相应的处理,帮助调试程序。

11.set命令

  • set args       设置命令行参数
  • set env environmentVarname=value 设置环境变量。如:set env USER=benben

12.call命令

  • call function     强制调用某函数
    强制调用某函数,它会显示函数返回值(如果函数返回值不是void)。print命令也可以完成该功能。

13.disassemble命令

    反汇编命令,查看执行时源代码的机器码。 GDB常用调试命令以及多进程多线程调试


二:多进程调试


1.单独调试子进程

    我们可以先运行程序,然后再另一终端使用ps -ef | grep "main"(main此处是可执行文件名)搜索到子进程pid,然后启动gdb,在将其附加(attach)到gdb调试器上。
  • attach child-pid        使用该命令后,直接run即可,和调试普通程序就没区别了
  • dettach                     脱离进程

2.使用调试器选项follow-fork-mode

    我们知道如果不设置任何选项,gdb默认调试父进程。调试器选项用法如下:
  • set follow-fork-mode mode     其中mode的可选值是parent和child,分别表示调试父进程和子进程。
  • info inferiors                            查询正在调试的进程
  • inferior processnum                切换进程
    默认设置下,在调试多进程程序时GDB只会调试主进程。但是GDB(>V7.0)支持多进程的分别以及同时调试,换句话说,GDB可以同时调试多个程序。只需要设置follow-fork-mode(默认值:parent)和detach-on-fork(默认值:on)即可。我们还可以使用catch fork指令,如果fork异常,会停止程序。
        follow-fork-mode  detach-on-fork   说明
        parent                     on               只调试主进程(GDB默认)
        child                        on               只调试子进程
        parent                     off               同时调试两个进程,gdb跟主进程,子进程block在fork位置
        child                        off               同时调试两个进程,gdb跟子进程,主进程block在fork位置
   设置方法:set follow-fork-mode [parent|child]   set detach-on-fork [on|off]
GDB常用调试命令以及多进程多线程调试

三:多线程调试


    gdb调试一般有两种模式:all-stop模式和no-stop模式(gdb7.0之前不支持no-stop模式)。     1.all-stop模式         在这种模式下,当你的程序在gdb由于任何原因而停止,所有的线程都会停止,而不仅仅是当前的线程。一般来说,gdb不能单步所有的线程。因为线程调度是gdb无法控制的。无论什么时候当gdb停止你的程序,它都会自动切换到触发断点的那个线程     2.no-stop模式(网络编程常用)         顾名思义,启动不关模式。当程序在gdb中停止,只有当前的线程会被停止,而其他线程将会继续运行。这时候step,next这些命令就只对当前线程起作用。         如果需要打开no-stop模式,可以向~/.gdbinit添加配置文件:
#Enable the async interface
set target-async 1
#If using the CLI, pagination breaks non-stop
set pagination off
#Finall, turn it on
set non-stop on
        gdb支持的命里有两种类型:前台的(同步的)和后台(异步 )的。区别很简单,同步的在输出提示符之前会等待程序report一些线程已经终止的信息,异步则是直接返回。所以我们需要set target-async 1。set pagination off不要出现 Type <return> to continue 的提示信息 。最后一步是打开。        下面是常用命令:
  • info threads                                       显示所有线程
  • thread id                                            切换到指定线程
  • break filename:linenum thread all     在所有线程相应行设置断点,注意如果主线程不会执行到该行,并且启动all-stop模式,主线程执行n或s会切换过去
  • set scheduler-locking off|on\step       默认off,执行s或c其它线程也同步执行。on,只有当前相称执行。step,只有当前线程执行
  • show scheduler-locking                     显示当前模式
  • thread apply all command                 每个线程执行同意命令,如bt。或者thread apply 1 3 bt,即线程1,3执行bt。
        主要是我们要用能用的上的,比如no-stop模式,一般多线程调试就很有用的。

四:core文件

  • ulimit -c unlimited           生成core文件,也可以是指定大小,然后使用gdb ./main core启动,bt查看调用栈即可。


    参考:https://segmentfault.com/a/1190000003733061                http://www.cnblogs.com/frankbadpot/archive/2010/06/23/1762916.html