ARM几条汇编指令的比较

从网上找到一些关于ARM伪指令的资料,现整理如下:

1. LDR
LDR R0, =0X3FF5000 ;伪指令,把0X3FF5000这个地址送给R0
LDR R0, 0XFF       ; 把立即数0xff送给R0
LDR R0, =&FF       ; &相当于0X
BIC R0, R0, #%1011 ;.#表示立即数,%表示二进制

LDR R1, =0x3ff5000    ;伪指令 R1=0X3FF5000
LDR R1, 0x3ff5000     ;存储器访问指令 R1= [0x3ff5000]

2. adr与ldr比较
adr r0, InitSystem  ;
ldr r1, =InitSystem ;
伪指令adr r0,InitSystem 编译时汇编成:sub r0,PC,#offset to InitSystem

LDR r1,=InitSystem ,这种方式读取的地址值在连接时已经被固定了,这种代码
不是位置无关的。遇到LDR伪指令时,汇编编译器将该地址值保存到一个缓冲区(l
iteral pool)中,然后将该LDR 伪指令处理成一条基于PC到该数据缓冲区单元的LD
R 指令,从而将该地址值读取到寄存器总,这时,要求该数据缓冲区到PC的距离小
于4KB。如果该目标地址值为一个外部地址值或者不在本数据段内,则汇编译器在目
标文件中插入一个地址重定位伪操作,当连接器进行连接时生成该地址值。
LDR r1,=InitSystem 汇编成:LDR R1,[PC,#offset to Litpool1]
------------------------------------------------------
adr用来加载地址,例如adr r0,var1
ldr用来加载地址处的内容,例如ldr r0,var1
上面的这种语法只能从.text段中加载
但ldr r0,=var1可从任意段中加载地址

ldr有伪指令和非伪指令,伪指令后面的立即数前加=
ADR在编译时会被替换成一条add或者sub指令,如果替换不了则报错。相对PC寻址
ADRL会被替换成两条指令,替换不了报错误。相对PC或者积存器寻址
这两条指令依据立即数的对齐方式不同,允许的立即数范围也不同。
LDR则是产生文字池的方式加载常量,基于PC的相对寻址,专用加载32bit立即数.

通过反汇编可以很容易看出LDR和ADR区别:
假设入口点地址为0x8000
AREA LDRlabel, CODE, READONLY
ENTRY ; Mark first instruction to execute
start
BL func1 ; Branch to first subroutine
stop
MOV r0, #0x18 ; angel_SWIreason_ReportException
LDR r1, =0x20026 ; ADP_Stopped_ApplicationExit
LDR r1,=0xff
SWI 0x123456 ; ARM semihosting SWI
func1
LDR r0, =start ; => LDR R0,[PC, #offset to Litpool 1]
ADR r2,start
LDR r1, =Darea +12 ; => LDR R1,[PC, #offset to Litpool 1]
ADR r3,Darea+12
MOV pc,lr ; Return
LTORG ; Literal Pool 1
Darea SPACE 8000 ; Clears a 8000 byte area of memory,
; starting at the current location,
; to zero.
END
反汇编后:
start [0xeb000003] bl func1
stop [0xe3a00018] mov r0,#0x18
00008008 [0xe59f1018] ldr r1,0x00008028 ; = #0x00020026
0000800c [0xe3a010ff] mov r1,#0xff
00008010 [0xef123456] swi 0x123456
func1 [0xe59f0010] ldr r0,0x0000802c ; = #0x00008000
00008018 [0xe24f2020] sub r2,pc,#0x20 ; #0x8000
0000801c [0xe59f100c] ldr r1,0x00008030 ; = #0x00008040
00008020 [0xe28f3018] add r3,pc,#0x18 ; #0x8040
00008024 [0xe1a0f00e] mov pc,r14
00008028 [0x00020026] dcd 0x00020026&...
0000802c [0x00008000] dcd 0x00008000....
00008030 [0x00008040] dcd 0x00008040@...
Darea [0x00000000] dcd 0x00000000....
00008038 [0x00000000] dcd 0x00000000....
0000803c [0x00000000] dcd 0x00000000....
00008040 [0x00000000] dcd 0x00000000....
00008044 [0x00000000] dcd 0x00000000....
……….
3. MOV
MOV加载8位立即数
8位立即数即第2操作数,必须可由一个8位常数循环移位偶数位得到,如0xf0000000,0xf00000001都是合法的

4. 数据回写
例如:
ldr r0,[r1]!
stmdb sp!,{r0,r4}
!用于前索引方式中表示数据回写,例如:ldr r0,[r1,#4]!
后索引方式不用!, 数据始终回写,例如:ldr r0,[r1],#4
对堆栈方式中用!, 表示堆栈自动增加或者减少

访问内存的LDR/STR指令索引方式
ldr r0,[r1,#4]前索引, 先加
ldr r0,[r1],#4后索引, 后加

5. DCD和SPACE
DCD
分配一个地址并初始化为指定的表达式
如 DCD 0x8000
就是分配一个32位的地址,其内容是0x8000
如果用上标号label,那么这个label相当于一个变量,如
label DCD 0X8000
这样引用label就是使用了0x8000这个数值。
SPACE
分配一段指定长度的内存空间并初始化为0

label SPACE 0x8000
就是分配一段长度为0x8000的空间,并初始化为0
Data1 DCD 1,2,3
Data2 SPACE 12
反汇编后的结果:
Data1 [0x00000001] dcd 0x00000001 ....
00008040 [0x00000002] dcd 0x00000002 ....
00008044 [0x00000003] dcd 0x00000003 ....
Data2 [0x00000000] dcd 0x00000000 ....
0000804c [0x00000000] dcd 0x00000000 ....
00008050 [0x00000000] dcd 0x00000000 ....

6. LTORG与LDR
LTORG是与LDR联合使用的literal pool,可以在函数尾部声明, 这样相对PC偏移最少, 如果不用LTORG, 则编译器自动在(整个)程序末尾声名,但这样偏移有可能太大而编译不通过.

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值