gdb 常用参数

1. 调试带参数的程序:

  a. gdb --args 程序名 arg1,arg2...

       b. gdb 程序名

           然后 set args arg1,arg2...

           这里的 arg1, arg2... 不再包含程序名

    show args 命令可以查看设置好的运行参数

2. 显示源代码

   a. list

          list <linenum>         显示程序第linenum行的周围的源程序。   
          list <function>         显示函数名为function的函数的源程序。       
          list        显示当前行后面的源程序。   
          list -         显示当前行前面的源程序。

          一般是打印当前行的上5行和下5行,如果显示函数是是上2行下8行,默认是10行,当然,你也可以定制显示的范围,使用下面命令可以设置一次显示源程序的行数。

          set listsize <count>        设置一次显示源代码的行数。       
          show listsize       查看当前listsize的设置。       

          list命令还有下面的用法:

          list <first>, <last>        显示从first行到last行之间的源代码。   
          list , <last>        显示从当前行到last行之间的源代码。       
          list +         往后显示源代码。       

         一般来说在list后面可以跟以下这们的参数:

         <linenum>   行号。
         <+offset>   当前行号的正偏移量。
         <-offset>   当前行号的负偏移量。
         <filename:linenum> 哪个文件的哪一行。
         <function> 函数名。
         <filename:function> 哪个文件中的哪个函数。
         <*address> 程序运行时的语句在内存中的地址。 

 

 

 

 

 

 

 

 

 

 

******gdb 使用范例************************
-----------------
清单 一个有错误的 C 源程序 bugging.c
代码:

-----------------
1 #i nclude
2
3 static char buff [256];
4 static char* string;
5 int main ()
6 {
7   printf ("Please input a string: ");
8   gets (string);  
9   printf ("\nYour string is: %s\n", string);
10 }


-----------------
  上面这个程序非常简单,其目的是接受用户的输入,然后将用户的输入打印出来。该程序使用了一个未经过初始化的字符串地址 string,因此,编译并运行之后,将出现 Segment Fault 错误:
$ gcc -o bugging -g bugging.c
$ ./bugging
Please input a string: asfd
Segmentation fault (core dumped)
为了查找该程序中出现的问题,我们利用 gdb,并按如下的步骤进行:
1.运行 gdb bugging 命令,装入 bugging 可执行文件;
2.执行装入的 bugging 命令 run;
3.使用 where 命令查看程序出错的地方;
4.利用 list 命令查看调用 gets 函数附近的代码;
5.唯一能够导致 gets 函数出错的因素就是变量 string。用print命令查看 string 的值;
6.在 gdb 中,我们可以直接修改变量的值,只要将 string 取一个合法的指针值就可以了,为此,我们在第8行处设置断点 break 8;
7.程序重新运行到第 8行处停止,这时,我们可以用 set variable 命令修改 string 的取值;
8.然后继续运行,将看到正确的程序运行结果。

 

 

 

 

一、多线程调试

 

多线程调试可能是问得最多的。其实,重要就是下面几个命令:

 

  • info thread 查看当前进程的线程。
  • thread <ID> 切换调试的线程为指定ID的线程。
  • break file.c:100 thread all  在file.c文件第100行处为所有经过这里的线程设置断点。
  • set scheduler-locking off|on|step,这个是问得最多的。在使用step或者continue命令调试当前被调试线程的时候,其他线程也是同时执行的,怎么只让被调试程序执行呢?通过这个命令就可以实现这个需求。
    • off 不锁定任何线程,也就是所有线程都执行,这是默认值。
    • on 只有当前被调试程序会执行。
    • step 在单步的时候,除了next过一个函数的情况(熟悉情况的人可能知道,这其实是一个设置断点然后continue的行为)以外,只有当前线程会执行。

 

二、调试宏

 

这个问题超多。在GDB下,我们无法print宏定义,因为宏是预编译的。但是我们还是有办法来调试宏,这个需要GCC的配合。

 

在GCC编译程序的时候,加上-ggdb3参数,这样,你就可以调试宏了。

 

另外,你可以使用下述的GDB的宏调试命令 来查看相关的宏。

 

  • info macro – 你可以查看这个宏在哪些文件里被引用了,以及宏定义是什么样的。
  • macro – 你可以查看宏展开的样子。

 

 

 

三、源文件

 

这个问题问的也是很多的,太多的朋友都说找不到源文件。在这里我想提醒大家做下面的检查:

 

  1. 编译程序员是否加上了-g参数以包含debug信息。
  2. 路径是否设置正确了。使用GDB的directory命令来设置源文件的目录。

 

下面给一个调试/bin/ls的示例(ubuntu下)

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
$ apt-get source coreutils
$ sudo apt-get install coreutils-dbgsym
$ gdb /bin/ls
GNU gdb (GDB) 7.1-ubuntu
(gdb) list main
1192    ls.c: No such file or directory.
in ls.c
(gdb) directory ~/src/coreutils-7.4/src/
Source directories searched: /home/hchen/src/coreutils-7.4:$cdir:$cwd
(gdb) list main
1192        }
1193    }
1194
1195    int
1196    main (int argc, char **argv)
1197    {
1198      int i;
1199      struct pending *thispend;
1200      int n_files;
1201

 

四、条件断点

 

条件断点是语法是:break  [where] if [condition],这种断点真是非常管用。尤其是在一个循环或递归中,或是要监视某个变量。注意,这个设置是在GDB中的,只不过每经过那个断点时GDB会帮你检查一下条件是否满足。

 

五、命令行参数

 

有时候,我们需要调试的程序需要有命令行参数,很多朋友都不知道怎么设置调试的程序的命令行参数。其实,有两种方法:

 

  1. gdb命令行的 –args 参数
  2. gdb环境中 set args命令。

 

六、gdb的变量

 

有时候,在调试程序时,我们不单单只是查看运行时的变量,我们还可以直接设置程序中的变量,以模拟一些很难在测试中出现的情况,比较一些出错,或是switch的分支语句。使用set命令可以修改程序中的变量。

 

另外,你知道gdb中也可以有变量吗?就像shell一样,gdb中的变量以$开头,比如你想打印一个数组中的个个元素,你可以这样:

 

1
2
3
4
5
(gdb) set $i = 0
 
(gdb) p a[$i++]
 
...  #然后就一路回车下去了

 

当然,这里只是给一个示例,表示程序的变量和gdb的变量是可以交互的。

 

七、x命令

 

也许,你很喜欢用p命令。所以,当你不知道变量名的时候,你可能会手足无措,因为p命令总是需要一个变量名的。x命令是用来查看内存的,在gdb中 “help x” 你可以查看其帮助。

 

  • x/x 以十六进制输出
  • x/d 以十进制输出
  • x/c 以单字符输出
  • x/i  反汇编 – 通常,我们会使用 x/10i $ip-20 来查看当前的汇编($ip是指令寄存器)
  • x/s 以字符串输出

 

八、command命令

 

有一些朋友问我如何自动化调试。这里向大家介绍command命令,简单的理解一下,其就是把一组gdb的命令打包,有点像字处理软件的“宏”。下面是一个示例:

 

1
2
3
4
5
6
7
8
9
10
(gdb) break func
Breakpoint 1 at 0x3475678: file test.c, line 12.
(gdb) command 1
Type commands for when breakpoint 1 is hit, one per line.
End with a line saying just "end".
>print arg1
>print arg2
>print arg3
>end
(gdb)

 

当我们的断点到达时,自动执行command中的三个命令,把func的三个参数值打出来。

 

posted @ 2012-09-03 20:09  alxe_yu  阅读(1224)  评论(0)    收藏  举报