; RUN: opt -mtriple=x86_64-pc-windows-msvc -S -winehprepare < %s | FileCheck %s declare i32 @__CxxFrameHandler3(...) declare i32 @__C_specific_handler(...) declare void @ProcessCLRException(...) declare void @f() declare void @llvm.foo(i32) nounwind declare void @llvm.bar() nounwind declare i32 @llvm.qux() nounwind declare i1 @llvm.baz() nounwind define void @test1() personality i32 (...)* @__CxxFrameHandler3 { entry: ; %x def colors: {entry} subset of use colors; must spill %x = call i32 @llvm.qux() invoke void @f() to label %noreturn unwind label %catch.switch catch.switch: %cs = catchswitch within none [label %catch] unwind to caller catch: %cp = catchpad within %cs [] br label %noreturn noreturn: ; %x use colors: {entry, cleanup} call void @llvm.foo(i32 %x) unreachable } ; Need two copies of the call to @h, one under entry and one under catch. ; Currently we generate a load for each, though we shouldn't need one ; for the use in entry's copy. ; CHECK-LABEL: define void @test1( ; CHECK: entry: ; CHECK: %x = call i32 @llvm.qux() ; CHECK: invoke void @f() ; CHECK: to label %[[EntryCopy:[^ ]+]] unwind label %catch ; CHECK: catch.switch: ; CHECK: %cs = catchswitch within none [label %catch] unwind to caller ; CHECK: catch: ; CHECK: catchpad within %cs [] ; CHECK-NEXT: call void @llvm.foo(i32 %x) ; CHECK: [[EntryCopy]]: ; CHECK: call void @llvm.foo(i32 %x) define void @test2() personality i32 (...)* @__CxxFrameHandler3 { entry: invoke void @f() to label %exit unwind label %cleanup cleanup: cleanuppad within none [] br label %exit exit: call void @llvm.bar() ret void } ; Need two copies of %exit's call to @f -- the subsequent ret is only ; valid when coming from %entry, but on the path from %cleanup, this ; might be a valid call to @f which might dynamically not return. ; CHECK-LABEL: define void @test2( ; CHECK: entry: ; CHECK: invoke void @f() ; CHECK: to label %[[exit:[^ ]+]] unwind label %cleanup ; CHECK: cleanup: ; CHECK: cleanuppad within none [] ; CHECK: call void @llvm.bar() ; CHECK-NEXT: unreachable ; CHECK: [[exit]]: ; CHECK: call void @llvm.bar() ; CHECK-NEXT: ret void define void @test3() personality i32 (...)* @__CxxFrameHandler3 { entry: invoke void @f() to label %invoke.cont unwind label %catch.switch invoke.cont: invoke void @f() to label %exit unwind label %cleanup catch.switch: %cs = catchswitch within none [label %catch] unwind to caller catch: catchpad within %cs [] br label %shared cleanup: cleanuppad within none [] br label %shared shared: call void @llvm.bar() br label %exit exit: ret void } ; Need two copies of %shared's call to @f (similar to @test2 but ; the two regions here are siblings, not parent-child). ; CHECK-LABEL: define void @test3( ; CHECK: invoke void @f() ; CHECK: invoke void @f() ; CHECK: to label %[[exit:[^ ]+]] unwind ; CHECK: catch: ; CHECK: catchpad within %cs [] ; CHECK-NEXT: call void @llvm.bar() ; CHECK-NEXT: unreachable ; CHECK: cleanup: ; CHECK: cleanuppad within none [] ; CHECK: call void @llvm.bar() ; CHECK-NEXT: unreachable ; CHECK: [[exit]]: ; CHECK: ret void define void @test4() personality i32 (...)* @__CxxFrameHandler3 { entry: invoke void @f() to label %shared unwind label %catch.switch catch.switch: %cs = catchswitch within none [label %catch] unwind to caller catch: catchpad within %cs [] br label %shared shared: %x = call i32 @llvm.qux() %i = call i32 @llvm.qux() %zero.trip = icmp eq i32 %i, 0 br i1 %zero.trip, label %exit, label %loop loop: %i.loop = phi i32 [ %i, %shared ], [ %i.dec, %loop.tail ] %b = call i1 @llvm.baz() br i1 %b, label %left, label %right left: %y = call i32 @llvm.qux() br label %loop.tail right: call void @llvm.foo(i32 %x) br label %loop.tail loop.tail: %i.dec = sub i32 %i.loop, 1 %done = icmp eq i32 %i.dec, 0 br i1 %done, label %exit, label %loop exit: call void @llvm.foo(i32 %x) unreachable } ; Make sure we can clone regions that have internal control ; flow and SSA values. Here we need two copies of everything ; from %shared to %exit. ; CHECK-LABEL: define void @test4( ; CHECK: entry: ; CHECK: to label %[[shared_E:[^ ]+]] unwind label %catch.switch ; CHECK: catch: ; CHECK: catchpad within %cs [] ; CHECK: [[x_C:%[^ ]+]] = call i32 @llvm.qux() ; CHECK: [[i_C:%[^ ]+]] = call i32 @llvm.qux() ; CHECK: [[zt_C:%[^ ]+]] = icmp eq i32 [[i_C]], 0 ; CHECK: br i1 [[zt_C]], label %[[exit_C:[^ ]+]], label %[[loop_C:[^ ]+]] ; CHECK: [[shared_E]]: ; CHECK: [[x_E:%[^ ]+]] = call i32 @llvm.qux() ; CHECK: [[i_E:%[^ ]+]] = call i32 @llvm.qux() ; CHECK: [[zt_E:%[^ ]+]] = icmp eq i32 [[i_E]], 0 ; CHECK: br i1 [[zt_E]], label %[[exit_E:[^ ]+]], label %[[loop_E:[^ ]+]] ; CHECK: [[loop_C]]: ; CHECK: [[iloop_C:%[^ ]+]] = phi i32 [ [[i_C]], %catch ], [ [[idec_C:%[^ ]+]], %[[looptail_C:[^ ]+]] ] ; CHECK: [[b_C:%[^ ]+]] = call i1 @llvm.baz() ; CHECK: br i1 [[b_C]], label %[[left_C:[^ ]+]], label %[[right_C:[^ ]+]] ; CHECK: [[loop_E]]: ; CHECK: [[iloop_E:%[^ ]+]] = phi i32 [ [[i_E]], %[[shared_E]] ], [ [[idec_E:%[^ ]+]], %[[looptail_E:[^ ]+]] ] ; CHECK: [[b_E:%[^ ]+]] = call i1 @llvm.baz() ; CHECK: br i1 [[b_E]], label %[[left_E:[^ ]+]], label %[[right_E:[^ ]+]] ; CHECK: [[left_C]]: ; CHECK: [[y_C:%[^ ]+]] = call i32 @llvm.qux() ; CHECK: br label %[[looptail_C]] ; CHECK: [[left_E]]: ; CHECK: [[y_E:%[^ ]+]] = call i32 @llvm.qux() ; CHECK: br label %[[looptail_E]] ; CHECK: [[right_C]]: ; CHECK: call void @llvm.foo(i32 [[x_C]]) ; CHECK: br label %[[looptail_C]] ; CHECK: [[right_E]]: ; CHECK: call void @llvm.foo(i32 [[x_E]]) ; CHECK: br label %[[looptail_E]] ; CHECK: [[looptail_C]]: ; CHECK: [[idec_C]] = sub i32 [[iloop_C]], 1 ; CHECK: [[done_C:%[^ ]+]] = icmp eq i32 [[idec_C]], 0 ; CHECK: br i1 [[done_C]], label %[[exit_C]], label %[[loop_C]] ; CHECK: [[looptail_E]]: ; CHECK: [[idec_E]] = sub i32 [[iloop_E]], 1 ; CHECK: [[done_E:%[^ ]+]] = icmp eq i32 [[idec_E]], 0 ; CHECK: br i1 [[done_E]], label %[[exit_E]], label %[[loop_E]] ; CHECK: [[exit_C]]: ; CHECK: call void @llvm.foo(i32 [[x_C]]) ; CHECK: unreachable ; CHECK: [[exit_E]]: ; CHECK: call void @llvm.foo(i32 [[x_E]]) ; CHECK: unreachable define void @test5() personality i32 (...)* @__C_specific_handler { entry: invoke void @f() to label %exit unwind label %outer outer: %o = cleanuppad within none [] %x = call i32 @llvm.qux() invoke void @f() [ "funclet"(token %o) ] to label %outer.ret unwind label %catch.switch catch.switch: %cs = catchswitch within %o [label %inner] unwind to caller inner: %i = catchpad within %cs [] catchret from %i to label %outer.post-inner outer.post-inner: call void @llvm.foo(i32 %x) br label %outer.ret outer.ret: cleanupret from %o unwind to caller exit: ret void } ; Simple nested case (catch-inside-cleanup). Nothing needs ; to be cloned. The def and use of %x are both in %outer ; and so don't need to be spilled. ; CHECK-LABEL: define void @test5( ; CHECK: outer: ; CHECK: %x = call i32 @llvm.qux() ; CHECK-NEXT: invoke void @f() ; CHECK-NEXT: to label %outer.ret unwind label %catch.switch ; CHECK: inner: ; CHECK-NEXT: %i = catchpad within %cs [] ; CHECK-NEXT: catchret from %i to label %outer.post-inner ; CHECK: outer.post-inner: ; CHECK-NEXT: call void @llvm.foo(i32 %x) ; CHECK-NEXT: br label %outer.ret define void @test10() personality i32 (...)* @__CxxFrameHandler3 { entry: invoke void @f() to label %unreachable unwind label %inner inner: %cleanup = cleanuppad within none [] ; make sure we don't overlook this cleanupret and try to process ; successor %outer as a child of inner. cleanupret from %cleanup unwind label %outer outer: %cs = catchswitch within none [label %catch.body] unwind to caller catch.body: %catch = catchpad within %cs [] catchret from %catch to label %exit exit: ret void unreachable: unreachable } ; CHECK-LABEL: define void @test10( ; CHECK-NEXT: entry: ; CHECK-NEXT: invoke ; CHECK-NEXT: to label %unreachable unwind label %inner ; CHECK: inner: ; CHECK-NEXT: %cleanup = cleanuppad within none [] ; CHECK-NEXT: cleanupret from %cleanup unwind label %outer ; CHECK: outer: ; CHECK-NEXT: %cs = catchswitch within none [label %catch.body] unwind to caller ; CHECK: catch.body: ; CHECK-NEXT: %catch = catchpad within %cs [] ; CHECK-NEXT: catchret from %catch to label %exit ; CHECK: exit: ; CHECK-NEXT: ret void define void @test11() personality i32 (...)* @__C_specific_handler { entry: invoke void @f() to label %exit unwind label %cleanup.outer cleanup.outer: %outer = cleanuppad within none [] invoke void @f() [ "funclet"(token %outer) ] to label %outer.cont unwind label %cleanup.inner outer.cont: br label %merge cleanup.inner: %inner = cleanuppad within %outer [] br label %merge merge: call void @llvm.bar() unreachable exit: ret void } ; merge.end will get cloned for outer and inner, but is implausible ; from inner, so the call @f() in inner's copy of merge should be ; rewritten to call @f() ; CHECK-LABEL: define void @test11() ; CHECK: %inner = cleanuppad within %outer [] ; CHECK-NEXT: call void @llvm.bar() ; CHECK-NEXT: unreachable define void @test12() personality i32 (...)* @__CxxFrameHandler3 !dbg !5 { entry: invoke void @f() to label %cont unwind label %left, !dbg !8 cont: invoke void @f() to label %exit unwind label %right left: cleanuppad within none [] br label %join right: cleanuppad within none [] br label %join join: ; This call will get cloned; make sure we can handle cloning ; instructions with debug metadata attached. call void @llvm.bar(), !dbg !9 unreachable exit: ret void } ; CHECK-LABEL: define void @test13() ; CHECK: ret void define void @test13() personality i32 (...)* @__CxxFrameHandler3 { entry: ret void unreachable: cleanuppad within none [] unreachable } define void @test14() personality void (...)* @ProcessCLRException { entry: invoke void @f() to label %cont unwind label %cleanup cont: invoke void @f() to label %exit unwind label %switch.outer cleanup: %cleanpad = cleanuppad within none [] invoke void @f() [ "funclet" (token %cleanpad) ] to label %cleanret unwind label %switch.inner switch.inner: %cs.inner = catchswitch within %cleanpad [label %pad.inner] unwind to caller pad.inner: %cp.inner = catchpad within %cs.inner [i32 1] catchret from %cp.inner to label %join cleanret: cleanupret from %cleanpad unwind to caller switch.outer: %cs.outer = catchswitch within none [label %pad.outer] unwind to caller pad.outer: %cp.outer = catchpad within %cs.outer [i32 2] catchret from %cp.outer to label %join join: %phi = phi i32 [ 1, %pad.inner ], [ 2, %pad.outer ] call void @llvm.foo(i32 %phi) unreachable exit: ret void } ; Both catchrets target %join, but the catchret from %cp.inner ; returns to %cleanpad and the catchret from %cp.outer returns to the ; main function, so %join needs to get cloned and one of the cleanuprets ; needs to be updated to target the clone ; CHECK-LABEL: define void @test14() ; CHECK: catchret from %cp.inner to label %[[Clone1:.+]] ; CHECK: catchret from %cp.outer to label %[[Clone2:.+]] ; CHECK: [[Clone1]]: ; CHECK-NEXT: call void @llvm.foo(i32 1) ; CHECK-NEXT: unreachable ; CHECK: [[Clone2]]: ; CHECK-NEXT: call void @llvm.foo(i32 2) ; CHECK-NEXT: unreachable ;; Debug info (from test12) ; Make sure the DISubprogram doesn't get cloned ; CHECK-LABEL: !llvm.module.flags ; CHECK-NOT: !DISubprogram ; CHECK: !{{[0-9]+}} = distinct !DISubprogram(name: "test12" ; CHECK-NOT: !DISubprogram !llvm.module.flags = !{!0} !llvm.dbg.cu = !{!1} !0 = !{i32 2, !"Debug Info Version", i32 3} !1 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !2, producer: "compiler", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, enums: !3) !2 = !DIFile(filename: "test.cpp", directory: ".") !3 = !{} !5 = distinct !DISubprogram(name: "test12", scope: !2, file: !2, type: !6, isLocal: false, isDefinition: true, scopeLine: 3, flags: DIFlagPrototyped, isOptimized: true, unit: !1, retainedNodes: !3) !6 = !DISubroutineType(types: !7) !7 = !{null} !8 = !DILocation(line: 1, scope: !5) !9 = !DILocation(line: 2, scope: !5)