50 lines
1.7 KiB
C++
50 lines
1.7 KiB
C++
|
// RUN: %clang_cc1 %s -triple=x86_64-apple-darwin10 -emit-llvm -o - -fcxx-exceptions -fexceptions -std=c++11 | FileCheck %s
|
||
|
|
||
|
// rdar://11904428
|
||
|
// Ensure that we call __cxa_begin_catch before calling
|
||
|
// std::terminate in a noexcept function.
|
||
|
namespace test0 {
|
||
|
void foo();
|
||
|
|
||
|
struct A {
|
||
|
A();
|
||
|
~A();
|
||
|
};
|
||
|
|
||
|
void test() noexcept {
|
||
|
A a;
|
||
|
foo();
|
||
|
}
|
||
|
}
|
||
|
// CHECK-LABEL: define void @_ZN5test04testEv()
|
||
|
// CHECK: [[EXN:%.*]] = alloca i8*
|
||
|
// This goes to the terminate lpad.
|
||
|
// CHECK: invoke void @_ZN5test01AC1Ev(
|
||
|
// This goes to the cleanup-and-then-terminate lpad.
|
||
|
// CHECK: invoke void @_ZN5test03fooEv()
|
||
|
// Destructors don't throw by default in C++11.
|
||
|
// CHECK: call void @_ZN5test01AD1Ev(
|
||
|
// Cleanup lpad.
|
||
|
// CHECK: [[T0:%.*]] = landingpad
|
||
|
// CHECK-NEXT: catch i8* null
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i8*, i32 } [[T0]], 0
|
||
|
// CHECK-NEXT: store i8* [[T1]], i8** [[EXN]]
|
||
|
// (Calling this destructor is not technically required.)
|
||
|
// CHECK: call void @_ZN5test01AD1Ev(
|
||
|
// CHECK-NEXT: br label
|
||
|
// The terminate landing pad jumps in here for some reason.
|
||
|
// CHECK: [[T0:%.*]] = landingpad
|
||
|
// CHECK-NEXT: catch i8* null
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i8*, i32 } [[T0]], 0
|
||
|
// CHECK-NEXT: call void @__clang_call_terminate(i8* [[T1]])
|
||
|
// CHECK-NEXT: unreachable
|
||
|
// The terminate handler chained to by the cleanup lpad.
|
||
|
// CHECK: [[T0:%.*]] = load i8** [[EXN]]
|
||
|
// CHECK-NEXT: call void @__clang_call_terminate(i8* [[T0]])
|
||
|
// CHECK-NEXT: unreachable
|
||
|
|
||
|
// CHECK-LABEL: define linkonce_odr hidden void @__clang_call_terminate(
|
||
|
// CHECK: call i8* @__cxa_begin_catch(
|
||
|
// CHECK-NEXT: call void @_ZSt9terminatev()
|
||
|
// CHECK-NEXT: unreachable
|