暂停机制: 有3种方式可以通知GDB暂停程序的执行。 a.断点: 通知GDB在程序中的特定位置暂停执行; b.监视点:通知GDB当特定内存位置(或者涉及一个或多个位置的表达式)的值发生变化时暂停执行; c.捕获点: 通知GDB当特定事件发生时暂停执行; 容易混淆的是,在GDB文档中将这3个机制都称为断点。这可能是因为它们的很多属性和命令都相同; GDB关于删除断点的delete命令:删除断点、监视点和捕获点; GDB中关于“位置”的含义非常灵活,它可以指各种源代码行、代码地址、源代码文件中的行号或者函数的入口等; 设置格式: 文件名:行号 断点可以通过函数名,当前文件内的行号来设置,也可以先指定文件名再指定行号,还可以指定与暂停位置的偏移量,或者用地址来设置;
程序员创建的每个断点(包括断点、监视点、和捕获点)都被标识为从1开始的唯一整数标识符;这个标识符用来执行该断点上的各种操作, break function 在函数function() 的入口(第一行可执行代码)处设置断点, 例如: break main 在源文件filename的line处设置断点,如果filename不在当前目录中,则可以给出相对路径名或者完全路径名来帮助GDB查找该文件。 例如: break source/bed.c:35 break bed.c:function 临时断点(只生效一次): 使用tbreak命令设置,它与break采用相同类型的参数, 在任何给定时间,GDB都有一个焦点,可以将它看作当前“活动”文件,这意味着除非对命令做了限定。否则都是在具有GDB的焦点的文件上执行命令。 默认情况下,具有GDB的初始焦点的文件是包含main()函数的文件,但是当发生如下任一动作时,焦点会转移到不同的文件上; 1、向不同的源文件应用list命令; list function 2、进入位于不同的源文件文件中的代码; 3、当在不同的源代码文件中执行代码时GDB遇到断点; 在调试会话期间不应退出GDB, 例如,当发现并修复了一个程序错误,但是其他程序错误仍然存在时,不应当退出GDB然后重新进入来使用程序的新版本。这样做有些不必要地繁琐, 而且还会不得不重新进入断点; 如果在修改和重新编译代码时没有退出GDB,那么下次执行GDB是的run命令时,GDB会感知到代码修改,并自动重新加载新版本; 假如要关机了,又想保存GDB的断点等信息;那怎么办呢? 可以将断点放在源代码所在目录的.gdbinit 启动文件中; 例如,当发现并修复了一个程序错误,但是其他程序错误仍然存在时,不应当退出GDB然后重新进入来使用程序的新版本。这样做有些不必要地繁琐, 而且还会不得不重新进入断点; 如果在修改和重新编译代码时没有退出GDB,那么下次执行GDB是的run命令时,GDB会感知到代码修改,并自动重新加载新版本; 如果要波啊流断点以便用户使用,暂时又不希望GDB停止执行; 可以禁用它们,在以后需要时再启用。 使用disable breakpoint-list命令禁用断点。 enable breakpoint-list命令启用断点; 其中breakpoint_list是使用空格分隔的列表,其中有一个或多个断点标识符; disable不带参数执行,将禁用所有现有断点,enable也一样; i b命令也能指出特定断点引起GDB停止程序执行多是次; 使用commands命令设置命令列表: 其中breakpoint-number是要将命令添加到其上的断点的标识符,commands是用新行分隔的任何有效GDB命令列表。 逐条输入命令,然后键入end表示命令完毕。 从那以后,每当GDB在这个断点处中断时,它都会执行你输入的任何命令。 commands breakpoint-number ...... commands ...... end 例如: commands 1 printf "var value is %d\n", n end 对比: commands 1 silent printf "var value is %d\n", n end GDB的define命令创建宏 define print_and_go printf "%d\n", n continue end 监视点: 它是一种特殊类型的断点,是要求GDB暂停程序执行的指令。 区别在于监视点没有“住在”某一行源代码中,而是,监视点是指示GDB每当某个表达式改变了值就暂停执行的指令。 例如: watch i 它会使得每当i改变值时GDB就暂停; display命令(简写disp):这个命令要求GDB在执行中每次有暂停(由于有断点,使用next、step命令等)时就输出指定条目; disp i 查看i的值; 通过GDB中的call命令来调用源代码中的函数: 例如: commands 2 printf "************" call function(参数) end 人工数组(artificial array) *pointer@number_of_elements p p *x@25 GDB中, (监视局部变量) info locals 命令得到当前栈帧中哦所有局部变量的值列表; GDB中检查内存 在有些情况下,可能希望检测给定地址的内存,而不是通过变量的名称。 x命令; p/x var GDB中设置当前的某个变量的值; set x = 12 set args命令设置程序的命令行参数: set args 1 2 4 检查当前函数参数的info args命令: 全局变量:errno 在文件/usr/include/linux/errno.h 或 /usr/include/asm-generic/errno.h 中有对错误数值的详细解释; 使用strace,跟踪程序做过的所有系统调用; 关于网络的调试,Ethereal程序跟踪单个TCP/IP分组; 与线程相关的GDB命令用法汇总: 确定每个线程在做什么,可以通过 GDB的info threads来确定: 检查线程1: thread 1 break 88 thread 3 (当线程3达到源代码行88时停止执行) break 88 thread 3 if x==y (当线程3到达3源代码行88,并且变量x和y相等时停止执行)
layout next 命令: 源代码、汇编、二进制之间转换 还有一组专用的gdb变量可以用来检查和修改计算机的通用寄存器,gdb提供了每一台计算机中实际使用的4个寄存器的标准名字: $pc : 程序计数器 $fp : 帧指针(当前堆栈帧) $sp : 栈指针 $ps : 处理器状态
|
|