| // RUN: %clang_cc1 -std=c++14 -fcoroutines-ts -triple=x86_64-pc-windows-msvc18.0.0 -emit-llvm %s -o - -fexceptions -fcxx-exceptions -disable-llvm-passes | FileCheck %s |
| // RUN: %clang_cc1 -std=c++14 -fcoroutines-ts -triple=x86_64-unknown-linux-gnu -emit-llvm -o - %s -fexceptions -fcxx-exceptions -disable-llvm-passes | FileCheck --check-prefix=CHECK-LPAD %s |
| |
| #include "Inputs/coroutine.h" |
| |
| namespace coro = std::experimental::coroutines_v1; |
| |
| namespace std { |
| using exception_ptr = int; |
| exception_ptr current_exception(); |
| } |
| |
| struct coro_t { |
| struct promise_type { |
| coro_t get_return_object() { |
| coro::coroutine_handle<promise_type>{}; |
| return {}; |
| } |
| coro::suspend_never initial_suspend() { return {}; } |
| coro::suspend_never final_suspend() { return {}; } |
| void return_void(){} |
| void unhandled_exception() noexcept; |
| }; |
| }; |
| |
| struct Cleanup { ~Cleanup(); }; |
| void may_throw(); |
| |
| coro_t f() { |
| Cleanup x; |
| may_throw(); |
| co_return; |
| } |
| |
| // CHECK: @"\01?f@@YA?AUcoro_t@@XZ"( |
| // CHECK: invoke void @"\01?may_throw@@YAXXZ"() |
| // CHECK: to label %{{.+}} unwind label %[[EHCLEANUP:.+]] |
| // CHECK: [[EHCLEANUP]]: |
| // CHECK: %[[INNERPAD:.+]] = cleanuppad within none [] |
| // CHECK: call void @"\01??_DCleanup@@QEAAXXZ"( |
| // CHECK: cleanupret from %[[INNERPAD]] unwind label %[[CATCHSW:.+]] |
| // CHECK: [[CATCHSW]]: |
| // CHECK: %[[CATCHSWTOK:.+]] = catchswitch within none [label %[[CATCH:.+]]] unwind label |
| // CHECK: [[CATCH]]: |
| // CHECK: %[[CATCHTOK:.+]] = catchpad within [[CATCHSWTOK:.+]] |
| // CHECK: call void @"\01?unhandled_exception@promise_type@coro_t@@QEAAXXZ" |
| // CHECK: catchret from %[[CATCHTOK]] to label %[[CATCHRETDEST:.+]] |
| // CHECK: [[CATCHRETDEST]]: |
| // CHECK-NEXT: br label %[[TRYCONT:.+]] |
| // CHECK: [[TRYCONT]]: |
| // CHECK-NEXT: br label %[[COROFIN:.+]] |
| // CHECK: [[COROFIN]]: |
| // CHECK-NEXT: invoke void @"\01?final_suspend@promise_type@coro_t@@QEAA?AUsuspend_never@coroutines_v1@experimental@std@@XZ"( |
| |
| // CHECK-LPAD: @_Z1fv( |
| // CHECK-LPAD: invoke void @_Z9may_throwv() |
| // CHECK-LPAD: to label %[[CONT:.+]] unwind label %[[CLEANUP:.+]] |
| // CHECK-LPAD: [[CLEANUP]]: |
| // CHECK-LPAD: call void @_ZN7CleanupD1Ev(%struct.Cleanup* %x) #2 |
| // CHECK-LPAD: br label %[[CATCH:.+]] |
| |
| // CHECK-LPAD: [[CATCH]]: |
| // CHECK-LPAD: call i8* @__cxa_begin_catch |
| // CHECK-LPAD: call void @_ZN6coro_t12promise_type19unhandled_exceptionEv(%"struct.coro_t::promise_type"* %__promise) #2 |
| // CHECK-LPAD: invoke void @__cxa_end_catch() |
| // CHECK-LPAD-NEXT: to label %[[CATCHRETDEST:.+]] unwind label |
| // CHECK-LPAD: [[CATCHRETDEST]]: |
| // CHECK-LPAD-NEXT: br label %[[TRYCONT:.+]] |
| // CHECK-LPAD: [[TRYCONT]]: |
| // CHECK-LPAD-NEXT: br label %[[COROFIN:.+]] |
| // CHECK-LPAD: [[COROFIN]]: |
| // CHECK-LPAD-NEXT: invoke void @_ZN6coro_t12promise_type13final_suspendEv( |