GDB 调试程序

  GDB 调试程序

无论是多么资深的程序员在编写的程序时,都不大可能一次性就会成功,在程序运行时,会出现许许多多意想不到的错误,一味 地只是查看程序用处不大,最有效的方法通过一些手段进入到程序内部进行调试。通常在调试程序的时候如果能够得到以下一些信息,对于开发者找到错误所在是很 有帮助的。

1 . 程序是运行到哪个语句或者表达式就发生了错误?

2 . 如果错误是在执行一个函数的时候出现的,那么是程序的哪一行包含了这个函数的调用语句,在调用该函数的时候传递的实参是什么?

3 . 在程序执行到某处时,所关心的某一个变量值为多少?

4 .某个表达式最终运行的结果为何值?

调试器 ( 更准确地说应该称为符号调试器 ) 能够完成上述目标。它是一个能够运行其他程序的应用程序,它和普通意义上的程序的唯一不同之处在于,调试器能够进入到程序源码中,允许开发者进行逐行单步 运行,了解程序代码执行顺序,和每条语句执行的结果,可以在程序运行的同时,查看甚至是改变任一变量值。在程序运行出错时,它为程序开发者提供程序运行时 的详细细节,从而找到出错的原因。在 Linux 系统中,最常用到的就是 GDB(GNU Degugger) 。 GDB 是 GNU 自带的调试工具。

7.2.3.1 GDB 常用命令

要想使用 gdb ,必须在对源码进行编译的时候,使用 -g 编译选项开关,来通知编译器,开发者希望进行程序调试。用了 -g 选项后,程序在编译的时候就会包含调试信息,这些调试信息存在目标文件中,它描述了每个函数或变量的数据类型以及源码行号和可执行代码地址间对应关系, gdb 正是通过这些信息使源码和机器码相关联的,它实现了源码级的调试。

为了使用 gdb 调试,只需要在命令行中输入 gdb filename(filename 是用 gcc 编译生成的最终可执行文件名 ) ,该语句启动与调试器的文本接口。就在上一小节中所举 makefile 例子来说,就是键入gdb tune1 ,则在屏幕上会出现

[nie@uClinux mysrc]$ gdb tune1

GNU gdb Red Hat Linux 7.x (5.0rh-15) (MI_OUT)

Copyright 2001 Free Software Foundation, Inc.

GDB is free software, covered by the GNU General Public License, and you are

welcome to change it and/or distribute copies of it under certain conditions.

Type "show copying" to see the conditions.

There is absolutely no warranty for GDB. Type "show warranty" for details.

This GDB was configured as "i386-redhat-linux"...

(gdb)

        gdb 虽然运行起来,但是可执行程序 tune1 并没有运行,此时在 gdb 提示符下直接键入 run 命令即可,如果可执行程序在运行的时候需要输入命令行参数,则在 gdb 提示符下可以这样键入命令:

run command-line-arguments ,就如同是输入命令: tune1 command-line-arguments 一样,启动了可执行程序的运行。

有时候,我们希望能够断点调试程序,让程序执行到代码某处时停止继续执行下去,此时可以使用命令 break ,该命令的格式为 break place ,这里 place 可以是程序代码的行号,某函数名,甚至可以是用 break main ,让程序断点设置在代码一开始执行的地方,比如对于上面举的可执行文件名为 tune1 的例子,它调用了一个函数名为 rtExtModeCheckInit 的子函数,如果想让程序执行到该函数处停止,可以在 gdb 提示符下输入: break rtExtModeCheckInit ,此时屏幕上出现下列信息:

Breakpoint 1 at 0x8049a28: file grt_main.c, line 604. 。当然,也可以使用行号设置中断位置,上面设置中断的语句可以等价为 break 604 ,可以在屏幕上看到相同的效果。

当设置了断点后,程序会运行到断点处停下来,此时从屏幕上可以得到类似下面的信息:

Breakpoint 1, main (argc=4, argv=0xbffffb84) at grt_main.c:604

604 rtExtModeCheckInit();

(gdb)

当想将某个断点除去,可以在 gdb 提示符下输入命令: delete N ,这里 N 表示第几个中断,第一个设置的中断序号为 1 ,第二个设置的序号为 2 ,依次类推。如果 delete 后不跟任何序号,在表示把设置的所有断点都删除。如果想查看目前设置断点的情况,可以使用命令 info break ,屏幕会显示出每一个设置的断点信息。

在 gdb 提示符下使用 help 命令,会给出有关 gdb 命令的一个简短描述和命令分类。

如果开发者想进入到程序内部进行单步调试, gdb 提供两种命令供选择, step 和 next 命令,两者的区别在于 step 执行每一条语句,如果遇到函数调用,会跳转到到该函数定义的开始行去执行,而 next 则不进入到函数内部,它把函数调用语句当作普通一条语句执行完成。 continue 命令是继续运行程序,直到遇到下一个断点或程序结束。

有时候使用者仅仅是在 linux 的 bash 提示符下输入命令 gdb 后,启动了 gdb 而已,此时,如果要加载可执行文件,需要在 gdb 提示符下键入命令: file filename(filename 为可执行文件名 ) ,注意是可执行文件的名字而不是源文件名。

当在调试过程中,想查看一个变量值的时候,可以在 gdb 环境下输入命令: watch variablename , 这里的 variablename 是你想观察的变量名。

还有一个可以显示表达式值的命令 print ,其使用规则为 print expressionname ,其中expressionname 为要显示的表达式名。

7.2.3.2 GDB 具体调试实例

下面通过使用一个简单的程序使读者进一步熟悉用 gdb 的调试方法。

源程序名为 example1.c ,代码如下:

/*******************************************************

* Institute of Automation, Chinese Academy of Sciences

* File Name : example.c

* Description : introduce how to use gdb

* Author : Xueyuan Nie

* Date :

*******************************************************/

#include <stdio.h>

static void display(int i, int *ptr);

int main(void)

{

int x = 5;

int *xptr = &x;

printf("In main():/n");

printf(" x is %d and is stored at %p./n", x, &x);

printf(" xptr holds %p and points to %d./n", xptr, *xptr);

display(x, xptr);

return 0;

}

void display(int z, int *zptr)

{

printf("In display():/n");

printf(" z is %d and is stored at %p./n", z, &z);

printf(" zptr holds %p and points to %d./n", zptr, *zptr);

}

要使用 gdb 调试程序,一定要在编译程序时,使用 -g 编译选项,以生成参数符号表 ( augmented symbol table) ,提供调试信息。 首先使用 gcc – g – o example1 example.c 对源代码进行编译,这样就可以使用 gdb 监视 example1 的执行细节。在 bash 提示符下,键入命令 :gdb example1 ,启动了对可执行文件 example1 的调试,

在屏幕上会出现下面的信息:

[nie@uClinux nie]$ gdb example1

GNU gdb Red Hat Linux 7.x (5.0rh-15) (MI_OUT)

Copyright 2001 Free Software Foundation, Inc.

GDB is free software, covered by the GNU General Public License, and you are

welcome to change it and/or distribute copies of it under certain conditions.

Type "show copying" to see the conditions.

There is absolutely no warranty for GDB. Type "show warranty" for details.

This GDB was configured as "i386-redhat-linux"...

(gdb)

最后一行 (gdb) 就是进入到 gdb 调试中的提示符,此时可以在提示符下输入任何想键入的命令。 现在如果要进行断点调试的话,就需要显示一下要调试的源码,以便知道在哪个地方进行断点设置。在 gdb 下, Linux 最常用文本编辑命令 vi 不能使用,可以使用 list 命令列出可执行文件的源代码的一部分,为了列出源代码的全部,只要多键入几次 list 命令即可。具体操作如下:

(gdb) list

1 #include <stdio.h>

2 static void display(int i, int *ptr);

3

4 int main(void) {

5 int x = 5;

6 int *xptr = &x;

7 printf("In main():/n");

8 printf(" x is %d and is stored at %p./n", x, &x);

9 printf(" xptr holds %p and points to %d./n", xptr, *xptr);

10 display(x, xptr);

(gdb) list

11 return 0;

12 }

13

14 void display(int z, int *zptr) {

15 printf("In display():/n");

16 printf(" z is %d and is stored at %p./n", z, &z);

17 printf(" zptr holds %p and points to %d./n", zptr, *zptr);

18 }

(gdb) list

Line number 19 out of range; example1.c has 18 lines.

(gdb)

屏幕上清楚显示出了每一个语句所在的具体行号,比如现在我们想在第五行设置断点,可以在 gdb 提示符下输入命令: break 5 ,可以看到下面的显示信息:

(gdb) break 5

Breakpoint 1 at 0x8048466: file example1.c, line 5.

断点已经设置好,现在开始让程序运行起来,键入命令 run ,也可以键入其缩写形式 r ,屏幕上出现的信息如下:

(gdb) r

Starting program: /home/nie/example1

Breakpoint 1, main () at example1.c:5

5 int x = 5;

上述信息表明, gdb 已经开始执行可执行程序,目前程序运行到 example1.c 程序中 main() 函数的第五行处停止,并且显示出即将要执行的第五行语句。

现在我们进行单步调试的工作,输入命令: next ,它表明单步执行程序的每一条语句,当用 next命令执行到函数 display 处时,即当屏幕出现如下所示信息时:

(gdb) next

6 int *xptr = &x;

(gdb) next

7 printf("In main():/n");

(gdb) next

In main():

8 printf(" x is %d and is stored at %p./n", x, &x);

(gdb) next

x is 5 and is stored at 0xbffffb44.

9 printf(" xptr holds %p and points to %d./n", xptr, *xptr);

(gdb) next

xptr holds 0xbffffb44 and points to 5.

10 display(x, xptr);

为了进入到函数 display 内部进行调试,输入命令 step ,即:

(gdb) step

display (z=5, zptr=0xbffffb44) at example1.c:15

15 printf("In display():/n");

step 命令使执行进入到函数内部,此时在该函数内部,可以继续使用 step 命令或者是 next 命令进行单步执行,如果不想单步执行,而是直接将程序一次执行完毕,可以输入命令 continue 即可。

要退出 gdb ,请键入命令 quit ,如果程序此时仍在进行, gdb 会让你确认是否真的要退出,屏幕会出现类似下面的提示信息:

(gdb) quit

The program is running. Exit anyway? (y or n)

按下 'y' 即退出调试程序,如果程序本身已经运行完毕,则 quit 命令键入后,会直接退出 gdb, 而不出现任何提示信息。

当然除了使用 gdb 进行程序调试外,如果程序比较简短,逻辑又比较简单,此时完全可以不用gdb ,采用 printf 语句在程序当中输出中间变量的值来调试程序,也是一个不错的调试方法。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值