• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1; RUN: opt -inline -S %s | FileCheck %s
2; RUN: opt -passes='cgscc(inline)' -S %s | FileCheck %s
3
4declare void @g()
5
6
7;;; Test with a call in a funclet that needs to remain a call
8;;; when inlined because the funclet doesn't unwind to caller.
9;;; CHECK-LABEL: define void @test1(
10define void @test1() personality void ()* @g {
11entry:
12; CHECK-NEXT: entry:
13  invoke void @test1_inlinee()
14    to label %exit unwind label %cleanup
15cleanup:
16  %pad = cleanuppad within none []
17  call void @g() [ "funclet"(token %pad) ]
18  cleanupret from %pad unwind to caller
19exit:
20  ret void
21}
22
23define void @test1_inlinee() alwaysinline personality void ()* @g {
24entry:
25  invoke void @g()
26    to label %exit unwind label %cleanup.inner
27; CHECK-NEXT:  invoke void @g()
28; CHECK-NEXT:    unwind label %[[cleanup_inner:.+]]
29
30cleanup.inner:
31  %pad.inner = cleanuppad within none []
32  call void @g() [ "funclet"(token %pad.inner) ]
33  cleanupret from %pad.inner unwind label %cleanup.outer
34; CHECK: [[cleanup_inner]]:
35; The call here needs to remain a call becuase pad.inner has a cleanupret
36; that stays within the inlinee.
37; CHECK-NEXT:  %[[pad_inner:[^ ]+]] = cleanuppad within none
38; CHECK-NEXT:  call void @g() [ "funclet"(token %[[pad_inner]]) ]
39; CHECK-NEXT:  cleanupret from %[[pad_inner]] unwind label %[[cleanup_outer:.+]]
40
41cleanup.outer:
42  %pad.outer = cleanuppad within none []
43  call void @g() [ "funclet"(token %pad.outer) ]
44  cleanupret from %pad.outer unwind to caller
45; CHECK: [[cleanup_outer]]:
46; The call and cleanupret here need to be redirected to caller cleanup
47; CHECK-NEXT: %[[pad_outer:[^ ]+]] = cleanuppad within none
48; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[pad_outer]]) ]
49; CHECK-NEXT:   unwind label %cleanup
50; CHECK: cleanupret from %[[pad_outer]] unwind label %cleanup{{$}}
51
52exit:
53  ret void
54}
55
56
57
58;;; Test with an "unwind to caller" catchswitch in a parent funclet
59;;; that needs to remain "unwind to caller" because the parent
60;;; doesn't unwind to caller.
61;;; CHECK-LABEL: define void @test2(
62define void @test2() personality void ()* @g {
63entry:
64; CHECK-NEXT: entry:
65  invoke void @test2_inlinee()
66    to label %exit unwind label %cleanup
67cleanup:
68  %pad = cleanuppad within none []
69  call void @g() [ "funclet"(token %pad) ]
70  cleanupret from %pad unwind to caller
71exit:
72  ret void
73}
74
75define void @test2_inlinee() alwaysinline personality void ()* @g {
76entry:
77  invoke void @g()
78    to label %exit unwind label %cleanup1
79; CHECK-NEXT:   invoke void @g()
80; CHECK-NEXT:     unwind label %[[cleanup1:.+]]
81
82cleanup1:
83  %outer = cleanuppad within none []
84  invoke void @g() [ "funclet"(token %outer) ]
85    to label %ret1 unwind label %catchswitch
86; CHECK: [[cleanup1]]:
87; CHECK-NEXT: %[[outer:[^ ]+]] = cleanuppad within none
88; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[outer]]) ]
89; CHECK-NEXT:   unwind label %[[catchswitch:.+]]
90
91catchswitch:
92  %cs = catchswitch within %outer [label %catch] unwind to caller
93; CHECK: [[catchswitch]]:
94; The catchswitch here needs to remain "unwind to caller" since %outer
95; has a cleanupret that remains within the inlinee.
96; CHECK-NEXT: %[[cs:[^ ]+]] = catchswitch within %[[outer]] [label %[[catch:.+]]] unwind to caller
97
98catch:
99  %inner = catchpad within %cs []
100  call void @g() [ "funclet"(token %inner) ]
101  catchret from %inner to label %ret1
102; CHECK: [[catch]]:
103; The call here needs to remain a call since it too is within %outer
104; CHECK:   %[[inner:[^ ]+]] = catchpad within %[[cs]]
105; CHECK-NEXT: call void @g() [ "funclet"(token %[[inner]]) ]
106
107ret1:
108  cleanupret from %outer unwind label %cleanup2
109; CHECK: cleanupret from %[[outer]] unwind label %[[cleanup2:.+]]
110
111cleanup2:
112  %later = cleanuppad within none []
113  cleanupret from %later unwind to caller
114; CHECK: [[cleanup2]]:
115; The cleanupret here needs to get redirected to the caller cleanup
116; CHECK-NEXT: %[[later:[^ ]+]] = cleanuppad within none
117; CHECK-NEXT: cleanupret from %[[later]] unwind label %cleanup{{$}}
118
119exit:
120  ret void
121}
122
123
124;;; Test with a call in a cleanup that has no definitive unwind
125;;; destination, that must be rewritten to an invoke.
126;;; CHECK-LABEL: define void @test3(
127define void @test3() personality void ()* @g {
128entry:
129; CHECK-NEXT: entry:
130  invoke void @test3_inlinee()
131    to label %exit unwind label %cleanup
132cleanup:
133  %pad = cleanuppad within none []
134  call void @g() [ "funclet"(token %pad) ]
135  cleanupret from %pad unwind to caller
136exit:
137  ret void
138}
139
140define void @test3_inlinee() alwaysinline personality void ()* @g {
141entry:
142  invoke void @g()
143    to label %exit unwind label %cleanup
144; CHECK-NEXT:  invoke void @g()
145; CHECK-NEXT:    unwind label %[[cleanup:.+]]
146
147cleanup:
148  %pad = cleanuppad within none []
149  call void @g() [ "funclet"(token %pad) ]
150  unreachable
151; CHECK: [[cleanup]]:
152; The call must be rewritten to an invoke targeting the caller cleanup
153; because it may well unwind to there.
154; CHECK-NEXT: %[[pad:[^ ]+]] = cleanuppad within none
155; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[pad]]) ]
156; CHECK-NEXT:   unwind label %cleanup{{$}}
157
158exit:
159  ret void
160}
161
162
163;;; Test with a catchswitch in a cleanup that has no definitive
164;;; unwind destination, that must be rewritten to unwind to the
165;;; inlined invoke's unwind dest
166;;; CHECK-LABEL: define void @test4(
167define void @test4() personality void ()* @g {
168entry:
169; CHECK-NEXT: entry:
170  invoke void @test4_inlinee()
171    to label %exit unwind label %cleanup
172cleanup:
173  %pad = cleanuppad within none []
174  call void @g() [ "funclet"(token %pad) ]
175  cleanupret from %pad unwind to caller
176exit:
177  ret void
178}
179
180define void @test4_inlinee() alwaysinline personality void ()* @g {
181entry:
182  invoke void @g()
183    to label %exit unwind label %cleanup
184; CHECK-NEXT: invoke void @g()
185; CHECK-NEXT:   unwind label %[[cleanup:.+]]
186
187cleanup:
188  %clean = cleanuppad within none []
189  invoke void @g() [ "funclet"(token %clean) ]
190    to label %unreachable unwind label %dispatch
191; CHECK: [[cleanup]]:
192; CHECK-NEXT: %[[clean:[^ ]+]] = cleanuppad within none
193; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[clean]]) ]
194; CHECK-NEXT:   unwind label %[[dispatch:.+]]
195
196dispatch:
197  %cs = catchswitch within %clean [label %catch] unwind to caller
198; CHECK: [[dispatch]]:
199; The catchswitch must be rewritten to unwind to %cleanup in the caller
200; because it may well unwind to there.
201; CHECK-NEXT: %[[cs:[^ ]+]] = catchswitch within %[[clean]] [label %[[catch:.+]]] unwind label %cleanup{{$}}
202
203catch:
204  catchpad within %cs []
205  br label %unreachable
206unreachable:
207  unreachable
208exit:
209  ret void
210}
211
212
213;;; Test with multiple levels of nesting, and unwind dests
214;;; that need to be inferred from ancestors, descendants,
215;;; and cousins.
216;;; CHECK-LABEL: define void @test5(
217define void @test5() personality void ()* @g {
218entry:
219; CHECK-NEXT: entry:
220  invoke void @test5_inlinee()
221    to label %exit unwind label %cleanup
222cleanup:
223  %pad = cleanuppad within none []
224  call void @g() [ "funclet"(token %pad) ]
225  cleanupret from %pad unwind to caller
226exit:
227  ret void
228}
229
230define void @test5_inlinee() alwaysinline personality void ()* @g {
231entry:
232  invoke void @g()
233    to label %cont unwind label %noinfo.root
234; CHECK-NEXT: invoke void @g()
235; CHECK-NEXT:   to label %[[cont:[^ ]+]] unwind label %[[noinfo_root:.+]]
236
237noinfo.root:
238  %noinfo.root.pad = cleanuppad within none []
239  call void @g() [ "funclet"(token %noinfo.root.pad) ]
240  invoke void @g() [ "funclet"(token %noinfo.root.pad) ]
241    to label %noinfo.root.cont unwind label %noinfo.left
242; CHECK: [[noinfo_root]]:
243; Nothing under "noinfo.root" has a definitive unwind destination, so
244; we must assume all of it may actually unwind, and redirect unwinds
245; to the cleanup in the caller.
246; CHECK-NEXT: %[[noinfo_root_pad:[^ ]+]] = cleanuppad within none []
247; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_root_pad]]) ]
248; CHECK-NEXT:   to label %[[next:[^ ]+]] unwind label %cleanup{{$}}
249; CHECK: [[next]]:
250; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_root_pad]]) ]
251; CHECK-NEXT:   to label %[[noinfo_root_cont:[^ ]+]] unwind label %[[noinfo_left:.+]]
252
253noinfo.left:
254  %noinfo.left.pad = cleanuppad within %noinfo.root.pad []
255  invoke void @g() [ "funclet"(token %noinfo.left.pad) ]
256    to label %unreachable unwind label %noinfo.left.child
257; CHECK: [[noinfo_left]]:
258; CHECK-NEXT: %[[noinfo_left_pad:[^ ]+]] = cleanuppad within %[[noinfo_root_pad]]
259; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_left_pad]]) ]
260; CHECK-NEXT:   unwind label %[[noinfo_left_child:.+]]
261
262noinfo.left.child:
263  %noinfo.left.child.cs = catchswitch within %noinfo.left.pad [label %noinfo.left.child.catch] unwind to caller
264; CHECK: [[noinfo_left_child]]:
265; CHECK-NEXT: %[[noinfo_left_child_cs:[^ ]+]] = catchswitch within %[[noinfo_left_pad]] [label %[[noinfo_left_child_catch:[^ ]+]]] unwind label %cleanup{{$}}
266
267noinfo.left.child.catch:
268  %noinfo.left.child.pad = catchpad within %noinfo.left.child.cs []
269  call void @g() [ "funclet"(token %noinfo.left.child.pad) ]
270  br label %unreachable
271; CHECK: [[noinfo_left_child_catch]]:
272; CHECK-NEXT: %[[noinfo_left_child_pad:[^ ]+]] = catchpad within %[[noinfo_left_child_cs]] []
273; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_left_child_pad]]) ]
274; CHECK-NEXT:   unwind label %cleanup{{$}}
275
276noinfo.root.cont:
277  invoke void @g() [ "funclet"(token %noinfo.root.pad) ]
278    to label %unreachable unwind label %noinfo.right
279; CHECK: [[noinfo_root_cont]]:
280; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_root_pad]]) ]
281; CHECK-NEXT:   unwind label %[[noinfo_right:.+]]
282
283noinfo.right:
284  %noinfo.right.cs = catchswitch within %noinfo.root.pad [label %noinfo.right.catch] unwind to caller
285; CHECK: [[noinfo_right]]:
286; CHECK-NEXT: %[[noinfo_right_cs:[^ ]+]] = catchswitch within %[[noinfo_root_pad]] [label %[[noinfo_right_catch:[^ ]+]]] unwind label %cleanup{{$}}
287
288noinfo.right.catch:
289  %noinfo.right.pad = catchpad within %noinfo.right.cs []
290  invoke void @g() [ "funclet"(token %noinfo.right.pad) ]
291    to label %unreachable unwind label %noinfo.right.child
292; CHECK: [[noinfo_right_catch]]:
293; CHECK-NEXT: %[[noinfo_right_pad:[^ ]+]] = catchpad within %[[noinfo_right_cs]]
294; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_right_pad]]) ]
295; CHECK-NEXT:   unwind label %[[noinfo_right_child:.+]]
296
297noinfo.right.child:
298  %noinfo.right.child.pad = cleanuppad within %noinfo.right.pad []
299  call void @g() [ "funclet"(token %noinfo.right.child.pad) ]
300  br label %unreachable
301; CHECK: [[noinfo_right_child]]:
302; CHECK-NEXT: %[[noinfo_right_child_pad:[^ ]+]] = cleanuppad within %[[noinfo_right_pad]]
303; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[noinfo_right_child_pad]]) ]
304; CHECK-NEXT:   unwind label %cleanup{{$}}
305
306cont:
307  invoke void @g()
308    to label %exit unwind label %implicit.root
309; CHECK: [[cont]]:
310; CHECK-NEXT: invoke void @g()
311; CHECK-NEXT:   unwind label %[[implicit_root:.+]]
312
313implicit.root:
314  %implicit.root.pad = cleanuppad within none []
315  call void @g() [ "funclet"(token %implicit.root.pad) ]
316  invoke void @g() [ "funclet"(token %implicit.root.pad) ]
317    to label %implicit.root.cont unwind label %implicit.left
318; CHECK: [[implicit_root]]:
319; There's an unwind edge to %internal in implicit.right, and we need to propagate that
320; fact down to implicit.right.grandchild, up to implicit.root, and down to
321; implicit.left.child.catch, leaving all calls and "unwind to caller" catchswitches
322; alone to so they don't conflict with the unwind edge in implicit.right
323; CHECK-NEXT: %[[implicit_root_pad:[^ ]+]] = cleanuppad within none
324; CHECK-NEXT: call void @g() [ "funclet"(token %[[implicit_root_pad]]) ]
325; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[implicit_root_pad]]) ]
326; CHECK-NEXT:   to label %[[implicit_root_cont:[^ ]+]] unwind label %[[implicit_left:.+]]
327
328implicit.left:
329  %implicit.left.pad = cleanuppad within %implicit.root.pad []
330  invoke void @g() [ "funclet"(token %implicit.left.pad) ]
331    to label %unreachable unwind label %implicit.left.child
332; CHECK: [[implicit_left]]:
333; CHECK-NEXT: %[[implicit_left_pad:[^ ]+]] = cleanuppad within %[[implicit_root_pad:[^ ]+]]
334; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[implicit_left_pad]]) ]
335; CHECK-NEXT:   unwind label %[[implicit_left_child:.+]]
336
337implicit.left.child:
338  %implicit.left.child.cs = catchswitch within %implicit.left.pad [label %implicit.left.child.catch] unwind to caller
339; CHECK: [[implicit_left_child]]:
340; CHECK-NEXT: %[[implicit_left_child_cs:[^ ]+]] = catchswitch within %[[implicit_left_pad]] [label %[[implicit_left_child_catch:[^ ]+]]] unwind to caller
341
342implicit.left.child.catch:
343  %implicit.left.child.pad = catchpad within %implicit.left.child.cs []
344  call void @g() [ "funclet"(token %implicit.left.child.pad) ]
345  br label %unreachable
346; CHECK: [[implicit_left_child_catch]]:
347; CHECK-NEXT: %[[implicit_left_child_pad:[^ ]+]] = catchpad within %[[implicit_left_child_cs]]
348; CHECK-NEXT: call void @g() [ "funclet"(token %[[implicit_left_child_pad]]) ]
349
350implicit.root.cont:
351  invoke void @g() [ "funclet"(token %implicit.root.pad) ]
352    to label %unreachable unwind label %implicit.right
353; CHECK: [[implicit_root_cont]]:
354; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[implicit_root_pad]]) ]
355; CHECK-NEXT:   unwind label %[[implicit_right:.+]]
356
357implicit.right:
358  %implicit.right.cs = catchswitch within %implicit.root.pad [label %implicit.right.catch] unwind label %internal
359; CHECK: [[implicit_right]]:
360; This is the unwind edge (to %internal) whose existence needs to get propagated around the "implicit" tree
361; CHECK-NEXT: %[[implicit_right_cs:[^ ]+]] = catchswitch within %[[implicit_root_pad]] [label %[[implicit_right_catch:[^ ]+]]] unwind label %[[internal:.+]]
362
363implicit.right.catch:
364  %implicit.right.pad = catchpad within %implicit.right.cs []
365  invoke void @g() [ "funclet"(token %implicit.right.pad) ]
366    to label %unreachable unwind label %implicit.right.child
367; CHECK: [[implicit_right_catch]]:
368; CHECK-NEXT: %[[implicit_right_pad:[^ ]+]] = catchpad within %[[implicit_right_cs]]
369; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[implicit_right_pad]]) ]
370; CHECK-NEXT:   unwind label %[[implicit_right_child:.+]]
371
372implicit.right.child:
373  %implicit.right.child.pad = cleanuppad within %implicit.right.pad []
374  invoke void @g() [ "funclet"(token %implicit.right.child.pad) ]
375    to label %unreachable unwind label %implicit.right.grandchild
376; CHECK: [[implicit_right_child]]:
377; CHECK-NEXT: %[[implicit_right_child_pad:[^ ]+]] = cleanuppad within %[[implicit_right_pad]]
378; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[implicit_right_child_pad]]) ]
379; CHECK-NEXT:   unwind label %[[implicit_right_grandchild:.+]]
380
381implicit.right.grandchild:
382  %implicit.right.grandchild.cs = catchswitch within %implicit.right.child.pad [label %implicit.right.grandchild.catch] unwind to caller
383; CHECK: [[implicit_right_grandchild]]:
384; CHECK-NEXT: %[[implicit_right_grandchild_cs:[^ ]+]] = catchswitch within %[[implicit_right_child_pad]] [label %[[implicit_right_grandchild_catch:[^ ]+]]] unwind to caller
385
386implicit.right.grandchild.catch:
387  %implicit.right.grandhcild.pad = catchpad within %implicit.right.grandchild.cs []
388  call void @g() [ "funclet"(token %implicit.right.grandhcild.pad) ]
389  br label %unreachable
390; CHECK: [[implicit_right_grandchild_catch]]:
391; CHECK-NEXT: %[[implicit_right_grandhcild_pad:[^ ]+]] = catchpad within %[[implicit_right_grandchild_cs]]
392; CHECK-NEXT: call void @g() [ "funclet"(token %[[implicit_right_grandhcild_pad]]) ]
393
394internal:
395  %internal.pad = cleanuppad within none []
396  call void @g() [ "funclet"(token %internal.pad) ]
397  cleanupret from %internal.pad unwind to caller
398; CHECK: [[internal]]:
399; internal is a cleanup with a "return to caller" cleanuppad; that needs to get redirected
400; to %cleanup in the caller, and the call needs to get similarly rewritten to an invoke.
401; CHECK-NEXT: %[[internal_pad:[^ ]+]] = cleanuppad within none
402; CHECK-NEXT: invoke void @g() [ "funclet"(token %internal.pad.i) ]
403; CHECK-NEXT:   to label %[[next:[^ ]+]] unwind label %cleanup{{$}}
404; CHECK: [[next]]:
405; CHECK-NEXT: cleanupret from %[[internal_pad]] unwind label %cleanup{{$}}
406
407unreachable:
408  unreachable
409exit:
410  ret void
411}
412
413;;; Test with funclets that don't have information for themselves, but have
414;;; descendants which unwind to other descendants (left.left unwinds to
415;;; left.right, and right unwinds to far_right).  Make sure that these local
416;;; unwinds don't trip up processing of the ancestor nodes (left and root) that
417;;; ultimately have no information.
418;;; CHECK-LABEL: define void @test6(
419define void @test6() personality void()* @ProcessCLRException {
420entry:
421; CHECK-NEXT: entry:
422  invoke void @test6_inlinee()
423    to label %exit unwind label %cleanup
424cleanup:
425  %pad = cleanuppad within none []
426  call void @g() [ "funclet"(token %pad) ]
427  cleanupret from %pad unwind to caller
428exit:
429  ret void
430}
431
432define void @test6_inlinee() alwaysinline personality void ()* @ProcessCLRException {
433entry:
434  invoke void @g()
435    to label %exit unwind label %root
436    ; CHECK-NEXT:  invoke void @g()
437    ; CHECK-NEXT:    unwind label %[[root:.+]]
438root:
439  %root.pad = cleanuppad within none []
440  invoke void @g() [ "funclet"(token %root.pad) ]
441    to label %root.cont unwind label %left
442; CHECK: [[root]]:
443; CHECK-NEXT: %[[root_pad:.+]] = cleanuppad within none []
444; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[root_pad]]) ]
445; CHECK-NEXT:   to label %[[root_cont:.+]] unwind label %[[left:.+]]
446
447left:
448  %left.cs = catchswitch within %root.pad [label %left.catch] unwind to caller
449; CHECK: [[left]]:
450; CHECK-NEXT: %[[left_cs:.+]] = catchswitch within %[[root_pad]] [label %[[left_catch:.+]]] unwind label %cleanup
451
452left.catch:
453  %left.cp = catchpad within %left.cs []
454  call void @g() [ "funclet"(token %left.cp) ]
455  invoke void @g() [ "funclet"(token %left.cp) ]
456    to label %unreach unwind label %left.left
457; CHECK: [[left_catch:.+]]:
458; CHECK-NEXT: %[[left_cp:.+]] = catchpad within %[[left_cs]] []
459; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[left_cp]]) ]
460; CHECK-NEXT:   to label %[[lc_cont:.+]] unwind label %cleanup
461; CHECK: [[lc_cont]]:
462; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[left_cp]]) ]
463; CHECK-NEXT:   to label %[[unreach:.+]] unwind label %[[left_left:.+]]
464
465left.left:
466  %ll.pad = cleanuppad within %left.cp []
467  cleanupret from %ll.pad unwind label %left.right
468; CHECK: [[left_left]]:
469; CHECK-NEXT: %[[ll_pad:.+]] = cleanuppad within %[[left_cp]] []
470; CHECK-NEXT: cleanupret from %[[ll_pad]] unwind label %[[left_right:.+]]
471
472left.right:
473  %lr.pad = cleanuppad within %left.cp []
474  unreachable
475; CHECK: [[left_right]]:
476; CHECK-NEXT: %[[lr_pad:.+]] = cleanuppad within %[[left_cp]] []
477; CHECK-NEXT: unreachable
478
479root.cont:
480  call void @g() [ "funclet"(token %root.pad) ]
481  invoke void @g() [ "funclet"(token %root.pad) ]
482    to label %unreach unwind label %right
483; CHECK: [[root_cont]]:
484; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[root_pad]]) ]
485; CHECK-NEXT:   to label %[[root_cont_cont:.+]] unwind label %cleanup
486; CHECK: [[root_cont_cont]]:
487; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[root_pad]]) ]
488; CHECK-NEXT:   to label %[[unreach]] unwind label %[[right:.+]]
489
490right:
491  %right.pad = cleanuppad within %root.pad []
492  invoke void @g() [ "funclet"(token %right.pad) ]
493    to label %unreach unwind label %right.child
494; CHECK: [[right]]:
495; CHECK-NEXT: %[[right_pad:.+]] = cleanuppad within %[[root_pad]] []
496; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[right_pad]]) ]
497; CHECK-NEXT:   to label %[[unreach]] unwind label %[[right_child:.+]]
498
499right.child:
500  %rc.pad = cleanuppad within %right.pad []
501  invoke void @g() [ "funclet"(token %rc.pad) ]
502    to label %unreach unwind label %far_right
503; CHECK: [[right_child]]:
504; CHECK-NEXT: %[[rc_pad:.+]] = cleanuppad within %[[right_pad]] []
505; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[rc_pad]]) ]
506; CHECK-NEXT:   to label %[[unreach]] unwind label %[[far_right:.+]]
507
508far_right:
509  %fr.cs = catchswitch within %root.pad [label %fr.catch] unwind to caller
510; CHECK: [[far_right]]:
511; CHECK-NEXT: %[[fr_cs:.+]] = catchswitch within %[[root_pad]] [label %[[fr_catch:.+]]] unwind label %cleanup
512
513fr.catch:
514  %fr.cp = catchpad within %fr.cs []
515  unreachable
516; CHECK: [[fr_catch]]:
517; CHECK-NEXT: %[[fr_cp:.+]] = catchpad within %[[fr_cs]] []
518; CHECK-NEXT: unreachable
519
520unreach:
521  unreachable
522; CHECK: [[unreach]]:
523; CHECK-NEXT: unreachable
524
525exit:
526  ret void
527}
528
529
530;;; Test with a no-info funclet (right) which has a cousin (left.left) that
531;;; unwinds to another cousin (left.right); make sure we don't trip over this
532;;; when propagating unwind destination info to "right".
533;;; CHECK-LABEL: define void @test7(
534define void @test7() personality void()* @ProcessCLRException {
535entry:
536; CHECK-NEXT: entry:
537  invoke void @test7_inlinee()
538    to label %exit unwind label %cleanup
539cleanup:
540  %pad = cleanuppad within none []
541  call void @g() [ "funclet"(token %pad) ]
542  cleanupret from %pad unwind to caller
543exit:
544  ret void
545}
546
547define void @test7_inlinee() alwaysinline personality void ()* @ProcessCLRException {
548entry:
549  invoke void @g()
550    to label %exit unwind label %root
551; CHECK-NEXT:  invoke void @g()
552; CHECK-NEXT:    unwind label %[[root:.+]]
553
554root:
555  %root.cp = cleanuppad within none []
556  invoke void @g() [ "funclet"(token %root.cp) ]
557    to label %root.cont unwind label %child
558; CHECK: [[root]]:
559; CHECK-NEXT: %[[root_cp:.+]] = cleanuppad within none []
560; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[root_cp]]) ]
561; CHECK-NEXT:   to label %[[root_cont:.+]] unwind label %[[child:.+]]
562
563root.cont:
564  cleanupret from %root.cp unwind to caller
565; CHECK: [[root_cont]]:
566; CHECK-NEXT: cleanupret from %[[root_cp]] unwind label %cleanup
567
568child:
569  %child.cp = cleanuppad within %root.cp []
570  invoke void @g() [ "funclet"(token %child.cp) ]
571    to label %child.cont unwind label %left
572; CHECK: [[child]]:
573; CHECK-NEXT: %[[child_cp:.+]] = cleanuppad within %[[root_cp]] []
574; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[child_cp]]) ]
575; CHECK-NEXT:   to label %[[child_cont:.+]] unwind label %[[left:.+]]
576
577left:
578  %left.cp = cleanuppad within %child.cp []
579  invoke void @g() [ "funclet"(token %left.cp) ]
580    to label %left.cont unwind label %left.left
581; CHECK: [[left]]:
582; CHECK-NEXT: %[[left_cp:.+]] = cleanuppad within %[[child_cp]] []
583; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[left_cp]]) ]
584; CHECK-NEXT:   to label %[[left_cont:.+]] unwind label %[[left_left:.+]]
585
586left.left:
587  %ll.cp = cleanuppad within %left.cp []
588  cleanupret from %ll.cp unwind label %left.right
589; CHECK: [[left_left]]:
590; CHECK-NEXT: %[[ll_cp:.+]] = cleanuppad within %[[left_cp]] []
591; CHECK-NEXT: cleanupret from %[[ll_cp]] unwind label %[[left_right:.+]]
592
593left.cont:
594  invoke void @g() [ "funclet"(token %left.cp) ]
595    to label %unreach unwind label %left.right
596; CHECK: [[left_cont]]:
597; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[left_cp]]) ]
598; CHECK-NEXT:   to label %[[unreach:.+]] unwind label %[[left_right]]
599
600left.right:
601  %lr.cp = cleanuppad within %left.cp []
602  unreachable
603; CHECK: [[left_right]]:
604; CHECK-NEXT: %[[lr_cp:.+]] = cleanuppad within %[[left_cp]] []
605; CHECK-NEXT: unreachable
606
607child.cont:
608  invoke void @g() [ "funclet"(token %child.cp) ]
609    to label %unreach unwind label %right
610; CHECK: [[child_cont]]:
611; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[child_cp]]) ]
612; CHECK-NEXT:   to label %[[unreach]] unwind label %[[right:.+]]
613
614right:
615  %right.cp = cleanuppad within %child.cp []
616  call void @g() [ "funclet"(token %right.cp) ]
617  unreachable
618; CHECK: [[right]]:
619; CHECK-NEXT: %[[right_cp:.+]] = cleanuppad within %[[child_cp]]
620; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[right_cp]]) ]
621; CHECK-NEXT:   to label %[[right_cont:.+]] unwind label %cleanup
622; CHECK: [[right_cont]]:
623; CHECK-NEXT: unreachable
624
625unreach:
626  unreachable
627; CHECK: [[unreach]]:
628; CHECK-NEXT: unreachable
629
630exit:
631  ret void
632}
633
634declare void @ProcessCLRException()
635
636; Make sure the logic doesn't get tripped up when the inlined invoke is
637; itself within a funclet in the caller.
638; CHECK-LABEL: define void @test8(
639define void @test8() personality void ()* @ProcessCLRException {
640entry:
641  invoke void @g()
642    to label %exit unwind label %callsite_parent
643callsite_parent:
644  %callsite_parent.pad = cleanuppad within none []
645; CHECK: %callsite_parent.pad = cleanuppad within none
646  invoke void @test8_inlinee() [ "funclet"(token %callsite_parent.pad) ]
647    to label %ret unwind label %cleanup
648ret:
649  cleanupret from %callsite_parent.pad unwind label %cleanup
650cleanup:
651  %pad = cleanuppad within none []
652  call void @g() [ "funclet"(token %pad) ]
653  cleanupret from %pad unwind to caller
654exit:
655  ret void
656}
657
658define void @test8_inlinee() alwaysinline personality void ()* @ProcessCLRException {
659entry:
660  invoke void @g()
661    to label %exit unwind label %inlinee_cleanup
662; CHECK-NEXT: invoke void @g() [ "funclet"(token %callsite_parent.pad) ]
663; CHECK-NEXT:   unwind label %[[inlinee_cleanup:.+]]
664
665inlinee_cleanup:
666  %inlinee.pad = cleanuppad within none []
667  call void @g() [ "funclet"(token %inlinee.pad) ]
668  unreachable
669; CHECK: [[inlinee_cleanup]]:
670; CHECK-NEXT: %[[inlinee_pad:[^ ]+]] = cleanuppad within %callsite_parent.pad
671; CHECK-NEXT: invoke void @g() [ "funclet"(token %[[inlinee_pad]]) ]
672; CHECK-NEXT:   unwind label %cleanup{{$}}
673
674exit:
675  ret void
676}
677