一、GDB调试命令

 

描述

backtrace(或bt)

查看各级函数调用及参数

finish

连续运行到当前函数返回为止,然后停下来等待命令

frame(或f) 帧编号

选择栈帧

info(或i) locals

查看当前栈帧局部变量的值

list(或l)

列出源代码,接着上次的位置往下列,每次列10行

list 行号

列出从第几行开始的源代码

list 函数名

列出某个函数的源代码

next(或n)

执行下一行语句

print(或p)

打印表达式的值,通过表达式可以修改变量的值或者调用函数

quit(或q)

退出gdb调试环境

set var

修改变量的值

start

开始执行程序,停在main函数第一行语句前面等待命令

step(或s)

执行下一行语句,如果有函数调用则进入到函数中

比如还有几个命令如下

wath     观察一个变量

current 跳转到下个断点,或则跳转到观察点

quit  退出gdb调试

特别地,对于段错误的解决:

1.编译的时候加上-g选项
2.在ARM开发板上:
2.1.ulimit -c unlimited
2.2.执行应用程序:程序出错时会在当前目录下生成名为core的文件
3.在PC上
3.1. arm-linux-gdb App core文件所在目录/core
3.2.在gdb中输入:backtrace 就可以知道在哪里出错

注意:

1.如果输入bt没有结果,或者提示无法运行,一般是没有权限导致的,chmod 777 core即可

2.出现0x40000780 in ?? ()

warning: Unable to find dynamic linker breakpoint function.
GDB will be unable to debug shared library initializers
and track explicitly loaded dynamic code.
and all your stack frames look like this:
0x40000780 in ?? ()

一般是找不到动态库的原因:
解决办法
首先输入info sharedlibrary 查询缺少的动态库
然后把动态库拷贝到PC端的当前目录
最后设置路径为当前目录 set solib-search-path .
或者使用其他方式设置

http://blog.chinaunix.net/uid-24922718-id-3489839.html
http://www.fayewilliams.com/2013/01/31/gdb-unable-to-find-dynamic-linker-breakpoint-function/
http://bbs.chinaunix.net/thread-3611739-1-1.html
//可能动态库要加上-g才行

安装GDB:

  1. sudo apt-get install gdb

二、分析例子

代码如下:

  1. #include <stdio.h>
  2. int add_range(int low, int high);
  3. int add (int a, int b);
  4. int add (int a, int b) {
  5. int result = a + b;
  6. return result;
  7. }
  8. int add_range(int low, int high)
  9. {
  10. int i = 0, sum = 0;
  11. for (i = low; i <= high; i++)
  12. sum = sum + i;
  13. return sum;
  14. }
  15. int main(void)
  16. {
  17. int result[100];
  18. result[0] = add_range(1, 10);
  19. result[1] = add_range(1, 100);
  20. printf("result[0]=%d\nresult[1]=%d\n", result[0], result[1]);
  21. int a = 9, b = 10;
  22. int result1 = a + b;
  23. int result2 = a * b;
  24. printf("result1 = %d, result2 = %d", result1, result2);
  25. int result3 = add(result1, result2) ;
  26. printf("result3=%d", result3);
  27. if (result3 == 10) {
  28. printf("result3 = %d", result3);
  29. } else {
  30. printf("result != 10");
  31. }
  32. return 0;
  33. }

1.在编译时要加上-g选项,生成的可执行文件才能用gdb进行源码级调试:

命令如下:

  1. gcc -g main.c -o hello1
  2. gdb hello1

如下图

2.然后我们怎么查看源代码呢?命令如下

  1. list

然后我们还要看下面的代码可以怎么样操作,你可以  回车 或则 输入l  之后回车(因为list 简写 l)注意很多gdb命令都可以简写成首字母,或则输入 list 2 回车也可以

我们退出gdb命令如下

  1. quit

我们这里先不退出

3.我们先运行程序用下面命令

  1. start

如下图

4.我们用next命令执行下一条语句,next命令简写为n,输入下民命令

  1. n

5.我们也可是使用step命令进入函数内入跟踪,简写命令s,再输入命令n执行下一步,输入下面命令

  1. s

6.在函数中有几种查看状态的办法,backtrace命令(简写为bt)可以查看函数调用的栈帧:

可见当前的add_range函数是被main函数调用的,main传进来的参数是low=1, high=100main函数的栈帧编号为1,add_range的栈帧编号为0。

7.现在可以用info命令(简写为i)查看函数局部变量的值:

i locals

add_range函数局部变量的值:

v:shapes="_x0000_i1032">

如果想查看main函数当前局部变量的值也可以做到,先用frame命令(简写为f)选择1号栈帧然后再查看局部变量:

(gdb) f 1

输入下面命令

  1. i locals

8.我们还可以用p打印出当前值

9.我们也可以修改sum的值

  1. set var sum = 0

v:shapes="_x0000_i1036">

10.比如我们不要运行这个函数了,要运行这个函数下面的程序,我们可以用finish

  1. finish

11.quit    
退出命令

v:shapes="_x0000_i1038">

三、断点调试

举例分析:

hello2.c源文件如下

  1. #include<stdio.h>
  2. int main() {
  3. int a = 1, b = 2, c = 3;
  4. int i;
  5. printf("hello chenyu");
  6. for (i = 0 ; i < 100; i++) {
  7. ++a;
  8. if (i > 90) {
  9. ++b;
  10. } else {
  11. ++c;
  12. }
  13. }
  14. printf("a = %d, b = %d, c = %d", a, b, c);
  15. int cc = 0;
  16. for (cc; cc < 9; cc++) {
  17. printf("c %d", i);
  18. }
  19. return 0;
  20. }

比如我们在 ++b (第10行)这行和printf("c %d", i)(第18行)这两行打断点示例:

1、编译

---在编译时要加上-g选项,生成的可执行文件才能用gdb进行源码级调试

gcc -g hello2.c

2、gdb调试

  1. gdb hello2

3、显示代码行数命令如下

  1. list

list(或l)

列出源代码,接着上次的位置往下列,每次列10行

list 行号

列出从第几行开始的源代码

list 函数名

列出某个函数的源代码

回车 往下查看

4 、然后输入打断点命令

1) 、源文件的某一行设置断点

  1. break 行号

2)、一个特定的函数设置断点

  1. break 函数名

3)、设置条件断点

  1. break 行号 if 条件

示例:

  1. break 10
  2. break 18

图片如下:

5 、 运行并且跳转到断点

1)、运行

  1. start

2) 、跳转到断点命令(countine 简写c)

  1. c

程序就到++b 这行来,上面2个步骤也可以用这个run (简写 r)命令代码

  1. r

我们这里代码会跳到这个断点里面来,我么就避免了手动循环这么多次,因为这个断点会打很多次,所以我们可以用c命令,还是会在这行,所以打断点一般不要打在循环里面,当这个代码执行完之后,我们再用c命令,就会打印18行的代码

6、看函数局部变量的值或单个变量的值可以用下面命令

  1. i locals
  2. p c

如果想查看其他函数当前局部变量的值也可以做到,先用frame命令(简写为f)选择其他函数的栈帧号(backtrace命令(简写为bt)获取)然后再查看局部变量

f 1

i locals

7、进入函数内部用下面命令

  1. s

8、单步调试往下执行用下面命令

  1. n

9、删除断点

  1. delete 行号

10、我们也可以用watch命令来观察变量,如果变量的值发生变化,程序就会停止

  1. watch b

11、退出gdb命令

  1. quit

四、如何保存断点

有时候运行时,上次打的断点没了,怎么保存呢?

1、查看断点信息并保存

  1. info b

2
、保存断点

  1. save breakpoint fig.dp

断点保存在fig.dp文件里面

3、调试时候读取保存断点文件

  1. gdb hello2 -x fig.dp

要记住加上参数 -x

然后再去查看是否有断点

  1. info b

可以看到是有2个断点的

原文:http://blog.csdn.net/u011068702/article/details/53925415

   http://blog.csdn.net/u011068702/article/details/53931648

五、实例:

示例1:

book@book-desktop:/work/projects/test$ gcc -o
LinkListClearOprNoHeadNode -g LinkListClearOprNoHeadNode.c

book@book-desktop:/work/projects/test$ gcc -g -o
LinkListClearOprNoHeadNode LinkListClearOprNoHeadNode.c

book@book-desktop:/work/projects/test$ gdb
LinkListClearOprNoHeadNode

GNU gdb (GDB)
7.0-ubuntu

Copyright (C) 2009 Free Software Foundation, Inc.

License GPLv3+: GNU GPL version 3 or later
<http://gnu.org/licenses/gpl.html>

This is free software: you are free to change and
redistribute it.

There is NO WARRANTY, to the extent permitted by
law.  Type "show copying"

and "show warranty" for details.

This GDB was configured as "i486-linux-gnu".

For bug reporting instructions, please see:

<http://www.gnu.org/software/gdb/bugs/>...

Reading symbols from /work/projects/test/LinkListClearOprNoHeadNode...done.

(gdb) list

35     
-Input                  :

36     
-Output                 :

37     
-Return                 :

38      * Modify Date     Version                Author                   Modification

39      *
-----------------------------------------------

40      *
2017/03/29      V1.0.0                 Yu Weifeng       Created

41     
******************************************************************************/

42      int main(int argc,char **argv)

43      {

44              PT_LinkList ptLinkListHead=NULL;

(gdb)

45              T_LinkListElement
tLinkListElement;

46              tLinkListElement.cData=1;

47             
InsertNodeToLinkList(&ptLinkListHead,1,&tLinkListElement);

48              tLinkListElement.cData=23;

49             
InsertNodeToLinkList(&ptLinkListHead,2,&tLinkListElement);

50              tLinkListElement.cData=24;

51             
InsertNodeToLinkList(&ptLinkListHead,3,&tLinkListElement);

52              tLinkListElement.cData=26;

53             
InsertNodeToLinkList(&ptLinkListHead,4,&tLinkListElement);

54              tLinkListElement.cData=66;

(gdb)

55             
InsertNodeToLinkList(&ptLinkListHead,5,&tLinkListElement);

56              tLinkListElement.cData=99;

57              InsertNodeToLinkList(&ptLinkListHead,6,&tLinkListElement);

58

59              printf("插入结果:");

60              TraverseLinkList(ptLinkListHead);

61             
DeleteNodeFromLinkList(ptLinkListHead,6,&tLinkListElement);

62              DeleteNodeFromLinkList(ptLinkListHead,4,&tLinkListElement);

63

64              printf("删除结果:");

(gdb)

65              TraverseLinkList(ptLinkListHead);

66

67              printf("插入结果:");

68              tLinkListElement.cData=5;

69              InsertNodeToLinkList(&ptLinkListHead,3,&tLinkListElement);

70              TraverseLinkList(ptLinkListHead);

71              return 0;

72      }

73     
/*****************************************************************************

74     
-Fuction                :
GetLinkListLength

(gdb) info b

No breakpoints or watchpoints.

(gdb) break 47

Breakpoint 1 at 0x80484ca: file
LinkListClearOprNoHeadNode.c, line 47.

(gdb) start

Temporary breakpoint 2 at 0x80484bd: file
LinkListClearOprNoHeadNode.c, line 44.

Starting program: /work/projects/test/LinkListClearOprNoHeadNode

Temporary breakpoint 2, main (argc=1, argv=0xbffff784)
at LinkListClearOprNoHeadNode.c:44

44             
PT_LinkList ptLinkListHead=NULL;

(gdb) c

Continuing.

Breakpoint 1, main (argc=1, argv=0xbffff784) at LinkListClearOprNoHeadNode.c:47

47             
InsertNodeToLinkList(&ptLinkListHead,1,&tLinkListElement);

(gdb) s

InsertNodeToLinkList (i_pptLinkListHead=0xbffff6c8,
i_dwPosition=1, i_ptElement=0xbffff6cf) at LinkListClearOprNoHeadNode.c:114

114             int ret=-1;

(gdb) i locals

ret = 134513528

ptLinkListNode = 0xbffff6a8

ptInsertListNode = 0x8049ff4

iLen = 2424820

(gdb) p iLen

$1 = 2424820

(gdb) n

115             T_LinkList
*ptLinkListNode=*i_pptLinkListHead;

(gdb)

116             T_LinkList *ptInsertListNode=NULL;

(gdb)

117             int iLen=0;

(gdb)

118            
iLen=GetLinkListLength(ptLinkListNode);

示例2:

book@book-desktop:/work/projects/test$
gdb LinkListClearOprNoHeadNode

GNU gdb (GDB)
7.0-ubuntu

Copyright (C) 2009
Free Software Foundation, Inc.

License GPLv3+: GNU
GPL version 3 or later <http://gnu.org/licenses/gpl.html>

This is free software:
you are free to change and redistribute it.

There is NO WARRANTY,
to the extent permitted by law.  Type
"show copying"

and "show
warranty" for details.

This GDB was
configured as "i486-linux-gnu".

For bug reporting
instructions, please see:

<http://www.gnu.org/software/gdb/bugs/>...

Reading symbols from
/work/projects/test/LinkListClearOprNoHeadNode...done.

(gdb) list
InsertNodeToLinkList

108     * Modify Date     Version                Author                   Modification

109     * -----------------------------------------------

110     * 2017/03/29      V1.0.0                 Yu Weifeng       Created

111    
******************************************************************************/

112     static int InsertNodeToLinkList(T_LinkList
**i_pptLinkListHead,int i_dwPosition,T_LinkListElement *i_ptElement)

113     {

114             int ret=-1;

115             T_LinkList
*ptLinkListNode=*i_pptLinkListHead;

116             T_LinkList *ptInsertListNode=NULL;

117             int iLen=0;

(gdb)

118            
iLen=GetLinkListLength(ptLinkListNode);

119            
if(i_dwPosition<1||i_dwPosition-1>iLen)

120             {

121                     ret=-1;

122                    
printf("InsertPositionErr,Len:%d,Pos:%d\r\n",iLen,i_dwPosition);

123             }

124             else

125             {

126                    
ptInsertListNode=(T_LinkList *)malloc(sizeof(T_LinkList));

127                     if(NULL==ptInsertListNode)

(gdb)

128                     {

129                             printf("pInsertListNodeMallocErr\r\n");

130                             ret=-1;

131                     }

132                     else

133                     {

134                            
ptInsertListNode->ptNext=NULL;

135                             memcpy(&ptInsertListNode->tData,i_ptElement,sizeof(T_LinkListElement));

136                            
if(NULL==*i_pptLinkListHead)

137                             {

(gdb)

138                                    
*i_pptLinkListHead=ptInsertListNode;

139                             }

140                             else

141                             {

142                                    
while(--i_dwPosition)

143                                     {

144                                            
ptLinkListNode=ptLinkListNode->ptNext;

145                                     }

146                                    
ptInsertListNode->ptNext=ptLinkListNode;

147                                     ptLinkListNode=ptInsertListNode;

(gdb) info b

No breakpoints or
watchpoints.

(gdb) break 144

Breakpoint 1 at
0x804873f: file LinkListClearOprNoHeadNode.c, line 144.

(gdb) start

Temporary breakpoint 2
at 0x80484bd: file LinkListClearOprNoHeadNode.c, line 44.

Starting program:
/work/projects/test/LinkListClearOprNoHeadNode

Temporary breakpoint
2, main (argc=1, argv=0xbffff784) at LinkListClearOprNoHeadNode.c:44

44              PT_LinkList ptLinkListHead=NULL;

(gdb) c

Continuing.

ListNull

Breakpoint 1,
InsertNodeToLinkList (i_pptLinkListHead=0xbffff6c8, i_dwPosition=1,
i_ptElement=0xbffff6cf) at LinkListClearOprNoHeadNode.c:144

144                                            
ptLinkListNode=ptLinkListNode->ptNext;

(gdb) i locals

ret = -1

ptLinkListNode =
0x804b008

ptInsertListNode =
0x804b018

iLen = 1

Linux之GDB调试介绍与应用20170601的更多相关文章

  1. 一文入门Linux下gdb调试(二)

    作者:良知犹存 转载授权以及围观:欢迎添加微信号:Conscience_Remains 总述     今天我们介绍一下core dump文件,Core dump叫做核心转储,它是进程运行时在突然崩溃的 ...

  2. Linux知识(5)----LINUX下GDB调试

    命令 解释 示例   file 加载被调试的可执行程序文件.因为一般都在被调试程序所在目录下执行GDB,因而文本名不需要带路径. (gdb) file gdb-sample     r c Run的简 ...

  3. Linux下GDB调试简单示例

    这里介绍对文件first.c的基本GDB调试操作,只有部分命令,只是一个示例,运行环境为装有gcc编译器和gdb调试器的Linux环境,基本GDB调试命令如下表: 命令                 ...

  4. Linux嵌入式GDB调试环境搭建

    ======================= 我的环境 ==========================PC 端: CPU:x86_64, 系统:Ubuntu,IP:172.16.2.212开发 ...

  5. Linux+eclipse+gdb调试postgresql源码

    pg内核源码解析课上用的vs调试pg源码, VS用起来确实方便,但是配置调试环境着实有点麻烦.首先得装个windows系统,最好是xp,win7稍微麻烦点:最好使用vs05,08和10也可以,但是比0 ...

  6. 一文入门Linux下gdb调试(一)

    作者:良知犹存 转载授权以及围观:欢迎添加微信号:Conscience_Remains 总述 在window下我们习惯了IDE的各种调试按钮,说实话确实挺方便的,但到了Linux下,没有那么多的IDE ...

  7. Linux下GDB调试

    GDB 是一个强大的命令行调试工具.大家知道命令行的强大就是在于,其可以形成执行 序列,形成脚本.UNIX 下的软件全是命令行的,这给程序开发提供了极大的便利,命令行 软件的优势在于, 他们可以非常容 ...

  8. Linux下GDB调试C/C++

    首先先编译程序并生成调试符号: gcc -g -c main.cpp gcc -o exefile main.o 以上的exefile为可执行程序的文件名 然后: gdb exefile 可以开始gd ...

  9. Linux学习----gdb调试(指针的指针)

    昨天遇到一个很奇怪的问题,如下: 按照理论,最后*p的值应该是99,不知为什么是15了,所以今天记录用gdb调试的过程,并熟悉gdb的使用. (调试过程参考:http://www.cnblogs.co ...

随机推荐

  1. 大O算法

    大O计法:根据执行次数计算#sum = (1+n)*n/2://执行了一次,即为O(1)#for(i=0;i<n;i++);//执行了n次,即为O(n)#算法的时间复杂度:T(n) = O(f( ...

  2. 袋鼠云研发手记 | 袋鼠云EasyManager的TypeScript重构纪要

    作为一家创新驱动的科技公司,袋鼠云每年研发投入达数千万,公司80%员工都是技术人员,袋鼠云产品家族包括企业级一站式数据中台PaaS数栈.交互式数据可视化大屏开发平台Easy[V]等产品也在迅速迭代.在 ...

  3. access数据库频繁读取操作会出现 System.Data.OleDb.OleDbException 的异常解决

    asp.net access数据库 本来想着打开一个access数据库连接后,不关闭,下次操作数据了,直接拿来用,谁知道连着测试64次后(大概这么多次),就会出现System.Data.OleDb.O ...

  4. 第八次ScrumMeeting博客

    第八次ScrumMeeting博客 本次会议于11月2日(四)22时整在3公寓725房间召开,持续20分钟. 与会人员:刘畅.辛德泰.窦鑫泽.张安澜.赵奕.方科栋. 1. 每个人的工作(有Issue的 ...

  5. 数据库与数据仓库的比较Hbase——Hive

    数据仓库(Data Warehouse)是一个面向主题的(Subject Oriented).集成的(Integrate).相对稳定的(Non-Volatile).反映历史变化(Time Varian ...

  6. 2016-6-2-第二个sprint

    1.开始一个新的冲刺: 起止:2016.6.1~2016.6.14 ProductBacklog:继续向下细化 Sprint 计划会议:确定此次冲刺要完成的目标 Sprint Backlog:新的冲刺 ...

  7. PSP Daily软件Alpha版本——基于NABCD评论,及改进建议

    1.根据(不限于)NABCD评论作品的选题: 此软件的用户人群较为明确,即:用户(软件工程课上学生)记录例行报告.写每周PSP表格和统计的需求.潜在用户还有未来该课堂的学生和需要用PSP方法记录任务完 ...

  8. 团队展示(I know)

    一.队员姓名与学号 姓名 学号 组长 陈家权 031502107 赖晓连 031502118 ★ 雷晶 031502119 林巧娜 031502125 庄加鑫 031502147 二.队名 I kno ...

  9. 栈和队列在python中的实现

    栈和队列是两种基本的数据结构,同为容器类型.两者根本的区别在于: stack:后进先出 queue:先进先出 PS:stack和queue是不能通过查询具体某一个位置的元素而进行操作的.但是他们的排列 ...

  10. 领悟JavaScript面向对象

    JavaScript 是面向对象的.但是不少人对这一点理解得并不全面. 在 JavaScript 中,对象分为两种.一种可以称为“普通对象”,就是我们所普遍理解的那些:数字.日期.用户自定义的对象(如 ...