; RUN: mlir-translate -import-llvm %s | FileCheck %s @_ZTIi = external dso_local constant ptr @_ZTIii= external dso_local constant ptr declare void @foo(ptr) declare void @vararg_foo(ptr, ...) declare ptr @bar(ptr) declare i32 @__gxx_personality_v0(...) ; CHECK-LABEL: @invokeLandingpad define i32 @invokeLandingpad() personality ptr @__gxx_personality_v0 { ; CHECK: %[[a1:[0-9]+]] = llvm.mlir.addressof @_ZTIii : !llvm.ptr ; CHECK: %[[a3:[0-9]+]] = llvm.alloca %{{[0-9]+}} x i8 {alignment = 1 : i64} : (i32) -> !llvm.ptr %1 = alloca i8 ; CHECK: llvm.invoke @foo(%[[a3]]) to ^[[bb1:.*]] unwind ^[[bb4:.*]] : (!llvm.ptr) -> () invoke void @foo(ptr %1) to label %bb1 unwind label %bb4 ; CHECK: ^[[bb1]]: bb1: ; CHECK: %{{[0-9]+}} = llvm.invoke @bar(%[[a3]]) to ^[[bb2:.*]] unwind ^[[bb4]] : (!llvm.ptr) -> !llvm.ptr %2 = invoke ptr @bar(ptr %1) to label %bb2 unwind label %bb4 ; CHECK: ^[[bb2]]: bb2: ; CHECK: llvm.invoke @vararg_foo(%[[a3]], %{{.*}}) to ^[[bb3:.*]] unwind ^[[bb4]] vararg(!llvm.func) : (!llvm.ptr, i32) -> () invoke void (ptr, ...) @vararg_foo(ptr %1, i32 0) to label %bb3 unwind label %bb4 ; CHECK: ^[[bb3]]: bb3: ; CHECK: llvm.invoke %{{.*}}(%[[a3]], %{{.*}}) to ^[[bb5:.*]] unwind ^[[bb4]] vararg(!llvm.func) : !llvm.ptr, (!llvm.ptr, i32) -> () invoke void (ptr, ...) undef(ptr %1, i32 0) to label %bb5 unwind label %bb4 ; CHECK: ^[[bb4]]: bb4: ; CHECK: %{{[0-9]+}} = llvm.landingpad (catch %{{[0-9]+}} : !llvm.ptr) (catch %[[a1]] : !llvm.ptr) (filter %{{[0-9]+}} : !llvm.array<1 x i1>) : !llvm.struct<(ptr, i32)> %3 = landingpad { ptr, i32 } catch ptr @_ZTIi catch ptr @_ZTIii filter [1 x i1] [i1 1] resume { ptr, i32 } %3 ; CHECK: ^[[bb5]]: bb5: ; CHECK: llvm.return %{{[0-9]+}} : i32 ret i32 1 } declare i32 @foo2() ; CHECK-LABEL: @invokePhi ; CHECK-SAME: (%[[cond:.*]]: i1) -> i32 define i32 @invokePhi(i1 %cond) personality ptr @__gxx_personality_v0 { entry: ; CHECK: %[[c0:.*]] = llvm.mlir.constant(0 : i32) : i32 ; CHECK: llvm.cond_br %[[cond]], ^[[bb1:.*]], ^[[bb2:.*]] br i1 %cond, label %call, label %nocall ; CHECK: ^[[bb1]]: call: ; CHECK: %[[invoke:.*]] = llvm.invoke @foo2() to ^[[bb3:.*]] unwind ^[[bb5:.*]] : () -> i32 %invoke = invoke i32 @foo2() to label %bb0 unwind label %bb1 ; CHECK: ^[[bb2]]: nocall: ; CHECK: llvm.br ^[[bb4:.*]](%[[c0]] : i32) br label %bb0 ; CHECK: ^[[bb3]]: ; CHECK: llvm.br ^[[bb4]](%[[invoke]] : i32) ; CHECK: ^[[bb4]](%[[barg:.*]]: i32): bb0: %ret = phi i32 [ 0, %nocall ], [ %invoke, %call ] ; CHECK: llvm.return %[[barg]] : i32 ret i32 %ret ; CHECK: ^[[bb5]]: bb1: ; CHECK: %[[lp:.*]] = llvm.landingpad cleanup : i32 %resume = landingpad i32 cleanup ; CHECK: llvm.resume %[[lp]] : i32 resume i32 %resume } ; CHECK-LABEL: @invokePhiComplex ; CHECK-SAME: (%[[cond:.*]]: i1) -> i32 define i32 @invokePhiComplex(i1 %cond) personality ptr @__gxx_personality_v0 { entry: ; CHECK: %[[c0:.*]] = llvm.mlir.constant(0 : i32) : i32 ; CHECK: %[[c1:.*]] = llvm.mlir.constant(1 : i32) : i32 ; CHECK: %[[c2:.*]] = llvm.mlir.constant(2 : i32) : i32 ; CHECK: %[[c20:.*]] = llvm.mlir.constant(20 : i32) : i32 ; CHECK: llvm.cond_br %[[cond]], ^[[bb1:.*]], ^[[bb2:.*]] br i1 %cond, label %call, label %nocall ; CHECK: ^[[bb1]]: call: ; CHECK: %[[invoke:.*]] = llvm.invoke @foo2() to ^[[bb3:.*]] unwind ^[[bb5:.*]] : () -> i32 %invoke = invoke i32 @foo2() to label %bb0 unwind label %bb1 ; CHECK: ^[[bb2]]: nocall: ; CHECK: llvm.br ^[[bb4:.*]](%[[c0]], %[[c1]], %[[c2]] : i32, i32, i32) br label %bb0 ; CHECK: ^[[bb3]]: ; CHECK: llvm.br ^[[bb4]](%[[invoke]], %[[c20]], %[[invoke]] : i32, i32, i32) ; CHECK: ^[[bb4]](%[[barg0:.*]]: i32, %[[barg1:.*]]: i32, %[[barg2:.*]]: i32): bb0: %a = phi i32 [ 0, %nocall ], [ %invoke, %call ] %b = phi i32 [ 1, %nocall ], [ 20, %call ] %c = phi i32 [ 2, %nocall ], [ %invoke, %call ] ; CHECK: %[[add0:.*]] = llvm.add %[[barg0]], %[[barg1]] : i32 ; CHECK: %[[add1:.*]] = llvm.add %[[barg2]], %[[add0]] : i32 %d = add i32 %a, %b %e = add i32 %c, %d ; CHECK: llvm.return %[[add1]] : i32 ret i32 %e ; CHECK: ^[[bb5]]: bb1: ; CHECK: %[[lp:.*]] = llvm.landingpad cleanup : i32 %resume = landingpad i32 cleanup ; CHECK: llvm.resume %[[lp]] : i32 resume i32 %resume } declare void @f0(ptr) declare void @f1(i32) declare void @f2({ptr, i32}) ; CHECK-LABEL: @landingpad_dominance define void @landingpad_dominance() personality ptr @__gxx_personality_v0 { entry: ; CHECK: %[[null:.*]] = llvm.mlir.zero : !llvm.ptr ; CHECK: %[[c1:.*]] = llvm.mlir.constant(0 : i32) : i32 ; CHECK: %[[undef:.*]] = llvm.mlir.undef : !llvm.struct<(ptr, i32)> ; CHECK: %[[tmpstruct:.*]] = llvm.insertvalue %[[null]], %[[undef]][0] : !llvm.struct<(ptr, i32)> ; CHECK: %[[struct:.*]] = llvm.insertvalue %[[c1]], %[[tmpstruct]][1] : !llvm.struct<(ptr, i32)> ; CHECK: llvm.call @f0(%[[null]]) : (!llvm.ptr) -> () call void @f0(ptr null) ; CHECK: llvm.call @f1(%[[c1]]) : (i32) -> () call void @f1(i32 0) ; CHECK: llvm.invoke @f0(%[[null]]) to ^[[block2:.*]] unwind ^[[block1:.*]] : (!llvm.ptr) -> () invoke void @f0(ptr null) to label %exit unwind label %catch ; CHECK: ^[[block1]]: catch: ; CHECK: %[[lp:.*]] = llvm.landingpad (catch %[[null]] : !llvm.ptr) : !llvm.struct<(ptr, i32)> %lp = landingpad { ptr, i32 } catch ptr null ; CHECK: llvm.call @f2(%[[struct]]) : (!llvm.struct<(ptr, i32)>) -> () call void @f2({ptr, i32} {ptr null, i32 0}) ; CHECK: llvm.resume %[[lp]] : !llvm.struct<(ptr, i32)> resume {ptr, i32} %lp ; CHECK: ^[[block2]]: exit: ; CHECK: llvm.return ret void }