1 ; RUN: opt -mtriple=x86_64-pc-windows-msvc -S -winehprepare < %s | FileCheck %s
3 declare i32 @__CxxFrameHandler3(...)
4 declare i32 @__C_specific_handler(...)
5 declare void @ProcessCLRException(...)
9 declare void @llvm.foo(i32) nounwind
10 declare void @llvm.bar() nounwind
11 declare i32 @llvm.qux() nounwind
12 declare i1 @llvm.baz() nounwind
14 define void @test1() personality ptr @__CxxFrameHandler3 {
16 ; %x def colors: {entry} subset of use colors; must spill
17 %x = call i32 @llvm.qux()
19 to label %noreturn unwind label %catch.switch
21 %cs = catchswitch within none [label %catch] unwind to caller
23 %cp = catchpad within %cs []
26 ; %x use colors: {entry, cleanup}
27 call void @llvm.foo(i32 %x)
30 ; Need two copies of the call to @h, one under entry and one under catch.
31 ; Currently we generate a load for each, though we shouldn't need one
32 ; for the use in entry's copy.
33 ; CHECK-LABEL: define void @test1(
35 ; CHECK: %x = call i32 @llvm.qux()
36 ; CHECK: invoke void @f()
37 ; CHECK: to label %[[EntryCopy:[^ ]+]] unwind label %catch
38 ; CHECK: catch.switch:
39 ; CHECK: %cs = catchswitch within none [label %catch] unwind to caller
41 ; CHECK: catchpad within %cs []
42 ; CHECK-NEXT: call void @llvm.foo(i32 %x)
43 ; CHECK: [[EntryCopy]]:
44 ; CHECK: call void @llvm.foo(i32 %x)
47 define void @test2() personality ptr @__CxxFrameHandler3 {
50 to label %exit unwind label %cleanup
52 cleanuppad within none []
58 ; Need two copies of %exit's call to @f -- the subsequent ret is only
59 ; valid when coming from %entry, but on the path from %cleanup, this
60 ; might be a valid call to @f which might dynamically not return.
61 ; CHECK-LABEL: define void @test2(
63 ; CHECK: invoke void @f()
64 ; CHECK: to label %[[exit:[^ ]+]] unwind label %cleanup
66 ; CHECK: cleanuppad within none []
67 ; CHECK: call void @llvm.bar()
68 ; CHECK-NEXT: unreachable
70 ; CHECK: call void @llvm.bar()
71 ; CHECK-NEXT: ret void
74 define void @test3() personality ptr @__CxxFrameHandler3 {
77 to label %invoke.cont unwind label %catch.switch
80 to label %exit unwind label %cleanup
82 %cs = catchswitch within none [label %catch] unwind to caller
84 catchpad within %cs []
87 cleanuppad within none []
95 ; Need two copies of %shared's call to @f (similar to @test2 but
96 ; the two regions here are siblings, not parent-child).
97 ; CHECK-LABEL: define void @test3(
98 ; CHECK: invoke void @f()
99 ; CHECK: invoke void @f()
100 ; CHECK: to label %[[exit:[^ ]+]] unwind
102 ; CHECK: catchpad within %cs []
103 ; CHECK-NEXT: call void @llvm.bar()
104 ; CHECK-NEXT: unreachable
106 ; CHECK: cleanuppad within none []
107 ; CHECK: call void @llvm.bar()
108 ; CHECK-NEXT: unreachable
113 define void @test4() personality ptr @__CxxFrameHandler3 {
116 to label %shared unwind label %catch.switch
118 %cs = catchswitch within none [label %catch] unwind to caller
120 catchpad within %cs []
123 %x = call i32 @llvm.qux()
124 %i = call i32 @llvm.qux()
125 %zero.trip = icmp eq i32 %i, 0
126 br i1 %zero.trip, label %exit, label %loop
128 %i.loop = phi i32 [ %i, %shared ], [ %i.dec, %loop.tail ]
129 %b = call i1 @llvm.baz()
130 br i1 %b, label %left, label %right
132 %y = call i32 @llvm.qux()
135 call void @llvm.foo(i32 %x)
138 %i.dec = sub i32 %i.loop, 1
139 %done = icmp eq i32 %i.dec, 0
140 br i1 %done, label %exit, label %loop
142 call void @llvm.foo(i32 %x)
145 ; Make sure we can clone regions that have internal control
146 ; flow and SSA values. Here we need two copies of everything
147 ; from %shared to %exit.
148 ; CHECK-LABEL: define void @test4(
150 ; CHECK: to label %[[shared_E:[^ ]+]] unwind label %catch.switch
152 ; CHECK: catchpad within %cs []
153 ; CHECK: [[x_C:%[^ ]+]] = call i32 @llvm.qux()
154 ; CHECK: [[i_C:%[^ ]+]] = call i32 @llvm.qux()
155 ; CHECK: [[zt_C:%[^ ]+]] = icmp eq i32 [[i_C]], 0
156 ; CHECK: br i1 [[zt_C]], label %[[exit_C:[^ ]+]], label %[[loop_C:[^ ]+]]
157 ; CHECK: [[shared_E]]:
158 ; CHECK: [[x_E:%[^ ]+]] = call i32 @llvm.qux()
159 ; CHECK: [[i_E:%[^ ]+]] = call i32 @llvm.qux()
160 ; CHECK: [[zt_E:%[^ ]+]] = icmp eq i32 [[i_E]], 0
161 ; CHECK: br i1 [[zt_E]], label %[[exit_E:[^ ]+]], label %[[loop_E:[^ ]+]]
163 ; CHECK: [[iloop_C:%[^ ]+]] = phi i32 [ [[i_C]], %catch ], [ [[idec_C:%[^ ]+]], %[[looptail_C:[^ ]+]] ]
164 ; CHECK: [[b_C:%[^ ]+]] = call i1 @llvm.baz()
165 ; CHECK: br i1 [[b_C]], label %[[left_C:[^ ]+]], label %[[right_C:[^ ]+]]
167 ; CHECK: [[iloop_E:%[^ ]+]] = phi i32 [ [[i_E]], %[[shared_E]] ], [ [[idec_E:%[^ ]+]], %[[looptail_E:[^ ]+]] ]
168 ; CHECK: [[b_E:%[^ ]+]] = call i1 @llvm.baz()
169 ; CHECK: br i1 [[b_E]], label %[[left_E:[^ ]+]], label %[[right_E:[^ ]+]]
171 ; CHECK: [[y_C:%[^ ]+]] = call i32 @llvm.qux()
172 ; CHECK: br label %[[looptail_C]]
174 ; CHECK: [[y_E:%[^ ]+]] = call i32 @llvm.qux()
175 ; CHECK: br label %[[looptail_E]]
176 ; CHECK: [[right_C]]:
177 ; CHECK: call void @llvm.foo(i32 [[x_C]])
178 ; CHECK: br label %[[looptail_C]]
179 ; CHECK: [[right_E]]:
180 ; CHECK: call void @llvm.foo(i32 [[x_E]])
181 ; CHECK: br label %[[looptail_E]]
182 ; CHECK: [[looptail_C]]:
183 ; CHECK: [[idec_C]] = sub i32 [[iloop_C]], 1
184 ; CHECK: [[done_C:%[^ ]+]] = icmp eq i32 [[idec_C]], 0
185 ; CHECK: br i1 [[done_C]], label %[[exit_C]], label %[[loop_C]]
186 ; CHECK: [[looptail_E]]:
187 ; CHECK: [[idec_E]] = sub i32 [[iloop_E]], 1
188 ; CHECK: [[done_E:%[^ ]+]] = icmp eq i32 [[idec_E]], 0
189 ; CHECK: br i1 [[done_E]], label %[[exit_E]], label %[[loop_E]]
191 ; CHECK: call void @llvm.foo(i32 [[x_C]])
194 ; CHECK: call void @llvm.foo(i32 [[x_E]])
198 define void @test5() personality ptr @__C_specific_handler {
201 to label %exit unwind label %outer
203 %o = cleanuppad within none []
204 %x = call i32 @llvm.qux()
205 invoke void @f() [ "funclet"(token %o) ]
206 to label %outer.ret unwind label %catch.switch
208 %cs = catchswitch within %o [label %inner] unwind to caller
210 %i = catchpad within %cs []
211 catchret from %i to label %outer.post-inner
213 call void @llvm.foo(i32 %x)
216 cleanupret from %o unwind to caller
220 ; Simple nested case (catch-inside-cleanup). Nothing needs
221 ; to be cloned. The def and use of %x are both in %outer
222 ; and so don't need to be spilled.
223 ; CHECK-LABEL: define void @test5(
225 ; CHECK: %x = call i32 @llvm.qux()
226 ; CHECK-NEXT: invoke void @f()
227 ; CHECK-NEXT: to label %outer.ret unwind label %catch.switch
229 ; CHECK-NEXT: %i = catchpad within %cs []
230 ; CHECK-NEXT: catchret from %i to label %outer.post-inner
231 ; CHECK: outer.post-inner:
232 ; CHECK-NEXT: call void @llvm.foo(i32 %x)
233 ; CHECK-NEXT: br label %outer.ret
236 define void @test10() personality ptr @__CxxFrameHandler3 {
239 to label %unreachable unwind label %inner
241 %cleanup = cleanuppad within none []
242 ; make sure we don't overlook this cleanupret and try to process
243 ; successor %outer as a child of inner.
244 cleanupret from %cleanup unwind label %outer
246 %cs = catchswitch within none [label %catch.body] unwind to caller
249 %catch = catchpad within %cs []
250 catchret from %catch to label %exit
256 ; CHECK-LABEL: define void @test10(
259 ; CHECK-NEXT: to label %unreachable unwind label %inner
261 ; CHECK-NEXT: %cleanup = cleanuppad within none []
262 ; CHECK-NEXT: cleanupret from %cleanup unwind label %outer
264 ; CHECK-NEXT: %cs = catchswitch within none [label %catch.body] unwind to caller
266 ; CHECK-NEXT: %catch = catchpad within %cs []
267 ; CHECK-NEXT: catchret from %catch to label %exit
269 ; CHECK-NEXT: ret void
271 define void @test11() personality ptr @__C_specific_handler {
274 to label %exit unwind label %cleanup.outer
276 %outer = cleanuppad within none []
277 invoke void @f() [ "funclet"(token %outer) ]
278 to label %outer.cont unwind label %cleanup.inner
282 %inner = cleanuppad within %outer []
285 call void @llvm.bar()
290 ; merge.end will get cloned for outer and inner, but is implausible
291 ; from inner, so the call @f() in inner's copy of merge should be
292 ; rewritten to call @f()
293 ; CHECK-LABEL: define void @test11()
294 ; CHECK: %inner = cleanuppad within %outer []
295 ; CHECK-NEXT: call void @llvm.bar()
296 ; CHECK-NEXT: unreachable
298 define void @test12() personality ptr @__CxxFrameHandler3 !dbg !5 {
301 to label %cont unwind label %left, !dbg !8
304 to label %exit unwind label %right
306 cleanuppad within none []
309 cleanuppad within none []
312 ; This call will get cloned; make sure we can handle cloning
313 ; instructions with debug metadata attached.
314 call void @llvm.bar(), !dbg !9
320 ; CHECK-LABEL: define void @test13()
322 define void @test13() personality ptr @__CxxFrameHandler3 {
327 cleanuppad within none []
331 define void @test14() personality ptr @ProcessCLRException {
334 to label %cont unwind label %cleanup
337 to label %exit unwind label %switch.outer
339 %cleanpad = cleanuppad within none []
340 invoke void @f() [ "funclet" (token %cleanpad) ]
341 to label %cleanret unwind label %switch.inner
343 %cs.inner = catchswitch within %cleanpad [label %pad.inner] unwind to caller
345 %cp.inner = catchpad within %cs.inner [i32 1]
346 catchret from %cp.inner to label %join
348 cleanupret from %cleanpad unwind to caller
350 %cs.outer = catchswitch within none [label %pad.outer] unwind to caller
352 %cp.outer = catchpad within %cs.outer [i32 2]
353 catchret from %cp.outer to label %join
355 %phi = phi i32 [ 1, %pad.inner ], [ 2, %pad.outer ]
356 call void @llvm.foo(i32 %phi)
361 ; Both catchrets target %join, but the catchret from %cp.inner
362 ; returns to %cleanpad and the catchret from %cp.outer returns to the
363 ; main function, so %join needs to get cloned and one of the cleanuprets
364 ; needs to be updated to target the clone
365 ; CHECK-LABEL: define void @test14()
366 ; CHECK: catchret from %cp.inner to label %[[Clone1:.+]]
367 ; CHECK: catchret from %cp.outer to label %[[Clone2:.+]]
369 ; CHECK-NEXT: call void @llvm.foo(i32 1)
370 ; CHECK-NEXT: unreachable
372 ; CHECK-NEXT: call void @llvm.foo(i32 2)
373 ; CHECK-NEXT: unreachable
375 ;; Debug info (from test12)
377 ; Make sure the DISubprogram doesn't get cloned
378 ; CHECK-LABEL: !llvm.module.flags
379 ; CHECK-NOT: !DISubprogram
380 ; CHECK: !{{[0-9]+}} = distinct !DISubprogram(name: "test12"
381 ; CHECK-NOT: !DISubprogram
382 !llvm.module.flags = !{!0}
385 !0 = !{i32 2, !"Debug Info Version", i32 3}
386 !1 = distinct !DICompileUnit(language: DW_LANG_C_plus_plus, file: !2, producer: "compiler", isOptimized: false, runtimeVersion: 0, emissionKind: FullDebug, enums: !3)
387 !2 = !DIFile(filename: "test.cpp", directory: ".")
389 !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)
390 !6 = !DISubroutineType(types: !7)
392 !8 = !DILocation(line: 1, scope: !5)
393 !9 = !DILocation(line: 2, scope: !5)