汇编中的LDR,MOV,ADR指令的区别

在看BootLoader的第一个引导阶段时,细心的同学会发现代码中有使用mov r1,#0x3;ldr r0 ,=INTMSK;adr r0, _start指等令来获取一个值给指定的寄存器,那为什么要使用不同的指令来获取值呢,而不直接使用MOV指令呢? 解释前首先把这几个指令的功能简单的介绍下。

  • MOV是一条数据传送指令 格式:MOV {< cond >} { s } < Rd >, < shifter_operand >,作用是把< shift_operand >表示的数据传送到目标寄存器< Rd >中
  • LDR 伪指令,格式:LDR R0, =XXX。作用是把数值传送到R0目标寄存器中。
  • ADR R0 ,_start 伪指令的作用是把_start 的值传送到目标寄存器R0中。
    即这三条指令都是获取数值传送到目标寄存器中,但是后面两条是伪指令,并不是真正的汇编语言,编译器看到的时候会转换成真正的汇编语言。
    MOV 指令后面的操作数据可以是寄存器,立即数(不是所有的数都是立即数)。
    LDR 指令后面接的数可以随意的,可以不是立即数。
    而ADR指令是把_start的地址传输到R0中,转换为真的汇编指令类似于:SUB R0,PC,#172。是相对于PC偏移量的来寻址的,即指令可以通到PC指令来寻找到,在PC的前后小范围内。这样就生成了位置无关代码,即不管程序移动什么位置,都能正确获取到_start的指令地址。而要是写成了MOV R0,_start则r0中的值会随着程序连接地址而变。这个指令是在代码重定位中使用到,刚上电的时候,引导代码在物理地址的0处。需要把代码搬移到链接器指定的物理位置中。

ADR指令主要用于生成地址无关代码,以及不用判断一个数是否是立即数;LDR伪指令的作用是后面接的数据可以不是立即数;而MOV指令后面接数值的话只能是立即数,MOV后还可以接其它的形式的操作数。

发布了35 篇原创文章 · 获赞 1 · 访问量 1870

猜你喜欢

转载自blog.csdn.net/lzj_linux188/article/details/103645814