ARM day2、day3 汇编

news2024/10/5 13:49:25

一、汇编学习:可以向上理解软件、向下感知硬件

二、符号(注释)

@注释
#注释(放在行首表示注释一行)
/*    */注释
#+数字立即数
一种标号(比如main:        loop:)
.text        .end+换行固定格式

三、ARM指令格式和立即数

ARM指令构成

ARM 指令包含操作码和一些其他的信息,只剩下8 位存放数据

具体来说,一个 ARM 指令通常由以下部分组成:

  1. 操作码(Opcode):这指定了指令的基本操作,例如 ADD、MOV、SUB 等。
  2. 条件码(Condition):这指定了该指令在什么条件下执行。
  3. 寄存器索引(Register Index):这指定了要操作的寄存器。
  4. 位移量(Offset):这是一个相对于某个基址的偏移量,常用于内存操作。
  5. 立即数(Immediate):这是一个直接嵌入到指令中的数值。

其中,操作码、条件码、寄存器索引和位移量总共占据了大部分的指令位数,因此,一个 ARM 指令中只有少部分(通常为 8 位)是用来表示立即数的。

例如,一个 ARM 指令可能是这样的格式:

Opcode    Condition   Register1  Register2  Offset/Displacement   #0

一般指令格式

操作码  目的操作数   源操作数  

mov      r0,        #5   
mov      r0,           r1 

立即数的优点取指的时候,可以直接读到cpu  不需要单独去内存读取,速度快
立即数的缺点不能是任意的32位数字,有局限,且只能放在指令右边

 四、如何判断立即数合法

转化为二进制编码,观察第一个1和最后一个1之间的位数(包含这两个1)

1、将数据转换为 二进制编码   4 个一组 
2、数1 的位数, 如果超过8 个 那就不是立即数
3、如果数据中有连续  大于等于 24 个 0 ,通过循环位移 偶数位, 使其高位全部为0 
4、找到 高位1 去掉前边偶数个0 
5、找到低位的1, 去掉末尾偶数个0 
6、剩下的位数小于等于8 那么就是立即数,否则 不是 

eg1:0xF000 000F

1111  0000 0000 0000 0000 0000 0000  1111 -- 1111 1111合法

eg2:0x123

... 0000    0001 0010 0011 -- 01 0010 0011不合法

eg3:0x234

....     0010 0011 0100合法

eg4:0x3F0

...      0011 1111 0000合法

五、伪指令

伪指令不是指令

伪指令和指令的根本区别编译后,是否生成机器码(伪指令不会生成)
伪指令的意义

指导编译的过程(如果需要执行伪指令,必须先翻译成在指令)

比如.text  .end 也是伪指令

伪指令和编译器相关我们用到是gnu工具链,因此我们学习的是gnu 环境下的汇编伪指令
常用伪指令

    ldr  //大范围地址的加载指令
    adr  //小范围地址的加载指令
    adrl //中范围地址的加载指令
    nop  //空操作

    ldr (load register) 将内存内容加载进入通用寄存器 
    str (store register) 将寄存器内容存入内存空间

eg:

ldr r0, =0x12345678

num : .word   0x15300000     //int  num = 0x15300000 

buf1: .byte 1, 2, 3, 4        //char  buf1[4] = {1,2,3,4}; 

六、指令--搬移、偏移指令

mov

搬移

mov r2, r1   // r2 = r1

mov r0, #7   //r1  = 7 

mvn

取反搬移

mvn r0, r1   // r0 = ~r1

ldr向寄存器存入非立即数可以用伪指令
LSL左移位
LSR右移位
ASL算术左移位(位数不够补符号位)
ASR算术右移位(位数不够补符号位)
ROR循环移位
RRX带扩展的循环右移
偏移指令总结    mov r0,r2,  lsr #3   //r0 = r2  >> 3 
    mov r1 ,r0,lsl #1       //r1 = r0 << 1 
    add   r2, r1,r0,lsr #2  // r2 = r1 + r0 >> 2 
        
        LSL '逻辑左移' (Logical   shift  left)
        LSR '逻辑右移'(Logical   shift  right)
        ASL '算术左移' (Arithmet  shift  left )
        ASR '算术右移'(Arithmet  shift right )
        ROR '循环右移'
        RRX '带扩展的循环右移'
重要例子

tst  r0, #0x8    @判断某个数的某几位 是否为 0
    @ movne  r11, #1
    moveq  r11, #1   @当上一条判断结果相等时  执行 mov
    movne  r12, #2   @当上一条判断结果不相等时执行mov 

tst  r0, #0x8    表示判断r0的第三位(8的二进制1000,也就是从右往左的第四位(0 1 2 3))是否为0

eq  ne表示相等/不相等

 搬移

练习:
	 a) r0 = 16
    b) r0 = r1/16
    c) r1 = r2 *2
    d) r0 = -4
    e) r1 = r0/2
 答:   
	mov r0, #16 	
	mov r1, #32
	mov r2,r1, LSR #4
	mov r2, #4
	mov r3, r2, LSL #1	
	mov r0, #-4 	
	mov r1,r0, ASR #1 

偏移(移位)

.text   @基本格式的头
@搬移指令 
	mov r0, #3			@搬移指令:   r0   = 3 ; #3 立即数 
	ldr r0, =0xfff00000 @ldr 伪指令   r0   = 0xfff 
	
	mov r1,r0			@搬移指令	  r1   = r0 
	
@移位
	mov  r0, #0x4	
	mov  r1, r0,  LSL #1 @r1=r0 << 1 , 逻辑左移: 高位移除部分舍去,低位不够补0

	ldr  r0, =0xffffffff
	mov  r2, r0, LSR #1  @r2=r0 >>1  逻辑右移,低位移除部分舍去, 高位部分不够补 0

	mov  r3, r0, ASR #1  @r3 = r0 >> 1  算术右移,低位移除部分舍去 高位不够 补 符号位
	
	ldr  r0, =0x7000000f 
	mov  r4,r0, ROR #1	  @r4 = r0 >>1 , 循环右移, 低位移除部分补到 高位 
	
	ldr  r0, =0xff0000ff
	mvn  r5,r0			  @r5 = ~r0
	


.end    @基本格式的尾巴  通常 该行下边 加空行

七、指令--位操作  AND ORR BIC EOR

and
orr
bic

按位清零(比如说0x4 就表示第2位(从0开始数)清零)

bic  r0,r1,#0x4  //r0  = r1 &(~0x4  )

eor异或
.text   @基本格式的头
	@逻辑指令
	mov r0, #6
	mov r1, #5
	and r2, r0,r1	@r2 = r1 & r0
	orr r3, r0,r1	@r3 = r1 | r0
	eor	r4, r0,r1	@r4 = r1 ^ r0 
	
	ldr  r0,=0x12345678
	bic  r1, r0, #0xff
	bic  r2,r0, #0xf0000000
	
	tst  r0, #0x8	@判断某个数的某几位 是否为 0
	@ movne  r11, #1
	moveq  r11, #1   @当上一条判断结果相等时  执行 mov
	movne  r12, #2   @当上一条判断结果不相等时执行mov 
	
	@使能中断和快速中断(设置cpsr的值)
	mrs r0, cpsr  @ r0 = cpsr
	bic r1, r0, #0xc0
	msr cpsr, r1  @cpsr = r1
	

.end    @基本格式的尾巴  通常 该行下边 加空行

 八、指令--比较指令、测试指令

用法注意:  

        cmp

        moveq

        movne

cmp

cmp   r1, r0           //改变 cpsr 的 NZ 

moveq  r2, #0    //相当于  if (r1 == r0)  r2 = 0(这是两行代码)

cmn
关系符号

eq ==

ge >=

gt >

le <=

lt <

ne !=

tst

实质是做 与运算, 通常用于 测试某一位或几位是0 还是1 结果 CPSR  'Z'  位来判断, Z 位位 1 表明结果为0 

tst  r0,  #0x10  //测试 第4位是否为 0,  if((r0  &(0x10)) == 0)
//00010000

teq

实质是异或运算  测试两个 数是否相等,如果两个数相等或者异或结果 为 0  修改 cpsr 的z 位 判断

teq r0, r1   // if((r0 ^ r1 ) == 0) 

.text   @基本格式的头
@比较指令
	/*
		int  a = 5;
		if(a >= 5)
			a =  0
			else 
			a = 100
		
	*/
	/*
	mov r0, #5
	cmp r0, #5   		@比较指令 , 比较 r0 和 5 
	movge   r0 , #0		@当cmp的结果 >= 时, r0 = 0 
	movlt	r0, #100	@当cmp的结果 是 < 时, r0 = 100
	*/

.end    @基本格式的尾巴  通常 该行下边 加空行

九、指令--算术指令 :ADD ADC SUB SBC RSB RSC

add

相加

add r2, r1,#3     //r2 = r1 + 4 

adds

会改变条件位的相加(配合adc使用)

adds  r0, r1, r2   //  r0 = r1 + r2  &&  (cpsr ) v c 

CPSR:进位 -- c置为1    溢出 -- v置为1

补充--溢出:在两个正数相加时,如果结果超过了机器所能表示的最大正数,那么就发生了上溢。同样地,在两个负数相加时,如果结果小于机器所能表示的最小负数,那么就发生了下溢

adc

带进位的加法(常用于64位加法,配合adds使用)

adc  r0, r1, r2  //r0 = r1 + r2  + (cpsr)  c

sub

相减

sub r1, r2, r3 

subs

影响cpsr的条件位 

没有借位 时 cpsr  'c' 位 置1  当有借位 c = 0 

sbcsbc  带借位的减法
rsbrsb   逆向减法
rsb r1,r2, r3  //r1  = r3 - r2 
mul

乘法指令 

mul r2, r0, r1  // r2 = r0 * r1 

练习:64位加减法(高32位 低32位)

.text   @基本格式的头
	@算术指令
	mov r0, #10
	mov r1, #5
	add r2, r0, r1 	@r2 = r0 + r1 
	 
	ldr  r0, =0xff000000
	ldr  r1, =0xf0000000
	@add  r2, r0, r1    @(不带进位加法)
	
	adds  r2, r0, r1  @r2 = r0 + r1 , 如果有进位 那么会修改cpsr的进位值

	mov r3, #1
	mov r4, #1
	adc r5,r3,r4    	@r5 = r3 + r4 + 进位值

	@64 bit 两个数的加法  r0, r1 表示被加数, r2, r3表示加数 
	@64 bit 两个数的减法 
	ldr r0, =0xffffffff			@低32 bit
	ldr r1, =0x1				@高32bit
	
	ldr  r2,=0x1				@低32bit 
	ldr	 r3,=0x5				@高32bit 
	
	adds  r4, r0, r2
	adc   r5, r1, r3
	
	subs  r6, r2,r0
	sbc   r7, r3,r1
	@add  r2, r0, r1 			@不带进位 不能计算 64 bit 
	
	

.end    @基本格式的尾巴  通常 该行下边 加空行

十、条件码(eq、ne、lt小于、gt大于)

十一、跳转

b 类似 goto 
    bl  跳转之前 先保存 下一条指令的地址
    bl: lr(r14) = pc (r15) -4    (由cpu 帮我们执行 )

练习:函数跳转

int  main(void )
{
    
    int ret = 0;
    func1(2);
    while(1) ; 
}
func1(int a)
{
    if(a == 2)  return func2(a)
        else   return func3 (a)        
}

func2(int a)
{
    return a+3;       
}
func3(int a)
{
    return a-1; 
}    

.text   @基本格式的头
	@跳转练习
	
main:
	mov r0, #0
	mov r1, #2
	bl func1

main_end:
	b  main_end  	@死循环
	
func1:
	cmp r1, #2
	bleq  func2
	blne  func3
func1_end:

func2:
	add r1, #3
	b func2_end
func2_end:
	mov pc, lr

func3:

@.....字节补充一下
func3_end:

	

.end    @基本格式的尾巴  通常 该行下边 加空行

练习:1加到100

.text   @基本格式的头
	@跳转指令 
	/*
	mov  r0, #1
	mov  r1, #10
	
	b  t			@跳转指令, 跳到 t 后执行 
	add  r2, r0, r1
	
t:	
	sub r3, r0, r1
	
	@b  t 
	*/
	
	@实现 1-100的累加 和 
	/*
		for(int i=1; i<=100; i++)
		{
			static int sum += i; 
		}
	*/
	
	mov r0, #0    	@类似 sum 累加和
	mov r1, #1		@类似 i   循环变量
loop:
	@循环计算
	cmp r1, #100
	bgt	loop_end	@这句 相当于 判断 i 大于 100 跳出循环 
	
	add r0,r1		@相当于 sum += i; 
	add r1, #1		@相当于 i++
	b loop			@继续下次循环
	
loop_end:
	@结束 
	mov r12,r0   @把计算结果 放到 r12 
	
	

.end    @基本格式的尾巴  通常 该行下边 加空行

练习:延时1s

.text   @基本格式的头
	@跳转指令 
	@实现延时 1 s
	ldr r0, = 0x1f1fffff   @具体多少为 1秒 需要计算
loop:
	cmp r0, #0
	beq  loop_end

	sub r0,#1
	b loop
loop_end:
	mov r12, #0
	

.end    @基本格式的尾巴  通常 该行下边 加空行

十二、修改ARM状态和模式 ???

.text   @基本格式的头

	@判断 当前工作状态是否 是ARM 状态 ,如何切换到 user 模式或其他模式 
	mrs  r0, cpsr  				@r0 = cpsr 
	mov r1, #0x20  				@0010 0000   T 位    
	orr  r1, r1, #0x1000000 	@J 位, 结果r1把 T位和 J位置为1 ,其余为0
	
	tst r0, r1				@判断 r0中的J和 T 是否为 0 
	biceq  r0, #0x1f		@把 mode 清0 
	orreq  r0, #0x10		@把mode 变为 user模式: 10000
	msr  cpsr, r0  			@cpsr = r0
	 	
.end    

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

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

相关文章

[足式机器人]Part2 Dr. CAN学习笔记-Advanced控制理论 Ch04-9 可观测性与分离原理

本文仅供学习使用 本文参考&#xff1a; B站&#xff1a;DR_CAN Dr. CAN学习笔记-Advanced控制理论 Ch04-9 可观测性与分离原理

【Java JVM】栈帧

执行引擎是 Java 虚拟机核心的组成部分之一。 在《Java虚拟机规范》中制定了 Java 虚拟机字节码执行引擎的概念模型, 这个概念模型成为各大发行商的 Java 虚拟机执行引擎的统一外观 (Facade)。 不同的虚拟机的实现中, 通常会有 解释执行 (通过解释器执行)编译执行 (通过即时编…

Servlet项目教学(附实例代码)

【员工信息管理】 1.员工信息管理 1.1 介绍 用户进行登录后,可以对员工信息进行管理(增删查改),等操作.如果用户没有登录,不能访问员工操作页面.并且员工操作页面显示当前登录用户信息. 1.2 技术点 使用VueElementUI充当前端界面,使用ServletJDBCMysql提供数据管理控制.后端统…

嵌入式学习-网络编程-Day3

思维导图 多进程并发服务器通信模型如下 void handler(int signo) {while(waitpid()>0); //以非阻塞配合信号完成僵尸进程的回收 } int main() {//将信号与信号处理函数绑定signal(SIGCHLD, handler);socket(); //创建用于连接的套接字bind(); //绑定IP…

Python和Java代码实现:切线法求解一维最优化问题

Python和Java代码实现&#xff1a;切线法求解一维最优化问题 代码实现Python代码Java代码 求解实例 根据概念查询&#xff0c;切线法定义如下&#xff1a; 切线法&#xff08;Tangent Method&#xff09;是一种用于求解非线性方程的数值方法。它也被称为牛顿法&#xff08;Newt…

C++ 程序文档生成器(doxygen)使用说明

程序文档&#xff0c;是每个程序员必看文档&#xff0c;在日常业务开发中&#xff0c;难免会封装一些组件。没有很好的组件文档&#xff0c;再好的组件都是废物&#xff0c;。因此大型业务中&#xff0c;文档和思维导图&#xff0c;两个都是必备&#xff01; 一、注释风格 …

IPv6隧道--GRE隧道

GRE隧道 通用路由封装协议GRE(Generic Routing Encapsulation)可以对某些网络层协议(如IPX、ATM、IPv6、AppleTalk等)的数据报文进行封装,使这些被封装的数据报文能够在另一个网络层协议(如IPv4)中传输。 GRE提供了将一种协议的报文封装在另一种协议报文中的机制,是一…

各种设备上恢复已删除的文件和文件夹的数据恢复软件清单

最好的数据恢复软件可以简单轻松地恢复计算机、移动设备或存储介质上已删除的文件和文件夹。 询问任何经历过数据丢失的人这是否是一种有趣的经历&#xff0c;他们会告诉您数据丢失&#xff0c;无论是由于硬件或软件故障、意外删除还是网络犯罪&#xff0c;都会带来极大的压力…

美国 SEC 批准比特币现货 ETF 上市,SEC 告诉我们的风险包含哪些?

撰文&#xff1a;Will 阿望 查看原文&#xff1a;美国 SEC 批准比特币现货 ETF 上市&#xff0c;SEC 告诉我们的风险包含哪些&#xff1f; 历经十年的 BTC ETF 艰辛审批之路终于迎来了胜利的曙光&#xff0c;2024 年 1 月 11 日凌晨 4 时&#xff0c;美国证监会&#xff08;S…

AIGC实战——像素卷积神经网络(PixelCNN)

AIGC实战——像素卷积神经网络 0. 前言1. PixelCNN 工作原理1.1 掩码卷积层 1.2 残差块2. 训练 PixelCNN3. PixelCNN 分析4. 使用混合分布改进 PixelCNN小结系列链接 0. 前言 像素卷积神经网络 (Pixel Convolutional Neural Network, PixelCNN) 是于 2016 年提出的一种图像生成…

礼贺新春,徐坊大曲新品【中国红】

梁山徐坊大曲新推出中国风礼盒&#xff0c;以中国红为主题&#xff0c;为即将到来的新春佳节增添了浓厚的节日气氛。为您呈现一场视觉与味觉的盛宴。从礼盒的颜色到图案设计&#xff0c;无不体现出中国红的热情与活力&#xff0c;象征着吉祥、喜庆与团圆。梁山徐坊大曲&#xf…

ubuntu qt 运行命令行

文章目录 1.C实现2.python实现 1.C实现 下面是封装好的C头文件&#xff0c;直接调用run_cmd_fun()即可。 #ifndef GET_CMD_H #define GET_CMD_H#endif // GET_CMD_H #include <iostream> #include<QString> using namespace std;//system("gnome-terminal -…

USB8814动态信号采集卡——声音振动类信号处理的理想之选!

背景介绍&#xff1a; 科技的发展在一定程度上依赖于对信号的处理&#xff0c;信号处理技术的先进性在很大程度上决定了科技发展的速度和方向。数字信号处理技术的崛起&#xff0c;彻底改变了传统的信息与信号处理方式&#xff0c;使得数据采集这一前期工作在数字系统中发挥着…

FTP文件传输协议 、多种方式安装yum仓库

一、网络文件共享服务 1.存储类型分三种&#xff1a; 直连式存储&#xff1a;Direct-Attached Storage&#xff0c;简称DAS 存储区域网络&#xff1a;Storage Area Network&#xff0c;简称SAN&#xff08;可以使用空间&#xff0c;管理也是你来管理&#xff09; 网络附加存储…

ImageNet Classification with Deep Convolutional 论文笔记

✅作者简介&#xff1a;人工智能专业本科在读&#xff0c;喜欢计算机与编程&#xff0c;写博客记录自己的学习历程。 &#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&…

Leetcode23-数组能形成多少数对(2341)

1、题目 给你一个下标从 0 开始的整数数组 nums 。在一步操作中&#xff0c;你可以执行以下步骤&#xff1a; 从 nums 选出 两个 相等的 整数 从 nums 中移除这两个整数&#xff0c;形成一个 数对 请你在 nums 上多次执行此操作直到无法继续执行。 返回一个下标从 0 开始、长…

SpringMVC参数接收见解4

# 4.参数接收Springmvc中&#xff0c;接收页面提交的数据是通过方法形参来接收&#xff1a; 处理器适配器调用springmvc使用反射将前端提交的参数传递给controller方法的形参 springmvc接收的参数都是String类型&#xff0c;所以spirngmvc提供了很多converter&#xff08;转换…

第二证券:大盘探底回升走出底部还看成交量配合

持续震动数日后&#xff0c;大盘再现探底上升走势。 上证指数周二小幅低开后窄幅震动&#xff0c;午后快速回落改写本轮回调新低后&#xff0c;有资金开始出手介入&#xff0c;尾盘指数翻红。深证成指同样是在午后呈现探底上升走势&#xff0c;最终重回5日均线上方。截至收盘&…

mysql 下载和安装和修改MYSQL8.0 数据库存储文件的路径

一、第一步:下载步骤 下载链接&#xff1a;MySQL :: Download MySQL Installer 选择版本8.0.35&#xff0c;社区版&#xff0c; 点击 Download 下载 安装包 二、第二步:安装步骤 添加环境变量&#xff0c;C:\Program Files\MySQL\MySQL Server 8.0\bin 可以点开MySQL 8.0 Co…

如何用AI提高论文阅读效率?

已经2024年了&#xff0c;该出现一个写论文解读AI Agent了。 大家肯定也在经常刷论文吧。 但真正尝试过用GPT去刷论文、写论文解读的小伙伴&#xff0c;一定深有体验——费劲。其他agents也没有能搞定的&#xff0c;今天我发现了一个超级厉害的写论文解读的agent &#xff0c…