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.mzph.cn/bicheng/3707.shtml

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

相关文章

强固型车载电脑在智能轨道安全解决方案的应用

智能轨道安全解决方案 信迈提供一系列具有传感、诊断、人工智能和无线功能的车载列车解决方案。它们提供全面的可扩展性和面向未来的车辆、路旁、信号、电力、障碍物检测和数据收集功能。 应用程序: 铁路供电监控车载列车安全保护铁路轨道监控驾驶行为分析 智能车载解决方案…

Django连接数据库

数据库登录命令 mysql -u root -p show databases; Django连接数据库 在settings.py文件中进行配置和修改 DATABASES {default: {ENGINE: django.db.backends.mysql,HOST: 127.0.0.1, # 数据库主机PORT: 3306, # 数据库端口USER: root, # 数据库用户名PASSWORD: 12345…

flutter release 报错 Error: SocketException: Failed host lookup:

flutter 的 debug 模式没有任何问题 &#xff0c;打了release 包后一直报下面的错&#xff0c;查了一下是 因为没有网络权限 Error: SocketException: Failed host lookup: yomi-test-aws-sg.yomigame.games (OS Error: No address associated with hostname, errno 7) 按照下…

win10加入域环境

win10加入域环境 导航 文章目录 win10加入域环境导航一、关闭防火墙二、使客户端的电脑指向于域控服务器三、检验是否加入了域 一、关闭防火墙 在进行加入域服务之前,我们需要先关闭防火墙(为了不必要的麻烦) 按 winr调出运行窗口,输入 control打开控制面板 点击系统和安全点…

python基础之元组、集合和函数的定义与返回值

1.元祖 1.元祖的定义 元组的数据结构跟列表相似 特征&#xff1a;有序、 有序&#xff1a;有&#xff08;索引/下标/index&#xff09; 正序、反序标识符&#xff1a; ( ) 里面的元素是用英文格式的逗号分割开来关键字&#xff1a;tuple 列表和元组有什么区别&#xff1f; 元组…

异常风云:解码 Java 异常机制

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互相学习&#xff0c;一个人虽可以走的更快&#xff0c;但一群人可以走的更远。 我是一名后…

C语言数据类型的介绍,类型的基本归类,整型在内存中的存储,原码、反码、补码,大小端等介绍

文章目录 前言一、数据类型的介绍类型的意义 1. 类型的基本归类&#xff08;1&#xff09;. 整型家族&#xff08;2&#xff09;. 浮点数家族&#xff08;3&#xff09;. 构造类型&#xff08;4&#xff09;. 指针类型&#xff08;5&#xff09;. 空类型 二、整型在内存中的存储…

[Collection与数据结构] PriorityQueue与堆

1. 优先级队列 1.1 概念 前面介绍过队列&#xff0c;队列是一种先进先出(FIFO)的数据结构&#xff0c;但有些情况下&#xff0c;操作的数据可能带有优先级&#xff0c;一般出队列时&#xff0c;可能需要优先级高的元素先出队列&#xff0c;该中场景下&#xff0c;使用队列显然…

自动化机器学习流水线:基于Spring Boot与AI机器学习技术的融合探索

&#x1f9d1; 作者简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟&#xff0c;欢迎关注。提供嵌入式方向…

Python常用包介绍

数据处理 1.numpy&#xff08;数据处理和科学计算&#xff09; import numpy as np np.set_printoptions(precision2, suppressTrue) # 设置打印选项&#xff0c;保留两位小数&#xff0c;禁止科学计数法arr np.arange(1, 6) # 使用arange函数创建数组 print(arr)# 输出&…

深度学习下的视觉SLAM综述

作者&#xff1a;黄泽霞&#xff0c;邵春莉 来源&#xff1a;《机器人》 编辑&#xff1a;东岸因为一点人工一点智能 深度学习下的视觉SLAM综述到目前为止&#xff0c;深度学习与SLAM的结合已经在视觉里程计、场景识别与全局优化等各种任务中取得了显著的成果。同时&#xf…

Prompt Engineering,提示工程

什么是提示工程&#xff1f; 提示工程也叫【指令工程】。 Prompt发送给大模型的指令。比如[讲个笑话]、[用Python编个贪吃蛇游戏]、[给男/女朋友写情书]等看起来简单&#xff0c;但上手简单精通难 [Propmpt]是AGI时代的[编程语言][Propmpt]是AGI时代的[软件工程][提示工程]是…

线上申报开放时间!2024年阜阳市大数据企业培育认定申报条件、流程和材料

2024年阜阳市大数据企业培育认定申报条件、流程和材料&#xff0c;线上申报开放时间整理如下 一、2024年阜阳市大数据企业培育认定申报要求 &#xff08;一&#xff09;经营范围 申请认定的企业应当从事以下生产经营活动&#xff1a; 1.从事数据收集、存储、使用、加工、传输、…

使用linux,c++,创作一个简单的五子棋游戏

#include <iostream> #include <vector> #include <unordered_map> using namespace std; // 棋盘大小 const int BOARD_SIZE 15; // 棋子类型 enum ChessType { EMPTY, BLACK, WHITE }; // 棋盘类 class ChessBoard { private: vect…

CHiME-8多通道远场语音识别Baseline介绍

语音领域每年都有很多比赛&#xff0c;每个比赛都有自己的侧重点&#xff0c;其中CHiME系列比赛的侧重点就是多通道远场语音识别&#xff0c;与其他的语音识别比赛有所区别的是&#xff0c;CHiME提供分布式麦克风和麦克风阵列数据&#xff0c;这样可以选择合适的前端算法以降低…

创建Spring Boot项目

选择Maven Archetype,之后再Archetype选择webapp 两个都打勾 这是当前的打勾 这个是以后都默认勾上 打开对应的路径&#xff0c;用vscode打开settings.xml 加入国内源 阿里云 若没有此文件可上网查找 若jar包出现问题&#xff0c;可在repostitory文件内全删除 之后在Maven刷…

第12章 最佳的UI体验——Material Design实战

第12章 最佳的UI体验——Material Design实战 其实长久以来&#xff0c;大多数人都认为Android系统的UI并不算美观&#xff0c;至少没有iOS系统的美观。以至于很多IT公司在进行应用界面设计的时候&#xff0c;为了保证双平台的统一性&#xff0c;强制要求Android端的界面风格必…

HarmonyOS开发案例:【 自定义弹窗】

介绍 基于ArkTS的声明式开发范式实现了三种不同的弹窗&#xff0c;第一种直接使用公共组件&#xff0c;后两种使用CustomDialogController实现自定义弹窗&#xff0c;效果如图所示&#xff1a; 相关概念 [AlertDialog]&#xff1a;警告弹窗&#xff0c;可设置文本内容和响应回…

了解HTTP代理服务器:优势、分类及应用实践

在我们日常的网络使用中&#xff0c;我们经常听到HTTP代理服务器这个术语。那么&#xff0c;HTTP代理服务器到底是什么&#xff1f;它有什么优势和分类&#xff1f;又如何应用于实践中呢&#xff1f;让我们一起来了解一下。 HTTP代理服务器是一种位于客户端和服务器之间的中间…

图像处理基础知识

图像处理基础知识 图像 1、模拟图像 模拟图像&#xff0c;又称连续图像&#xff0c;是指在二维坐标系中连续变化的图像&#xff0c;即图像的像点是无限稠密的&#xff0c;同时具有灰度值&#xff08;即图像从暗到亮的变化值&#xff09;。 2、数字图像 数字图像&#xff0…