当前位置:首页 > 芯闻号 > 充电吧
[导读]        感谢原著作者对人类文化的传播做出的努力!以下内容直译为主,意译为辅,同时笔者可能会加入个人观点以方便理解。如有翻译不当的地方希望各位同仁积极指出,如有必要的话请做出引证,以助于笔者翻译

        感谢原著作者对人类文化的传播做出的努力!以下内容直译为主,意译为辅,同时笔者可能会加入个人观点以方便理解。如有翻译不当的地方希望各位同仁积极指出,如有必要的话请做出引证,以助于笔者翻译水平的提高,同时也有利于其它读者的学习,进步。天朗气清,惠风和畅,祝各位生活愉快!

   

问题:我刚开始学习ARM汇编语言,不太清楚如何利用MOV指令将立即数传入到寄存器。在ARM参考手册及我的课本中,都说MOV指令后跟的立即数变化范围是0~255。但是当我在ADS1.2集成开发环境中进行测试时,下述指令语句却正常运行。

MOV   R2, #0xFFFFFFFF

根据上述说法数字0xFFFFFFFF不是超出了范围吗?为什么会出现这种情况呢?

回答:ARM可以对立即数进行一些特定的操作,因为ARM核内集成了桶形移位器,ARM操作码可以借此完成一些特定操作。下文介绍ARM汇编器使用了哪些技巧,从而使一个大的立即数达到ARM指令可处理的小空间的。

—————————————————————————————————————————————————

不是任意一个32位数都可以表示成32位指令字。ARM数据处理指令在指令字中有12位空间用于存数值。如下图所示它是由4位循环移位值和8位立即数组成:


4位循环移位值保存在11-8位上,它乘以2从而得到循环移位范围0-30。

根据这种设计原则,我们可以表示的立即数如下:

0x000000FF 0x00000FF0 0xFF000000 0xF000000F

而像下面这样的立即数

0x000001FE 0xF000F000 0x55550000

是不符合ARM指令处理规范的,所以无法直接处理。

汇编器会把大的数转化成循环移位形式,即由8位二进制数循环右移偶数次得到,循环移位范围是0-30。不合法的立即数会产生错误。

有些汇编器会使用一些其它技巧,如使用MVN代替MOV得到一些数的按位取反数。例如指令MOV  r0, #0xFFFFFFFF可以被汇编为MVN  r0, #0。

以上这种问题导致有些常数是ARM友好的(ARM friendly),有些则不是。所以仔细研究一下你正在使用的数,也许还有再进一步优化的余地。

你可以使用指令序列创建一些单条指令无法操作的常数,如:


MOV   r2,#0x55              ; R2 = 0x00000055
ORR   r2, r2, r2, LSL #8    ; R2= 0x00005555
ORR   r2, r2, r2, LSL #16   ; R2 =0x55555555


或者从存储器中装载数值:


LDR  r2, =0x55555555


如果可能的话,伪指令LDR  Rx,=const会尝试用一条指令创建常数,否则会生成一个LDR。

—————————————————————————————————————————————————


你所提到的例子可能使用了上文讨论的这些技巧,如生成了MVN操作码来装载立即数的按位取反值。这些操作并不适用于所有立即数,但是ARM汇编器很机智地知道该如何处理(C编译器当然也是)。如果有些数无法用移位/取反方法表示,这些数通常会从PC相关的位置装载或者用几条指令来创建。


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读
关闭