1、启动gdb
【root】gdb
2、设置调试子进程
(gdb) set follow-fork-mode child
3、执行调试文件
(gdb) file test
4、设置子进程断点
(gdb) b son_thread_fun
5、 程序运行到断点处
(gdb) r
6、单步跟踪
(gdb) n 用n单步跟踪,不进入函数调用
(gdb) s 用s单步跟踪,会进入函数调用
7、查看变量的值
(gdb) print c->addr_text
(gdb) print &c->addr-text 查看变量地址
8、查看栈信息
(gdb) bt
9、查看内存
(gdb) x /40xw 0x80801245 从地址0x80801245查看40个十六进制的四字节
用gdb查看内存
格式: x /nfu
说明
x 是 examine 的缩写
n表示要显示的内存单元的个数
f表示显示方式, 可取如下值
x 按十六进制格式显示变量。
d 按十进制格式显示变量。
u 按十进制格式显示无符号整型。
o 按八进制格式显示变量。
t 按二进制格式显示变量。
a 按十六进制格式显示变量。
i 指令地址格式
c 按字符格式显示变量。
f 按浮点数格式显示变量。
u表示一个地址单元的长度
b表示单字节,
h表示双字节,
w表示四字节,
g表示八字节
举例
x/3uh buf
表示从内存地址buf读取内容,
h表示以双字节为一个单位,
3表示三个单位,
u表示按十六进制显示
例子:
n是个局部变量
Breakpoint 1, main (argc=1, argv=0xbffff3a4) at calc.c:7
7 int n = atoi(argv[1]);
(gdb) print &n
$1 = (int *) 0xbffff2ec
(gdb) x 0xbffff2ec
0xbffff2ec: 0x00282ff4
(gdb) print * (int *) 0xbffff2ec
$2 = 2633716
(gdb) x /4xw 0xbffff2ec
0xbffff2ec: 0x00282ff4 0x080484e0 0x00000000 0xbffff378
(gdb) x /4dw 0xbffff2ec
0xbffff2ec: 2633716 134513888 0 -1073745032
(gdb)
10、注意事项
(1) 如果要使用gdb调试,则在编译文件时需要带上-g选项
例如:
gcc -c -g test.c -o test.o
gcc test.o -lnginx -lpthread -o test
(2)如果在print变量值时,提示值已经优化而无法显示时,说明文件在编译时使用了-O 编译优化,如要要正常显示,则需要把-O 修改为 -O0
例如:nginx的编译选项
CFLAGS = -pipe -O -W -Wall -Wpointer-arith -Wno-unused-parameter -Werror -g
可修改为
CFLAGS = -pipe -O0 -W -Wall -Wpointer-arith -Wno-unused-parameter -Werror -g