1 info 用来描述你的程序的状态,比如info b就是显示出当前的程序的所有断点.
2 set 用来设置一些环境变量的值,比如set prompt $.
3 show用来描述gdb自己的状态.
编译要用-g选项.
然后用gdb +程序名,或者直接gdb后,用file + 文件名加载程序.
1 run/r 运行程序.
2 set args 设置程序的参数.
3 path directory 加一个目录到环境变量path
4 set directory 设置gdb的工作目录
5 pwd 当前的工作目录
6 attch process-id 调试 运行的进程
dettach 当调试进程完毕,release 掉gdb的控制.
调试多线程.
1 thread threadno 选择当前的线程.
2 info thread 查看当前程序的线程.
debug多进程:
默认情况下,当fork一个子进程之后,gdb会继续debug父进程,而子进程会运行下去.不过我们能够改变这个.
set follow-fork-mode mode 这里mode可以为parent或者child. parent是默认值,而child的话就是gdb继续debug子进程,而父进程会运行下去.
如果你想要同时debug父子进程,也可以设置:
set detach-on-fork mode 默认是on,也就是只能debug一个进程,如果改为off则可以同时debug父子进程.
保存一个书签稍后返回.
checkpoint 保存当前的程序的状态.
restart checkpoint-id 返回到checkpoint-id那个点.这个值可以用info checkpoint来查看
breakpoint,watchpoint以及catchpoint
breakpoint 就是断点.
watchpoint 就是用来检测变量的改变,他可以看做是特殊的断点,也就是当变量改变时停止程序.
catchpoint 是另外一种特殊的断点,用来监测某一事件的发生,有点类似其它语言中的异常.
1 设置Breakpoint
break location 设置断点.location可以为行号,函数名或者指令地址.
break 设置断点,不过这个断点为当前栈帧的下一条指令.
break location if condition 当condition为真时,程序到达这个断点才起作用.
tbreaks args, 一次性的断点.
rbreaks regex 设置断点在所有与regex匹配的函数.这个正则表达式的语义与grep的相同.
2 设置watchpoint
watch expr[thread threadno] 设置检测变量expr,后面可以跟着改变这个变量的线程.如果跟着线程号,则说明只有当这个线程改变变量时,程序才会stop.
rwatch expr[thread threadno] 上面是监测变量改变,而这个命令是监测程序读取变量.
awatch expr[thread threadno] 当expr要么被读,要么被写时,程序直接break.
3 设置catchpoint
catch event 当event发生的时候程序停止.
event可以是下面的几种类型: throw ,catch,exception,exception unhandled,assert,exec,fork,vfork.
tcatch event 一次性的监测事件.
删除断点
clear 删除在当前的栈帧的将要被执行的下一条指令断点.
clear location 删除location位置的断点.其实更有用的是下面几个命令:
clear function,clear filename:function ,clear linenum,clear filename:linenum.
delete [breakpoints][range...] 其实也就是删除多个断点,如果没有range,咋就是删除全部断点.
关闭断点
一个断点(包括 watchpoint和catchpoint)可以有下面四种状态.
打开,关闭,enabled once,enabled for deletion
disable [breakpoints][range..] 关闭指定的断点或者全部断点(如果没有range)
enable [breakpoints][range..] 打开指定的断点或者全部断点(如果没有range)
enable [breakpoints] once range 临时打开指定的断点(也就是说是一次性的,.
enable [breakpoints] delete range 临时打开指定的断点并只工作一次,也就是一次之后这个断点将会被删除.
break condition
contidition bnum expression 当expression为真的时候,程序到达这个断点才会停止.
contidition bnum 从断点bnum删除掉一个condition.
ignore bnum count 设置一个断点bnum的忽略次数为count.也就是只有count次数后,这个断点才会起作用.
断点命令列表
这个主要是用来当到达这个断点,程序停止后,你想要执行一连串的命令.格式为:
- command [bnum]
- ... command-list ..
- end
指定一堆命令给断点bnum.如果想删除命令的话就把command-list置为空就行了.
如果没有bnum,则这个command指的是最后设置的一个断点.
这里有个例子:
- break 403
- commands
- ///不输出任何东西
- silent
- ///改变x的值
- set x = y + 4
- ///然后continue
- cont
- end
接下来来看continue和step
continue表示让程序继续执行,直到下一个断点或者执行完毕。
step表示让程序执行一行代码或者说一条机器指令(依赖于你选择的命令)。
下面来看命令:
- continue [ignore-count]
- c [ignore-count]
- fg [ignore-count]
这几个命令都是resume一个程序,然后参数ignore-count表示忽略当前这个断点的次数。
step [count]
继续运行程序直到抵达一个新的代码行(它会跟入函数).这里要记住step只会停止在source line的第一条指令。
如果加上count参数则表示它会step count次。如果遇到断点则会停止。
next [count]
和step很类似,区别就是不会跟进函数。
- set step-mode
- set step-mode on
- set step-mode off
on就可以使step停止在没有debug信息的函数的第一条指令上。off则是直接执行完这个函数。
这里要注意上面的命令都只是跳一行代码。而不是一条指令。
util 继续运行直到source line通过了当前的行。这个命令主要是针对循环语句中的step。比如你在循环结尾设置util,则只有当循环退出时才会在这个断点停止。而不是每次都停止。
util location 继续运行直到指定的location,或者当前的栈帧返回。
advance location 继续运行直到给定的location,这个相比与上面的命令,它就象全局的。
stepi [arg]
执行一条机器指令。 arg表示次数。
nexti [arg]
和next类似只不过执行的是一条机器指令。
下来来看信号。
gdb可以监测在你的程序中的任何信号。
来看命令。
handle signal [keywords...]
这个命令用来改变信号signal(名字或者数字)在gdb中的行为。
其中关键就是keywords.在这里keywords可以为下面几种类型:
1 nostop gdb接收到信号不会停止程序,而只是打印出一段message
2 stop 和上面类似只不过会停止程序。
3 print 当信号发生必须打印一条消息通知。
4 noprint 信号发生,gdb将不会打印任何东西。
5 pass和noignore 这两个是同义的。表示信号对你的程序是可见的。
6 nopass和ignore 这两个也是同义的。和上面相反。。
在gdb中,当你的程序由于一个信号而停止后,直到你继续执行,否则信号对你的程序是不可见得。也就是说当gdb捕捉到信号,我们可以用nopass或者ignore来使信号对我们的程序为不可见。
最后来看下多线程程序的调试。
首先来看多线程调试的几种模式。
1 all-stop模式。
在这种模式中,当你的程序在gdb由于任何原因而停止,此时所有的线程都会停止。而不仅仅是当前的线程。一般来说gdb不能单步所有的线程。因为线程调度gdb是无法控制的。无论什么时候当gdb停止你的程序,它都会自动切换到触发断点的那个线程。
在一些os中我们可以通过lock线程调度器,从而达到只有一个线程在运行。
set scheduler-locking mode
设置模式,如果mode是off,则表示没有lock,则任何线程在任何时候都有可能在运行。当mode为on的时候,锁定其他的线程,也就是只有当前线程在执行。也就是你单步的时候其他线程是不会运行的。这个对我们只关注本线程比较重要。
默认情况下,当你键入step或者next命令时,gdb只允许当前进程的线程运行。我们可以通过命令来修改这个默认值。
set schedule-multiple mode
当mode为on则所有进程的所有线程都匀许运行。否则只有当前的进程的线程能够resume。
2 none-stop模式。
顾名思义,当程序在gdb中停止,只有当前的线程会被停止,而其他的线程将会继续运行。
这个时候step,next这些命令就只对当前的线程起作用。
我们要打开这个模式需要这样操作:
- # Enable the async interface.
- set target-async 1
- # If using the CLI, pagination breaks non-stop.
- set pagination off
- # Finally, turn it on!
- set non-stop on
这里要注意打开这个模式必须得在你attach或运行这个程序或者进程之前才能进行。
Background Execution
gdb执行命令有两种类型:前台的(同步)和后台(异步)的。
区别很简单,前台的话,gdb在输出提示符之前会等待程序report一些线程已经终止的信息。而异步的则是直接返回。
我们需要显式打开异步模式。
set target-async on
下面就是支持异步的命令:
- run
- attach
- step
- stepi
- next
- nexti
- continue
- finish
- until
通过上面我们可以看到异步模式主要用在none-stop模式中。
如果你想停止后台运行的程序,那么使用interrupt
在all-stop模式中interrupt将会停止所有的线程。而在none-stop中只会停止当前线程。interrupt -a此时就能停止所有线程。’
当你有多个线程,你此时只想给某个线程设置断点,这个时候可以用这个命令:
break linespec thread threadno
break linespec thread threadno if ...
linespec为源码行号,threadno为线程id。
最后来看下多线程调试中可能会遇到的一个问题:
如果一个线程在一个断点,或者由于其他什么原因停止,此时另外的线程阻塞在一个系统调用。这个时候这个系统调用就有可能会过早的返回。因此我们在调用系统调用,最好都要检测它的返回值。
举个例子:
sleep (10);
这个我们应该改成这样:
- int unslept = 10;
- while (unslept > 0)
- unslept = sleep (unslept);