x86 64位的ubuntu环境下汇编(无优化)及函数调用栈的详解

1. 引言

为了深入理解c++,决定学习一些简单的汇编语言。使用ubuntu系统下g++很容易将一个c++的文件编译成汇编语言。本文使用此方法,对一个简单的c++文件编译成汇编语言进行理解。

2.示例

文件名:reorder_demo.cpp

#include<stdio.h>typedef unsigned char uint8;uint8 a = 0U;
uint8 b = 0U;int main(int argn, char* argv[])
{a = b + 1;b = 1;return 0;
}

转化成汇编语言的编译命令如下

g++ -S reorder_demo.cpp

转化后生成reorder_demo.s,汇编语言内容如下:

	.file	"reorder_demo.cpp".text.globl	a.bss.type	a, @object.size	a, 1
a:.zero	1.globl	b.type	b, @object.size	b, 1
b:.zero	1.text.globl	main.type	main, @function
main:
.LFB0:.cfi_startprocendbr64pushq	%rbp.cfi_def_cfa_offset 16.cfi_offset 6, -16movq	%rsp, %rbp.cfi_def_cfa_register 6movl	%edi, -4(%rbp)movq	%rsi, -16(%rbp)movzbl	b(%rip), %eaxaddl	$1, %eaxmovb	%al, a(%rip)movb	$1, b(%rip)movl	$0, %eaxpopq	%rbp.cfi_def_cfa 7, 8ret.cfi_endproc
.LFE0:.size	main, .-main.ident	"GCC: (Ubuntu 9.4.0-1ubuntu1~20.04.2) 9.4.0".section	.note.GNU-stack,"",@progbits.section	.note.gnu.property,"a".align 8.long	 1f - 0f.long	 4f - 1f.long	 5
0:.string	 "GNU"
1:.align 8.long	 0xc0000002.long	 3f - 2f
2:.long	 0x3
3:.align 8
4:

以上汇编代码的格式是AT&T 格式的汇编,如果想生成intel格式的汇编可以使用以下命令:

g++ -S -masm=intel  reorder_demo.cpp

3.内存分区

为了更容易理解,从别处找了张内存分区的图供参考,如下

4.汇编分析(包括函数栈帧变化)

.file    "reorder_demo.cpp"

标识汇编文件的源文件为reorder_demo.cpp

.text

标识代码存放段开始。上面汇编中2个.text都是如此,第一个是存放全局变量的代码开始,第二个是存放main函数代码段的开始。

.zero    1

这是一条伪汇编指令,就是把CPU中通用寄存器R0的值设置为0.

伪指令的作用在汇编过程中起作用,一旦汇编结束,就没有实际的作用。

.globl    a                   

标识全局符号a
.bss                             

bss段一般标识未手动初始化的数据,并不给该段的数据分配实际的内存空间,只是记录数据所需空间的大小。
.type    a, @object

定义全局符号a的类型为object,意思就是全局符号a是一个变量。
.size    a, 1

全局变量a的尺寸为1个字节

a:

给全局变量a分配地址

globl b和a类似,不在赘述

.LFB0:  

局部函数开始 Local Function Bigin

.LFE0:

局部函数结束 Local Function END

.cfi_startproc

汇编语言中的一条伪指令,用于声明起始过程,为调试生成调试信息。

endbr64

汇编伪指令,空操作,主要由处理器流水线用作标记指令,以检测控制流违规。

pushq    %rbp

把寄存器%rbp的值压入调用栈中。%rbp标识函数栈帧的基地址。

模型如下(此处约定灰色表示命令执行前状态,红色表示当前块引用内汇编指令执行后的状态)

  补充x86-64 指令架构要求 64Bit 通用寄存器如下:

寄存器全称用途
%raxregister a extended存储过程调用返回值(return value)
%rbxregister b extended/
%rcxregister c extended存储过程调用的第四个参数
%rdxregister d extended存储过程调用的第三个参数
%rbpregister base pointer存储当前栈帧的基地址
%rspregister stack pointer存储栈顶地址
%rsiregister source index存储过程调用的第二个参数
%rdiregister destination index存储过程调用的第一个参数
%r8register 8存储过程调用的第五个参数
%r9register 9存储过程调用的第六个参数
%r10-%r11register 10 ~ register 11/
%r12-%r15register 12 ~ register 15/
%rip永远指向下一条即将执行的地址

可以看出一条规律:r打头的寄存器表示的是64bit的寄存器。 

其中,rbp和rsp模型表示

movq    %rsp, %rbp

直译:把rsp寄存器的值(函数基地址)移动到rbp寄存器(当前栈顶)中。

PS: movq表示移动的是64位寄存器,需要注意的是,和大多数所知的不同,此处移动的方向从第一参数移动到第二参数,以下也是如此。

理解:此条汇编指令代表调用新函数(main函数)前,函数栈帧基地址指向当前栈顶指针。

模型如下:

.cfi_def_cfa_offset 16

cfi_offset 6, -16

.cfi主要用于添加调试信息,功能意义可忽略。

movl    %edi, -4(%rbp)    
movq    %rsi, -16(%rbp)

函数把edi和rsi寄存器中值放入%rbp-4和%rbp-16的栈中,如上通用寄存器表格,rsi存放着函数第二参数edi除了是32bit的通用寄存器,其他方面的作用和rdi类似,存放函数第一参数。其中movl是表示移动的是long类型的数,而movq移动的是64位的数。对应于c++源码中的main函数参数,int argn(可知int占用32bit)和char* argv(因为是64位操作系统编译的,指针占用64bit)。模型如下(其中rsb寄存器的内容隐式地发生变化)

 movzbl    b(%rip), %eax

eax寄存器称为x86架构下32位(解析这段汇编调查中发现一个规律e打头的寄存器是32位寄存器)累加寄存器,主要用于算术运算,逻辑操作,其分布模型如下

AH占高8位,AL占低8位,AX占低16位。

movzbl 是把8bit的数填充0后移动到32位寄存器。

直译:把b的值放入到eax寄存器中,其中rip寄存器里放的是全局变量b的地址。模型如下

addl    $1, %eax

直译:把数字1和eax里面的值相加后放入到eax寄存器中。其中addl表示操作的数据类型是long类型的(32bit)
movb    %al, a(%rip)

直译:把寄存器AL(是EAX寄存器的低8bit)的值放入到全局变量a的地址里面。模型可以参考之前的图形。其中movb表示操作的对象是byte大小的数据(8bit)。

汇总起来,这里3条汇编指令对应的源代码为:

a = b +  1;

movb    $1, b(%rip)

直译:把1移动给全局变量b,其中rip寄存器里存放的是全局变量b的内存地址。
movl    $0, %eax

直译:把0移动给寄存器eax

popq    %rbp

直译:把寄存器rbp指向的地址从栈中弹出去。模型如下:

.cfi_def_cfa 7, 8  -> 伪指令,不讲解
ret

函数退出
.cfi_endproc 
-> 伪指令,不讲解

.size    main, .-main  

此处.size指令提示汇编器在目标文件中记录某种size的信息,此处是记录main函数的尺寸。

.-main 标识main函数的尺寸。理解这句话,就得先理解“.”的含义。

“.”标识内存中当前地址,main标识main开头地址,所以,尾地址“.” - 头地址“main”,表示main函数在内存中占用的空间尺寸。

.ident    "GCC: (Ubuntu 9.4.0-1ubuntu1~20.04.2) 9.4.0"

GCC编译器记录的追踪信息,目标文件结束时常常伴随记录,在链接时此信息会被去除。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/651057.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

常见的工业路由器访问问题

A&#xff1a;工业路由器已经设置了pptp怎么访问路由下面的电脑 1. 确认PPTP VPN设置&#xff1a;首先&#xff0c;确保PPTP VPN服务器在工业路由器上已正确设置&#xff0c;并且处于活动状态。这包括确保VPN服务器的IP地址、端口、用户名和密码等设置正确无误。 2. 连接到VP…

indexDB 大图缓存

背景 最近在项目中遇到了一个问题&#xff1a;由于大屏背景图加载速度过慢&#xff0c;导致页面黑屏时间过长&#xff0c;影响了用户的体验。从下图可以看出加载耗时将近一分钟 IndexDB 主要的想法就是利用indexDB去做缓存&#xff0c;优化加载速度&#xff1b;在这之前&am…

安卓NetworkStatsManager使用及demo

目录 一、TrafficStats类简介二、demo示例 一、TrafficStats类简介 TrafficStats Android API 8提供了android.net.TrafficStats类。 通过此类能获取设备重启以来网络信息&#xff0c;部分函数如下所示&#xff1a; static long getMobileRxBytes() //获取通过移动数据网络…

一张图带你理解 绝对路径 和 相对路径

绝对路径和相对路径是用于定位文件或目录位置的两种不同方式。 1、绝对路径&#xff1a; 绝对路径是从文件系统的根目录开始的完整路径&#xff0c;可以唯一地标识文件或目录的位置。 绝对路径是以根目录开始的 在Unix/Linux系统中&#xff0c;绝对路径是类似于/home/user/do…

释放Stable Diffusion 无限可能

最近在整理大语言模型的系列内容&#xff0c;Stable Diffusion 是我下一篇博客的主题。关注 Stable Diffusion&#xff0c;是因为它是目前最受欢迎和影响力最大的多模态生成模型之一。Stable Diffusion 于 2022 年 8 月发布&#xff0c;主要用于根据文本的描述产生详细图像&…

GraphMAE 简介

GraphMAE: Self-Supervised Masked Graph Autoencoders 前言 当前的自监督学习领域主要分为对比式自监督学习以及生成式自监督学习&#xff0c;这篇文章主要聚焦于生成式对比学习。事实上后者在图领域的进展是远远不如前者的&#xff0c;但是在cv&#xff0c;nlp领域却是以生…

EaseUS RecExperts for Mac/Win:你的专属屏幕录像专家

在信息爆炸的时代&#xff0c;屏幕录像软件已经成为我们工作和生活中的得力助手。无论是教学演示、产品介绍&#xff0c;还是游戏录制、会议记录&#xff0c;一款功能强大的屏幕录像软件都能轻松应对。而EaseUS RecExperts&#xff0c;正是这样一款值得你信赖的屏幕录像专家。 …

JavaEE 初阶篇-深入了解 I/O 高级流(缓冲流、交换流、数据流和序列化流)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 文章目录 1.0 缓冲流概述 1.1 缓冲流的工作原理 1.2 使用缓冲流的步骤 1.3 字节缓冲流于字符缓冲流的区别 1.4 字节缓冲流的实例 1.5 字符缓冲流的实例 2.0 转换流概述 2.1 字符…

去年十八,初识Java 2

我的Python和PHP是怎么学的&#xff1f;是直接写项目&#xff0c;在项目中学的。 不过…这招到 java 里好像不好使了QAQ 零、前置基础 1、类的继承 在 Java 中&#xff0c;extends 用于创建类的继承关系。当一个类继承另一个类时&#xff0c;它会获得父类的属性和方法&…

AS-V1000 视频监控平台,如何实现设备上线、下线时产生告警,及时通知管理人员

目录 一、客户需求 &#xff08;一&#xff09;客户需求 &#xff08;二&#xff09;掌握设备状况的意义 1、实时故障检测与预警 2、提升系统可靠性 3、优化资源配置 4、增强安全保障 5、提升管理效率 二、产品介绍 三、系统配置 &#xff08;一&#xff09;实现告警…

python基础——正则表达式

&#x1f4dd;前言&#xff1a; 这篇文章主要想讲解一下python中的正则表达式&#xff1a; 1&#xff0c;什么是正则表达式 2&#xff0c;re模块三匹配 3&#xff0c;元字符匹配 4&#xff0c;具体示例 &#x1f3ac;个人简介&#xff1a;努力学习ing &#x1f4cb;个人专栏&am…

什么是用户体验(UX)文案,为什么它很重要?

网上购物如今比以往任何时候都更加相关。所以我们将以此为例说明什么是用户体验&#xff08;UX&#xff09;文案&#xff0c;以及为什么它很重要。 假设你去了一个在线商店。你需要执行一系列操作&#xff1a; 找到合适的部分选择你感兴趣的产品弄清楚它们是什么&#xff0c;…