GDB 反向调试

2024-09-07 11:32
文章标签 调试 gdb 反向

本文主要是介绍GDB 反向调试,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

使用调试器时最常用的功能就是step, next, continue,这几个调试命令都是“往下执行”的, 但是很多时候会有这种需求:你在调试的过程中多跳过了几步而错过中间过程,这时候不得不重头调试一遍,非常麻烦。而GDB从7.0版本开始支持反向调试功能,也就是允许你倒退着运行程序,或者说撤销程序执行的步骤从而会到以前的状态。

 

直观地来看,加入你正在使用GDB7.0以上版本的调试器并且运行在支持反向调试的平台,你就可以用以下几条命令来调试程序:

reverse-continue

反向运行程序知道遇到一个能使程序中断的事件(比如断点,观察点,异常)。

 

reverse-step

反向运行程序到上一次被执行的源代码行。

 

reverse-stepi

反向运行程序到上一条机器指令

 

reverse-next

反向运行到上一次被执行的源代码行,但是不进入函数。

 

reverse-nexti

反向运行到上一条机器指令,除非这条指令用来返回一个函数调用、整个函数将会被反向执行。

 

reverse-finish

反向运行程序回到调用当前函数的地方。

 

set exec-direction [forward | reverse]

设置程序运行方向,可以用平常的命令step和continue等来执行反向的调试命令。

 

上面的反向运行也可以理解为撤销后面运行的语句所产生的效果,回到以前的状态。

 

好的,接下来我们来试试看如何反向调试。

首先确认自己的平台支持进程记录回放(Process Record and Replay),当在调试器启用进程记录回放功能时,调试器会记录下子进程,也就是被调试进程的每一步的运行状态与上一步运行状态的差异,需要撤销的时候就可以很方便回到上一步。

假设我们有以下C程序:

[cpp] view plain copy print?
  1. int main(int argc, const char *argv[])  
  2. {  
  3.   int a = 0;  
  4.   a = 1;  
  5.   a = 2;  
  6.   return 0;  
  7. }  
 

 

将它编译并加上调试符号:

[python] view plain copy print?
  1. $ gcc -Wall -g a.c  
 

开始调试:

[python] view plain copy print?
  1. $ gdb a.out  
 

查看一下源代码:

[python] view plain copy print?
  1. (gdb) l  
  2. 1   int main(int argc, const char *argv[])  
  3. 2   {  
  4. 3     int a = 0;  
  5. 4     a = 1;  
  6. 5     a = 2;  
  7. 6     return 0;  
  8. 7   }  
 

接下来设置一个断点在第三行:

[python] view plain copy print?
  1. (gdb) b 3  
  2. Breakpoint 1 at 0x804839a: file a.c, line 3.  
 

运行,程序会在第三行的地方停下来:

[python] view plain copy print?
  1. (gdb) r  
  2. Starting program: /home/cheryl/a.out   
  3. Breakpoint 1, main (argc=1, argv=0xbffff3e4) at a.c:3  
  4. 3     int a = 0;  

给变量a设置监视点方便我们观察:

[python] view plain copy print?
  1. (gdb) watch a  
  2. Hardware watchpoint 2: a  
 

启动进程记录回放:

[python] view plain copy print?
  1. (gdb) record  
 

现在每运行一步调试器都会记录下变化,以便回溯。我们连续执行3条语句。

[python] view plain copy print?
  1. (gdb) n  
  2. 4     a = 1;  
  3. (gdb)   
  4. Hardware watchpoint 2: a  
  5. Old value = 0  
  6. New value = 1  
  7. main (argc=1, argv=0xbffff3e4) at a.c:5  
  8. 5     a = 2;  
  9. (gdb)   
  10. Hardware watchpoint 2: a  
  11. Old value = 1  
  12. New value = 2  
  13. main (argc=1, argv=0xbffff3e4) at a.c:6  
  14. 6     return 0;  
 

可以看到,a的值先是从0变为了1,然后变为2,如果想让程序倒退回到以前的状态怎么办?可以用reverse-next命令:

[python] view plain copy print?
  1. (gdb) reverse-next  
  2. Hardware watchpoint 2: a  
  3. Old value = 2  
  4. New value = 1  
  5. main (argc=1, argv=0xbffff3e4) at a.c:5  
  6. 5     a = 2;  
  7. (gdb)   
  8. Hardware watchpoint 2: a  
  9. Old value = 1  
  10. New value = 0  
  11. main (argc=1, argv=0xbffff3e4) at a.c:4  
  12. 4     a = 1;  
  13. (gdb)   
  14. No more reverse-execution history.  
  15. main (argc=1, argv=0xbffff3e4) at a.c:3  
  16. 3     int a = 0;  
  17. (gdb)   

这样程序就倒退到了我们启动进程记录回放的地方,a的值经过两步回到了最初的状态。

若需要关闭进程记录回放,可以使用record stop:

 

 

[python] view plain copy print?
  1. (gdb) record stop  
  2. Process record is stoped and all execution log is deleted.  
 

 

参考:《Reverse Debugging with GDB》 --- http://sourceware.org/gdb/wiki/ReverseDebug

 From: http://blog.csdn.net/CherylNatsu/article/details/6436570

附注:因多次调试感觉很浪费时间 故此寻找gdb的该项技术,幸好系统的gdb版本是支持的:)

这篇关于GDB 反向调试的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1144936

相关文章

IDEA如何实现远程断点调试jar包

《IDEA如何实现远程断点调试jar包》:本文主要介绍IDEA如何实现远程断点调试jar包的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录问题步骤总结问题以jar包的形式运行Spring Boot项目时报错,但是在IDEA开发环境javascript下编译

Python MCPInspector调试思路详解

《PythonMCPInspector调试思路详解》:本文主要介绍PythonMCPInspector调试思路详解,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录python-MCPInspector调试1-核心知识点2-思路整理1-核心思路2-核心代码3-参考网址

Linux系统调试之ltrace工具使用与调试过程

《Linux系统调试之ltrace工具使用与调试过程》:本文主要介绍Linux系统调试之ltrace工具使用与调试过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、ltrace 定义与作用二、ltrace 工作原理1. 劫持进程的 PLT/GOT 表2. 重定

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

使用Python自建轻量级的HTTP调试工具

《使用Python自建轻量级的HTTP调试工具》这篇文章主要为大家详细介绍了如何使用Python自建一个轻量级的HTTP调试工具,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录一、为什么需要自建工具二、核心功能设计三、技术选型四、分步实现五、进阶优化技巧六、使用示例七、性能对比八、扩展方向建

前端bug调试的方法技巧及常见错误

《前端bug调试的方法技巧及常见错误》:本文主要介绍编程中常见的报错和Bug,以及调试的重要性,调试的基本流程是通过缩小范围来定位问题,并给出了推测法、删除代码法、console调试和debugg... 目录调试基本流程调试方法排查bug的两大技巧如何看控制台报错前端常见错误取值调用报错资源引入错误解析错误

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

ASIO网络调试助手之一:简介

多年前,写过几篇《Boost.Asio C++网络编程》的学习文章,一直没机会实践。最近项目中用到了Asio,于是抽空写了个网络调试助手。 开发环境: Win10 Qt5.12.6 + Asio(standalone) + spdlog 支持协议: UDP + TCP Client + TCP Server 独立的Asio(http://www.think-async.com)只包含了头文件,不依