安卓逆向之ARM汇编寻址,汇编指令
一:ARM汇编寻址
1. 立即数寻址 (Immediate Addressing)
指令中直接给出一个常数值(立即数),并对其进行操作。
MOV R0, #5 ; 将立即数5载入寄存器R0
2. 直接寻址 (Direct Addressing)
指令中给出的地址直接指定了内存中的一个位置,通常在数据段中。
LDR R0, =0x1000 ; 将地址0x1000的值加载到R0寄存器
3. 间接寻址 (Indirect Addressing)
使用寄存器中的地址来访问内存中的数据。
LDR R0, [R1] ; 将R1寄存器中存储的地址所指向的值载入R0
4. 基址寻址 (Base Addressing)
通过一个基址寄存器,加上一个偏移量来访问内存。
LDR R0, [R1, #4] ; 从R1寄存器所指向的地址加上4的偏移量处加载数据到R0
5. 偏移寻址 (Offset Addressing)
在基址的基础上,通过增加一个偏移值来访问数据。
STR R0, [R1, R2] ; 将R0的值存储到地址为R1 + R2的内存位置
6. 自增和自减寻址 (Pre-Indexed & Post-Indexed Addressing)
通过预增或后增来访问数据。
- 预索引 (Pre-Indexed): 在访问内存前,先修改基址寄存器的值。
LDR R0, [R1, #4]! ; 加载R1地址加4处的值到R0,并将R1更新为R1 + 4
- 后索引 (Post-Indexed): 在访问后修改基址寄存器的值。
LDR R0, [R1], #4 ; 首先加载R1地址处的值到R0,然后将R1增加4
7. 结构体寻址 (Struct Addressing)
通常在处理结构体时,利用基址和字段偏移量进行访问。
LDR R0, [R1, #OFFSET] ; 读取R1寄存器指向的位置加上字段偏移量存入R0
8. 无条件跳转和条件跳转
无条件跳转 B 和带条件的跳转指令 BEQ、BNE 等,这些也是寻址模式的体现,因为它们影响程序计数器 (PC) 的值。
B label ; 跳转到 label
BEQ label ; 如果 ZF (零标志位) 为 1,跳转到 label
示例汇编程序
以下是一个简单的 ARM 汇编程序示例,展示了不同寻址方式的使用:
.section .data
value: .word 10.section .text
.global _start_start:LDR R0, =value ; 使用直接寻址加载 value 地址LDR R1, [R0] ; 使用间接寻址加载 value 的值 (10)ADD R1, R1, #1 ; R1 = R1 + 1,结果 R1 = 11STR R1, [R0] ; 使用间接寻址将 R1 的值存回 memory locationMOV R7, #1 ; 系统调用号 (exit)SWI 0 ; 触发系统调用,退出程序
在 ARM 汇编语言中,寄存器移位寻址、加载/存储寻址方式以及栈寻址是非常重要的寻址模式。
1. 寄存器移位寻址 (Register Shift Addressing)
寄存器移位寻址允许在操作数中对寄存器的值进行位移操作。这种模式通常用于算术和逻辑操作中,可以通过左移、右移等方式改变寄存器中的值。
示例
MOV R0, #1 ; 将立即数1加载到R0
LSL R1, R0, #2 ; 将R0左移2位,结果存储在R1中 (R1 = 4)
LSR R2, R1, #1 ; 将R1右移1位,结果存储在R2中 (R2 = 2)
- LSL (Logical Shift Left): 左移操作,低位补零。
- LSR (Logical Shift Right): 右移操作,最高位补零。
2. 加载/存储寻址方式 (Load/Store Addressing)
在 ARM 中,加载(LDR)和存储(STR)指令用于将数据从内存加载到寄存器,或将寄存器中的数据存储到内存。常见的加载/存储寻址方式包括:
- 直接寻址: 直接指定地址。
- 基址寻址: 使用基址寄存器和偏移量。
- 间接寻址: 使用寄存器中的地址。
示例
.data
value: .word 10.text
.global _start_start:LDR R0, =value ; 直接寻址,加载 value 的地址LDR R1, [R0] ; 从 R0 指向的地址加载值 (10) 到 R1ADD R1, R1, #1 ; R1 = R1 + 1 (R1 = 11)STR R1, [R0] ; 将 R1 的值存储回 value 的地址
3. 栈寻址 (Stack Addressing)
栈是用于存储局部变量、函数参数和返回地址的特殊内存区域。ARM 使用向下增长的栈,通常通过 PUSH 和 POP 指令进行操作。栈寻址通常涉及到 SP(栈指针寄存器)。
示例
.text
.global _start_start:MOV R0, #5 ; 将5放入R0PUSH {R0} ; 将R0的值压入栈中POP {R1} ; 从栈中弹出值到R1 (R1 = 5)MOV R7, #1 ; 系统调用号 (exit)SWI 0 ; 触发系统调用,退出程序
4. 栈的使用
栈通常用于函数调用和返回。以下是一个简单的示例,演示如何在函数中使用栈:
.text
.global mainmain:PUSH {LR} ; 保存返回地址MOV R0, #10 ; 将10放入R0PUSH {R0} ; 将R0的值压入栈中BL subroutine ; 调用子程序POP {R0} ; 从栈弹出值到R0POP {PC} ; 从栈弹出返回地址,返回到调用者; 程序结束subroutine:POP {R1} ; 从栈弹出值到R1ADD R1, R1, #1 ; R1 = R1 + 1PUSH {R1} ; 将结果压入栈中BX LR ; 返回到调用者
总结
- 寄存器移位寻址 允许对寄存器的值进行位移操作,适用于算术和逻辑运算。
- 加载/存储寻址方式 是 ARM 汇编中最常用的方式,用于在寄存器和内存之间传输数据。
- 栈寻址 是通过栈实现局部变量和函数调用的管理,使用
PUSH和POP指令进行操作。
二:汇编指令详细介绍
在 ARM 汇编语言中,指令集非常丰富,涵盖了数据处理、控制流、内存访问等多个方面。以下是一些常见的 ARM 汇编指令的详细介绍,包括它们的功能、语法和示例。
1. 数据处理指令
1.1 MOV (Move)
将一个值移动到寄存器中。
语法:
MOV <destination>, <source>
示例:
MOV R0, #5 ; 将立即数5加载到R0寄存器
1.2 ADD (Addition)
将两个寄存器的值相加,并将结果存储在目标寄存器中。
语法:
ADD <destination>, <operand1>, <operand2>
示例:
ADD R1, R0, R2 ; R1 = R0 + R2
1.3 SUB (Subtraction)
从一个寄存器中减去另一个寄存器的值。
语法:
SUB <destination>, <operand1>, <operand2>
示例:
SUB R3, R1, R0 ; R3 = R1 - R0
1.4 MUL (Multiply)
将两个寄存器的值相乘。
语法:
MUL <destination>, <operand1>, <operand2>
示例:
MUL R4, R0, R1 ; R4 = R0 * R1
2. 逻辑指令
2.1 AND (Bitwise AND)
对两个寄存器的值进行按位与操作。
语法:
AND <destination>, <operand1>, <operand2>
示例:
AND R5, R0, R1 ; R5 = R0 & R1
2.2 ORR (Bitwise OR)
对两个寄存器的值进行按位或操作。
语法:
ORR <destination>, <operand1>, <operand2>
示例:
ORR R6, R0, R1 ; R6 = R0 | R1
2.3 EOR (Bitwise Exclusive OR)
对两个寄存器的值进行按位异或操作。
语法:
EOR <destination>, <operand1>, <operand2>
示例:
EOR R7, R0, R1 ; R7 = R0 ^ R1
3. 移位指令
3.1 LSL (Logical Shift Left)
将寄存器的值左移指定的位数。
语法:
LSL <destination>, <source>, #<shift>
示例:
LSL R8, R0, #2 ; R8 = R0 << 2
3.2 LSR (Logical Shift Right)
将寄存器的值右移指定的位数。
语法:
LSR <destination>, <source>, #<shift>
示例:
LSR R9, R1, #1 ; R9 = R1 >> 1
4. 加载和存储指令
4.1 LDR (Load Register)
从内存加载数据到寄存器。
语法:
LDR <destination>, [<address>]
示例:
LDR R0, [R1] ; 从R1指向的地址加载数据到R0
4.2 STR (Store Register)
将寄存器中的数据存储到内存。
语法:
STR <source>, [<address>]
示例:
STR R0, [R1] ; 将R0的值存储到R1指向的地址
5. 控制流指令
5.1 B (Branch)
无条件跳转到指定标签。
语法:
B <label>
示例:
B loop ; 跳转到 loop 标签
5.2 BL (Branch with Link)
跳转到指定标签并保存返回地址。
语法:
BL <label>
示例:
BL function ; 跳转到 function 标签并保存返回地址
5.3 BX (Branch and Exchange)
跳转到指定寄存器的地址,并根据地址的最低位决定是 ARM 还是 Thumb 模式。
语法:
BX <register>
示例:
BX R0 ; 跳转到R0寄存器指向的地址
6. 栈操作指令
6.1 PUSH
将寄存器的值压入栈中。
语法:
PUSH {<registers>}
示例:
PUSH {R0, R1} ; 将R0和R1的值压入栈中
6.2 POP
从栈中弹出值到寄存器。
语法:
POP {<registers>}
示例:
POP {R0, R1} ; 从栈中弹出值到R0和R1
7. 条件执行指令
ARM 汇编指令可以通过条件后缀来实现条件执行。例如,BEQ 表示“如果相等则跳转”。
7.1 BEQ (Branch if Equal)
如果零标志位(ZF)为1,则跳转。
语法:
BEQ <label>
示例:
BEQ equal_label ; 如果相等则跳转到 equal_label
7.2 BNE (Branch if Not Equal)
如果零标志位(ZF)为0,则跳转。
语法:
BNE <label>
示例:
BNE not_equal_label ; 如果不相等则跳转到 not_equal_label
三:补充:数据处理指令
在 ARM 汇编语言中,数据处理指令是用于对寄存器中的数据进行算术运算、逻辑运算、位操作等的指令。除了之前提到的基本指令,ARM 还提供了许多其他强大的数据处理指令。
1. 数据处理指令详细介绍
1.1 ADC (Add with Carry)
将两个寄存器的值相加,并加上进位标志(C)。
语法:
ADC <destination>, <operand1>, <operand2>
示例:
ADC R0, R1, R2 ; R0 = R1 + R2 + C
1.2 SBC (Subtract with Carry)
从一个寄存器中减去另一个寄存器的值,并减去进位标志(C)。
语法:
SBC <destination>, <operand1>, <operand2>
示例:
SBC R0, R1, R2 ; R0 = R1 - R2 - C
1.3 ROR (Rotate Right)
将寄存器的值右旋转指定的位数。
语法:
ROR <destination>, <source>, #<shift>
示例:
ROR R0, R1, #2 ; R0 = R1 右旋转2位
1.4 ROL (Rotate Left)
将寄存器的值左旋转指定的位数(在 ARM 中通常使用 LSL 来实现)。
语法:
ROL <destination>, <source>, #<shift>
示例:
MOV R0, R1 ; 将R1复制到R0
LSL R0, R0, #1 ; 左移1位
ORR R0, R0, R1, LSR #31 ; 将R1的最高位移到最低位
1.5 CMP (Compare)
比较两个寄存器的值,设置状态标志,但不存储结果。
语法:
CMP <operand1>, <operand2>
示例:
CMP R0, R1 ; 比较R0和R1的值
1.6 CMN (Compare Negative)
比较两个寄存器的值,并设置状态标志,类似于 CMP,但相当于 R0 + R1。
语法:
CMN <operand1>, <operand2>
示例:
CMN R0, R1 ; 计算R0 + R1并设置状态标志
1.7 TST (Test)
对两个寄存器进行按位与操作,设置状态标志,但不存储结果。
语法:
TST <operand1>, <operand2>
示例:
TST R0, R1 ; R0 和 R1 进行按位与操作
1.8 TEQ (Test Equivalence)
对两个寄存器进行按位异或操作,设置状态标志,但不存储结果。
语法:
TEQ <operand1>, <operand2>
示例:
TEQ R0, R1 ; R0 和 R1 进行按位异或操作
1.9 BIC (Bit Clear)
将一个寄存器的值与另一个寄存器进行按位与,清除指定的位。
语法:
BIC <destination>, <operand1>, <operand2>
示例:
BIC R0, R1, #0xFF ; 清除R1中最低的8位,结果存入R0
1.10 MVN (Move Not)
将一个寄存器的值取反,并存储到目标寄存器中。
语法:
MVN <destination>, <source>
示例:
MVN R0, R1 ; R0 = ~R1
2. 其他重要指令
2.1 SEL (Select)
根据条件选择一个寄存器的值。虽然在 ARM 指令集中并不常见,但在某些 ARM 扩展中存在。
语法:
SEL <destination>, <condition>, <operand1>, <operand2>
示例:
SEL R0, R1, R2, R3 ; 根据条件选择 R1 或 R2 存入 R0
2.2 CLZ (Count Leading Zeros)
计算一个寄存器中前导零的数量。
语法:
CLZ <destination>, <source>
CLZ R0, R1 ; R0 = R1 中前导零的数量
相关文章:
安卓逆向之ARM汇编寻址,汇编指令
一:ARM汇编寻址 1. 立即数寻址 (Immediate Addressing) 指令中直接给出一个常数值(立即数),并对其进行操作。 MOV R0, #5 ; 将立即数5载入寄存器R02. 直接寻址 (Direct Addressing) 指令中给出的地址直接指定了内存中的一…...
Idea常见插件(超级实用)
文章目录 Idea好用的插件推荐Idea插件安装Chinese(中文版)Alibaba Java Coding Guidelines(代码规范)Auto Filling Java Arguments(自动补全参数)CamelCase(变量名称格式转换)CodeGeeX(智能&…...
C++中如何获取时间并格式化为字符串?
在C中,你可以使用标准库中的 <chrono> 和 <iomanip> 头文件来获取当前时间并将其格式化为字符串。以下是一个简单的示例,展示了如何获取当前时间并将其格式化为一个可读的字符串(例如:YYYY-MM-DD HH:MM:SS)…...
项目1 yolov5鱼苗检测计数
yolov5鱼苗检测 1. yolov5鱼苗检测1.1. 环境配置1.2 Predict1.3 Validate1.4 Train1.5 生成 ONNX 2 代码解析2.1 模型2.2 数据集2.3 损失函数2.4 训练2.5 预测 之前做的项目,再回顾一下 环境:GPU1卡,CPU4核,每显卡12GB,…...
GPU 学习笔记三:GPU多机多卡组网和拓扑结构分析(基于数据中心分析)
文章目录 一、概述二、数据中心(DC)2.1 数据中心简介2.2 传统数据中心的网络模型2.3 脊叶网络模型(Spine-Leaf)2.4 Facebook的Fabric网络架构 三、基于数据中心的多机多卡拓扑3.1 Spine-Leaf 架构网络规模测算方法3.2 NVIDIA多机多…...
各编程语言处理HTTP状态码的库推荐
Http 状态码用那个库 备注 Spring 的状态码库为 org.springframework.http.HttpStatus Apache 的状态码库为: org.apache.http.HttpStatus 通常这 2 个库都差不多。 如你的项目中已经用了 Spring 的代码的话,那么就用 Spring 的库吧。 不管是那个库…...
【Mac】Python 环境管理工具
一、pyenv 1、安装 (1)安装 brew install pyenv(2)环境配置 查看系统使用 shell 是 bash 还是 zsh bash 配置文件:~/.bash_profile zsh 配置文件:~/.zshrc userMac ~ % echo $SHELL /bin/zsh userMa…...
大语言模型数据流程源码解读(基于llama3模型)
文章目录 前言一、数据进入LlamaForCausalLM(LlamaPreTrainedModel)类二、数据进入LlamaModel(LlamaPreTrainedModel)类1、input_ids的embedding编码2、position_ids位置获取3、causal_mask因果mask构建1、causal_mask调用2、因果mask代码解读(_update_causal_mask)4、hidden_s…...
[蓝桥杯 2015 省 A] 饮料换购
题目描述 乐羊羊饮料厂正在举办一次促销优惠活动。乐羊羊 C 型饮料,凭 3 个瓶盖可以再换一瓶 C 型饮料,并且可以一直循环下去(但不允许暂借或赊账)。 请你计算一下,如果小明不浪费瓶盖,尽量地参加活动,那么ÿ…...
K8S测试pod内存和CPU资源不足
只设置requests参数 mysql主从pod启动后监控 读压测之后 同时设置limits和requests,只调低内存值 监控 压力测试 同时设置limits和requests,只调低CPU值 初始状态 开始压测 结论 对于CPU,如果pod中服务使用CPU超过设置的limits&…...
rabbitmq 使用注意事项
1,注意开启的端口号,一共四个端口号,1883是mqtt连接的端口号,如果没开,是连接不上的需要手动起mqtt插件。 //开始mqtt插件服务 rabbitmq-plugins enable rabbitmq_mqtt 2,15672端口是http网页登录的管理后…...
<项目代码>YOLOv8 夜间车辆识别<目标检测>
YOLOv8是一种单阶段(one-stage)检测算法,它将目标检测问题转化为一个回归问题,能够在一次前向传播过程中同时完成目标的分类和定位任务。相较于两阶段检测算法(如Faster R-CNN),YOLOv8具有更高的…...
xterm.js 库作用
前言:xterm.js 是一个用于在网页上模拟终端的强大 JavaScript 库。 一、在网页中实现终端模拟 1. 提供类似终端的界面 xterm.js可以在浏览器中创建一个看起来和行为都类似于传统终端的界面。这包括显示命令行提示符、接受用户输入、显示命令输出等。 例如&#…...
在Excel中如何快速筛选非特定颜色
Excel中的自动筛选是个非常强大的工具,不仅可以筛选内容,而且可以筛选颜色,例如筛选A列红色单元格。但是有时希望筛选除了红色之外的单元格(下图右侧所示),其他单元格的填充色不固定,有几种颜色…...
kotlin定时器和主线程定时器
场景 最近要用kotlin写一个每隔一段时间切视频并截图 刷刷的就写出来了,很快啊 timerTask object : TimerTask() {override fun run() {captureWindow()if ((group 1) * 4 > urls.size) {showDialog()timerTask.cancel()timer.cancel()}groupupdatePlayers(…...
vscode不能执行vue命令/ vue : 无法加载文件
问题: 解决: 1. 在Windows应用中找到Windows PowerShell,以管理员运行: 2. 在命令框输入: set-ExecutionPolicy RemoteSigned, 然后输入A即可解决...
1.4 STL C++面试问题
1.4.1 说说STL的基本组成部分 总结 STL 的基本组成部分包括容器、算法、迭代器、函数对象和仿函数和适配器。通过这些组件,STL 提供了高效、灵活和可复用的代码结构,极大地提高了 C 的开发效率和程序的可维护性。STL 的设计思想使得算法和数据结构的使…...
Bash、sh 和 Shell都弄混了?
在Linux和Unix系统中,Bash、sh 和 Shell 都与命令行解释器相关,但它们各自的含义和作用略有不同。以下是它们之间的关系和区别: Shell Shell 是一个通用术语,指的是操作系统中负责解释和执行用户命令的程序。它是用户与操作系统…...
架构师备考专栏-导航页
简介 架构师备考专栏——软考系统架构师考试的学习宝典,集合了全面覆盖架构师考试大纲的精华文章。每篇文章都为本人手输,并校对数遍后发表,在此我保障每篇文章的质量绝对过关。诚邀对架构师软考感兴趣的朋友们收藏此页面,并根据个人所需高效…...
STM32-Cube定时器TIM
一、内部时钟源 1、创建项目 File → New → STM32 project选择STM32F103C8T6单片机,命名TIM 2、配置单片机 1.打开USART1,方便我们与电脑连接查看数据 开启UART1并开启中断。 2、设置时钟源 开启外部高速晶振 将时钟频率设置为72MHz 设置调试模…...
Linux 文件类型,目录与路径,文件与目录管理
文件类型 后面的字符表示文件类型标志 普通文件:-(纯文本文件,二进制文件,数据格式文件) 如文本文件、图片、程序文件等。 目录文件:d(directory) 用来存放其他文件或子目录。 设备…...
stm32G473的flash模式是单bank还是双bank?
今天突然有人stm32G473的flash模式是单bank还是双bank?由于时间太久,我真忘记了。搜搜发现,还真有人和我一样。见下面的链接:https://shequ.stmicroelectronics.cn/forum.php?modviewthread&tid644563 根据STM32G4系列参考手…...
为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?
在建筑行业,项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升,传统的管理模式已经难以满足现代工程的需求。过去,许多企业依赖手工记录、口头沟通和分散的信息管理,导致效率低下、成本失控、风险频发。例如&#…...
CentOS下的分布式内存计算Spark环境部署
一、Spark 核心架构与应用场景 1.1 分布式计算引擎的核心优势 Spark 是基于内存的分布式计算框架,相比 MapReduce 具有以下核心优势: 内存计算:数据可常驻内存,迭代计算性能提升 10-100 倍(文档段落:3-79…...
全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
质量体系的重要
质量体系是为确保产品、服务或过程质量满足规定要求,由相互关联的要素构成的有机整体。其核心内容可归纳为以下五个方面: 🏛️ 一、组织架构与职责 质量体系明确组织内各部门、岗位的职责与权限,形成层级清晰的管理网络…...
如何在看板中有效管理突发紧急任务
在看板中有效管理突发紧急任务需要:设立专门的紧急任务通道、重新调整任务优先级、保持适度的WIP(Work-in-Progress)弹性、优化任务处理流程、提高团队应对突发情况的敏捷性。其中,设立专门的紧急任务通道尤为重要,这能…...
在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?
uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件,用于在原生应用中加载 HTML 页面: 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...
Yolov8 目标检测蒸馏学习记录
yolov8系列模型蒸馏基本流程,代码下载:这里本人提交了一个demo:djdll/Yolov8_Distillation: Yolov8轻量化_蒸馏代码实现 在轻量化模型设计中,**知识蒸馏(Knowledge Distillation)**被广泛应用,作为提升模型…...
C语言中提供的第三方库之哈希表实现
一. 简介 前面一篇文章简单学习了C语言中第三方库(uthash库)提供对哈希表的操作,文章如下: C语言中提供的第三方库uthash常用接口-CSDN博客 本文简单学习一下第三方库 uthash库对哈希表的操作。 二. uthash库哈希表操作示例 u…...
