Chinaunix首页 | 论坛 | 博客
  • 博客访问: 86485
  • 博文数量: 53
  • 博客积分: 2500
  • 博客等级: 少校
  • 技术积分: 390
  • 用 户 组: 普通用户
  • 注册时间: 2008-05-06 20:01
文章分类

全部博文(53)

文章存档

2008年(53)

我的朋友

分类:

2008-05-15 11:17:08

 

2汇编指令的一些总结

ARM汇编指令很多,但是真正常用的不是很多,而且需要认真琢磨的又更少了。

比较有用的是MOV B BL LDR STR

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

 

      @ disable watch dog timer      

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

   movr2, #0x0

   strr2, [r1]         

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

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

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

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

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有两种跳转方式。

1mov 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指令集。


阅读(856) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~