【GDB】

GDB

  • 1. GDB调试器
    • 1.1 前言
    • 1.2 GDB编译程序
    • 1.3 启动GDB
    • 1.4 载入被调试程序
      • 1.4.1 本地调试
      • 1.4.2 远程调试
    • 1.5 查看源码
    • 1.6 运行程序
    • 1.7 断点设置
      • 1.7.1 通过行号设置断点
      • 1.7.2 通过函数名设置断点
      • 1.7.3 通过条件设置断点
      • 1.7.4 查看断点信息
      • 1.7.5 删除断点
    • 1.8 单步调试
    • 1.9 查看变量
    • 1.10 修改变量
    • 1.11 设置观察点
    • 1.12 打印表达式
    • 1.13 查看运行信息
    • 1.14
    • 1.15
    • 1.16 分割窗口
  • 2. GDB调试core文件
    • 2.1 设定core文件的最大值
    • 2.2 临时修改core文件的生成路径
      • 2.2.1 示例一:除数设为0
      • 2.2.2 示例二:这里scanf后的参数没有加&
    • 2.3
    • 2.4

1. GDB调试器

1.1 前言

GDB是GNU调试器(GNU Debugger)的缩写,是一种功能强大的调试工具,可以用于调试多种编程语言的程序,如C、C++、汇编等。GDB可以帮助程序员在程序运行时进行调试,包括查看变量的值、跟踪程序执行流程、设置断点等。
GDB支持多种操作系统,包括Linux、Unix、Windows等。

1.2 GDB编译程序

  • 文件编译
    g++ main.cpp -o main
  • 带有GDB debug功能,编译
    g++ -g main.cpp -o main
  • 查看可执行文件是否带有debug功能
    readelf -S main | grep debug
    在这里插入图片描述

1.3 启动GDB

【GDB】_第1张图片

1.4 载入被调试程序

1.4.1 本地调试

~#gdb ./build

在这里插入图片描述

1.4.2 远程调试

远程调试需要两个GDB程序,运行在远程设备(target)上的程序称之为gdbserver,运行在本地主机host上的gdb程序为交叉编译器,即在x86平台上运行arm平台的gdb程序,记作gdb_client
远程设备 target gdbserver
本地主机 host gdb

  1. 嵌入式设备执行:~#./gdbserver 192.168.10.2:1234 ./build
    其中192.168.10.2表示允许从这个IP地址登录道嵌入式设备,一般我们的PC服务器地址,也尅省略,表示允许从任何IP连入,:1234为端口号,build为即将调试的应用程序。

  2. 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位端口,必须与嵌入式设备运行时指定的端口一直,两者均为必填项,不能省略。
    注:

  • 设备端无需输入build_Debug ,否则加载会非常慢
  • 由于设备端内存等的原因,一般情况下我们都是使用远程调试

1.5 查看源码

list :简记为 l ,其作用就是列出程序的源代码,默认每次显示10行。
list 行号:将显示当前文件以“行号”为中心的前后10行代码,如:list 12
list 函数名:将显示“函数名”所在函数的源代码,如:list main
list :不带参数,将接着上一次 list 命令的,输出下边的内容。

list / l [first, last](行号)
【GDB】_第2张图片

1.6 运行程序

运行命令
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
【GDB】_第3张图片
若程序运行时需要参数,在run命令后加上参数即可.

1.7 断点设置

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 行号/函数名/条件

1.7.1 通过行号设置断点

在这里插入图片描述

1.7.2 通过函数名设置断点

在这里插入图片描述

1.7.3 通过条件设置断点

当n > 1时会在OddOrEven函数中停止执行
在这里插入图片描述
【GDB】_第4张图片

1.7.4 查看断点信息

info breakpoints
【GDB】_第5张图片

1.7.5 删除断点

delete breakpointNum
【GDB】_第6张图片

1.8 单步调试

step / s: 每次执行一步,碰到函数会进入函数内部执行
next / n: 每次执行一步,碰到函数会一步将函数执行完成(不进入函数内部)
finish: 在处于函数内部时,使用该命令会直接将该函数执行完,若函数内部还有其他断点则执行到断点处
continue / c: 执行到下一个断点处
skip: 设置跳过执行某一个函数,即设置之后使用step不会进入函数内部
在这里插入图片描述
删除skip: skip delete [num]

1.9 查看变量

print / p
命令格式:p/格式 变量;

格式 说明
x 显示为16进制
d 显示为10进制
u 显示为无符号10进制
o 显示为8进制
t 显示为2进制数,t表示two
a 地址
c 显示为字符
f 浮点小数
s 显示为字符串

1.10 修改变量

set / print 变量名 = 值

1.11 设置观察点

(当变量发生变化后,程序暂停执行并将变量的原值和新值都打印出来)
watch [变量名]
【GDB】_第7张图片

1.12 打印表达式

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: 显示当前堆栈页的所有变量

1.13 查看运行信息

where/bt :当前运行的堆栈列表;
bt backtrace 显示当前调用堆栈
up/down 改变堆栈显示的深度
set args 参数:指定运行时的参数
show args:查看设置好的参数
info program: 来查看程序的是否在运行,进程号,被暂停的原因。

1.14

1.15

1.16 分割窗口

layout:用于分割窗口,可以一边查看代码,一边测试:
layout src:显示源代码窗口
layout asm:显示反汇编窗口
layout regs:显示源代码/反汇编和CPU寄存器窗口
layout split:显示源代码和反汇编窗口
Ctrl + L:刷新窗口


2. GDB调试core文件

2.1 设定core文件的最大值

ulimit -c [n]
有参数n为设置,没有参数n为查看,若core文件最大值为0则不会生成coredump
在这里插入图片描述

2.2 临时修改core文件的生成路径

echo /root/achen/project/coredump/core.%e.%p> /proc/sys/kernel/core_pattern
%e为程序名,%p为进程id

2.2.1 示例一:除数设为0

【GDB】_第8张图片
编译后运行,并将除数设为0
提示核心已转储(coredump),查看coredump目录下,生成了core文件
在这里插入图片描述
使用GDB对core进行调试
core-file core文件
在这里插入图片描述
这里提示信号终止了程序,发成了arithmetic异常(由于除数为0)
查看函数的调用栈帧和层级关系
backtrace / bt
【GDB】_第9张图片
这里和上面core-file都有一串问号,原因是没加载符号表,解决方案为在core-file之前先载入被调试程序
【GDB】_第10张图片
调用bt命令
在这里插入图片描述
结果为函数调用的栈帧和层级关系,因为该程序中没有函数调用,所以只有显示main函数,且错误行数为第9行

2.2.2 示例二:这里scanf后的参数没有加&

【GDB】_第11张图片
这里scanf后的参数没有加&
编译后运行
在这里插入图片描述
发生coredump,查看coredump目录,生成了core文件
在这里插入图片描述
使用GDB进行调试
【GDB】_第12张图片
这里显示信号终止了服务,发生了段错误
查看栈帧和层级关系发现函数调用关系为main->scanf->_IO_vfscanf,函数错误为第6行

2.3

2.4

你可能感兴趣的:(#,开发工具,c语言)