wiki页:http://code.google.com/p/ldd6410/wiki/GDBDebug
不论是调试Linux内核空间的驱动仍是调试用户空间的应用程序,掌握gdb的用法都是必须。并且,调试内核和调试应用程序时使用的gdb命令是彻底相同的,下面以代码清单22.2的应用程序为例演示gdb调试器的用法。
1 int add(int a, int b)
2 {
3 return a + b;
4 }
5
6 main()
7 {
8 int sum[10] =
9 {
10 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
11 } ;
12 int i;
13
14 int array1[10] =
15 {
16 48, 56, 77, 33, 33, 11, 226, 544, 78, 90
17 };
18 int array2[10] =
19 {
20 85, 99, 66, 0x199, 393, 11, 1, 2, 3, 4
21 };
22
23 for (i = 0; i < 10; i++)
24 {
25 sum[i] = add(array1[i], array2[i]);
26 }
27 }
[root@localhost driver_study]# gdb gdb_example
GNU gdb Red Hat Linux (5.3post-0.20021129.18rh)
Copyright 2003 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-gnu"...
(gdb)
(gdb) list 15
10
11 int array1[10] =
12 {
13 48, 56, 77, 33, 33, 11, 226, 544, 78, 90
14 };
15 int array2[10] =
16 {
17 85, 99, 66, 0x199, 393, 11, 1, 2, 3, 4
18 };
19
list <function> ,显示函数名为function的函数的源程序,如:
(gdb) list main
2 {
3 return a + b;
4 }
5
6 main()
7 {
8 int sum[10];
9 int i;
10
11 int array1[10] =
list,显示当前行后面的源程序。
list - ,显示当前行前面的源程序。
(gdb) break add
Breakpoint 1 at 0x80482f7: file gdb_example.c, line 3.
(gdb) run
Starting program: /driver_study/gdb_example
Breakpoint 1, add (a=48, b=85) at gdb_example.c:3
warning: Source file is more recent than executable.
3 return a + b;
(gdb) next
4 }
(gdb) next
main () at gdb_example.c:23
23 for (i = 0; i < 10; i++)
(gdb) next
25 sum[i] = add(array1[i], array2[i]);
(gdb) print sum
$1 = {133, 0, 0, 0, 0, 0, 0, 0, 0, 0}
set args 可指定运行时参数,如:set args 10 20 30 40 50;show args 命令能够查看设置好的运行参数。
path <dir> 可设定程序的运行路径;how paths可查看程序的运行路径;set environment varname [=value]用于设置环境变量,如set env USER=baohua;show environment [varname]则用于查看环境变量。
cd <dir> 至关于shell的cd命令;pwd 显示当前所在的目录。
info terminal 用于显示程序用到的终端的模式;gdb中也可使用重定向控制程序输出,如run > outfile;tty命令能够指定输入输出的终端设备,如:tty /dev/ttyS1。
在进入指定函数时停住,C++中可使用class::function或function(type, type)格式来指定函数名。
在指定行号停住。
在当前行号的前面或后面的offset行停住,offiset为天然数。
在源文件filename的linenum行处停住。
在源文件filename的function函数的入口处停住。
在程序运行的内存地址处停住。
break命令没有参数时,表示在下一条指令处停住。
“...”能够是上述的break <linenum>、break +offset / break –offset中的参数,condition表示条件,在条件成立时停住。好比在循环体中,能够设置break if i=100,表示当i为100时停住程序。 查看断点时,可以使用info命令,如info breakpoints [n]、info break [n](n表示断点号)。
(gdb) break 25
Breakpoint 1 at 0x8048362: file gdb_example.c, line 25.
(gdb) run
Starting program: /driver_study/gdb_example
Breakpoint 1, main () at gdb_example.c:25
25 sum[i] = add(array1[i], array2[i]);
(gdb) step
add (a=48, b=85) at gdb_example.c:3
3 return a + b;
单步跟踪,若是有函数调用,它不会进入该函数。一样地,next后面不加count表示一条条地执行,加表示执行后面的count条指令,而后再停住。
set step-mode on用于打开step-mode模式,这样,在进行单步跟踪时,程序不会由于没有debug信息而不停住,这个参数的设置可便于查看机器码。set step-mod off用于关闭step-mode模式。
运行程序,直到当前函数完成返回,并打印函数返回时的堆栈地址和返回值及参数值等信息。
一直在循环体内执行单步,退不出来是一件使人烦恼的事情,until命令能够运行程序直到退出循环体。
stepi和nexti用于单步跟踪一条机器指令,一条程序代码有可能由数条机器指令完成,stepi和nexti能够单步执行机器指令。 另外,运行“display/i $pc”命令后,单步跟踪会在打出程序代码的同时打出机器指令,即汇编代码。
continue [ignore-count]
c [ignore-count]
fg [ignore-count]
(gdb) continue
Continuing.
Hardware watchpoint 3: i
Old value = 2
New value = 3
0x0804838d in main () at gdb_example.c:23
23 for (i = 0; i < 10; i++)
(gdb) continue
Continuing.
Breakpoint 1, main () at gdb_example.c:25
25 sum[i] = add(array1[i], array2[i]);
(gdb) continue
Continuing.
Hardware watchpoint 3: i
Old value = 3
New value = 4
0x0804838d in main () at gdb_example.c:23
23 for (i = 0; i < 10; i++)
print <expr>
print /<f> <expr>
<expr>是表达式,是被调试的程序中的表达式,<f>是输出的格式,好比,若是要把表达式按16进制的格式输出,那么就是/x。在表达式中,有几种GDB所支持的操做符,它们能够用在任何一种语言中,“@”是一个和数组有关的操做符,“::”指定一个在文件或是函数中的变量,“{<type>} <addr>”表示一个指向内存地址<addr>的类型为type的一个对象。
(gdb) print sum
$2 = {133, 155, 0, 0, 0, 0, 0, 0, 0, 0}
(gdb) next
Breakpoint 1, main () at gdb_example.c:25
25 sum[i] = add(array1[i], array2[i]);
(gdb) next
23 for (i = 0; i < 10; i++)
(gdb) print sum
$3 = {133, 155, 143, 0, 0, 0, 0, 0, 0, 0}
int *array = (int *) malloc (len * sizeof (int));
p *array@len
咱们可用display命令设置一些自动显示的变量,当程序停住时,或是单步跟踪时,这些变量会自动显示。 若是要修改变量,如x的值,可以使用以下命令:
print x=4
(gdb) watch i
Hardware watchpoint 3: i
(gdb) next
23 for (i = 0; i < 10; i++)
(gdb) next
Hardware watchpoint 3: i
Old value = 0
New value = 1
0x0804838d in main () at gdb_example.c:23
23 for (i = 0; i < 10; i++)
(gdb) next
Breakpoint 1, main () at gdb_example.c:25
25 sum[i] = add(array1[i], array2[i]);
(gdb) next
23 for (i = 0; i < 10; i++)
(gdb) next
Hardware watchpoint 3: i
Old value = 1
New value = 2
0x0804838d in main () at gdb_example.c:23
23 for (i = 0; i < 10; i++)
x/<n/f/u> <addr>
return
return <expression>
info line tst.c:func
(gdb) disassemble func
Dump of assembler code for function func:
0x8048450 <func>: push %ebp
0x8048451 <func+1>: mov %esp,%ebp
0x8048453 <func+3>: sub $0x18,%esp
0x8048456 <func+6>: movl $0x0,0xfffffffc(%ebp)
...
End of assembler dump.