详解gdb的使用技巧

01 概念

    GDB是一个由GNU开源组织发布的、UNIX/LINUX操作系统下的、基于命令行的、功能强大的程序调试工具。

    在实际应用中,有两种调试方法:在线调试和离线调试。

    离线调试适用于开发测试环境,可以自由启停进程,设置断点;在线调试一般用于现场问题分析,不能随便启停进程,对于技术要求较高。

02 前提条件

2.1 编译

    若想执行gdb调试,在Makefile文件中需要增加编译调试选项-g,例如:

    gdb dup_file.c –o dum_file_elf –g –lpthread

    说明:-g选项的作用是在可执行文件(ELF)中加入源代码的相关信息,比如ELF中第几条机器指令对应源代码的行数。但不是把整个源文件嵌入到可执行文件中,所以在调试时必须保证gdb能找到源文件。

    -g完整格式是-glevel,其中,level中指定了调试信息中包含了调试信息的多少,默认的是2,level=1最少,level=3最多。

2.2 readelf查看段信息

    例如:

    readelf -S helloWorld|grep debug

    注:helloWorld为文件名,如果没有任何debug信息,则不能被调试。

2.3 file查看strip状况

    下面的情况也是不可调试的:

    file helloWorld

    helloWorld: (省略前面内容) stripped

    注:如果最后是stripped,则说明该文件的符号表信息和调试信息已被去除,不能使用gdb调试。但是not stripped的情况并不能说明能够被调试。

03 使用方法

3.1 启动调试

    在开发中可以将源码和可执行文件拷贝到某一目录下,使用gdb启动进程进行调试,也可以不拷贝源码和可执行文件,使用NFS挂载到编译环境执行调试;在现场环境中使用ps获取进程的pid,然后gdb –p pid执行在线调试。

    离线调试:

    gdb 进程名

    gdb –tui 进程名

    在线调试:

    ps –A | grep 进程名

    gdb –p pid/gdb attach pid

    说明:使用-tui参数可以将调试窗口分为两部分:上面是源码,下面是调试信息,使用Ctrl+n/Ctrl+p或者方向键进行翻页。

    带参数调试:

    1、启动的时候带上参数

    gdb --args xxx 参数

    2、启动之后 run 带上参数

    # gdb xxx

    (gdb)run 参数

    3、启动之后 set args 设置参数

    # gdb xxx

    (gdb)set args 参数

    core文件调试

    当程序core dump时,可能会产生core文件,它能够很大程序帮助我们定位问题。但前提是系统没有限制core文件的产生。可以使用命令limit -c查看:

    $ ulimit -c

    如果结果是0,即便程序core dump了也不会有core文件留下。我们需要让core文件能够产生:

    ulimit -c unlimied  #表示不限制core文件大小

ulimit -c 10  #设置最大大小,单位为块,一块默认为512字节

    上面两种方式可选其一。第一种无限制,第二种指定最大产生的大小。

    针对生成core文件进行调试,可以采用在线加载和离线加载的方式,如下:

  gdb 可执行文件 core文件

3.2 SET命令

    注:有时候使用p打印调试信息不完整或者不便于阅读,可以使用set print elelent 0和setprint pretty on设置。

3.3 handle命令

    handle命令

    handle SIGUSR1 nostop noprint

    handle SIGUSR2 nostop noprint

    handle SIGPIPE nostop noprint

    handle SIGALARM nostop

    handle SIGHUP nostop

    handle SIGTERM nostop noprint

    注:设置GDB调试时对信号的相关动作。

3.4 设置断点

    打断点还是比较有技巧的,虽然有很多打断点的方法,但是实际调试中一般就使用以下几种:

   函数打断点:b 函数名

   某一行打断点:b 源文件:行号

    条件断点:

    break 断点 if 条件

    continue 断点编号(执行一次表示设定,再次执行表示取消)

    continue 断点编号 条件

    注:条件断点非常有用,实际调试中往往需要调试特定场景下函数调用关系,此时就需要设置断点触发的条件。

    查看断点:info breakpoint/info break/info b

    删除断点:delete 断点号/delete(删除所有断点)

    禁用/开启断点:disable/enable breakpoint

    ignore:

    断点条件的一个特殊用法是,程序只有在到达断点一定次数之后才会停止,此时可以使用指令:

    ignore 断点编号 次数

    ignore 2 10触发断点10次后才会停止,每次触发断点count自动减1

    说明:打完断点是不是执行continue就可以等待着运行到断点了呢?不一定,有时候断点处代码的执行需要外部出发,比如web发送特定消息后才可以触发执行,如果一直等待没有消息出发永远执行不到断点处,此时就需要结合自己的业务逻辑,手动设置出发条件。

3.5 执行程序

    执行程序的方法有两种:一种是从main函数开始执行逐步分析,一种是执行到断点处。

    重新运行:r/run

    继续执行:c/continue

    单步执行:n/next/next N(执行N次next)

    单步进入:step(遇到函数进入函数内部,退出函数时使用finish)

    结束函数:finish

    强制返回:return(忽略当前未执行的部分,强制返回)

3.6 显示堆栈

    (gbd) backstrace/bt

    有时候跳转的次数太多,不知道具体调用的层级关系了,可以使用bt查看堆栈,该命令会产生一张列表,包含着运行过程和相关的参数。

3.7 变量操作

  设置变量:set 变量=表达式

    在调试的时候,有时候需要设置一些假数据查看对应输出,比如根据布尔值查看流程执行情况,此时就需要在执行到指定位置时手动设置一下数据的取值。

   监控变量:

    watch 变量 (数值改变时暂停运行)

    awatch <表达式> (被访问或改变时暂停运行)

    rwatch <表达式>  (被访问时暂停运行)

    有时候我们需要观察一个变量的变化过程,比如一个全局变量如何初始化,如何调用的,这就需要使用watch监控变量。

    变量类型:

    ptype var 变量类型

    whatis var 显示一个变量var的类型

    打印变量/表达式:

    打印变量:p 变量      

    打印字符/表达式:p “%s”,字符/表达式

    格式化输出:p/格式控制符 打印内容

    说明:

    gdb可支持的变量显示格式有:

    x:按16进制格式显示变量

    d:按10进制格式显示变量

    u:按16进制格式显示无符号整型

    o:按8进制格式显示变量

    t:按2进制格式显示变量

    c:按字符格式显示变量

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

    也可以使用x(Examination)来打印需要显示的字符信息,格式如下:

    x/格式 地址

    格式(可选)一般是NFU:

    1、N表示重复次数(表示显示内存的长度,也就是说从当前向后显示几个地址的内容)

    2、F表示显示格式

    3、U表示单位(b:字节,h:半字[2字节],w:字[4字节,默认],g:双字[8字节])。表示多少个字节作为一个值取出来,如果不指定的话,GDB默认是1个byte,当我们指定了字节长度后,GDB会从指定内存的地址开始,读取指定字节,并把其作为一个值取出来。

    参数u可使用下面字符代替:

    b:表示单字节

    h:表示双字节

    w:表示四字节

    g:表示八字节

3.8 调试函数

    disassemble    

    可以使用反汇编的指令disassemble去探究究竟在函数中发生了哪些操作,具体如下:

    1、disassemble

    2、disassemble 程序计数器

    3、disassemble 开始地址 结束地址

    格式1表示反汇编当前整个函数,格式2表示反汇编计数器所在函数的整个函数,格式3表示反汇编从开始地址到结束地址的部分。

    call

    强制调用函数:call 表达式

3.9 退出调试

    q/quit

    在执行到断点后,采用q/quit指令退出。

04 多进程调试

4.1 配置

    detach-on-fork

    该属性决定了gdb是同时调试父子进程,还是在fork了子进程之后,将子进程分离出去。

    on:子进程(或者父进程,取决于gdb在初始时,要调试的进程,也就是follow-fork-mode的值)

    off:同时调试父子进程,一个进程处于被调试的状态,而另一个则被gdb挂起

设置:set detach-on-fork on/off

    follow-fork-mode

    该属性决定了gdb在进程调用fork之后的行为。

    set follow-fork-mode parent:默认情况下,在调用fork之后,gdb选择跟随(也就是调试)父进程,而子进程则在处于运行的状态(此时父进程处于阻塞的状态)。

    set follow-fork-mode child:fork之后gdb选择调试子进程,而父进程处于运行的状态。

4.2 查看进程

    查看当前调试的进程:info inferiors

05 多线程调试

5.1查看线程

    查看线程:info threads

    注:输出信息前面有“*”表示调试的当前线程(一般thread切换线程后查看)。

    有的程序会在运行过程中主线程创建多个子线程,所以前后执行info threads显示的线程数是会动态变化的。

5.2 查看线程堆栈

    查看所有线程堆栈:thread apply all bt

    查看指定线程堆栈:thread apply thread1 thread2... bt

5.3 切换线程

    切换线程:thread N

    注:通过打印counter,可以看到多个线程都是在运行的,如果想要让其他线程处于停止状态,只有当前调试的线程执行,可以采用set scheduler-locking on。

5.4 阻塞线程

    阻塞其他线程,仅调试当前线程工作:

 set scheduler-locking [on|off|step]

    运行指定线程并允许其他线程并行执行:

    thread apply N command

06 总结

    对于C语言开发,必须熟练使用gdb进行调试,这可以帮助我们快速定位问题并解决问题,在开发中可以帮助我们及时找到测试出现的问题,在现场问题中如果日志打印不是很充分,日志信息量不够的情况下,gdb调试显得非常重要。

    在实际应用中,我们通常是利用gdb分析core文件,这就需要结合寄存器,汇编,内存相关知识综合分析,后面会详细介绍相关分析技巧。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • 一.GDB命令详解 在linux程序调成中gdb是最常用的工具。 关于gdb的说明可以在linux下直接 man ...
    pengzhixiang阅读 56,873评论 1 10
  • 本文来自个人博客 sunyongfeng.com。博客的文章保持更新,此文可能不是最新状态。 简介 GDB,GNU...
    sunnogo阅读 13,185评论 0 8
  • 1.gdb 可执行文件 例子:gdb one 2.gdb attach pid 例子:gdb attach 123...
    crazyfolish阅读 849评论 0 0
  • 程序开发离不开调试,可以断点调试,也可以打log调试,linux下断点调试c,c++程序用gdb。 断点调试虽然很...
    程序大飞阅读 2,270评论 1 2
  • gdb简介 gdb命令行调试工具非常强大,是linux下调试的神器。可以完成四个方面的功能 启动程序,按照自定义的...
    ermaot阅读 143评论 0 0