如果我使用以下方式编译-O3
#include <vector>
int foo() {
std::vector<int> const v{17,2,3};
return v[0] + v[2];
}
我得到的组装是
foo():
mov eax, 20
ret
并且我了解当被其他 TU 调用时它将如何工作。
但是如果我用 进行编译-O0
,我会得到这个:
_Z3foov:
pushq %rbp
movq %rsp, %rbp
subq $112, %rsp
movl $17, -84(%rbp)
movl $2, -80(%rbp)
movl $3, -76(%rbp)
leaq -84(%rbp), %rax
movq %rax, -72(%rbp)
movq $3, -64(%rbp)
leaq -85(%rbp), %rcx
movq %rcx, -32(%rbp)
movq -32(%rbp), %rax
movq %rax, -8(%rbp)
movq -72(%rbp), %rsi
movq -64(%rbp), %rdx
leaq -56(%rbp), %rdi
callq _ZNSt6vectorIiSaIiEEC2ESt16initializer_listIiERKS0_
jmp .LBB0_1
.LBB0_1:
leaq -85(%rbp), %rax
movq %rax, -24(%rbp)
leaq -56(%rbp), %rdi
xorl %eax, %eax
movl %eax, %esi
callq _ZNKSt6vectorIiSaIiEEixEm
movl (%rax), %eax
movl %eax, -108(%rbp)
leaq -56(%rbp), %rdi
movl $2, %esi
callq _ZNKSt6vectorIiSaIiEEixEm
movq %rax, %rcx
movl -108(%rbp), %eax
addl (%rcx), %eax
movl %eax, -104(%rbp)
leaq -56(%rbp), %rdi
callq _ZNSt6vectorIiSaIiEED2Ev
movl -104(%rbp), %eax
addq $112, %rsp
popq %rbp
retq
movq %rax, %rcx
movl %edx, %eax
movq %rcx, -96(%rbp)
movl %eax, -100(%rbp)
leaq -85(%rbp), %rax
movq %rax, -16(%rbp)
movq -96(%rbp), %rdi
callq _Unwind_Resume@PLT
__clang_call_terminate:
pushq %rbp
movq %rsp, %rbp
callq __cxa_begin_catch@PLT
callq _ZSt9terminatev@PLT
.L.str:
.asciz "cannot create std::vector larger than max_size()"
.L.str.1:
.asciz "/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/15.0.0/../../../../include/c++/15.0.0/bits/stl_vector.h"
.L__PRETTY_FUNCTION__._ZNKSt6vectorIiSaIiEEixEm:
.asciz "const_reference std::vector<int>::operator[](size_type) const [_Tp = int, _Alloc = std::allocator<int>]"
.L.str.2:
.asciz "__n < this->size()"
DW.ref.__gxx_personality_v0:
.quad __gxx_personality_v0
在计算和_ZNKSt6vectorIiSaIiEEixEm
时调用的在哪里。但我真的不明白那些 2跳到了哪里。我的意思是,字符串只出现了 3 次,其中 2 次是,还有一次是这样的:v[0]
v[2]
call
_ZNKSt6vectorIiSaIiEEixEm
calls
.L__PRETTY_FUNCTION__._ZNKSt6vectorIiSaIiEEixEm:
.asciz "const_reference std::vector<int>::operator[](size_type) const [_Tp = int, _Alloc = std::allocator<int>]"
这怎么可能导致元素从向量中拉出来?
或者换句话说,如果我将上述 TU 与另一个调用它并返回值的 TU 链接起来,例如
int foo();
int main() {
return foo();
}
我在哪里可以找到执行的代码v[0]
?v[2]
如果它不在程序集中,它甚至不在二进制文件中,对吗?那么程序如何运行?
过滤器: Godbolt 编译器资源管理器中默认启用库函数。
在过滤器下拉菜单中禁用该选项,会出现
_ZNKSt6vectorIiSaIiEEixEm
. 的定义(std::vector<int, std::allocator<int>>::operator[](unsigned long) const:
启用名称分解)。(其定义包括边界检查和打印函数名称的断言失败,因此
.L__PRETTY_FUNCTION__._ZNKSt6vectorIiSaIiEEixEm:
库函数过滤器没有丢弃该定义)。