assembly Gnu Assembler(GAS)中的CFI指令有何用途?

uurv41yg  于 2022-11-13  发布在  其他
关注(0)|答案(5)|浏览(174)

似乎每一行后面都有一个.CFI指令,而且这些指令的种类也很广泛,例如.cfi_startproc.cfi_endproc等。更多信息请点击此处。

.file   "temp.c"
    .text
.globl main
    .type   main, @function
main:
.LFB0:
    .cfi_startproc
    pushq   %rbp
    .cfi_def_cfa_offset 16
    movq    %rsp, %rbp
    .cfi_offset 6, -16
    .cfi_def_cfa_register 6
    movl    $0, %eax
    leave
    ret
    .cfi_endproc
.LFE0:
    .size   main, .-main
.globl func
    .type   func, @function
func:
.LFB1:
    .cfi_startproc
    pushq   %rbp
    .cfi_def_cfa_offset 16
    movq    %rsp, %rbp
    .cfi_offset 6, -16
    .cfi_def_cfa_register 6
    movl    %edi, -4(%rbp)
    movl    %esi, %eax
    movb    %al, -8(%rbp)
    leave
    ret
    .cfi_endproc
.LFE1:
    .size   func, .-func
    .ident  "GCC: (Ubuntu 4.4.1-4ubuntu9) 4.4.1"
    .section    .note.GNU-stack,"",@progbits

我才明白这些的用意。

ukqbszuj

ukqbszuj1#

要禁用这些选项,请使用gcc选项

-fno-asynchronous-unwind-tables

也可能需要-fno-dwarf2-cfi-asm

hjzp0vay

hjzp0vay2#

我有一种感觉,它代表调用帧信息,是一个管理调用帧的GNU AS扩展。
在某些体系结构中,异常处理必须用调用帧信息指令来管理。这些指令在程序集中用来指导异常处理。如果由于某种原因(例如代码库的可移植性)GCC生成的异常处理信息不足,这些指令在POWER上的Linux上可用。
看起来这些是在某些平台上根据异常处理的需要生成的。
如果您希望禁用这些功能,请参阅David's answer

w7t8yxp5

w7t8yxp53#

CFI指令用于调试。它允许调试器展开堆栈。例如:如果过程A调用过程B,过程B又调用公共过程C,则过程C失败。现在您想知道实际上谁调用了C,然后您可能想知道谁调用了B。
调试器可以通过使用堆栈指针(%rsp)和寄存器%rbp来展开此堆栈,但它需要知道如何找到它们。这就是CFI指令的作用所在。

movq    %rsp, %rbp
.cfi_def_cfa_register 6

因此这里的最后一行告诉它“调用帧地址”现在在寄存器6(%rbp)中

zf2sa74q

zf2sa74q4#

要禁用这些异常,g++需要-fno-exceptions沿着前面提到的-fno-asynchronous-unwind-tables,前提是不使用异常。

v7pvogib

v7pvogib5#

好吧,它只是代表控制流完整性。它们本质上是传递给调试器和其他工具的信息项,用于描述程序的预期流。

相关问题