寻址方式是针对源操作数来说的
6.1 立即数寻址
- 源操作数是立即数
- 立即数:操作码以 # 号开头的数字为立即数
- 立即数寻址: MOV R0, #0x300
- 伪指令: LDR R0,=0x12345678
- 注意:
- 立即数是 8 位数据存储,用 X 表示(0~255),4 位存移位的次数,用 Y 表示(0~15),立即数 = X 循环右移 2 * Y 个位
- 立即数 0xf200 是由 0xcf2 间接表示的,即是由 8 位的 F2 循环右移 24 位(2 * 12)得到 X = 0xf2; Y = 0xC
6.2 寄存器寻址
- MOV R0,R1
- 源操作数是 寄存器
6.3 寄存器移位寻址
- 将寄存器寻址的源操作数进行移位
- MOV R0, R2, LSL, #3
- 将 R2 左移 3 位后,赋值给 R0
6.4 寄存器间接寻址
- ARM 中由 L/S 结构,即 load/store
- LOAD 是将内存的数据载入到寄存器中
- STROE 是将寄存器中的数据存储到内存中
- 指令:
- LDR R0,{R1} 把 {R1} 中的值取出来放入到 R0 中
- STR R0,{R1} 把 R0 中的值取出来放入到 R1 中
- {R1} 表示取 R1 存放的地址中的数据,换成 C 语言,即 *R1
1 MOV R1, #0x40000000 2 LDR R0, {R1} 3 STR R0, {R1} 4 5 //若 R1 中的数据 0x40000000 所代表的数据是 0x55,则 R0 的值为 0x55
6.5 基址变址寻址
- MOV R0, #44
- MOV R1, #0x40000008
- STR R0, [R1, #-4]
- [R1, #-4] 的意思是将 R1 存储的地址值 - 4
- #-4 表示偏移量
- STR R0, [R1, #-4]!
- ! 表示回写,即 C 语言中的 i--,上一句则先把 R1 的地址值赋值到 R0 中,再将 R0 中的值 - 4
- 上一句相当于 STR R0,[R1], #-4
6.6 多寄存器寻址
- STMIA R0!, {R1 - R3, R5}
- 将 R1 R2 R3 R5 中的值,存放在以 R0 为起使的地址空间中
- 大括号中的内容表示寄存器中的值,R0 对应的是存储器的地址
- STM: 操作多个存储器的值
- 对应的命令是 LDMIA
- I/D(increase/decrease) A/B(after/before)
6.7 堆栈寻址
- 进行栈的操作,栈的寄存器是 R13,即 SP 寄存器
- STMFD SP!, {R1-R3} 压栈,寄存器号大的先入栈,与书写顺序没关系
- LDMFD SP!, {R1-R3} 出栈 组合: F(FULL)/E(EMPTY) I/D
- 一般都写成 FD,因为 ARM 的地址空间是满递减的
- 满堆栈:堆栈指针指向最后压入的有效数据项
- 空堆栈:堆栈指针指向下一个待压入数据的空位置
- MOV SP, #0x40000010
- STMFD SP!, {R1-R4}
- 即将 R4 R3 R2 R1 的值存入 0x4000000c 0x40000008 0x40000004 0x40000000
6.8 相对寻址
- 51 中用的 jmp 和 call 指令
- 相对寻址就是一个跳转,相对寻址是相对于 PC 而言的,跳转指令:B BL BLX BX
- B:跳转指令
- BL:带返回的跳转指令
- BLX:带返回和状态切换的跳转指令
- BX:带状态切换的跳转指令