程序员求职经验分享与学习资料整理平台

网站首页 > 文章精选 正文

linux下GDB使用方法

balukai 2025-01-08 10:59:42 文章精选 11 ℃

?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等)

https://www.cnblogs.com/silentNight/p/5466418.html

Tags:

最近发表
标签列表