ARM中当立即数作为第二个操作数(源操作数)时如何使用MOV指令

翻译 2016年05月30日 22:07:19

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

       原文链接:点击打开链接

问题:我刚开始学习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相关的位置装载或者用几条指令来创建。

附:此链接可能有助于你对上文的理解 点击打开链接

ARM8位位图--第二操作数的立即数表示

在ARM处理器的汇编语言中,对指令语法格式中的的常数表达式有这样的规定:“该常数必须对应8位位图,即常数是由一个8位的常数循环移位偶数位得到的。”        首先从ARM指令系统的语法格式说起。...

嵌入式Linux系统中的.lds链接脚本基础

连接脚本的格式 ==================== 连接脚本是文本文件. 你写了一系列的命令作为一个连接脚本. 每一个命令是一个带有参数的关键字,或者是一个对符号的赋值. 你可以用分号分隔...

漫谈程序员系列:怎样成为技术达人

作为程序员,身边总是有牛逼的前辈、后辈或者同辈。牛逼的人总是让人羡慕。比如自己苦搞一天的 BUG ,头发快抓掉完了,人家扫上两眼,改一行代码,问题就解决了;比如自己干了十年开发,薪水还不到 10K ,...
  • foruok
  • foruok
  • 2014年12月24日 07:23
  • 55038

ARM立即数,LDR和MOV的区别

ARM立即数,LDR和MOV的区别      Mov 是把立即数赋给一个寄存器,但对立即数的范围有要求。只能是由8bit连续有效位通过偶数次移位能得到的数。如果立即数超出这个范围,就没办法用一条MOV...

ARM立即数讲解--LDR和MOV的区别

http://blog.csdn.net/atec2000/article/details/6627909   Mov 是把立即数赋给一个寄存器,但对立即数的范围有要求。只能是由8bit连续有效位...

ARM立即数,LDR和MOV的区别

Mov 是把立即数赋给一个寄存器,但对立即数的范围有要求。只能是由8bit连续有效位通过偶数次移位能得到的数。如果立即数超出这个范围,就没办法用一条MOV指令给寄存器赋值。 LDR除了普通的读数...

ARM立即数讲解--LDR和MOV的区别

Mov 是把立即数赋给一个寄存器,但对立即数的范围有要求。只能是由8bit连续有效位通过偶数次移位能得到的数。如果立即数超出这个范围,就没办法用一条MOV指令给寄存器赋值。LDR除了普通的读数之外,也...

ARM 汇编的mov操作立即数的疑问

转自:http://blog.chinaunix.net/space.php?uid=20799298&do=blog&cuid=2055392 ARM 汇编的mov操作立即数的疑问 (2009...

mov指令的操作数的取值范围到底是多少

关于mov指令操作数的取值范围,网上看到一些人说是0x00-0xFF,也有人说是其他的值的,但是经过一番求证,发现这些说法都不对。下面就是来详细解释,mov指令的操作数的取指范围,到底是多少。 ...

ARM指令中第二操作数为常数表达式时8位位图的理解

一条典型的ARM指令语法格式分为如下几个部分:     {} {S}     ,{,}    其中,是指令助记符,是必须的,而{}为指令执行条件,是可选的,如果不写则使用默认条件AL(无条件执行)...
  • nolatin
  • nolatin
  • 2012年10月04日 20:50
  • 1312
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:ARM中当立即数作为第二个操作数(源操作数)时如何使用MOV指令
举报原因:
原因补充:

(最多只允许输入30个字)