打印

GDB常用的调试方法

[复制链接]
187|0
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
香菇选手|  楼主 | 2018-10-3 12:32 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
什么是GDB

GDB是GNU开源组织发布的一个强大的UNIX下的C/C++程序调试工具。虽然GDB自身并没有Windows上大多数IDE的调试工具的高可视化和图形化功能,但命令行式调试也具有其很大的优点,如果希望在GDB上提升可视化功能,可以使用vim的gdb插件对gdb进行加强。

顺便提一下调试程序的几大功能:

自定义方式启动程序;

可以设置断点,使程序在指定的断点处停住;

当程序停住时,可以查看程序的运行时数据;

可以修改程序的运行时数据;



GDB与编译器的-g和-ggdb参数

cc/gcc/g++使用-g参数可以在编译时利用操作系统的“原生格式(native format)”生成调试信息。这些信息GDB可以直接利用,当然其它的调试工具也能够直接利用。

-g参数是分级别的:

a) -g2

使用-g的默认的级别,此时产生的调试信息包括扩展的符号表、行号、局部或外部变量信息。如果不使用,则只有内存地址信息,没有响应的标识信息;

b)-g3

包含级别-g2中的所有调试信息,以及源代码中定义的宏;

c)-g1

级别1(-g1)不包含局部变量和与行号有关的调试信息,因此只能够用于回溯跟踪和堆栈转储之用。回溯跟踪指的是监视程序在运行过程中的函数调用历史,堆栈转储则是一种以原始的十六进制格式保存程序执行环境的方法,两者都是经常用到的调试手段。

-ggdb参数能够为GDB生成更为丰富的调试信息,与-g相同,也有3个级别,如果希望调试宏,则可以使用-ggdb3。但与-g不同的是,-ggdb生成的信息只能被GDB使用,而不能被其它调试工具使用。



使用GDB启动调试

使用GDB启动调试的方法有一下几种(只是关联上调试目标程序,并没有开始运行目标程序):

gdb <program>

<program>也就是你的执行文件,要在PATH环境变量下能搜索到,一般在当前目录下;

gdb <program> core_file

用gdb同时调试一个运行程序和core文件,core是程序非法执行后core dump后产生的文件。启动后,gdb将加载core文件保存的堆栈信息,使用bt命令可以查看堆栈信息,后面也将对此进行详细说明;

gdb <program> <PID>

用gdb调试正在运行的进程,<PID>是正在运行的进程ID。gdb能自动attach到这个进程上,其中<program>要在PATH环境变量下能够搜索到。也可以使用gdb attach <PID>;



查看源代码

使用list命令,也可以使用缩写l。

list [linenum]

查看指定行[linenum]附近的代码。        



设置断点(Breakpoint)

GDB提供break命令来设置断点,有以下几种设置断点的方法:

break <function>

在进入指定函数时停住。C++中也可以使用class::function,或者类似function(type,type)的方式来指定函数;



break <filename>:<function>

在进入指定源文件的函数时停住。



break <linenum>

在运行到指定行时停住。



break <filename>:<linenum>

在运行到指定源文件的指定行时停住。



break <offset>

在当前行号的前后指定offset处停住,例如当前行为10,运行break +100表示在110处设置断点,运行break -10则表示在90处设置断点;



break *<address>

在程序运行到内存地址<address>处停住。



break

break后什么都没有,表示在下一条指令处停住。



break [where] if <condition>

在条件成立时停住。[where]可以是上述任何break的参数,也可以不设置,不设置则表示条件成立的下一条指令处停住。例如,break if i=100,表示当i等于100时停住程序。



commands <breakpoint num>

>[command_list]

>end

为断点设置自动执行命令。例如:

(gdb) commands 2

>print n

>end

表示在第2个断点时,执行print n操作。



查看断点(Breakpoint)

info breakpoints [n]

info break [n]

info b [n]

以上[n]表示第n个断点,如果没有,则显示所有断点。



删除/修改断点(Breakpoint)

GDB提供四个命令来维护断点:delete、clear、disable、enable。

delete

删除指定断点。

delete [breakpoint num]

delete [breakpoint num range]

[breakpoint num]为断点号,[breakpoint num range]为断点号范围。如果不指定断点号,则为删除所有断点



clear

清除断点

clear         清除当前位置断点

clear <filename>:<function>

clear <function>                                        清除指定函数位置上的断点

clear <filename>:<linenum>

clear <linenum>                                        清除指定行位置上的断点



disable

禁用断点,不删除。



enable

将禁用的断点启动。



        对于条件断点,可以使用condition <breakpoint num> <condition> 修改断点条件。

        可以使用ignore <breakpoint num> <count> 来忽略指定断点<count>次



设置观察点(Watchpoint)

观察点一般来观察某个表达式(变量也是一种表达式)的值是否有变化了,如果有变化,马上停住程序。我们有下面的几种方法来设置观察点:

watch <expr>

为表达式(变量)expr设置一个观察点。一量表达式值有变化时,马上停住程序。



rwatch <expr>

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



awatch <expr>

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



运行调试程序

set args [params]

设置程序运行参数,例如set args param1 param2



run/r

运行调试程序。               



单步调试

step/s [count]

单步跟踪,如果有函数调用,他会进入该函数。进入函数的前提是,此函数被编译有debug信息。很像VC等工具中的step in。后面可以加count也可以不加,不加表示一条条地执行,加则表示执行后面的count条指令,然后再停住。



next/n [count]

同样单步跟踪,如果有函数调用,他不会进入该函数。很像VC等工具中的step over。后面可以加count也可以不加,不加表示一条条地执行,加则表示执行后面的count条指令,然后再停住。



finish [count]

运行程序,直到当前函数完成返回。并打印函数返回时的堆栈地址和返回值及参数值等信息。



until/u [count]

当你厌倦了在一个循环体内单步跟踪时,在循环指令上使用这个命令可以运行程序直到退出循环体。



continue/c [ignore-count]

当程序被停住了,你可以用continue恢复程序运行,直到程序结束,或是下一个断点到来。ignore-count表示忽略其后的断点次数。



查看运行时数据

print

打印变量,可以使用缩写p。

p <expr>

基本用法,打印变量或者表达式,如p i、p i+j。



p/<format> <expr>

使用指定格式打印。支持的格式包括:

x  按十六进制格式显示变量。

d  按十进制格式显示变量。

u  按十六进制格式显示无符号整型。

o  按八进制格式显示变量。

t  按二进制格式显示变量。

a  按十六进制格式显示变量。

c  按字符格式显示变量。

f  按浮点数格式显示变量。



p *<array>@<len>

指定长度打印数组。

如有int* array = new int[100];

则可以使用 p *array@10,查看数组前10个元素的数值。



examine

查看指定内存地址中的数据,可以使用缩写x,可以在gdb中使用help x查看帮助。

使用方式为:

x/<format> <addr>

支持的格式除了print中的格式外,还包括:

n                 是一个正整数,表示显示内存的长度,也就是说从当前地址向后显示几个地址的内容。

u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4个bytes。u参数可以用下面的字符来代替,b表示单字节,h表示双字节,w表示四字 节,g表示八字节。当我们指定了字节长度后,GDB会从指内存定的内存地址开始,读写指定字节,并把其当作一个值取出来。



display

用于自动显示, 当程序停住时,或是在你单步跟踪时,这些变量会自动显示。使用方式为:

display <expr>

display/<format> <expr>

display/<format> <addr>

支持的格式i和s,可以help display查看。



修改运行时数据

修改运行时变量

set <var>=<value>

set var <var>=<value>

如果变量名与gdb的环境变量名冲突时,则需要加var前缀来说明这是要设置一个变量。



强制函数返回

return

return <expression>

如果函数需要返回值,则指定<expression>



强制调用函数

call <expression>



调试多线程

多线程调试常用的命令如下:

info thread

查看线程信息。



thread <ID>

切换到指定ID的线程。



break <where> thread all

对所有经过的线程都设置断点。



set scheduler-locking off|on|step

在使用step或者continue命令调试当前被调试线程的时候,其他线程也是同时执行的,怎么只让被调试程序执行呢?通过这个命令就可以实现这个需求。

off                 不锁定任何线程,也就是所有线程都执行,这是默认值。

on                 只有当前被调试程序会执行。

step         在单步的时候,除了next过一个函数的情况(熟悉情况的人可能知道,这其实是一个设置断点然后continue的行为)以外,只有当前线程会执行。

使用特权

评论回复

相关帖子

发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

450

主题

462

帖子

0

粉丝