@清晰掉 GDB调试器中的战斗机

时间:2021-06-12 19:03:22
GDB 的命令很多,本文不会全部介绍,仅会介绍一些最常用的。在介绍之前,先介绍GDB中的一个非常有用的功能:补齐功能。它就如同Linux下SHELL中的命令补齐一样。当你输入一个命令的前几个字符,然后输入TAB键,如果没有其它命令的前几个字符与此相同,SHELL将补齐此命令。如果有其它命令的前几个字符与此相同,你会听到一声警告声,再输入TAB键,SHELL将所有前几个字符与此相同的命令全部列出。而GDB中的补齐功能不仅能补齐GDB命令,而且能补齐参数。

本文将先介绍常用的命令,然后结合一个具体的例子来演示如何实际使用这些命令。下面的所有命令除了第一条启动GDB命令是在SHELL下输入的,其余都是GDB内的命令。大部分GDB内的命令都可以仅输入前几个字符,只要不与其它指令冲突。如quit可以简写为q,因为以q打头的命令只有quit。 List可以简写为l,等等。

1.启动GDB

你可以输入GDB来启动GDB程序。GDB 程序有许多参数,在此没有必要详细介绍,但一个最为常用的还是要介绍的:如果你已经编译好一个程序,我们假设文件名为hello,你想用GDB调试它,可以输入gdb hello来启动GDB并载入你的程序。如果你仅仅启动了GDB,你必须在启动后,在GDB中再载入你的程序。

2.载入程序 === file

在GDB内,载入程序很简单,使用file命令。如file hello。当然,程序的路径名要正确。

退出GDB === quit

在GDB的命令方式下,输入quit,你就可以退出GDB。你也可以输入'C-d'来退出GDB。

3.运行程序 === run

当你在GDB中已将要调试的程序载入后,你可以用run命令来执行。如果你的程序需要参数,你可以在run指令后接着输入参数,就象你在SHELL下执行一个需要参数的命令一样。

4.查看程序信息 === info

info指令用来查看程序的信息,当你用help info查看帮助的话,info指令的参数足足占了两个屏幕,它的参数非常多,但大部分不常用。我用info指令最多的是用它来查看断点信息。

4.1 查看断点信息

info br

br是断点break的缩写,记得GDB的补齐功能吧。用这条指令,你可以得到你所设置的所有断点的详细信息。包括断点号,类型,状态,内存地址,断点在源程序中的位置等。

4.2 查看当前源程序

info source

4.3 查看堆栈信息

info stack

用这条指令你可以看清楚程序的调用层次关系。

4.4 查看当前的参数

info args

4.5 传递参数
run [arglist] 或者 set args [arglist]

5.列出源一段源程序 === list

5.1 列出某个函数

list FUNCTION

6. 给变量赋值 === set

6.1 set variable 变量=值

7. 使中断失效或有效 === disable / enable

7.1 disable 断点编号 可以使某个断点失效 如diable main 可使break main 命令失效

7.2 enable 断点编号 可以使某个断点恢复有效

8. 删除断点 ===clean / delete

disable只是让某个断点暂时失效,断点依然存在于程序中.如果要彻底删除某个断点,可以使用clear
或delete命令.

8.1 clear: 删除程序中所有的断点.

8.2 clear 行号: 删除此行的断点.

8.3 clear 函数名: 删除该函数的断点.

8.4 delete 断点编号: 删除指定编号的断点.如果一次要删除多个断点,各个断点编号以空格隔开

=========================================================================
当程序出错并产生core 时
快速定位出错函数的办法
gdb 程序名 core文件名(一般是core,也可能是core.xxxx)

调试程序使用的键
r run 运行.程序还没有运行前使用
c cuntinue 继续运行。运行中断后继续运行
q 退出
kill 终止调试的程序
h help 帮助
<tab> 命令补全功能

step 跟入函数
next 不跟入函数
b breakpoint 设置断点。
用法:
b 函数名 对此函数进行中断
b 文件名:行号 对此文件中指定行中断.假如是当前文件,那么文件名与:号可以
省略
看当前断点数使用info break.禁止断点disable 断点号.删除delete 断点号.

l list 列出代码行。一次列10 行。连接使用list将会滚动显示. 也可以在list 后面
跟上 文件名:行号
watch 观察一个变量的值。每次中断时都会显示这个变量的值
p print 打印一个变量的值。与watch不同的是print只显示一次
这里在顺便说说如何改变一个 value. 当你下指令 p 的时候,例如你用 p b,
这时候你会看到 b 的 value, 也就是上面的 $1 = 15.
你也同样可以用 p 来改变一个 value, 例如下指令 p b = 100 试试看,
这时候你会发现, b 的 value 就变成 100 了:$1 = 100.

网上抄录
基本的使用方法简介
前言
程序代码中的错误可分为数类,除了最轻易除错的语法错误,编译程序会告诉你错误所在外,大部分的错误都可以归类为执行时错误。GDB 的功能便是寻找执行时错误。假如没有除错程序,我们只能在程序中加入输出变量值的指令来了解程序执行的状态。有了 GDB 除错程序,我们可以设定在任何地方停止程序的执行,然后可以随意检视变量值及更动变量,并逐行执行程序。
一个除错程序执行的流程通常是这样的:

1. 进入除错程序并指定可执行文件。
2. 指定程序代码所在目录。
3. 设定断点后执行程序。
4. 程序于断点中断后,可以 (1)检视程序执行状态;检视变量值或变更变量值 (2) 逐步执行程序,或是全速执行程序到下一个断点或是到程序结束为止。
5. 离开除错程序。
以下将分为下列数项分别介绍:
1. 进入 GDB 及指定可执行档
2. 指定程序代码所在目录及检视程序代码
3. 断点的设定与清除
4. 全速及逐步执行程序
5. 检视及更改变量值
6. 检视程序执行状态
7. 读取 Core 文件信息
进入 GDB 及指定可执行档:
1. 进入 GDB 并读入可执行档 (档名为 'PROGRAM'),预备进行除错。
gdb PROGRAM
指定程序代码所在目录及检视程序代码
1. 增加目录 DIR 到收寻程序代码的目录列表 (假如你的程序代码和可执行档放在同一个目录下,就不须指定程序代码所在目录。):
(gdb) Directory DIR

2. 检视程序代码,格式计有:
(gdb) list => 显示目前执行程序代码前后各五行的程序代码;或是显示从上次 list 之后的程序代码
(gdb) list function => 显示该程序开始处前后五行的程序代码。
(gdb) list - =>上次显示程序代码的前面的十行。
断点的设定与清除
1. 设定断点(指令为 break,可简写为 (b),格式计有:
(gdb) break filename.c:30 => 在 filename.c 的第三十行处停止执行。
(gdb) break function => 在进入 function 时中断程序的执行。
(gdb) break filename.c:function => 在程序代码档 filename.c 中的函数 function 处设定断点。
(gdb) break => 在下一个将被执行的命令设定断点。
(gdb) break ... if cond => 只有当 cond 成立的时候才中断。cond 须以 C 语言的语法写成。

2. 显示各个断点的信息。
(gdb) info break

3. 清除断点(命令为 clear),格式同 break 。例如 :
(gdb) clear filename.c:30

4. 清除断点,NUM 是在 info break 显示出来的断点编号。
(gdb) delete NUM
全速及逐步执行程序
1. 从程序开头全速执行程序,直到碰到断点或是程序执行完毕为止。
(gdb) run

2. 在程序被中断后,全速执行程序到下一个断点或是程序结束为止 (continue 指令可简写为 c)。
(gdb) continue

3. 执行一行程序. 若呼叫函数, 则将该包含该函数程序代码视为一行程序 (next 指令可简写为 n)。

(gdb) next

4. 执行一行程序. 若呼叫函数, 则进入函数逐行执行 (step 指令可简写为 s)。
(gdb) step

5. 执行一行程序,若此时程序是在 for/while/do loop 循环的最后一行,则一直执行到循环结束后的第一行程序后停止 (until 指令可简写为 u)。
(gdb) until

6. 执行现行程序到回到上一层程序为止。
(gdb) finish
检视及更改变量值
1. print 叙述,显示该叙述执行的结果 (print 指令可简写为 p)。如
(gdb) print a => 显示 a 变量的内容.
(gdb) print sizeof(a) => 显示 a 变量的长度.

2. display 叙述,在每个断点或是每执行一步时显示该叙述值。如
(gdb) display a

3. 更改变量值:
(gdb) print (a=10) => 将变量 a 的值设定为 10.

检视程序执行状态
1. 查看程序执行到此时,是经过哪些函数呼叫的程序 (backtrace 指令可简写为 BT),也就是查看函数呼叫堆栈。
(gdb) backtrace
读取 Core 文件信息
1. 读入 PROGRAM 及 PROGRAM.CORE 档,可检视 Core Dump 时程序变量值及程序流程状态 。
gdb PROGRAM core
说明:'core' 档案是由 PROGRAM 档执行后,碰到 Core Dump 时产生的 Core 檔檔名。假如你还需要该 Core 档,我们建议你将该档案档名更改为 PROGRAM.core。在输入上述命令后,你可以用 GDB 提供的检视变量值以及检视程序执行状态来读取程序 Core Dump 时的状态。

补充一点儿
(最近我刚在用gdb,感觉还不错,不过有的时候还是眼睛加printf更好
使)
我更多是用gdb直接调试程序,而不是core

gdb 可执行文件名 (进入gdb环境)

set ags -? (设置参数,即向你要调试的程序传参)

另外,用break functionName 和break lineNum都可以设置断点
用info break 可以查看当前断点的信息
用clear清除断点的时候要和break相对应
假如是用functionName设置的断点就只能 clear functionName
用lineNum设置的断点就只能用clear lineNum

BTW:可以到www.sunfreeware.com去下载gdb

以上是我近几天学习gdb的一点经验,都是我犯过错误的地方
希望对与我等同类的gdb菜鸟们有所帮助。

假如是刚开始时也可以使用ddd
图形化的

另外对于断点可以使用
delete 断点号
删除
断点号可以使用info break 得到

我也说说我的体会:
r (程序运行的参数) 可以设置运行时的参数
shell 运行shell环境的东西,比如shell make,完了以后直接r重新启动程序,就可以带上次已经加过的参数运行,不用再敲参数了
b 文件名:行号 有的时候很有用,嘿嘿
info b 看断点信息
disable 断点号 暂停某个断点
enable 断点号 启动某个断点
delete (断点号) 删除某个断点,不带断点号,全部删除
x/32uxb 指针 以16进制显示该指针指向的32个字节的数据

在Linux的c编程中大家经常用gdb调试程序,以下是一些常用的指令
1.break FUNCTION
在某个函数上设置断点。函数重载时,有可能同时在几个重载的函数上设置了断点

break +OFFSET
break -OFFSET
在当前程序运行到的前几行或后几行设置断点

break LINENUM
在行号为LINENUM的行上设置断点

break FILENAME:LINENUM
在文件名为FILENAME的原文件的第LINENUM行设置断点

break FILENAME:FUNCTION
在文件名为FILENAME的FUNCTION函数上设置断点
当你的多个文件中可能含有相同的函数名时必须给出文件名。

break *ADDRESS
在地址ADDRESS上设置断点,这个命令答应你在没有调试信息的程序中设置断点

break
当break命令不包含任何参数时,break命令在当前执行到的程序运行栈中的
下一条指令上设置一个断点。除了栈底以外,这个命令使程序在一旦从当前
函数返回时停止。相似的命令是finish,但finish并不设置断点。这一点在
循环语句中很有用。gdb在恢复执行时,至少执行一条指令。

break ... if COND
这个命令设置一个条件断点,条件由COND指定;在gdb每次执行到此断点时
COND都被计算当COND的值为非零时,程序在断点处停止

ignore BNUM COUNT'
设置第BNUM号断点的被忽略的次数为'COUNT',即断点BNUM再执行到第COUNT+1
次时程序停止

tbreak ARGS 或者简写为 tb
设置断点为只有效一次。ARGS的使用同break中的参量的使用

hbreak ARGS
设置一个由硬件支持的断点。这个命令的主要目的是用于对EPROM/ROM程序的调试
因为这条命令可以在不改变代码的情况下设置断点。这可以同SPARCLite DSU一起
使用。当程序访问某些变量和代码时,DSU将设置“陷井”。注重:你只能一次使用
一个断点,在新设置断点时,先删除原断点

thbreak ARGS'
设置只有一次作用的硬件支持断点

rbreak REGEX
在所有满足表达式REGEX的函数上设置断点。这个命令在所有相匹配的函数上设置无
条件断点,当这个命令完成时显示所有被设置的断点信息。这个命令设置的断点和
break命令设置的没有什么不同。当调试C++程序时这个命令在重载函数上设置断点时
非常有用。

info breakpoints [N]
info break [N]
info watchpoints [N]
显示所有的断点和观察点的设置表,有下列一些列
*Breakpoint Numbers*----断点号
*Type*----断点类型(断点或是观察点)
*Disposition*---显示断点的状态
*Enabled or Disabled*---使能或不使能。'y'表示使能,'n'表示不使能。
*Address*----地址,断点在你程序中的地址(内存地址)
*What*---地址,断点在你程序中的行号。
假如断点是条件断点,此命令还显示断点所需要的条件。
带参数N的'info break'命令只显示由N指定的断点的信息。
此命令还显示断点的运行信息(被执行过几次),这个功能在使用'ignore'
命令时很有用。你可以'ignore'一个断点许多次。使用这个命令可以查看断点
被执行了多少次。这样可以更快的找到错误。

maint info breakpoints
显示所有的断点,无论是你设置的还是gdb自动设置的。
断点的含义:
breakpoint:断点,普通断点
watchpoint:普通观察点
longjmp:内部断点,用于处理'longjmp'调用
longjmp resume:内部断点,设置在'longjmp'调用的目标上
until:'until'命令所使用的内部断点
finish:finish'命令所使用的内部断点

2.watch EXPR
为表达式(变量)expr设置一个观察点。一量表达式值有变化时,马上停住程序。
这个命令使用EXPR作为表达式设置一个观察点。GDB将把表达式加入到程序中并监
视程序的运行,当表达式的值被改变时GDB就使程序停止。这个也可以被用在SPARClite
DSU提供的新的自陷工具中。当程序存取某个地址或某条指令时(这个地址在调试寄
存器中指定),DSU将产生自陷。对于数据地址DSU支持'watch'命令,然而硬件断点寄
存器只能存储两个断点地址,而且断点的类型必须相同。就是两个'rwatch'型断点

或是两个'awatch'型断点。

rwatch EXPR'
设置一个观察点,当EXPR被程序读时,程序被暂停。

awatch EXPR'
设置一个观察点,当EXPR被读出然后被写入时程序被暂停。

info watchpoints
在多线程的程序中,观察点的作用很有限,GDB只能观察在一个线程中的表达式的值
假如你确信表达式只被当前线程所存取,那么使用观察点才有效。GDB不能注重一个
非当前线程对表达式值的改变。

rwatch <expr>
当表达式(变量)expr被读时,停住程序。

awatch <expr>
当表达式(变量)的值被读或被写时,停住程序。

info watchpoints
列出当前所设置了的所有观察点。

3.catch EXCEPTIONS
使用这个命令在一个被激活的异常处理句柄中设置断点。EXCEPTIONS是一个你要抓住
的异常。你一样可以使用'info catch'命令来列出活跃的异常处理句柄。

GDB中对于异常处理由以下情况不能处理:
* 假如你使用一个交互的函数,当函数运行结束时,GDB将象普通情况一样把控制返
回给你。假如在调用中发生了异常,这个函数将继续运行直到碰到一个断点,一个信号
或是退出运行。
* 你不能手工产生一个异常( 即异常只能由程序运行中产生 )
* 你不能手工设置一个异常处理句柄。
有时'catch'命令不一定是调试异常处理的最好的方法。假如你需要知道异常产生的
确切位置,最好在异常处理句柄被调用以前设置一个断点,这样你可以检查栈的内容。
假如你在一个异常处理句柄上设置断点,那么你就不轻易知道异常发生的位置和原因。
要仅仅只在异常处理句柄被唤醒之前设置断点,你必须了解一些语言的实现细节。

3.cont N
第N次经过该断点时才停止程序运行

4.enable 断点编号
恢复暂时失活的断点,要恢复多个编号的断点,可用空格将编号分开

5.disable 断点编号
使断点失效,但是断点还在

6.delete 断点编号或者表达式
删除某断点

7.clear 断点所在行号
清除某断点

8.查看断点列表
info break

9.watch counter>15
当counter>15的时候程序终止

10.当程序崩溃的时候linux会生成一个core文件,可以用
gdb a.out core
where
查看导致崩溃的原因

11.continue
恢复程序运行,直到碰到下一个断点

12.run
程序开始运行,直到碰到断点

13.step
执行一行代码

14.next
和s不同的是他不跟踪到代码的内部,一步一步执行代码

15.直接回车为执行上一个命令

16.print 变量
打印某一变量的值

17.display 变量
每次运行到断点就显示变量的值,用于观察变量的变化

18.set 变量=
在程序执行中重新设置某变量的值

19.printf "%2.2s\n",(char*)0x120100fa0
结果打印出:He

20. 设置gdb的列宽,以下为将屏幕设置为70列
set width 70

21. info args 列出你程序所接受的命令行参数
info registers列出寄存器的状态
info breakpoint列出在程序中设的断点
要获得具体的关于info的信息用help info.

22. set
这个命令用来为你的程序设置一个运行环境(使用一个表达式)。
set prompt $把gdb的提示符设为$.
set args 可指定运行时参数。(如:set args 10 20 30 40 50)
show args 命令可以查看设置好的运行参数。

23. show
show命令用来显示gdb自身的状态。
使用'set'命令可以改变绝大多数由'show'显示的信息
使用show radix命令来显示基数
用不带任何参变量的'set'命令可以显示所有可以设置的变量的值
有三个变量是不可以用'set'命令来设置的:
show version显示gdb的版本号
show copying显示版权信息
show warranty显示担保信息

 
分享到微信
 
@清晰掉 GDB调试器中的战斗机
2010-03-13 22:36:15 小盐 (看偶销魂的架势!喂,要盐么?)

1.what is gdb? 
一个调试器,能让你观察到正在运行的程序的某个变量的值,和gcc, emacs组成构建起整个gnu system的3剑客。

2.How it work? 
首先要安装,然后就可以 gdb application,进入gdb模式,执行(run)启动例程。(list-显示源码)

3.一个优点 
GDB的函数调用是缓式的,也就是说只有当变量真正需要时才会被计算。 
int c = add(1,1) 
printf("%d",c);

gdb-->run-->print c -1222342. c的值直到打印的时候才会被计算。(这里可能有个开关来控制缓式计算)

4.一个前提 
gcc 的时候加上 -g 或者 -ggdb,主要是生成供调试使用的某些信息。(emacs好像默认是关闭这个选项)

5.存在图形调试界面

6.输入help可以查看帮助 
help,会显示命令的分类,help 类名

7.how use? 
debug 3 program yourself.

8.常用命令(From:http://my.donews.com/tangfl/2006/10/23/gdb-debug-example/) 
  break NUM 在指定的行上设置断点。 
  bt 显示所有的调用栈帧。该命令可用来显示函数的调用顺序。 
  clear 删除设置在特定源文件、特定行上的断点。其用法为clear FILENAME:NUM 
  continue 继续执行正在调试的程序。该命令用在程序由于处理信号或断点而 导致停止运行时。
  display EXPR 每次程序停止后显示表达式的值。表达式由程序定义的变量组成。 
  file FILE 装载指定的可执行文件进行调试。 
  help NAME 显示指定命令的帮助信息。 
  info break 显示当前断点清单,包括到达断点处的次数等。 
  info files 显示被调试文件的详细信息。 
  info func 显示所有的函数名称。 
  info local 显示当函数中的局部变量信息。 
  info prog 显示被调试程序的执行状态。 
  info var 显示所有的全局和静态变量名称。 
  kill 终止正被调试的程序。 
  list 显示源代码段。 
  make 在不退出 gdb 的情况下运行 make 工具。 
  next 在不单步执行进入其他函数的情况下,向前执行一行源代码。 
  print EXPR 显示表达式 EXPR 的值。 
backtrace 显示程序中的当前位置和表示如何到达当前位置的栈跟踪(同义词:where) 
breakpoint 在程序中设置一个断点 
cd 改变当前工作目录 
clear 删除刚才停止处的断点 
commands 命中断点时,列出将要执行的命令 
continue 从断点开始继续执行 
delete 删除一个断点或监测点;也可与其他命令一起使用 
display 程序停止时显示变量和表达时 
down 下移栈帧,使得另一个函数成为当前函数 
frame 选择下一条continue命令的帧 
info 显示与该程序有关的各种信息 
jump 在源程序中的另一点开始运行 
kill 异常终止在gdb 控制下运行的程序 
list 列出相应于正在执行的程序的原文件内容 
next 执行下一个源程序行,从而执行其整体中的一个函数 
print 显示变量或表达式的值 
pwd 显示当前工作目录 
pype 显示一个数据结构(如一个结构或C++类)的内容 
quit 退出gdb 
reverse-search 在源文件中反向搜索正规表达式 
run 执行该程序 
search 在源文件中搜索正规表达式 
set variable 给变量赋值 
signal 将一个信号发送到正在运行的进程 
step 执行下一个源程序行,必要时进入下一个函数 
undisplay display命令的反命令,不要显示表达式 
until 结束当前循环 
up 上移栈帧,使另一函数成为当前函数 
watch 在程序中设置一个监测点(即数据断点) 
whatis 显示变量或函数类型