GDB是一个极为强大的调试工具,绝对比那些IDE下辖的调试器强大得多。在GDB的学习过程中一个核心的问题就是Call Stack的理解,这是GDB安身立命之根本。进一步的,就要涉及到core dump的使用了。既然是只谈论GDB的基础,那么本文中就不涉及到core dump的问题,有兴趣的可以自己google一下。
准备工作
gdb需要的文件是可执行文件。要想使用gdb去调试某个可执行文件,必须在用gcc或g++生成该文件的时候使用-g选项加入调试信息。
载入可执行文件
产生了可调试的可执行文件后,必须将其载入到gdb中才可进行调试。载入文件有两种办法:
1> 使用命令 gdb execfilename 载入;
2> 在进入gdb后,使用
file execfilename 或是
target exec execfilename
断点管理
使用break(简写为b即可)命令设置程序断点。
break:可以在程序运行时利用frame命令选定某一个stack frame,然后执行命令break。这样使得一旦程序执行到该选中stack frame中时,就停下。
break location:其中的location可以指定为当前源文件的第几行,或是针对当前位置的偏移量,也可以指定为函数名,甚至可以指定内存中的位置。
break location if condition:这是一个条件断点设置语句。
除了break之外,还有几个断点设置的关键字:
tbreak:在程序的一次运行中只停止一次。
rbreak regex:利用正则表达式设置一系列符合regex的断点。
可以使用info命令来查看断点信息,使用disable和enable来禁用启用断点,使用clear和delete来删除指定断点。
观察点管理
使用watch命令设置程序观察点。
watch expr:其中的expr可以是参数名,也可以是表达式。当expr的值被改写的时候程序就会暂停执行。
除了watch之外,还有几个观察点设置的关键字:
rwatch:当参数或是表达式的值被程序读取的时候暂停。
awatch:当参数或是表达式的值被读取或改写的时候都会暂停。
可以使用info命令来查看断点信息,使用disable和enable来禁用启用观察点,使用delete来删除指定观察点。
设置程序运行参数
set args:可指定运行时参数。(如:set args 10 20 30 40 50);
show args:命令可以查看设置好的运行参数。
变量管理
变量值显示:print var
变量值修改:set var=value
按指定地址输出内存值:x /NFU ADDR
其中,U指定一个单元的大小,可以是b(单字节)、h(双字节)、w(四字节)、g(八字节);N表示输出的单元个数;F表示输出的数据形式,可以是x(16进制整数格式 )、d(有符号十进制整数格式)、u(无符号十进制整数格式)、f(浮点数格式 )。
调试过程控制
step:单步执行,进入函数内部;
next:单步执行,不进入函数内部;
continue:恢复正常连续执行。
信息查询
info:这个应该是调试时用得最多的信息显示命令。它的主要作用是显示正在被调试的程序的一些指定信息,如breakpoints、watchpoints,frame中的args等等。
print:一般用来打印变量的当前值。
show:显示当前gdb本身的一些参数信息。
list:一般用于源文件的显示。
Call Stack
stack frame:当某个函数被调用的时候,它的调用入口,参数,局部变量都将会被保存在栈中的一块内存中,这块内存就叫stack frame。
call stack:所有的stack frames所占据的空间就叫call stack。最近被调用的函数的stack frame总是位于call stack的最里面,依次向外扩散:最里面的stack frame编号为0.
有了以上两个概念之后,我们就可以讲述gdb调试实现的机制了。gdb工作的基本单元就是stack frame,简称frame。比如用print var来打印变量名时,gdb是会在当前选中的frame(默认是当前工作的frame,除非手动选择别的frame)中寻找并打印该变量。这就是理解gdb工作的关键!
以下是一些操作frame并显示frame信息的命令:
frame framlocation:指定gdb工作的frame。其中,framlocation可以是frame的编号,也可以是frame的内存地址。
bt:显示当前程序Call stack中的函数调用顺序,或者说是frame的顺序。命令where和info stack和bt是一样的。
info frame:显示当前frame的详细信息。
info args:显示当前frame所对应的函数的参数信息。
info locals:显示当前frame所对应的函数的局部变量信息。
执行Shell命令
直接在gdb提示符后输入 shell commandname 。而如果需要在gdb中直接make源文件的话,直接在gdb提示符后输入 make make_args 即可;也可输入shell make make_args。