第21部分- Linux ARM汇编adrp指令

3,635 阅读2分钟

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/90a37afe6923486eb5ca5eed9e557743~tplv-k3u1fbpfcp-zoom-1.image

ADR指令作用:小范围的地址读取指令。ADR 指令将基于PC 相对偏移的地址值读取到寄存器中。

原理:将有符号的21位的偏移,加上PC, 结果写入到通用寄存器,可用来计算+/-1MB范围的任意字节的有效地址。

 

ADRP作用:以页为单位的大范围的地址读取指令,这里的P就是page的意思。

原理:符号扩展一个21位的offset(immhi+immlo),  向左移动12位,PC的值的低12位清零,然后把这两者相加,结果写入到Xd寄存器,用来得到一块含有lable的4KB对齐内存区域的base地址(也就是说lable所在的地址,一定落在这个4KB的内存区域里,指令助记符里Page也就是这个意思), 可用来寻址 +/- 4GB的范围(2^33次幂)。

 通俗来讲,ADRP指令就是先进行PC+imm(偏移值)然后找到lable所在的一个4KB的页,然后取得label的基址,再进行偏移去寻址。

错误

hpre_dev_api.o(.text+0x7c): unresolvable R_AARCH64_ADR_PREL_PG_HI21 relocation against symbol `stderr@@GLIBC_2.17'

通过静态库编译成动态库。所有静态库都需要加上-fPIC。

答:通过静态库编译成动态库。所有静态库都需要加上-fPIC。

反汇编得到:

0000000000000068 <_print_buffer>:

      68:       a9bc7bfd        stp     x29, x30, [sp,#-64]!

      6c:       910003fd        mov     x29, sp

      70:       f90017e0        str     x0, [sp,#40]

      74:       f90013e1        str     x1, [sp,#32]

      78:       b9001fe2        str     w2, [sp,#28]

7c:       90000000        adrp    x0, 0 <stderr>

      80:       91000000        add     x0, x0, #0x0

      84:       f9400006        ldr     x6, [x0]

      88:       b9401fe5        ldr     w5, [sp,#28]

      8c:       f94017e4        ldr     x4, [sp,#40]

      90:       528004a3        mov     w3, #0x25     

 

答:

./configure CFLAGS="-fPIC"  CXXFLAGS="-fPIC"