adr用来加载地址,例如adr r0,var1
ldr用来加载地址处的内容,例如ldr r0,var1
1、LDR R0,=0X3FF5000 ;伪指令,把0X3FF5000这个地址送给R0
2、LDR R0,0XFF ; 把立即数0xff送给R0
3、LDR R0,=&FF ; &相当于0X
4、BIC R0,R0,#%1011 ;.#表示立即数,%表示二进制
LDR R1,=0x3ff5000 ;伪指令 R1=0X3FF5000
LDR R1,0x3ff5000 ;存储器访问指令 R1= [0x3ff5000]
=======================================================================
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立即数.
----------------------------------------------
MOV加载8位立即数
8位立即数即第2操作数,必须可由一个8位常数循环移位偶数位得到,如0xf0000000,0xf00000001都是合
法的。
========================================================================
表示数据回写
例如:
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后索引,后加
===========================================================================
Var EQU 0x1000
ldr r0, =Var
EQU生成的实际是一个标号(定义变量?),用于相对积存器寻址,与PC相对寻址相对。用EQU生成的标号
用于数据段寻址。
关于DCD指令,注意与EQU不同,相应的ldr指令用法也不同. 定义一段内存区
semantec
Label DCD expr
例如:
ldr pc,Vector
....
Vector DCD restVector
restVector
============================================================================
LTORG与LDR
LTORG是与LDR联合使用的literal pool,可以在函数尾部声明,这样相对PC偏移最少,如果不用LTORG,
则编译器自动在(整个)程序末尾声名,但这样偏移有可能太大而编译不通过.
=============================================================
文章评论(0条评论)
登录后参与讨论