X86汇编速成

news2025/1/12 23:28:44

平时用的电脑都是X86的,但是现在大家都在搞RISC-V,计组也都开始以RISC-V作为示例,所以专门回头来补一下X86的汇编,方便平时使用。

寄存器register

请添加图片描述
X86_64中一共有16个64位的通用寄存器,分别为:

  • RAX, RBX, RCX,RDX, RBP, RSI,RDI, RSP, R8–R15
    • RAX用来存储函数返回值,
    • RSP用来作为堆栈指针寄存器,RSP增大入站,减小出栈。
    • RBP,栈帧指针,标识当前栈帧的起始位置。
    • 其余的随便用
      Callee save表示当出现函数调用的时候,这些通用寄存器内的值由被调用者保存,即在进入被调用函数后由被调用函数存储到它的栈里面,并在返回前还原回去,与之对应的,Caller save则表示由调用者存储,在进入调用函数前就要自己提前push到自己的栈里面

32位的X86中只有8个通用寄存器,没有R8-R15

栈stack

请添加图片描述

指令

指令有两种形式,一种是AT&T的,一种是Intel的,我们用Intel风格的

opcode arg1,agr2

这是一段代码:

int add(int a, int b){
    return a+b;
}
int main(){
    int a = 1;
    int b = 2;
    int c = add(a,b);
    return c-a-b;
}

下面是使用gcc编译得到的汇编代码

.file   "test_asm.c"
        .text
        .globl  add
        .type   add, @function
add:
.LFB0:
        .cfi_startproc
        endbr64
        pushq   %rbp
        .cfi_def_cfa_offset 16
        .cfi_offset 6, -16
        movq    %rsp, %rbp
        .cfi_def_cfa_register 6
        movl    %edi, -4(%rbp)
        movl    %esi, -8(%rbp)
        movl    -4(%rbp), %edx
        movl    -8(%rbp), %eax
        addl    %edx, %eax
        popq    %rbp
        .cfi_def_cfa 7, 8
        ret
        .cfi_endproc
.LFE0:
        .size   add, .-add
        .globl  main
        .type   main, @function
main:
.LFB1:
        .cfi_startproc
        endbr64
        pushq   %rbp
        .cfi_def_cfa_offset 16
        .cfi_offset 6, -16
        movq    %rsp, %rbp
        .cfi_def_cfa_register 6
        subq    $16, %rsp
        movl    $1, -12(%rbp)
        movl    $2, -8(%rbp)
        movl    -8(%rbp), %edx
        movl    -12(%rbp), %eax
        movl    %edx, %esi
        movl    %eax, %edi
        call    add
        movl    %eax, -4(%rbp)
        movl    -4(%rbp), %eax
        subl    -12(%rbp), %eax
        subl    -8(%rbp), %eax
        leave
        .cfi_def_cfa 7, 8
        ret
        .cfi_endproc
.LFE1:
        .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:

下面是在X86_64下使用objdump得到的反汇编指令代码

test_asm.o:     file format elf64-x86-64

Disassembly of section .text:

0000000000000000 <add>:
   0:   f3 0f 1e fa             endbr64               ; 引入指令序列断点,并启用64位模式

   4:   55                      push   %rbp           ; 保存调用者的栈帧指针到栈中,如上文提到的Callee save
   5:   48 89 e5                mov    %rsp,%rbp       ; 设置当前栈帧指针为栈顶指针

   8:   89 7d fc                mov    %edi,-0x4(%rbp) ; 将第一个参数存储到相对于栈帧指针偏移为-4的位置
   b:   89 75 f8                mov    %esi,-0x8(%rbp) ; 将第二个参数存储到相对于栈帧指针偏移为-8的位置

   e:   8b 55 fc                mov    -0x4(%rbp),%edx ; 将第一个参数加载到寄存器edx中
  11:   8b 45 f8                mov    -0x8(%rbp),%eax ; 将第二个参数加载到寄存器eax中

  14:   01 d0                   add    %edx,%eax       ; 执行加法操作,将edx和eax的值相加,结果存储在eax中

  16:   5d                      pop    %rbp            ; 恢复调用者的栈帧指针
  17:   c3                      retq                   ; 返回至调用者

0000000000000018 <main>:
  18:   f3 0f 1e fa             endbr64               ; 引入指令序列断点,并启用64位模式

  1c:   55                      push   %rbp           ; 保存调用者的栈帧指针到栈中
  1d:   48 89 e5                mov    %rsp,%rbp       ; 设置当前栈帧指针为栈顶指针

  20:   48 83 ec 10             sub    $0x10,%rsp     ; 为局部变量分配16字节的栈空间

  24:   c7 45 f4 01 00 00 00    movl   $0x1,-0xc(%rbp) ; 将值1存储到相对于栈帧指针偏移为-12的位置
  2b:   c7 45 f8 02 00 00 00    movl   $0x2,-0x8(%rbp) ; 将值2存储到相对于栈帧指针偏移为-8的位置

  32:   8b 55 f8                mov    -0x8(%rbp),%edx ; 将第二个参数加载到寄存器edx中
  35:   8b 45 f4                mov    -0xc(%rbp),%eax ; 将第一个参数加载到寄存器eax中

  38:   89 d6                   mov    %edx,%esi       ; 将edx的值复制给esi寄存器,用作add函数的第二个参数
  3a:   89 c7                   mov    %eax,%edi       ; 将eax的值复制给edi寄存器,用作add函数的第一个参数

  3c:   e8 00 00 00 00          callq  41 <main+0x29>  ; 调用add函数

  41:   89 45 fc                mov    %eax,-0x4(%rbp) ; 将add函数返回值存储到相对于栈帧指针偏移为-4的位置

  44:   8b 45 fc                mov    -0x4(%rbp),%eax ; 将add函数返回值加载到寄存器eax中
  47:   2b 45 f4                sub    -0xc(%rbp),%eax ; 执行减法操作,将eax的值减去第一个参数的值
  4a:   2b 45 f8                sub    -0x8(%rbp),%eax ; 执行减法操作,将eax的值减去第二个参数的值

  4d:   c9                      leaveq                 ; 恢复栈帧并将栈顶指针设置为栈帧指针
  4e:   c3                      retq                   ; 返回至调用者

也确实是像大家说的,X86的手册太太太长了,X86为了向32位兼容,搞出来的很多机制令人头大

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

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

相关文章

vmware和ubuntu的问题与解决

1.问题与对策 最近使用vmware安装ubuntu16和ubuntu20&#xff0c;遇到了挺多的问题&#xff0c;如下 ubuntu在用过多次后&#xff0c;重启后登录用户名后会出现花屏的现象。 解决方案如下 在键盘上同时按键&#xff1a;Ctrl Alt F4&#xff0c;进入命令行模式&#xff0c;…

Ubuntu 20.04.06 PCL C++学习记录(十六)

[TOC]PCL中点云分割模块的学习 学习背景 参考书籍&#xff1a;《点云库PCL从入门到精通》以及官方代码PCL官方代码链接,&#xff0c;PCL版本为1.10.0&#xff0c;CMake版本为3.16 学习内容 用一组点云数据做简单的平面的分割 源代码及所用函数 源代码 #include<iostr…

C++设计模式:观察者模式(三)

1、定义与动机 观察者模式定义&#xff1a;定义对象间的一种1对多&#xff08;变化&#xff09;的依赖关系&#xff0c;以便当一个对象&#xff08;Subject&#xff09;的状态发生比改变时&#xff0c;所有依赖于它的对象都得到通知并且自动更新 再软件构建过程中&#xff0c…

STM32一个地址未对齐引起的 HardFault 异常

1. 概述 客户在使用 STM32G070 的时候&#xff0c;KEIL MDK 为编译工具&#xff0c;当编译优化选项设置为Level0 的时候&#xff0c;程序会出现 Hard Fault 异常&#xff0c;而当编译优化选项设置为 Level1 的时候&#xff0c;则程序运行正常。表面上看&#xff0c;这似乎是 K…

算法打卡day38|动态规划篇06| 完全背包理论基础|Leetcode 518.零钱兑换II、377. 组合总和 Ⅳ

目录 完全背包理论基础 完全背包问题描述 完全背包解法 算法题 Leetcode 518.零钱兑换II 个人思路 解法 动态规划 Leetcode 377. 组合总和 Ⅳ 个人思路 解法 动态规划 完全背包理论基础 完全背包问题描述 有N件物品和一个最多能背重量为W的背包。第i件物品的重量是…

基于SSE长连接的智能客服问答系统技术方案及完整项目源码

文章目录 一、项目背景二、项目演示三、项目介绍B系统主要功能1. 注册登录重置密码2. 权限管理3. 项目管理4.客服管理 C系统主要功能1. 问答组件2. 主题色定制3. 类微信时间显示控件及智能tip提示4. 无障碍阅读4. 丰富的输入框组件5. 人工客服6. 聊天记录分表记录与查询 四、项…

【拓扑的基】示例及详解

集合X的某拓扑的一个基是X的子集的一个族(其成员称为基元素)&#xff0c;满足条件&#xff1a; 1. 2. 由基生成拓扑 由生成的拓扑(满足以上两个条件&#xff09; 等价描述&#xff1a; 由所有可表示为的某些成员的井的那些集合组成 例1: 证明&#xff1a;由生成的族确实是拓扑…

零代码与低代码开发平台

1、什么是低代码开发平台&#xff1f;什么是零代码开发平台&#xff1f; 零代码开发平台&#xff1a; 指的是不需要写代码就能够快速开发出业务应用/系统的平台。我们在工作中使用的业务应用&#xff0c;主要提供数据收集、数据处理、数据流转和展示等功能。零代码开发平台能够…

2005-2023年各省国内生产总值指数分季度数据

2005-2023年各省国内生产总值指数分季度数据 1、时间&#xff1a;2005-2023年 2、来源&#xff1a;国家统计局、各省统计局 3、指标&#xff1a;地区生产总值指数(上年同期100)_累计值(%) 4、范围&#xff1a;31省 5、时间跨度&#xff1a;季度 6、缺失情况&#xff1a;无…

Qt | Q_PROPERTY属性和QVariant 类

一、属性基础 1、属性与数据成员相似,但是属性可使用 Qt 元对象系统的功能。他们的主要差别在于存取方式不相同,比如属性值通常使用读取函数(即函数名通常以 get 开始的函数)和设置函数(即函数名通常以 set 开始的函数)来存取其值,除此种方法外,Qt 还有其他方式存取属性值…

比亚迪,学历大于一切

比亚迪 今天逛学生聚集地&#xff08;牛客网&#xff09;的时候&#xff0c;看到一篇前几天的帖子。 标题为「比亚迪&#xff0c;学历大于一切」。 这位发帖的同学指出&#xff0c;他曾在比亚迪工作&#xff0c;当时入职时级别为 G3/F1&#xff0c;目测自己工作三年也不一定能升…

【频繁模式挖掘】FP-Tree算法(附Python实现)

一、实验内容简介 该实验主要使用频繁模式和关联规则进行数据挖掘&#xff0c;在已经使用过Apriori算法挖掘频繁模式后&#xff0c;这次使用FP-tree算法来编写和设计程序&#xff0c;依然使用不同规模的数据集来检验效果&#xff0c;最后分析和探讨实验结果&#xff0c;看其是…

HarmonyOS 应用开发-使用colorPicker实现背景跟随主题颜色转换

介绍 本示例介绍使用image库以及effectKit库中的colorPicker对目标图片进行取色&#xff0c;将获取的颜色作为背景渐变色&#xff0c;通过swiper组件对图片进行轮播&#xff0c; 效果图预览 使用说明 直接进入页面&#xff0c;对图片进行左右滑动&#xff0c;或者等待几秒&a…

2014最新AI智能创作系统ChatGPT网站源码+Midjourney绘画网站源码+搭建部署教程文档,支持最近火爆的Suno-v3-AI音乐生成大模型

一、文章前言 SparkAi创作系统是基于ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;那么如何搭建部署AI创作ChatGPT&#xff1f;小编这里写一个详细图文教程吧。已支持…

鼠标经过切换项,切换显示对应的内容(js)

一、功能描述 有多个切换项&#xff0c;鼠标移到每一项时对应切换下面对应的内容&#xff0c;一项对应一项内容&#xff0c;并且切换选中的样式。 二、实现效果 以这个例子为例。 三、实现思路 1.获取所有切换项。 2.获取所有切换项内容。 3.定义好“move”样式&#xff0…

链表之双向链表的实现

铁汁们大家好&#xff0c;我们上一篇博客学习了单链表&#xff0c;这节课让我们继续往深学习&#xff0c;学习一下双线链表&#xff0c;话不多说&#xff0c;我们开始吧&#xff01; 目录 1.双向链表 2.顺序表和链表的优缺点 3.双向链表的实现 1.双向链表 1.我们要实现的双线…

IDEA中修改git的作者、邮箱名称

目录 一、查看当前git信息 1、查看git作者名称 如下图&#xff1a; 2、查看git邮箱信息 二、修改git信息 1、修改git作者名称 如下图&#xff1a; 2、修改git邮箱名称 一、查看当前git信息 1、查看git作者名称 在git控制台 或者 Terminal 输入 git config user.name …

蓝桥杯物联网竞赛_STM32L071_16_EEPROM

仍然是没有考过的知识点 朴素的讲就是板子中一块不会因为断电重启而导致数值初始化的一片地址 要注意的是有时候容易把板子什么写错导致板子什么地址写坏了导致程序无法烧录&#xff0c;这个时候记得一直按flash键烧录&#xff0c;烧录时会报错&#xff0c;点击确定&#xff0…

飞鸟写作可靠吗 #职场发展#经验分享#经验分享

飞鸟写作是一个非常便捷的论文写作工具&#xff0c;不仅可以帮助用户高效地完成论文写作&#xff0c;还可以提供查重降重的功能&#xff0c;帮助用户确保论文的原创性。那么&#xff0c;飞鸟写作到底可靠吗&#xff1f;答案是肯定的。 首先&#xff0c;飞鸟写作提供的查重降重功…

经典算法-分治法由散点得出凸包-python实现

import copy import random import matplotlib import mathdef distance_p2l(point, line_point1, line_point2):if (line_point2[0] - line_point1[0]) 0:return abs(point[0] - line_point2[0])# 计算直线的斜率m (line_point2[1] - line_point1[1]) / (line_point2[0] - l…