ARM指令LDR和ADR的一些区别
一、adr和ldr的区别
同学们在学习
ARM指令时,多数都会对adr和ldr这两个命令产生疑惑,那他们究竟有什么区别呢?
其实这两个都是伪指 令:adr是小范围的地址读取伪指令,ldr是大范围的读取地址伪指令。可实际上adr是将基于PC相对偏移的地址值或基于寄存器相对地址值读取的为指 令,而ldr用于加载32为立即数或一个地址到指定的寄存器中。到这儿就会看到其中的区别了。如果在程序中想加载某个函数或者某个在联接时候指定的地址时 请使用adr,例如在lds中需要重新定位的地址。当加载32为的立即数或外部地址时请用ldr。
我给大家先举个例子:
AREA test,CODE,READONLY
ldr r0,_start
_start
这段代码并无实际意义,只是为了方便说明。我们反汇编一下看看:
4:
ldr
r0,_start
ldr
r0, _start
从内存地址 _start 的地方把值读入。执行这个后,r0 = 0xe1a00000
adr
r0, _start
取得 _start 的地址到 r0,但是请看反编译的结果,它是与位置无关的。其实取得的时相对的位置。例如这段代码在 0x00000000 运行,那么 adr r0, _start 得到 r0 = 0x00000010;
ldr
r0, =_start
这个取得标号 _start 的绝对地址。这个绝对地址是在 link 的时候确定的。看上去这只是一个指令,但是它要占用 2 个 32bit 的空间,一条是指令,另一条是 _start 的数据(因为在编译的时候不能确定 _start 的值,而且也不能用 mov 指令来给 r0 赋一个 32bit 的常量,所以需要多出一个空间存放 _start 的真正数据,在这里就是 0x0000000c)。
因此可以看出,这个是绝对的寻址,不管这段代码在什么地方运行,它的结果都是 r0 = 0x0000000c。
本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/linweig/archive/2010/03/24/5411655.aspx
二、ldr和adr在使用标号表达式作为操作数的区别
http://blog.sina.com.cn/s/blog_4b5210840100c80i.html
ARM汇编有ldr指令以及ldr、adr伪指令,他门都可以将标号表达式作为操作数,下面通过分析一段代码以及对应的反汇编结果来说明它们的区别。
_start:
编译的时候设置 RO 为 0x30000000(好像有问题),下面是反汇编的结果:
1.ldr
r0, _start
:读取指定地址中的值
在这里_start是一个标号(是一个相对程序的表达式),汇编程序计算相对于 PC 的偏移量,并生成相对于 PC的前索引指令:ldr r0, [pc, #4]。执行指令后,r0 = 0xeafffffe。
2.adr
r0, _start
:将指定地址赋到r0中
ADD 指令或SUB 指令来实现该ADR 伪指令的功能,若不能用一条指令实现,则产生错误,
编译失败.
举例:
GPFCON
EQU
0x56000050
ldr
r0,=GPFCON
GPFCON
:标号
0x56000050
:标号的值
http://blog.chinaunix.net/u2/72383/showart_1071068.html
ldr的确是个复杂的指令,现总结一下:
ldr pc, _start_armboot
_start_armboot: .wor
d arm_startboot
这样label _start_armboot就在指令下方,因此肯定是合法的。
评论