给出以下代码:
#include <stdexcept>
#include <string>
using namespace std;
class exception_base : public runtime_error {
public:
exception_base()
: runtime_error(string()) { }
};
class my_exception : public exception_base {
public:
};
int main() {
throw my_exception();
}
这在GNU / Linux和Windows上运行良好,并且在OSX 10.11.4版的最新更新之前也可以在OSX上运行。好的,我的意思是,既然没有什么能捕捉到异常,std::terminate
就被称为。
但是,在使用clang(LLVM 7.3.0)的OSX 10.11.4上,程序因分段错误而崩溃。堆栈跟踪没有帮助:
Program received signal SIGSEGV, Segmentation fault.
0x0000000100000ad1 in main () at test.cpp:17
17 throw my_exception();
(gdb) bt
#0 0x0000000100000ad1 in main () at test.cpp:17
(gdb)
valgrind对此也无话可说:
==6500== Process terminating with default action of signal 11 (SIGSEGV)
==6500== General Protection Fault
==6500== at 0x100000AD1: main (test.cpp:17)
我认为代码不会以任何方式违反标准。我在这里想念什么吗?
请注意,即使我throw
在代码周围添加了try-catch,由于SIGSEGV,代码仍然会崩溃。
如果您查看反汇编,您将看到SSEmovaps
指令上发生了通用保护(GP)异常:
a.out`main: 0x100000ad0:pushq%rbp 0x100000ad1:movq%rsp,%rbp 0x100000ad4:subq $ 0x20,%rsp 0x100000ad8:movl $ 0x0,-0x4(%rbp) 0x100000adf:movl $ 0x10,%eax 0x100000ae4:move%eax,%edi 0x100000ae6:callq 0x100000dea; __cxa_allocate_exception的符号存根 0x100000aeb:movq%rax,%rdi 0x100000aee:xorps%xmm0,%xmm0 -> 0x100000af1:移动%xmm0,(%rax) 0x100000af4:movq%rdi,-0x20(%rbp) 0x100000af8:movq%rax,%rdi 0x100000afb:呼叫0x100000b40; my_exception :: my_exception ...
在调用my_exception :: my_exception()构造函数之前,使用一条movaps
指令将__cxa_allocate_exception(size_t)返回的内存块清零。但是,不能保证此指针(在我的情况下为0x0000000100103498)是16字节对齐的。当movaps
指令的源或目标操作数是内存操作数时,该操作数必须在16字节边界上对齐,否则会生成GP异常。
临时解决问题的一种方法是不使用SSE指令(-mno-sse
)进行编译。这不是理想的解决方案,因为SSE指令可以提高性能。
我认为这与http://reviews.llvm.org/D18479有关:
r246985进行了更改,以使异常对象具有更高的对齐方式,理由是Itanium表示_Unwind_Exception应该“双字”对齐,并且通常在声明结构的同时
__attribute__((aligned))
保证16字节对齐。事实证明,libc ++ abi不会使用声明结构,__attribute__((aligned))
因此只能保证在32位和64位平台上进行8字节对齐。在某些情况下,当后端发出需要16字节对齐的SIMD存储指令(例如movaps
)时,这会导致崩溃。此修补程序使ItaniumCXXABI :: getAlignmentOfExnObject返回达尔文的8字节对齐以修复崩溃。
..哪个补丁于2016年3月31日提交,名称为r264998。
还有https://llvm.org/bugs/show_bug.cgi?id=24604和https://llvm.org/bugs/show_bug.cgi?id=27208看起来相关。
更新我安装了Xcode 7.3.1(昨天发布),问题似乎已经解决;现在生成的程序集为:
a.out`main: 0x100000ac0:pushq%rbp 0x100000ac1:movq%rsp,%rbp 0x100000ac4:subq $ 0x20,%rsp 0x100000ac8:movl $ 0x0,-0x4(%rbp) 0x100000acf:movl $ 0x10,%eax 0x100000ad4:%eax,%edi 0x100000ad6:callq 0x100000dea;__cxa_allocate_exception的符号存根 0x100000adb:movq%rax,%rdi 0x100000ade:movq $ 0x0,0x8(%rax) 0x100000ae6:movq $ 0x0,(%rax) 0x100000aed:movq%rdi,-0x20(%rbp) 0x100000af1:movq%rax,%rdi 0x100000af4:呼叫0x100000b40; my_exception :: my_exception ...
本文收集自互联网,转载请注明来源。
如有侵权,请联系[email protected] 删除。
我来说两句