128

每行之后似乎都有一个 .CFI 指令,并且这些示例的种类繁多.cfi_startproc.cfi_endproc等等。 更多here

    .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

我没有明白这些的目的。

4

5 回答 5

150

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

-fno-asynchronous-unwind-tables

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

于 2013-04-15T03:21:32.590 回答
72

我有一种感觉,它代表呼叫帧信息,是管理呼叫帧的 GNU AS 扩展。来自DeveloperWorks

在某些体系结构上,必须使用调用帧信息指令来管理异常处理。这些指令在程序集中用于指导异常处理。如果由于某种原因(例如代码库的可移植性),GCC 生成的异常处理信息不足,这些指令在 Linux on POWER 上可用。

看起来这些是在某些平台上根据异常处理的需要生成的。

如果您希望禁用这些,请参阅David 的回答

于 2010-03-27T12:24:45.383 回答
31

CFI 指令用于调试。它允许调试器展开堆栈。例如:如果过程 A 调用过程 B,过程 B 然后调用公共过程 C。过程 C 失败。您现在想知道谁实际呼叫了 C,然后您可能想知道谁呼叫了 B。

调试器可以通过使用堆栈指针 (%rsp) 并注册 %rbp 来展开此堆栈,但是它需要知道如何找到它们。这就是 CFI 指令的用武之地。

movq    %rsp, %rbp
.cfi_def_cfa_register 6

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

于 2014-05-08T00:20:31.363 回答
4

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

于 2020-05-04T03:52:24.227 回答
-1

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

于 2021-11-12T05:32:38.050 回答