linux gdb详解

一.GDB命令详解

在linux程序调成中gdb是最常用的工具。

关于gdb的说明可以在linux下直接 man gdb或者gdb --help。

GDB man手册翻译如下(英文水平有限,只能表达大体的意思):

名称:

      gdb-gnu调试器

概要:

      gdb    [-help] [-nx] [-q] [-batch] [-cd=dir] [-f] [-b bps] [-tty=dev] [-s symfile] [-e prog] [-se prog] [-c core] [-x cmds] [-d dir] [prog [core|procID]]

      gdb    [options] --args prog [arguments]

      gdbtui [options]

说明:

gdb的目的主要是让你知道程序内部正在运行什么代码,或者是当程序崩溃的时候正在执行什么代码

GDB主要可以做四种事情来帮助你找到bug:

            启动你的程序,指定任意可以影响程序行为的参数。

            让你的程序在指定的条件停住.

            测试你的程序停止的时候发生了什么。

            改变程序内部的变量,来改正程序的错误继续执行。

你可以用gdb调试C/C++和modula-2程序,fortran的支持需要有Fortran编译器

gdb使用shell命令gdb来调用,一旦开始,就从终端读取命令直接gdb退出,你可以通过help命令获取帮助信息。

可以不带任何参数或选项执行gdb命令,但是最常用的启动gdb的方式是带一个或者两个参数,指定一个可执行文件来作为参数:

      gdb program(gdb+可执行文件名称)

也可以再gdb文件后面指定可执行文件 和 core文件的名称:

      gdb program core(gdb + 可执行文件 +core文件)

      You can, instead, specify a process ID as a second argument, if you want to debug a running process:

也可以指定一个进程id作为第二个参数,如果你想调试一个正在运行的程序:

      gdb program 1234(gdb进程名+进程id)

    会附件gdb到进程1234上(除非有一个文件名“1234”;GDB首先检查core文件)

gdb调试命令:

      break [file:]function

            设置一个断点在函数中(在文件中)

      run [arglist]

             启动程序带上指定的参数

      bt    Backtrace: 

                显示堆栈

      print expr

                显示表达式的值

      c      

            继续执行你的程序(程序停住后,例如:在断点处停止)

      next  

                执行程序的下一行代码(程序停止以后);跨国任何当前行的函数调用。

      edit [file:]function

            查看当前程序停在哪。

      list [file:]function

               显示程序当前停住的代码行附近的代码

      step  单步调试

            执行程序的下一行(程序停住后),进入当前行的函数调用的内部

      help [name]

            显示gdb命令的相关信息。

      quit  

           退出gdb

需要详细的gdb说明,查看using gdb:一个gdb调试程序的指南,stallmain和roland写的,也有可用的在线文档作为gdb入口在gdb项目信息中。

选项:

    默认的情况下,第一个选项就是执行文件名(等同与"-se"),第二个选项就是core文件名(等同与"-c"),许多选项都有全称和简写,如下:

全称格式也是工人的,如果你截断他们,只要足够明确也可以使用。(如果你愿意的话,你可以“+”而不是“-”来标记选项,尽管“-”是更常用的)。

所有的选项和命令行参数都是按顺序处理的,当使用-x选项的时候这个选项会不同。

      -b bps  

        设置gdb远程调试的所有串行接口的速度

      -batch 

            批处理执行模式,处理完“-x”(和".gdbinit",如果不inhibited)指定的所有命令文件以返回码0退出,如果出错则返回非0,当把gdb作为过滤器的时候,批处理模式非常有用,来在远程电脑上测试下载和执行程序,为了确定它有用,程序退出的消息(当一个程序在gdb模式下发布到控制终端的消息)不发布当使用批处理模式的时候.

      -c FILE, -core=FILE

                使用文件作为一个coredump文件来测试

      -cd=directory

              指定工作目录,不指定则为当前目录

      -d DIRECTORY, -directory=DIRECTORY

              Add directory to the path to search for source files.

      -e FILE, -exec=FILE

            指定可执行文件的名称

      -f, -fullname

            emacs设置这些选项当把gdb作为一个子进程的时候,告诉gdb在每次堆栈显示时候以标准且可识别的方式输出完整的文件名和行号(包含每次程序中断的时候),可识别的形式看上去像两个'32'字符开始,接下来是文件名,行号和字符位置和新行,他们用冒号分割,emacs-to-gdb接口程序用两个‘32’字符作为信号来在一帧上显示源代码。

      -h, -help

              显示帮助信息

      -n, -nx

            不要从任何".gdbinit"结尾的初始化文件执行,正常情况下,这种文件的命令在所有的命令行参数和选项处理完成之后才执行。

      -s FILE, -symbols=FILE

            从文件中读取符号表

      -se=file

            读取符号表的文件和可执行文件

    q, -quiet

            退出,不打印介绍信息和版权信息,这些消息在批处理模式下也不打印

      -tty=device

            使用设备来作为你的标准输入和输出

      --args

           在你的程序名后面传递参数当程序执行的时候。

      -tui  

        运行gdb使用文档(控制台)用户接口。

      -write 

        允许写入执行文件和core文件

      -x FILE, -command=FILE

          从文件中执行gdb命令


二.实例

1.最简单的gdb调试

文件名:add.c

代码很简单,实现一个加法,两个数相加,输出结果

#include <stdio.h>

int add(int x,int y)

{

    return x+y;

}

int main()

{

    int x = 10;

    int y = 20;

    int num = add(x,y);

    printf(" num = %d\n",num);

    return 0;

}

编译gcc -g add.c生成a.out文件(注意编译的时候一定要加上-g选项才能使用gdb调试程序)

gdb ./a.out


上图中有一个警告信息,因为我是源代码安装的,拷贝gdb下的python库到指定的目录即可(警告中的目录)

输入help可以查看帮助信息


start启动程序


可以使用n,s命令跑完整个程序,很简单,可以通过这个简单的demo熟悉一下gdb的各种命令。



2.gdb多进程调试

gdb调试多进程的命令:

set follow-fork-mode mode设置调试器的模式

mode参数可以是

parent fork之后调试原进程,子进程不受影响,这是缺省的方式

child fork之后调试新的进程,父进程不受影响。

show follow-fork-mode 显示当前调试器的模式

set detach-on-fork mode 设置gdb在fork之后是否detach进程中的其中一个,或者继续保留控制这两个进程 

on 子进程(或者父进程,依赖于follow-fork-mode的值)会detach然后独立运行,这是缺省的mode

off 两个进程都受gdb控制,一个进程(子进程或父进程,依赖于follow-fork-mode)被调试,另外一个进程被挂起

info inferiors     显示所有进程

inferiors processid  切换进程

detach inferiors processid     detach 一个由指定的进程,然后从fork 列表里删除。这个进程会被

允许继续独立运行。

kill inferiors  processid  杀死一个由指定的进程,然后从fork 列表里删除。

catch fork  让程序在fork,vfork或者exec调用的时候中断


实例:fork.c

#include <stdio.h>

#include <unistd.h>

#include <stdlib.h>

#include

int add(int x,int y)

{

    return x+y;

}

int test()

{

    int num = 0;

    pid_t pid  = fork();

    if(pid == 0)

    {

    while(1)

    {

            num = add(1,2);

            printf("child:pid:%d num = %d\n",getpid(),num);

            sleep(1);

    }

        exit(0);

    }

    while(1){

        num = add(10,20);

        printf("parent:pid:%d num = %d\n",getpid(),num);

        sleep(1);

    }

    return 0;

}

int main()

{

    test();

    return 0;

}

编译 gcc -g fork.c

设置catch fork,查看 follow-fork-mode和detach-on-fork的默认值

设置detach-on-fork为off,start启动程序,会再fork断点处停止

再两个进程中的add函数分别设置端点:


继续执行程序,info inferiors显示进程


inferiors 2切换进程

detach inferiors  2

重新启动gdb来测试kill inferiors 2,如下图


3.gdb多线程调试

gdb调试多线程的命令:

info threads 查询线程信息 

thread threadno 切换线程

thread apply [threadno] [all] args 对线程列表执行命令

set print thread-events 控制线程开始和结束时的打印信息

show print thread-events 显示线程打印信息的开关状态

实例:文件名thread.c

代码很简单如下:

#include <stdio.h>

#include <pthread.h>

#include <unistd.h>

int add(int x,int y)

{

    return x+y;

}

void* thread(void *args)

{

    int num = 0;

    while(1)

    {

        num += add(1,2);

        sleep(1);

    }

}

int main()

{

    pthread_t tid;

    pthread_create(&tid, NULL, thread, NULL);

    int num = 0;

    while(1)

    {

        num += add(10,20);

        sleep(1);

    }

    return 0;

}

编译gcc -g thread.c -lpthread(链接pthread线程库)

gdb调试:gdb ./a.out

在线程中两个调用add函数的位置分别设置断点



start启动程序,c继续执行程序直到断点处

info threads显示当前程序中的线程


c继续执行程序,到达第二个程序再执行info threads命令,发现当前程序发生了线程切换,由线程2变为线程1


thread 2手动切换线程到线程2,再执行info threads查看当前线程


使用thread apply 命令的示例:


set print thread-events on/off  这里默认是开启的

show print thread-events



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

推荐阅读更多精彩内容

  • linux资料总章2.1 1.0写的不好抱歉 但是2.0已经改了很多 但是错误还是无法避免 以后资料会慢慢更新 大...
    数据革命阅读 12,127评论 2 34
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,717评论 6 342
  • 当你专注地做一件事情,你会发现那时候的你自己是陶醉的,迷恋的,因为你此时的状态特别好,你一心一意想要学好一样东西,...
    P尐c阅读 3,421评论 0 1