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到该数据缓冲区单元的LDR 指令,从而将该地址值读取到寄存器总,这时,要求该数据缓冲区到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, 则编译器自动在(整个)程序末尾声名,但这样偏移有可能太大而编译不通过. ============================================================================ |