似乎每一行后面都有一个.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
我才明白这些的用意。
5条答案
按热度按时间ukqbszuj1#
要禁用这些选项,请使用gcc选项
也可能需要
-fno-dwarf2-cfi-asm
。hjzp0vay2#
我有一种感觉,它代表调用帧信息,是一个管理调用帧的GNU AS扩展。
在某些体系结构中,异常处理必须用调用帧信息指令来管理。这些指令在程序集中用来指导异常处理。如果由于某种原因(例如代码库的可移植性)GCC生成的异常处理信息不足,这些指令在POWER上的Linux上可用。
看起来这些是在某些平台上根据异常处理的需要生成的。
如果您希望禁用这些功能,请参阅David's answer。
w7t8yxp53#
CFI指令用于调试。它允许调试器展开堆栈。例如:如果过程A调用过程B,过程B又调用公共过程C,则过程C失败。现在您想知道实际上谁调用了C,然后您可能想知道谁调用了B。
调试器可以通过使用堆栈指针(%rsp)和寄存器%rbp来展开此堆栈,但它需要知道如何找到它们。这就是CFI指令的作用所在。
因此这里的最后一行告诉它“调用帧地址”现在在寄存器6(%rbp)中
zf2sa74q4#
要禁用这些异常,g++需要
-fno-exceptions
沿着前面提到的-fno-asynchronous-unwind-tables
,前提是不使用异常。v7pvogib5#
好吧,它只是代表控制流完整性。它们本质上是传递给调试器和其他工具的信息项,用于描述程序的预期流。