ARMv8 Cortex-a 编程向导手册学习_4. A64指令集

2024-03-12 12:59

本文主要是介绍ARMv8 Cortex-a 编程向导手册学习_4. A64指令集,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

/* TODO 本系列文章是对 ARMv8 Cortex-a 系列编程向导手册拙劣的翻译和注解,若有出入,以官方文档为准 */

Chapter 6 A64指令集

大多数程序员并不需要使用汇编语言编写应用程序,但是汇编代码可以有效的优化代码性能。而且当编写编译器,或者使用 CPU 底层功能,或者编写启动代码、设备驱动以及操作系统中断相关的任务切换时,此时不能直接使用 C 语言,而需要使用汇编;当调试代码时,我们需要有效的理解汇编指令与 C 程序流的映射关系。

6.1 指令助记符

A64 的指令可以操作32位以及64位值,这两种情况下采用不同的编码方式,编译器根据寄存器名字,来区分当前操作的是 32 位值还是 64 位值。示例如下:

ADD W0, W1, W2 // 32 位寄存器的加法
ADD X0, X1, X2 // 64位寄存器的加法
ADD X0, X1, W2, SXTW // add sign extended 32-bit register to 64-bit extended register
ADD X0, X1, #42 // 64 位寄存器与立即数的加法
ADD V0.8H, V1.8H, V2.8H // NEON 16-bit add, in each of 8 lanes

6.2 数据处理指令

A64 提供一些重要的算数与逻辑操作指令,应用于寄存器与寄存器之间,或者寄存器与立即数之间,乘法与除法指令被当作特殊的数据处理指令。
绝大多数数据处理指令使用一个目的寄存器和两个源操作数,这些指令的通用格式如下:
Instruction Rd, Rn, Operand2
数据处理指令包括:

  • 算数与逻辑操作指令
  • 赋值与移位指令
  • 符号与0扩展指令
  • 位域操作指令
  • 条件比较与数据处理指令

6.3.1 算数与逻辑操作

在这里插入图片描述
上图中列出了:

  • 算数指令
  • 逻辑指令
  • 比较指令
  • 赋值指令

一些指令拥有 s 后缀,表示该指令会根据指令执行的结果设置 PSTATE 的 NZCV 标志位。
指令使用示例如下:

ADD W0, W1, W2, LSL #3 // W0 = W1 + (W2 << 3)
SUBS X0, X4, X3, ASR #2 // X0 = X4 - (X3 >> 2), set flags
MOV X0, X1 // Copy X1 to X0
CMP W3, W4 // Set flags based on W3 - W4
ADD W0, W5, #27 // W0 = W5 + 27
MOV X1, #0x800 // x1 = 0x8000
BIC X0, X0, X1 // x0 &= ~x1

比较指令只设置标志位,不会保存结果 ,在上述算符与逻辑指令中,能处理的立即数的范围是 12 位,这个12位可以是 64 位值中任意的连续的 12 位。

6.2.2 乘法与除法指令

MUL X0, X1, X2 // X0 = X1 * X2
UDIV W0, W1, W2 // W0 = W1 / W2 (unsigned, 32-bit divide)
SDIV X0, X1, X2 // X0 = X1 / X2 (signed, 64-bit divide)

6.2.3 移位操作指令

移位操作指令存在如下类型:

  • 逻辑左移:LSL,该指令等于乘以2的幂次
  • 逻辑右移:RSL, 该指令等于除以2的幂次
  • 算法右移:ASR,该指令等于除以2的幂次,但会保留符号位
  • 循环右移:ROR, 该指令的 LSB 位会填充到 MSB 位

4种移位操作的示例图如下:
在这里插入图片描述
移位操作可以使用32位或64位的寄存器,移动的位数量可以是通过一个立即数指定,或者寄存器指定,如果通过寄存器指定,那么针对32位值的移位操作,寄存器的低5位指定位移动数量,如果针对64位值的移位操作,寄存器的低6位指定位移动数量。

6.2.4 位域与字节操作指令

与 ARMv7 类似, ARMv8 也提供了位域插入指令 BFI(Bit Fieled Insert) 以及位域提取指令S/UBFX(Bit Fieled Extract)。,用法与示例如下图:
在这里插入图片描述

6.2.5 条件指令

AArch64 的程序状态位域 PSTATE 提供了4个条件标志位:NZCV(与 ARMv7 类似),标志位描述如下表:

标志位描述
NNegative, 负值标志位 ,表示指令结果为一个负值
ZZero, 0 标志位,如果指令结果是0,则置位该标志位,表示两个操作数拥有相同的值
CCarry, 进位标志位
VOverflow, 溢出标志位,上溢或下溢

存在如下条件码:
在这里插入图片描述
上图这些条件码,在 PSTATE 的对应 Condition Flag 为对应的值时为真。
比如,如下:

b.eq	__main	/* 如果此时 Z 置位,那么跳转到 __main() */
b.ne 	__secondary_init /* 如果此时 Z 没有置位,那么跳转到 __secondary_init()*/

着重注意的指令:
条件比较指令:cmp 和 cmn,如果比较的结果为真,cpu 会设置条件标志位。
条件选择指令:csel,会根据指令跟随的条件码,选择相对应赋值操作。示例如下:

CMP w0, #0 // if (i == 0)
SUB w2, w1, #1 // r = r - 1
ADD w1, w1, #2 // r = r + 2
CSEL w1, w1, w2, EQ // select between the two results

上述汇编对应如下的 C 语句

if (i == 0) r = r + 2; else r = r - 1;

6.3 内存访问指令

跟 ARMv7 类似,ARMv8 架构也是一个 Load/Store 架构,即,不提供直接访问内存地址的数据处理指令(6.2章节提供的指令都时不能直接访问内存地址)。
数据首先被加载到通用目的寄存器中,然后才能使用数据处理指令进行修改,最后再存储到指定的地址中。
内存地址的访问流程即是:加载-数据处理-存储。

6.3.1 内存加载指令格式

通用的内存加载指令格式如下:

LDR Rt, <addr>	//将 addr 地址的值加载到 Rt 寄存器中

我们可以在 LDR 指令后加后缀的形式,选择加载的数据宽度:

LSRB加载字节数据,目的寄存器的高位填充0
LSRSB加载字节数据,目的寄存器的剩余高位填充符号位
LSRH加载半字数据,目的寄存器的剩余高位填充0
LSRSH加载半字数据,目的寄存器的剩余高位填充符号位
LSRWS加载字数据,目的寄存器的剩余高位填充符号位

上述内存加载指令的用法示例图如下:

在这里插入图片描述

6.3.2 内存存储指令

Store 指令常用格式如下:

STR Rn, <addr>	//将 Rn 的值保存到 addr 地址处

如果要存储的数据小于寄存器的宽度,那么我们可以使用‘B’或‘H’后缀,此时会将寄存器的低位存储到指定的地址。

6.3.3 浮点与 NEON 精度加载与存储

内存加载与存储指令也可以访问浮点寄存器。
在这里插入图片描述

6.3.4 加载指定地址处的值

offset mode:偏移模式
偏移寻址模式将一个立即数值或一个可修改的寄存器值添加到目的寄存器中用于生成地址:
在这里插入图片描述
index mode :索引模式
索引模式与偏移寻址模式相似,但索引模式会更新基寄存器的值:
在这里插入图片描述
PC-relative mode:PC相关模式
A64 增加了其他的寻址模式,比如可以从一个标号地址加载数据,如下:
在这里插入图片描述

6.3.5 访问多个内存位置

A64 不支持 LDM 与 STM 指令,只支持 LDP 与 STP 指令。
LDP 与 STP 指令一次可以访问两个通用目的寄存器来读或写。
指令使用示例如下:
在这里插入图片描述
图解如下:
在这里插入图片描述

6.3.6 非特权访问

6.3.7 内存提取指令

6.3.8 非临时加载与存储对

6.3.9 内存访问原子性

当使用 LDR 与 STR 指令,通过通用目的寄存器访问地址对齐的内存时,硬件保证这个访问是原子性的。
当使用 LDP 与 STP 指令,通过通用目的寄存器对访问地址对齐的内存时,硬件保证这两个寄存器的访问是两个单独的原子操作。
浮点内存访问时,硬件不保证原子性。

6.3.10 内存屏障与栅栏指令

  • DMB, 数据内存屏障
  • DSB, 数据同步屏障
  • ISB, 指令同步屏障

6.3.11 同步?

6.4 程序流控指令

A64 提供大量的不同种类的程序跳转指令。
在这里插入图片描述
当使用 BL 或 BLR 调用函数时,会使用 X30 保存函数返回地址。

6.5 系统控制与其他指令

6.5.1 异常处理指令

存在3种异常生成指令:

异常生成指令描述
SVC #imm16生成 EL1 异常
HVC #imm16生成 EL2 异常
SMC #imm16生成 EL3 异常

上表中,立即数的值保存在 ESR_ELn 寄存器中,可供异常服务函数处理。
当从异常退出时,我们使用ERET 指令,这个指令会使用 SPSR_ELn 寄存器的值设置 PSTATE,并会跳转到 ELR_ELn 保存的地址处开始执行代码。

6.5.3 系统寄存器访问指令

系统寄存器的访问使用 MSRMRS 指令

这篇关于ARMv8 Cortex-a 编程向导手册学习_4. A64指令集的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/801338

相关文章

揭秘Python Socket网络编程的7种硬核用法

《揭秘PythonSocket网络编程的7种硬核用法》Socket不仅能做聊天室,还能干一大堆硬核操作,这篇文章就带大家看看Python网络编程的7种超实用玩法,感兴趣的小伙伴可以跟随小编一起... 目录1.端口扫描器:探测开放端口2.简易 HTTP 服务器:10 秒搭个网页3.局域网游戏:多人联机对战4.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Python异步编程中asyncio.gather的并发控制详解

《Python异步编程中asyncio.gather的并发控制详解》在Python异步编程生态中,asyncio.gather是并发任务调度的核心工具,本文将通过实际场景和代码示例,展示如何结合信号量... 目录一、asyncio.gather的原始行为解析二、信号量控制法:给并发装上"节流阀"三、进阶控制

Java进阶学习之如何开启远程调式

《Java进阶学习之如何开启远程调式》Java开发中的远程调试是一项至关重要的技能,特别是在处理生产环境的问题或者协作开发时,:本文主要介绍Java进阶学习之如何开启远程调式的相关资料,需要的朋友... 目录概述Java远程调试的开启与底层原理开启Java远程调试底层原理JVM参数总结&nbsMbKKXJx

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

PyCharm接入DeepSeek实现AI编程的操作流程

《PyCharm接入DeepSeek实现AI编程的操作流程》DeepSeek是一家专注于人工智能技术研发的公司,致力于开发高性能、低成本的AI模型,接下来,我们把DeepSeek接入到PyCharm中... 目录引言效果演示创建API key在PyCharm中下载Continue插件配置Continue引言

C#反射编程之GetConstructor()方法解读

《C#反射编程之GetConstructor()方法解读》C#中Type类的GetConstructor()方法用于获取指定类型的构造函数,该方法有多个重载版本,可以根据不同的参数获取不同特性的构造函... 目录C# GetConstructor()方法有4个重载以GetConstructor(Type[]

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用