; RUN: llc -mtriple=x86_64-windows-msvc < %s | FileCheck %s ; WinEH requires funclet tokens on nounwind intrinsics if they can lower to ; regular function calls in the course of IR transformations. ; ; Test that the code generator will emit the function call and not consider it ; an "implausible instruciton". In the past this silently truncated code on ; exception paths and caused crashes at runtime. ; ; Reduced IR generated from ObjC++ source: ; ; @class Ety; ; void opaque(void); ; void test_catch_with_objc_intrinsic(void) { ; @try { ; opaque(); ; } @catch (Ety *ex) { ; // Destroy ex when leaving catchpad. This would emit calls to two ; // intrinsic functions: llvm.objc.retain and llvm.objc.storeStrong ; } ; } ; ; llvm.objc.retain and llvm.objc.storeStrong both lower into regular function ; calls before ISel. We only need one of them to trigger funclet truncation ; during codegen: define void @test_catch_with_objc_intrinsic() personality ptr @__CxxFrameHandler3 { entry: %exn.slot = alloca ptr, align 8 %ex2 = alloca ptr, align 8 invoke void @opaque() to label %invoke.cont unwind label %catch.dispatch catch.dispatch: ; preds = %entry %0 = catchswitch within none [label %catch] unwind to caller invoke.cont: ; preds = %entry unreachable catch: ; preds = %catch.dispatch %1 = catchpad within %0 [ptr null, i32 64, ptr %exn.slot] %exn = load ptr, ptr %exn.slot, align 8 %2 = call ptr @llvm.objc.retain(ptr %exn) [ "funclet"(token %1) ] store ptr %2, ptr %ex2, align 8 catchret from %1 to label %catchret.dest catchret.dest: ; preds = %catch ret void } declare void @opaque() declare ptr @llvm.objc.retain(ptr) #0 declare i32 @__CxxFrameHandler3(...) attributes #0 = { nounwind } ; EH catchpad with SEH prologue: ; CHECK-LABEL: # %catch ; CHECK: pushq %rbp ; CHECK: .seh_pushreg %rbp ; ... ; CHECK: .seh_endprologue ; ; At this point the code used to be truncated (and sometimes terminated with an ; int3 opcode): ; CHECK-NOT: int3 ; ; Instead, the runtime call to retain should be emitted: ; CHECK: movq -8(%rbp), %rcx ; CHECK: callq objc_retain ; ... ; ; This is the end of the funclet: ; CHECK: popq %rbp ; CHECK: retq # CATCHRET ; ... ; CHECK: .seh_handlerdata ; ... ; CHECK: .seh_endproc