| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py |
| ; RUN: opt -mtriple=wasm32-unknown-unknown -S --passes=expand-variadics --expand-variadics-override=optimize < %s | FileCheck %s -check-prefixes=CHECK |
| ; RUN: not --crash opt -mtriple=wasm32-unknown-unknown -S --passes=expand-variadics --expand-variadics-override=lowering < %s 2>&1 | FileCheck %s -check-prefixes=ERROR |
| ; REQUIRES: webassembly-registered-target |
| target datalayout = "e-m:e-p:32:32-p10:8:8-p20:8:8-i64:64-n32:64-S128-ni:1:10:20" |
| |
| ; ERROR: LLVM ERROR: Cannot lower callbase instruction |
| |
| @_ZTIi = external constant ptr |
| |
| ; Function Attrs: mustprogress |
| define hidden void @test0(i32 noundef %x) #0 personality ptr @__gxx_wasm_personality_v0 { |
| ; CHECK-LABEL: @test0( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: invoke void (...) @may_throw(i32 noundef [[X:%.*]]) |
| ; CHECK-NEXT: to label [[TRY_CONT:%.*]] unwind label [[CATCH_DISPATCH:%.*]] |
| ; CHECK: catch.dispatch: |
| ; CHECK-NEXT: [[TMP0:%.*]] = catchswitch within none [label %catch.start] unwind to caller |
| ; CHECK: catch.start: |
| ; CHECK-NEXT: [[TMP1:%.*]] = catchpad within [[TMP0]] [ptr @_ZTIi] |
| ; CHECK-NEXT: [[TMP2:%.*]] = tail call ptr @llvm.wasm.get.exception(token [[TMP1]]) |
| ; CHECK-NEXT: [[TMP3:%.*]] = tail call i32 @llvm.wasm.get.ehselector(token [[TMP1]]) |
| ; CHECK-NEXT: [[TMP4:%.*]] = tail call i32 @llvm.eh.typeid.for.p0(ptr nonnull @_ZTIi) |
| ; CHECK-NEXT: [[MATCHES:%.*]] = icmp eq i32 [[TMP3]], [[TMP4]] |
| ; CHECK-NEXT: br i1 [[MATCHES]], label [[CATCH:%.*]], label [[RETHROW:%.*]] |
| ; CHECK: catch: |
| ; CHECK-NEXT: [[TMP5:%.*]] = call ptr @__cxa_begin_catch(ptr [[TMP2]]) [ "funclet"(token [[TMP1]]) ] |
| ; CHECK-NEXT: call void (...) @dont_throw(i32 noundef [[X]]) [ "funclet"(token [[TMP1]]) ] |
| ; CHECK-NEXT: call void @__cxa_end_catch() [ "funclet"(token [[TMP1]]) ] |
| ; CHECK-NEXT: catchret from [[TMP1]] to label [[TRY_CONT]] |
| ; CHECK: rethrow: |
| ; CHECK-NEXT: call void @llvm.wasm.rethrow() [ "funclet"(token [[TMP1]]) ] |
| ; CHECK-NEXT: unreachable |
| ; CHECK: try.cont: |
| ; CHECK-NEXT: ret void |
| ; |
| entry: |
| invoke void (...) @may_throw(i32 noundef %x) |
| to label %try.cont unwind label %catch.dispatch |
| |
| catch.dispatch: ; preds = %entry |
| %0 = catchswitch within none [label %catch.start] unwind to caller |
| |
| catch.start: ; preds = %catch.dispatch |
| %1 = catchpad within %0 [ptr @_ZTIi] |
| %2 = tail call ptr @llvm.wasm.get.exception(token %1) |
| %3 = tail call i32 @llvm.wasm.get.ehselector(token %1) |
| %4 = tail call i32 @llvm.eh.typeid.for.p0(ptr nonnull @_ZTIi) |
| %matches = icmp eq i32 %3, %4 |
| br i1 %matches, label %catch, label %rethrow |
| |
| catch: ; preds = %catch.start |
| %5 = call ptr @__cxa_begin_catch(ptr %2) #6 [ "funclet"(token %1) ] |
| call void (...) @dont_throw(i32 noundef %x) #6 [ "funclet"(token %1) ] |
| call void @__cxa_end_catch() #6 [ "funclet"(token %1) ] |
| catchret from %1 to label %try.cont |
| |
| rethrow: ; preds = %catch.start |
| call void @llvm.wasm.rethrow() #5 [ "funclet"(token %1) ] |
| unreachable |
| |
| try.cont: ; preds = %entry, %catch |
| ret void |
| } |
| |
| declare void @may_throw(...) |
| |
| declare i32 @__gxx_wasm_personality_v0(...) |
| |
| ; Function Attrs: mustprogress nocallback nofree nosync nounwind willreturn |
| declare ptr @llvm.wasm.get.exception(token) |
| |
| ; Function Attrs: mustprogress nocallback nofree nosync nounwind willreturn |
| declare i32 @llvm.wasm.get.ehselector(token) |
| |
| ; Function Attrs: nofree nosync nounwind memory(none) |
| declare i32 @llvm.eh.typeid.for.p0(ptr) |
| |
| declare ptr @__cxa_begin_catch(ptr) |
| |
| ; Function Attrs: nounwind |
| declare void @dont_throw(...) |
| |
| declare void @__cxa_end_catch() |
| |
| ; Function Attrs: noreturn |
| declare void @llvm.wasm.rethrow() |
| |
| |