×
单片机 > 单片机程序设计 > 详情

ARM汇编指令的一些总结

发布时间:2020-06-22 发布时间:
|
ARM汇编指令很多,但是真正常用的不是很多,而且需要认真琢磨的又更少了。

比较有用的是MOV BL LDR STR

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

 

 

 

      @ 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)通过B BL BLX BX可以完成在当前指令向前或者向后32MB的地址空间的跳转(为什么是32MB呢?寄存器是32位的,此时的值是24位有符号数,所以32MB)。

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

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

       bl NEXT  ; 跳转到NEXT

 

       ……

 

    NEXT

 

       ……

 

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

 

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

关键字:ARM  汇编指令

『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
C51 特殊功能寄存器SFR的名称和地址