热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

1.GDB调试以汇编语言为例

#rpm-qa|grepgdb下载:安装#tar-zxvf#.configure#make使用GDB以汇编语言调试为例汇编语言实现CPUID指令CP

#rpm -qa |grep  gdb

下载:

安装

#tar -zxvf

#./configure

#make


使用GDB

以汇编语言调试为例

汇编语言实现CPUID指令


CPUID

cpuid是Intel  Pentinum以上级CPU内置的一个指令(486级以下的CPU不支持),他用于识别某一类型的CPU,

它能返回CPU级别,型号,CPU步进以及CPU字串信息,从此命令也可以得到CPU的缓存和TLB信息

CPUID返回数据类型是在EAX寄存器里定义的,而指令返回的数值则存储在EAX,EBX,ECX和EDX寄存器中。

返回的信息分两部分:基本信息与扩展信息。

在EAX输入0-3参数时,它返回CPU基本信息;

而在EAX输入0x8000000至ox800000x,他返回的是CPU扩展信息。扩展信息只包括在Pentinum4及以后的CPU上。

 

CPU级别         基本信息   扩展信息
486及以前的CPU      不可用    不可用
Pentium        0x1    不可用
Pentium Pro,Pentium 2    0x2    不可用
Pentium 3      0x3    不可用
Pentium 4      0x2    0x80000004
Xeon(至强)      0x2    0x80000004

 


代码

cpuid.s

#cpuid.s Sample program to extract the processor Vendor ID                
.section .data                
output:                   .ascii "The processor Vendor ID is 'xxxxxxxxxxxx'\n"
.section .text                
.globl _start                
_start:                   movl $0, %eax                   cpuid                   movl $output, %edi                   movl %ebx, 28(%edi)                   movl %edx, 32(%edi)                   movl %ecx, 36(%edi)                   movl $4, %eax                   movl $1, %ebx                   movl $output, %ecx                   movl $42, %edx                   int $0x80                   movl $1, %eax                   movl $0, %ebx                   int $0x80

 
汇编


使用GNU汇编器

#as -o cpuid.o cpuid.s

使用GNU链接器

#ld -o cpuid cpuid.o

运行程序

#./cpuid

输出


调试

为了调试汇编语言程序,必须使用-gstabs参数重新汇编代码

#as -gstabs -o cpuid.o cpuid.s

#ld -o cpuid cpuid.o

使用-gstabs参数在可执行文件中添加了附加信息,所以新产生的文件会大些

如果没有必要不要使用调试信息


运行

在gdb内运行

#gdb cupid

GNU调试器启动,把程序加载到内存,使用run命令(简化r也可以)从gdb内运行程序

可见调试器内的运行和从命令行直接运行一样


断点

汇编语言中指定断点时,必须指定对于最近的标签的相对位置,上述代码中只有一个标签,所以每个断点必须依据_start指定。

break(简化的b)命令格式 break * label+offset

break 函数名

break 行号

break 文件名:行号

break 文件名:函数名

break +偏移量

break -偏移量

break *地址

label是被引用的源代码中的标签

offset是应该停止的地方距离标签的行数

break * _start

*_start参数指定了断点,该参数指定_start标签后的第一条指令码。

run或者(简化的r)启动程序,暂停在第一条指令码处。


单步

使用next(简化的n)或者step(简化的s)命令单步调试

注意:s会步入调用的函数,类似于VS中的F11,而n类似于VS中的F10

每个next或者step命令执行下一行代码


继续运行

使用continue(或者c)按正常方式继续运行,类似于VS中的F5


查看数据

info  registers   查看所有寄存器的值

print (或者p)   显示特定寄存器或者来自程序的变量值

p $eax            显示寄存器的内容

p/格式  变量

格式

p/t    显示为2进制数

p/o   显示为8进制数

p/d   显示为10进制数

p/u   显示为无符号10进制数

p/x   显示为16进制数

p/a   地址

p/c   显示为字符

p/s   显示为字符串

p/f    浮点小数

p/i    显示为机器语言(仅在显示内存的x命令中可用)

程序指针可以写为$pc或者$eip,因为Intel IA-32架构中的程序指针名为eip

p $pc

p $eip

x         显示特定的内存位置内容

x/格式 地址

x命令显示特定内存位置的值。

x命令的格式 x/nyz

n是要显示的字段数;

y是输出格式同之前的p

z是要显示的字段长度:

b        字节

h        半字(2字节)

w       字(4字节)默认值

g        双字(8字节)

例如:

x $pc

x/i $pc

此处x/i意为显示汇编指令

也有反汇编命令disassemble(或者disas)

格式:

disas                                    

disas       程序计数器

disas       开始地址  结束地址

由此可见,在cpuid指令执行之前,EBX,ECX,EDX寄存器都是0,之后他们包含从厂商ID字符串来的值。

使用x命令显示位于output变量前42个字节的内存位置的值(&符号表明是一个内存位置):


GDB的数据显示格式:

x 按十六进制格式显示变量。
d 按十进制格式显示变量。
u 按十六进制格式显示无符号整型。
o 按八进制格式显示变量。
t 按二进制格式显示变量。
a 按十六进制格式显示变量。
c 按字符格式显示变量。
f 按浮点数格式显示变量。


查看寄存器和内存

1) info args
打印出当前函数的参数名及其值。
2)info locals
打印出当前函数中所有局部变量及其值。
3)info catch
打印出当前的函数中的异常处理信息。
4)源代码的内存
你可以使用info line命令来查看源代码在内存中的地址。info line后面可以跟“行号”,“函数名”,“文件名:行号”,“文件名:函
数名”,这个命令会打印出所指定的源码在运行时的内存地址,如:
(gdb) info line tst.c:func
Line 5 of "tst.c" starts at address 0x8048456 and ends at 0x804845d .
5)info break
查看断点信息。
6)info threads
看正在运行程序中的线程信息
7)info registers
查看寄存器的情况。(除了浮点寄存器)
8)info all-registers
查看所有寄存器的情况。(包括浮点寄存器)
9)info registers
查看所指定的寄存器的情况。

例如:info registers ebp

10)info frame或者i f
这个命令会打印出更为详细的当前栈层的信息,只不过,大多数都是运行时的内内
地址。比如:函数地址,调用函数的地址,被调用函数的地址,目前的函数是由什么
样的程序语言写成的、函数参数地址及值、局部变量的地址等等

11)(gdb) disassemble func

disassemble可以查看源程序的当前执行时的机器码,这个命令
会把目前内存中的指令dump出来

12)list

用list命令来打印程序的源代码

13)x

可以使用examine命令(简写是x)来查看内存地址中的值.

x/3uh 0x54320表示,从内存地址0x54320读取内容,h表示以双字节为一个单位,3表示三个单位,u表示按十六进制显示。

x/48xw  $ebp   也可以一样显示改寄存器或者用上述方法按寄存器的地址

参考:GDB查看栈信息


查看进程

info proc


查看栈帧

bt

backtrace

info stack

where

都是一样的只是别名而已

bt  显示所有栈帧

bt N  显示开头N个栈帧

bt -N 显示最后N个栈帧

bt full 显示栈帧+局部变量

bt full N  N同前


监视

eatch <表达式>

<表达式>发生变化时暂停运行&#xff0c;此处<表达式>是常量或变量等

awatch <表达式>

<表达式>被访问&#xff0c;改变时暂停运行

nwatch <表达式>

<表达式>被访问时暂停运行

delete<编号>

删除监视点


改变变量的值

set variable <变量><表达式>

例如&#xff1a;

(gdb) p  options

$7&#61;1

(gdb)set variable options&#61;0

(gdb)print options

$8&#61;0


GDB命令

1)单步调试:

n (next),

ni(nexti) 执行下一行&#xff08;以汇编代码为单位&#xff09;

s(step 跟n的区别&#xff0c;s进入到函数内)

si(stepi) 执行下一行&#xff08;以汇编代码为单位&#xff09;

u(until)执行到指定行

2)恢复操作&#xff1a;c(continue或者cont) 直到遇到下个断点

3)临时断点: tbreak 有效期&#xff0c;第一次遇到

4)检查变量&#xff1a;p (printf)  显示表达式

x   显示内存内容

5)监视点&#xff1a;watch 当监视点的值发生变化时停止

6)查看栈&#xff1a;bt(backtrace&#xff0c;where) 显示整个栈的内容。

f(frame)选择要显示的栈帧

do(down) 在当前调用的栈帧中选择要显示的栈帧

7)看已经设的断点: ib(info break)

8)设置断点&#xff1a;break(b)

break function, break line_number, break filename:line_number, break filename:function

info break 查看断点信息。

9)删除断点: d(delete)&#xff0c;delete&#43;数值标识符(从第7点可得到) (不加参数&#xff0c;删除所有断点)&#xff0c; clear使用跟第8点对应

10)禁用断点&#xff1a;disable&#43;数值标识符 &#xff08;重新启用 enable&#xff09;

11)在单步时跳出函数&#xff1a;finish

12)在单步时跳出循环&#xff1a;until

13)条件断点&#xff1a;break break-arg if (condition),例: break main if argc > 1

14)断点命令列表(到断点自动执行)&#xff1a;

commands breakpoint-number 例子&#xff1a;commands 1

... >printf "i &#61; %d", i

commands >end

...

end

a) 在commands 中加入silent&#xff0c;过滤到其他无用的输出。

b) 最后一个commands是continue的话&#xff0c;自动continue。

例&#xff1a;comands 1

> silent

> printf "i &#61; %d", i

> continue

> end

15)查看局部变量&#xff1a;info locals 得到当前栈中所有局部变量的值列表

16)设置变量&#xff1a;set x&#61;12

17)GDB线程命令&#xff1a;

a) info threads(给出当前所有的线程信息)

b) thread 3(切换查看线程)

c) break 88 thread 3(当线程3到达源代码行88时停止执行)

d) break 88 thread 3 if x &#61;&#61; y

e) thread apply all bt&#xff0c;查看所有的线程的栈信息。

18) 您可以以进程ID作为第二个参数&#xff0c;以调式一个正在运行的进程

gdb 程序名 1234

19)finish运行到函数结束

20)forward-search(或者fo)  向前搜索

21)generate-core-file(或者gcore)  生成内核转储

22edit 编辑文件或函数

23)directory(或者dir)插入目录

24)list(或者l)显示函数或者行

25)info (或者i)显示信息

26)help (或者h)显示帮助

注意&#xff1a;

1)重新编译文件时不要退出gdb&#xff0c;断点可以保存着。

2)在调试时不要开启优化代码的选项&#xff0c;不然经过了优化&#xff0c;设置的断点的位置跟编译后的位置相差可能很大。

《深入理解计算机系统(原书第2版)》

更高级的GDB调试&#xff1a;

GDB attach到进程
————————————————
版权声明&#xff1a;本文为CSDN博主「unix21」的原创文章&#xff0c;遵循 CC 4.0 BY-SA 版权协议&#xff0c;转载请附上原文出处链接及本声明。
原文链接&#xff1a;https://blog.csdn.net/unix21/article/details/8450016


推荐阅读
  • VueCLI多页分目录打包的步骤记录
    本文介绍了使用VueCLI进行多页分目录打包的步骤,包括页面目录结构、安装依赖、获取Vue CLI需要的多页对象等内容。同时还提供了自定义不同模块页面标题的方法。 ... [详细]
  • 本文讨论了在openwrt-17.01版本中,mt7628设备上初始化启动时eth0的mac地址总是随机生成的问题。每次随机生成的eth0的mac地址都会写到/sys/class/net/eth0/address目录下,而openwrt-17.01原版的SDK会根据随机生成的eth0的mac地址再生成eth0.1、eth0.2等,生成后的mac地址会保存在/etc/config/network下。 ... [详细]
  • Commit1ced2a7433ea8937a1b260ea65d708f32ca7c95eintroduceda+Clonetraitboundtom ... [详细]
  • 本文讨论了在Windows 8上安装gvim中插件时出现的错误加载问题。作者将EasyMotion插件放在了正确的位置,但加载时却出现了错误。作者提供了下载链接和之前放置插件的位置,并列出了出现的错误信息。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • C++字符字符串处理及字符集编码方案
    本文介绍了C++中字符字符串处理的问题,并详细解释了字符集编码方案,包括UNICODE、Windows apps采用的UTF-16编码、ASCII、SBCS和DBCS编码方案。同时说明了ANSI C标准和Windows中的字符/字符串数据类型实现。文章还提到了在编译时需要定义UNICODE宏以支持unicode编码,否则将使用windows code page编译。最后,给出了相关的头文件和数据类型定义。 ... [详细]
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • MATLAB函数重名问题解决方法及数据导入导出操作详解
    本文介绍了解决MATLAB函数重名的方法,并详细讲解了数据导入和导出的操作。包括使用菜单导入数据、在工作区直接新建变量、粘贴数据到.m文件或.txt文件并用load命令调用、使用save命令导出数据等方法。同时还介绍了使用dlmread函数调用数据的方法。通过本文的内容,读者可以更好地处理MATLAB中的函数重名问题,并掌握数据导入导出的各种操作。 ... [详细]
  • 本文介绍了在Windows系统上使用C语言命令行参数启动程序并传递参数的方法,包括接收参数程序的代码和bat文件的编写方法,同时给出了程序运行的结果。 ... [详细]
  • 解决.net项目中未注册“microsoft.ACE.oledb.12.0”提供程序的方法
    在开发.net项目中,通过microsoft.ACE.oledb读取excel文件信息时,报错“未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程序”。本文提供了解决这个问题的方法,包括错误描述和代码示例。通过注册提供程序和修改连接字符串,可以成功读取excel文件信息。 ... [详细]
  • 本文介绍了操作系统的定义和功能,包括操作系统的本质、用户界面以及系统调用的分类。同时还介绍了进程和线程的区别,包括进程和线程的定义和作用。 ... [详细]
author-avatar
mobiledu2502909493
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有