一开始就搞嵌入式,在板子上开发,也没搞gdb,全都是printf,内核模块就printk,感觉也不是很麻烦,也许是印证了习惯就好这句话吧。
之前也有好几次去尝试用gdb,但工作中不天天用,总用不习惯,这回为了调运行的程序,所以干脆再补补写个总结。也不是很 完整,还有些命令不知道使用技巧。
慢慢的要把gdb熟悉起来,在Linux搞调试可能效果会好些,比如调试应用层的iptable,iproute,iputils,net-tools源码等
比较重要和有用的用颜色标出。
1, 编译的时候gcc需要加-g选项,否则可执行文件中不带调试信息,gdb也就无法调试。
2,启动调试
gdb a.out或者启动gdb后执行file a.out
gdb -d .
加入一个源文件的搜索路径。默认搜索路径是环境变量中PATH所定义的路径
3, 设置参数:set args aa bb cc
查看参数:show args
4,查看数组数据
display *arr@6
6为数据长度
5,运行环境。
path
可设定程序的运行路径。
show paths 查看程序的运行路径。
set environment varname [=value] 设置环境变量。如:set env USER=hchen
show environment [varname] 查看环境变量。
6, 调试已运行的程序
pidof vdcd 获取进程号
启动gdb
attach 刚获得的进程号
detach
7, info program 来查看程序的是否在运行,进程号,被暂停
的原因。
8, 设置断点
b func
b line
b file:line
b file:func
在指定行号停住:
break +offset
break -offset 在当前行号的前面或后面的offset行停住。offiset为自然数。
break *address 在程序运行的内存地址处停住。
break 命令没有参数时,表示在下一条指令处停住。
break ... if 可以是上述的参数,condition表示条件,在条件成立时停住。比如在循环境体中,可以设
置break if i=100,表示当i为100时停住程序。
9,查看断点
info break(简写:i b)
(gdb)clear 行号n:清除第n行的断点
(gdb)delete 断点号n:删除第n个断点
(gdb)disable 断点号n:暂停第n个断点
(gdb)enable 断点号n:开启第n个断点
10,continue,继续运行被断点中断的程序。
如果有多个断点,它将跑到下一个断点。
11, until:当你厌倦了在一个循环体内单步跟踪时,这个命令可以运行程序直到退出循环体。
12 ,finish: 运行程序,直到当前函数完成返回,并打印函数返回时的堆栈地址和返回值及参数值等信息。
13, n 执行下一条,不跟进函数。
s 跟进函数。
14, backtrac(bt)显示堆栈
15,print a:将显示整数 a 的值
print ++a:将把 a 中的值加1,并显示出来
print name:将显示字符串 name 的值
print gdb_test(22):将以整数22作为参数调用 gdb_test() 函数
print gdb_test(a):将以变量 a 作为参数调用 gdb_test() 函数
16,display 一直显示
17, call 函数(参数):调用“函数”,并传递“参数”,如:call gdb_test(55)
18, layout:用于分割窗口,可以一边查看代码,一边测试:
layout src:显示源代码窗口
layout asm:显示反汇编窗口
layout regs:显示源代码/反汇编和CPU寄存器窗口
layout split:显示源代码和反汇编窗口
19,kill:将强行终止当前正在调试的程序
20,程式的输出重定向。
run > outfile
21,clear
清除所有的已定义的停止点。
gdb watch跟踪使用实例:
01 #include <stdio.h>
02 #include <stdlib.h>
03 #include <string.h>
04
05 int main()
06 {
07 char * name = NULL;
08 int len = 10;
09
10 name = (char *)malloc(len);
11 strncpy(name, "zengxiaolong", len);
12
13 char ** wild_pointer;
14 wild_pointer = &name; // 野指针,指向了变量name
15 *wild_pointer = (char *)0x100; // 野指针,破坏了变量name
16
17 name[0] = 'a';
18 return 0;
19 }
# gcc test.c -o test -g
# gdb -q test
(gdb) l
1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <string.h>
4
5 int main()
6 {
7 char * name = NULL;
8 int len = 10;
9
10 name = (char *)malloc(len);
(gdb) b 7
Breakpoint 1 at 0x80483b5: file test.c, line 7.
(gdb) r
Starting program: /tftpboot/test
Breakpoint 1, main () at test.c:7
7 char * name = NULL;
(gdb) watch name // 跟踪变量name变化情况
Hardware watchpoint 2: name // 说明该系统结构支持硬件断点
(gdb) c
Continuing.
Hardware watchpoint 2: name
Old value = 0xb7f79dc0 "U\211?WVS??y" // 这里变量name发生了变化
New value = 0x804a008 ""
main () at test.c:11
11 strncpy(name, "zengxiaolong", len);
(gdb) c
Continuing.
Hardware watchpoint 2: name
Old value = 0x804a008 "zengxiaolo"
New value = 0x100 <Address 0x100 out of bounds> // 指针越界了
main () at test.c:17
17 name[0] = 'a';
(gdb) c
Continuing.
Program received signal SIGSEGV, Segmentation fault.
0x080483fd in main () at test.c:17
17 name[0] = 'a';
(gdb)
顺便把以前写的链接一起帖上,方便查阅
http://hi.baidu.com/rosettasky/blog/category/C%20World/index/1