微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > ARM汇编指令的一些总结

ARM汇编指令的一些总结

时间:11-28 来源:互联网 点击:
ARM汇编指令很多,但是真正常用的不是很多,而且需要认真琢磨的又更少了。

比较有用的是MOVBBLLDRSTR

还是通过具体汇编代码来学习吧。

@ disable watch dog timer

movr1, #0x53000000//立即数寻址方式

movr2, #0x0

strr2, [r1]

MOV没有什么好说的,只要掌握几个寻址方式就可以了,而且ARM的寻址方式比386的简单很多。立即数寻址方式,立即数要求以“#”作前缀,对于十六进制的数,还要求在#后面加上0x或者&。0x大家很好理解。有一次我碰到了&ff这个数,现在才明白跟0xff是一样的。

STR是比较重要的指令了,跟它对应的是LDR。ARM指令集是加载/存储型的,也就是说它只处理在寄存器中的数据。那么对于系统存储器的访问就经常用到STR和LDR了。STR是把寄存器上的数据传输到指定地址的存储器上。它的格式我个人认为很特殊:

STR(条件)源寄存器,<存储器地址>

比如STR R0, [R1],意思是R0-> [R1],它把源寄存器写在前面,跟MOV、LDR都相反。

LDR应该是非常常见了。LDR就是把数据从存储器传输到寄存器上。而且有个伪指令也是LDR,因此我有个百思不得其解的问题。看这段代码:

movr1, #GPIO_CTL_BASE

addr1, r1, #oGPIO_F

ldrr2,=0x55aa// 0x55aa是个立即数啊,前面加个=干什么?

strr2, [r1, #oGPIO_CON]

movr2, #0xff

strr2, [r1, #oGPIO_UP]

movr2, #0x00

strr2, [r1, #oGPIO_DAT]

对于当中的ldr那句,我就不明白了,如果你把=去掉,是不能通过编译的。我查了一些资料,个人感觉知道了原因:这个=应该表示LDR不是ARM指令,而是伪指令。作为伪指令的时候,LDR的格式如下:

LDR寄存器,=数字常量/Label

它的作用是把一个32位的地址或者常量调入寄存器。嗬嗬,那大家可能会问,

“MOV r2,#0x55aa”也可以啊。应该是这样的。不过,LDR是伪指令啊,也就是说编译时编译器会处理它的。怎么处理的呢?——规则如下:如果该数字常量在MOV指令范围内,汇编器会把这个指令作为MOV。如果不在MOV范围中,汇编器把该常量放在程序后面,用LDR来读取,PC和该常量的偏移量不能超过4KB。

这么一说,虽然似懂非懂,但是能够解释这个语句了。

然后说一下跳转指令。ARM有两种跳转方式。

(1)mov pc <跳转地址〉

这种向程序计数器PC直接写跳转地址,能在4GB连续空间内任意跳转。

(2)通过BBLBLXBX可以完成在当前指令向前或者向后32MB的地址空间的跳转(为什么是32MB呢?寄存器是32位的,此时的值是24位有符号数,所以32MB)。

B是最简单的跳转指令。要注意的是,跳转指令的实际值不是绝对地址,而是相对地址——是相对当前PC值的一个偏移量,它的值由汇编器计算得出。

BL非常常用。它在跳转之前会在寄存器LR(R14)中保存PC的当前内容。BL的经典用法如下:

blNEXT; 跳转到NEXT

……

NEXT

……

mov pc, lr; 从子程序返回。

最后提一下Thumb指令。ARM体系结构还支持16位的Thumb指令集。Thumb指令集是ARM指令集的子集,它保留了32位代码优势的同时还大大节省了存储空间。由于Thumb指令集的长度只有16位,所以它的指令比较多。它和ARM各有自己的应用场合。对于系统性能有较高要求,应使用32位存储系统和ARM指令集;对于系统成本和功耗有较高要求,应使用16位存储系统和ARM指令集。

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top