gdb调试技巧

Wesley13
• 阅读 587

gdb支持源码级调试,这是众所周知的事情。当然,你也可以进行汇编级的调试。理论上这个更应该有用。(因为很多程序的编译常常没有源码级别的调试信息)
那么怎么做?
先写一个最简单的源代码。
$ cat > a.c
int main(void)
{
int i = 0;
i = 2;
return 0;
}
Ctrl + D 保存,编译。
$ gcc a.c
开始调试。
$ gdb a.out
输入断点
(gdb) break main
开始运行
(gdb) run
显示汇编代码。
(gdb) disassemble
默认的汇编语言编码是att模式。你可以手动修改为intel风格。
(gdb) set disassembly-flavor intel
重新显示汇编代码。
(gdb) disassemble
最左端的 => 是当前即将执行的汇编指令。
因为编译的时候,没有加上-g选项,所以没有符号表信息,所以此时只能检测寄存器的值。想要检测某个寄存器的值。怎么做?
下面所示。比如要显示 rbp的值。
(gdb) print $rbp
要执行到下一行。怎么做?
(gdb) nexti
遇到一个函数调用的时候(汇编指令是 call),怎么进入函数?
(gdb) stepi
直接运行到这个函数结束(即跳出帧),怎么做?
(gdb) finish
然后,就可以一个指令,一个指令的运行了。仍然是nexti。
如何在任意汇编指令上添加断点?比如在执行 mov    eax,0x0 的位置加上断点。怎么做?
(gdb) break *0x0000000000400486
这个16进制数是这个指令的地址,也就是该行最左端的16进制数。
想要直接运行到上述断点处,怎么做?
(gdb) continue
在此处,查看eax寄存器的值。
(gdb) print $eax
然后,执行nexti 可以再次检查,eax的值是不是变为了0x0。
(当然,它肯定会变成0x0的,除非这个这个机器的中央处理器有重大缺陷)

注:以下是简单的演示脚本。(示例中没有函数调用)
------------------------------------------------------------------------------------------——-----------------------------------

$ cat a.c
int main()
{
int i = 0;
i = 1;
return 0;
}
$ gcc a.c
$ gdb -silent a.out
Reading symbols from /home/test/a.out...(no debugging symbols found)...done.
(gdb) break main
Breakpoint 1 at 0x400478
(gdb) set disassembly-flavor intel 
(gdb) run   
Starting program: /home/test/a.out 

Breakpoint 1, 0x0000000000400478 in main ()
Missing separate debuginfos, use: debuginfo-install glibc-2.12-1.107.el6.x86_64
(gdb) disassemble 
Dump of assembler code for function main:
   0x0000000000400474 <+0>: push   rbp
   0x0000000000400475 <+1>: mov    rbp,rsp
=> 0x0000000000400478 <+4>: mov    DWORD PTR [rbp-0x4],0x0
   0x000000000040047f <+11>: mov    DWORD PTR [rbp-0x4],0x1
   0x0000000000400486 <+18>: mov    eax,0x0
   0x000000000040048b <+23>: leave  
   0x000000000040048c <+24>: ret    
End of assembler dump.
(gdb) print $rbp

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Stella981 Stella981
2年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Easter79 Easter79
2年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
2年前
CMU
一、实验目的1.理解C语言程序的机器级表示。2.初步掌握GDB调试器的用法。3.阅读C编译器生成的x8664机器代码,理解不同控制结构生成的基本指令模式,过程的实现。二、实验工具1.SecureCRT2.Linux3.Objdump命令反汇编4.GDB调试工具
Wesley13 Wesley13
2年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
2年前
Nginx反向代理upstream模块介绍
!(https://oscimg.oschina.net/oscnet/1e67c46e359a4d6c8f36b590a372961f.gif)!(https://oscimg.oschina.net/oscnet/819eda5e7de54c23b54b04cfc00d3206.jpg)1.Nginx反
Wesley13 Wesley13
2年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
程序调试利器——GDB使用指南
GDB是GNUDebugger的简称,其作用是可以在程序运行时,检测程序正在做些什么。GDB程序自身是使用C和C程序编写的,但可以支持除C和C之外很多编程语言的调试。
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
金旋 金旋
2个月前
Linux C++ GDB调试入门到精通
//下仔のke:https://yeziit.cn/15086/GDB,全称GNU调试器,是GNU项目的一部分,是一个强大的Unix下的程序调试工具。它可以用来调试C、C、ObjectiveC以及其他支持的语言写的程序。使用GDB,你可以设置断点、单步