调试工具
GDB
启动GDB的方式
# 直接对可执行程序进行GDB调试
gdb program
# 调试运行程序和core文件
gdb program XX.core
#对于服务程序,指定服务运行时的进程ID,GDB会自动attach并调试
gdb program <program_pid>
查看堆栈信息
# 打印堆栈信息
bt
# 将当前栈帧设置为args(编号或Address)指定的栈帧,并打印该栈帧的简要信息。
frame args
# 例如:
frame 7
# 打印指定栈帧的详细信息。
info frame args
# 例如:
info frame 7
# 打印变量的值
print <var>
# 查看堆栈信息
(gdb) i f
Stack level 0, frame at 0x7fffffffe030:
rip = 0x400b58 in test1 (backtrace.c:9); saved rip = 0x400bbe
called by frame at 0x7fffffffe060
source language c.
Arglist at 0x7fffffffe020, args: arg=1
Locals at 0x7fffffffe020, Previous frame's sp is 0x7fffffffe030
Saved registers:
rbp at 0x7fffffffe020, rip at 0x7fffffffe028
- 当前桢的地址:0x7fffffffe030。
- rip的值:0x400b58,此处引申介绍一下rip是什么:它是指令地址寄存器,用来存储 CPU 即将要执行的- - 指令地址。每次 CPU 执行完相应的汇编指令之后,rip 寄存器的值就会自行累加,rip 无法直接赋值。
- 当前桢函数:test1 (backtrace.c:9)。
- 调用者的rip值:saved rip = 0x400bbe。
- 调用者的帧地址:0x7fffffffe060。
- 源代码所用的程序的语言: source language c。
- 当前桢的参数的地址及值:Arglist at 0x7fffffffe020, args: arg=1。
- 当前帧中局部变量的地址:Locals at 0x7fffffffe020, Previous frame’s sp is 0x7fffffffe030。
- 当前桢中存储的寄存器:rbp at 0x7fffffffe020, rip at 0x7fffffffe028。
打印内存信息
# x/<n/f/u> <addr>
x/128xb
其中,n、f、u是可选的参数,<addr>
表示一个内存地址。
- n是一个正整数,表示需要显示的内存单元的个数,也就是说从当前地址向后显示几个内存单元的内容,一个内存单元的大小由后面的u定义。
- f 表示显示的格式,参见下面。如果地址所指的是字符串,那么格式可以是s,如果地十是指令地址,那么格式可以是i。
- u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4个bytes。u参数可以用下面的字符来代替:
- b表示单字节
- h表示双字节
- w表示四字 节
- g表示八字节
断点操作相关
# 查看断点信息
info breakpoints
# 删除所有断点
delete
# 删除断点(组),其中的数字代表断点的编号
delete 1
delete 1-4
# 删除所有断点,其中的数字代表行号,无数字代表所有
clear
clear line_num
# 清除行N以上所有的断点
clear N
# 断点失效
disable b_id
# 断点有效
enable b_id
变量操作相关
# 打印变量,其中print可以简写为p
print var
# 查看变量类型
ptype var
# 设置变量
set var=value
窗口操作
# 显示源代码窗口,当前执行代码会高亮标志
layout src
参考资料
pstack
pstack命令可显示每个进程的栈跟踪。
addr2line
/**
命令用法:
addr2line [-a| --addresses ] [-b bfdname |
--target=bfdname] [-C |
--demangle[=style]] [-e filename |
--exe=filename] [-f | --function] [-s |
--basename] [-i | --inlines] [-p |
--pretty-print] [-j |
--section=name] [-H |
--help] [-V |
--version] [addr addr ...]
命令参数:
-a --addresses:在函数名、文件和行号信息之前,显示地址,以十六进制形式。
-b --target=<bfdname>:指定目标文件的格式为bfdname。
-e --exe=<executable>:指定需要转换地址的可执行文件名。
-i --inlines : 如果需要转换地址是一个内联函数,则输出信息包括其最近范围内的一个非内联函数信息。
-j --section=<name>:给出的地址代表指定section偏移,而非绝对地址。
-p --pretty-print:函数输出信息更加人性化:每一个地址信息占一行。
-s --basenames:仅仅显示每个文件名的基址(即不显示文件的具体路径,只显示文件名)。
-f --functions:在显示文件名、行号输出信息的同时显示函数名信息。
-C --demangle[=style]:将低级别的符号名解码为用户级别的名字。
-h --help:输出帮助信息。
-v --version:输出版本号。
*/
参考资料: