assembly - 什么是 cfi_adjust_cfa_offset 和 cfi_rel_offset?

标签 assembly arm instructions instruction-set eabi

我目前正在尝试了解glibc的syscall函数的内部代码。下面是代码(取自 from here )。

/* In the EABI syscall interface, we don't need a special syscall to
   implement syscall().  It won't work reliably with 64-bit arguments
   (but that is true on many modern platforms).  */
ENTRY (syscall)
        mov        ip, sp
        push        {r4, r5, r6, r7}
        cfi_adjust_cfa_offset (16)
        cfi_rel_offset (r4, 0)
        cfi_rel_offset (r5, 4)
        cfi_rel_offset (r6, 8)
        cfi_rel_offset (r7, 12)
        mov        r7, r0
        mov        r0, r1
        mov        r1, r2
        mov        r2, r3
        ldmfd        ip, {r3, r4, r5, r6}
        swi        0x0
        pop        {r4, r5, r6, r7}
        cfi_adjust_cfa_offset (-16)
        cfi_restore (r4)
        cfi_restore (r5)
        cfi_restore (r6)
        cfi_restore (r7)
        cmn        r0, #4096
        it        cc
        RETINSTR(cc, lr)
        b        PLTJMP(syscall_error)
PSEUDO_END (syscall)

对传递系统调用号和参数的代码有了一些了解,这是syscall函数的核心功能。但我不明白 cfi_adjust_cfa_offset 指令、cfi_rel_offset 指令和 cfi_restore 指令的作用。我知道这些指令与 syscall 函数的功能关系不大。但是,我仍然想知道这些指令的作用。

谢谢。

最佳答案

这些不是说明,而是 assembler directives (通常它们应该以 . 开头,但这里是 they are macros ,可能是为了处理汇编程序中的差异)。它们告诉汇编器对特殊的元数据进行编码,这有助于调试器和异常处理程序正确地展开堆栈。通常它们是由编译器发出的,因此您不会经常看到它们,但它们在低级代码或像这里这样的手写汇编中尤其重要。

让我们看一下代码。首先,一些背景知识。

CFA 代表“规范帧地址”,默认情况下等于调用站点 ( see here ) 处的 sp 值。在 ARM 上,调用不会将返回地址压入堆栈,因此函数入口处的 sp 是相同的。

mov ip, sp

这会将 sp 值(即 CFA,因此指向任何不适合 r0-r3 寄存器的其他参数)复制到 ip(又名 r12)注册。

推 {r4,r5,r6, r7}

这保存了很快就会被修改但假定不会被调用修改的寄存器(非 volatile 寄存器)。推送改变了sp值4*4=16字节,不再等于CFA。

cfi_adjust_cfa_offset(16)

这会发出一个操作码,告诉调试器 CFA 距假定的 CFA (sp) 的偏移量为 16。

cfi_rel_offset(r4, 0)

这告诉调试器,r4 的原始值可以在新调整的 CFA 的偏移 0 处找到。以下指令描述了其他三个保存的寄存器。

    mov        r7, r0
    mov        r0, r1
    mov        r1, r2
    mov        r2, r3
    ldmfd        ip, {r3, r4, r5, r6}
    swi        0x0

这会按照 EABI 调用约定所期望的方式设置系统调用参数(r7 中的系统调用编号,r0-r6 中的第一个参数),然后调用系统调用指令

    pop        {r4, r5, r6, r7}
    cfi_adjust_cfa_offset (-16)
    cfi_restore (r4)
    cfi_restore (r5)
    cfi_restore (r6)
    cfi_restore (r7)

这里我们恢复之前保存的寄存器,将CFA调整回-16(因为sppop指令改变,并指定寄存器已恢复到其原始值)。

关于assembly - 什么是 cfi_adjust_cfa_offset 和 cfi_rel_offset?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51962243/

相关文章:

c - 如何处理这个: selected processor does not support `qadd16 r1,r1,r0'

caching - clflush 指令是否仅从 1 级缓存刷新 block ?

c - 在 C 中使用内联汇编 : Assigning labels in data segment

arrays - 我的程序不会打印数组中第一个和最后一个数字的总和

c - 外部 C 代码无法从两阶段 x86 引导加载程序执行?

assembly - 将CPU寄存器中的所有位有效设置为1

c - 为什么 LD_PRELOAD 与系统调用一起工作?

assembly - 一般而言,对于 x86 的 SETCC,CPU 指令集背后的基本原理是什么?

assembly - 存储指令中数据总线的状态

c - 使用指令指针的偏移量反汇编目标文件