GDB是GNU调试器(GNU Debugger)的缩写,是一种功能强大的调试工具,可以用于调试多种编程语言的程序,如C、C++、汇编等。GDB可以帮助程序员在程序运行时进行调试,包括查看变量的值、跟踪程序执行流程、设置断点等。
GDB支持多种操作系统,包括Linux、Unix、Windows等。
~#gdb ./build
远程调试需要两个GDB程序,运行在远程设备(target)上的程序称之为gdbserver,运行在本地主机host上的gdb程序为交叉编译器,即在x86平台上运行arm平台的gdb程序,记作gdb_client
远程设备 target gdbserver
本地主机 host gdb
嵌入式设备执行:~#./gdbserver 192.168.10.2:1234 ./build
其中192.168.10.2表示允许从这个IP地址登录道嵌入式设备,一般我们的PC服务器地址,也尅省略,表示允许从任何IP连入,:1234为端口号,build为即将调试的应用程序。
PC端执行:
~#arm-linux-gdb ./build_Debug
handle SIGPIPE SIGUSR2 SIG32 nostop noprint
target remote 172.8.4.11:1234 //与服务端建立关联
其中172.8.4.11为当前嵌入式设备的IP地址,:1234位端口,必须与嵌入式设备运行时指定的端口一直,两者均为必填项,不能省略。
注:
list :简记为 l ,其作用就是列出程序的源代码,默认每次显示10行。
list 行号:将显示当前文件以“行号”为中心的前后10行代码,如:list 12
list 函数名:将显示“函数名”所在函数的源代码,如:list main
list :不带参数,将接着上一次 list 命令的,输出下边的内容。
运行命令
run:简记为 r ,其作用是运行程序,当遇到断点后,程序会在断点处停止运行,等待用户输入下一步的命令。
continue (简写c ):继续执行,到下一个断点处(或运行结束)
next:(简写 n),单步跟踪程序,当遇到函数调用时,也不进入此函数体;此命令同 step 的主要区别是,step 遇到用户自定义的函数,将步进到函数中去运行,而 next 则直接调用函数,不会进入到函数体内。
step (简写s):单步调试如果有函数调用,则进入函数;与命令n不同,n是不进入调用的函数的
until:当你厌倦了在一个循环体内单步跟踪时,这个命令可以运行程序直到退出循环体。
until+行号: 运行至某行,不仅仅用来跳出循环
finish: 运行程序,直到当前函数完成返回,并打印函数返回时的堆栈地址和返回值及参数值等信息。
call 函数(参数):调用程序中可见的函数,并传递“参数”,如:call gdb_test(55)
quit:简记为 q ,退出gdb
run / r
若程序运行时需要参数,在run命令后加上参数即可.
break n (简写b n):在第n行处设置断点(可以带上代码路径和代码名称: b OAGUPDATE.cpp:578)
b fn1 if a>b:条件断点设置
break func(break缩写为b):在函数func()的入口处设置断点,如:break cb_button
delete 断点号n:删除第n个断点
disable 断点号n:暂停第n个断点
enable 断点号n:开启第n个断点
clear 行号n:清除第n行的断点
info b (info breakpoints) :显示当前程序的断点设置情况
delete breakpoints:清除所有断点:
break / b 行号/函数名/条件
step / s: 每次执行一步,碰到函数会进入函数内部执行
next / n: 每次执行一步,碰到函数会一步将函数执行完成(不进入函数内部)
finish: 在处于函数内部时,使用该命令会直接将该函数执行完,若函数内部还有其他断点则执行到断点处
continue / c: 执行到下一个断点处
skip: 设置跳过执行某一个函数,即设置之后使用step不会进入函数内部
删除skip: skip delete [num]
print / p
命令格式:p/格式 变量;
格式 | 说明 |
---|---|
x | 显示为16进制 |
d | 显示为10进制 |
u | 显示为无符号10进制 |
o | 显示为8进制 |
t | 显示为2进制数,t表示two |
a | 地址 |
c | 显示为字符 |
f | 浮点小数 |
s | 显示为字符串 |
set / print 变量名 = 值
(当变量发生变化后,程序暂停执行并将变量的原值和新值都打印出来)
watch [变量名]
print 表达式:简记为 p ,其中“表达式”可以是任何当前正在被测试程序的有效表达式,比如当前正在调试C语言的程序,那么“表达式”可以是任何C语言的有效表达式,包括数字,变量甚至是函数调用。
print a:将显示整数 a 的值
print ++a:将把 a 中的值加1,并显示出来
print name:将显示字符串 name 的值
print gdb_test(22):将以整数22作为参数调用 gdb_test() 函数
print gdb_test(a):将以变量 a 作为参数调用 gdb_test() 函数
display 表达式:在单步运行时将非常有用,使用display命令设置一个表达式后,它将在每次单步进行指令后,紧接着输出被设置的表达式及值。如: display a
watch 表达式:设置一个监视点,一旦被监视的“表达式”的值改变,gdb将强行终止正在被调试的程序。如: watch a
whatis :查询变量或函数
info function: 查询函数
扩展info locals: 显示当前堆栈页的所有变量
where/bt :当前运行的堆栈列表;
bt backtrace 显示当前调用堆栈
up/down 改变堆栈显示的深度
set args 参数:指定运行时的参数
show args:查看设置好的参数
info program: 来查看程序的是否在运行,进程号,被暂停的原因。
layout:用于分割窗口,可以一边查看代码,一边测试:
layout src:显示源代码窗口
layout asm:显示反汇编窗口
layout regs:显示源代码/反汇编和CPU寄存器窗口
layout split:显示源代码和反汇编窗口
Ctrl + L:刷新窗口
ulimit -c [n]
有参数n为设置,没有参数n为查看,若core文件最大值为0则不会生成coredump
echo /root/achen/project/coredump/core.%e.%p> /proc/sys/kernel/core_pattern
%e为程序名,%p为进程id
编译后运行,并将除数设为0
提示核心已转储(coredump),查看coredump目录下,生成了core文件
使用GDB对core进行调试
core-file core文件
这里提示信号终止了程序,发成了arithmetic异常(由于除数为0)
查看函数的调用栈帧和层级关系
backtrace / bt
这里和上面core-file都有一串问号,原因是没加载符号表,解决方案为在core-file之前先载入被调试程序
调用bt命令
结果为函数调用的栈帧和层级关系,因为该程序中没有函数调用,所以只有显示main函数,且错误行数为第9行
这里scanf后的参数没有加&
编译后运行
发生coredump,查看coredump目录,生成了core文件
使用GDB进行调试
这里显示信号终止了服务,发生了段错误
查看栈帧和层级关系发现函数调用关系为main->scanf->_IO_vfscanf,函数错误为第6行