linux下GDB的使用方法

gdb是GNU开源组织发布的一个强大的Linux下的程序调试工具。一般来说,GDB主要帮助你完成下面四个方面的功能:
1、启动你的程序,可以按照你的自定义的要求随心所欲的运行程序。
2、可让被调试的程序在你所指定的调置的断点处停住。(断点可以是条件表达式)
3、当程序被停住时,可以检查此时你的程序中所发生的事。
4、你可以改变你的程序,将一个BUG产生的影响修正从而测试其他BUG。
 
一、普通命令
(gdb)help:查看命令帮助,具体命令查询在gdb中输入help + 命令,简写h
(gdb)run:重新开始运行文件(run-text:加载文本文件,run-bin:加载二进制文件),简写r
(gdb)start:单步执行,运行程序,停在第一执行语句。
(gdb)list:查看原代码(list-n,从第n行开始查看代码。list+ 函数名:查看具体函数),简写l
(gdb)set:设置变量的值
(gdb)next:单步调试(逐过程,函数直接执行),简写n
(gdb)step:单步调试(逐语句:跳入自定义函数内部执行),简写s
(gdb)backtrace:查看函数的调用的栈帧和层级关系,简写bt
(gdb)bt full:不仅显示backtrace,还显示局部变量。
(gdb)frame:切换函数的栈帧,简写f
(gdb)info:查看函数内部局部变量的数值,简写i
(gdb)finish:结束当前函数,返回到函数调用点
(gdb)continue:继续运行,简写c
(gdb)print:打印值及地址,简写p
(gdb)p *array@len :显示数组, p相当于print,array就是数组首地址,也可以是数组名,len是想要显示的数组的长度。
(gdb)quit:退出gdb,简写q
(gdb)break+num:在第num行设置断点,简写b
(gdb) break 9 if sum != 0 :满足条件才激活断点
(gdb)info breakpoints:查看当前设置的所有断点
(gdb)delete breakpoints num:删除第num个断点,简写d
(gdb)display:追踪查看具体变量值
(gdb)undisplay:取消追踪观察变量
(gdb)watch:被设置观察点的变量发生修改时,打印显示。比如:watch input[4]
(gdb)i watch:显示观察点
(gdb)enable breakpoints:启用断点
(gdb)disable breakpoints:禁用断点
(gdb) x/<n/f/u> <addr>:查看内存, n、f、u是可选的参数。 <addr>表示一个内存地址。(比如: x/3uh 0x54320 表示,从内存地址0x54320读取内容,h表示以双字节为一个单位,3表示输出三个单位,u表示按十六进制显示。
(gdb)run argv[1] argv[2]:调试时命令行传参。
 
二、高级命令
1、调试设置参数的方法,有两种:
1)run 参数1 参数2 参数n
2)set args 参数1 参数2 参数n
可用 show args 查看。
 
2、gdb调试已在运行的进程
gdb -p <进程号>
或者:
gdb attach <进程号>
 
1.1)通过命令:ps -aux | grep 进程名(比如:main) ,获取执行main的进程(pid),比如获取到pid:10001
1.2)启动gdb attach,执行gdb attach pid即可调试正在运行的程序,输入命令:gdb attach 10001
若执行gdb attach时提示:” ptrace: Operation not permitted”,说明权限不够,切换root用户运行,执行:sudo gdb attach 10001
1.3)设置源码映射,对源码路径重新设置。[可选]
set substitute-path /home/johnchen1/workspace/OldCode /home/johnchen2/other/NewCode
源码路径:/home/johnchen1/workspace/OldCode
实际代码路径:/home/johnchen2/other/NewCode
用实际代码替换源码,此过程需要包版本和代码版本一致。
1.4)设置断点
b xxx.cpp:100
//调试xxx.cpp文件,断点为100行。
1.5)已是运行中的程序不需要启动(r),继续(c)即可。
1.6)等待指令或报文触发断点,被击中会停留在断点行。
1.7)其他gdb操作(s单步调试(逐语句:跳入自定义函数内部执行);n单步调试(逐过程,函数直接执行);bt等)
 
3、使用GDB时对源码路径重新设置
有的时候一个程序不是在同一个机器上执行的,当调试的时候就可能无法找到源代码的路径,这时就需要使用下面的方法了:
当gdb程序找不到源码时,进行重新设置源码路径步骤:
(1). 使用如下命令查看目标文件中保存的是相对路径还是绝对路径
readelf -p .debug_str target_bin_file
(2). 如果是相对路径:
使用 gdb 的dir命令添加的源码搜索路径
(3). 如果是绝对路径:
进行路径重新映射:set substitute-path old_path new_path
查看当前环境设置的调试目录:show directories
 
4、examine
examine(x)可以按照一定的格式打印内存地址处的数据,详细文档可参考这里。
(gdb) x/{COUNT}{FMT}{SIZE} {ADDRESS}
{COUNT}: 打印的数目,默认为1。
{FMT}: 打印的格式1,默认为上次使用的{FMT}:
o(octal): 8进制整数
x(hex): 16进制整数
d(decimal): 10进制整数
u(unsigned decimal): 10进制非负整数
t(binary): 2进制整数
f(float): 浮点数
a(address): 输出相对于前面最近的符号的偏移
i(instruction): 输出地址处的汇编指令
c(char): 字符
s(string): c字符串(null-terminated string)
z(hex, zero padded on the left): 见说明
{SIZE}: 打印的字节数目,默认为上次使用的{SIZE}:
b(byte): 1个字节
h(halfword): 2个字节
w(word): 4个字节
g(giant, 8 bytes): 8个字节
{ADDRESS}: 目标地址
 
几个例子:
(gdb) x/a 0x401419
0x401419 <main()+113>:  0x55c3c900000000b8
 
(gdb) x/i 0x40138d
=> 0x40138d <crash(int, double)+41>:    mov    -0x10(%rbp),%eax
 
(gdb) x/1fg 140737488346064
0x7fffffffdbd0: 10.125
 
5、GDB堆栈跟踪
程序“调用堆栈”是当前函数之前的所有已调用函数的列表(包括当前函数)。每个函数及其变量都被分配了一个“帧”,
最近调用的函数在 0 号帧中(“底部”帧)。要打印堆栈,发出命令 'bt'('backtrace' [回溯] 的缩写):
(gdb) bt
#0  0x80483ea in wib (no1=8, no2=8) at eg1.c:7
#1  0x8048435 in main (argc=1, argv=0xbffff9c4) at eg1.c:21
此结果显示了在 main() 的第 21 行中调用了函数 wib()(只要使用 'list 21' 就能证实这一点),而且 wib() 在 0 号帧中,
main() 在 1 号帧中。由于 wib() 在 0 号帧中,那么它就是执行程序时发生算术错误的函数。
实际上,发出 'info locals' 命令时,gdb 会打印出当前帧中的局部变量,缺省情况下,这个帧中的函数就是被中断的函数(0 号帧)。
可以使用命令 'frame' 打印当前帧。要查看 main 函数(在 1 号帧中)中的变量,可以发出 'frame 1' 切换到 1 号帧,
然后发出 'info locals' 命令:
(gdb) frame 1
#1  0x8048435 in main (argc=1, argv=0xbffff9c4) at eg1.c:21
21          result = wib(value, div);
(gdb) info locals
value = 8
div = 8
result = 4
i = 2
total = 6
此信息显示了在第三次执行 "for" 循环时(i 等于 2)发生了错误,此时 "value" 等于 "div"。
可以通过如上所示在 'frame' 命令中明确指定号码,或者使用 'up' 命令在堆栈中上移以及 'down' 命令在堆栈中下移来切换帧。
要获取有关帧的进一步信息,如它的地址和程序语言,可以使用命令 'info frame'。
 
6、线程相关的命令
一些调试多线程程序时常用的命令:
info threads: 查看线程列表
thread 2: 切换到2号线程,线程编号可由info threads得知
thread apply all bt: 打印所有线程的堆栈
 
7、使用linux的GDB打印STL(vector,map,set等)
©️2020 CSDN 皮肤主题: 技术工厂 设计师: CSDN官方博客 返回首页
实付0元
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值