Home | History | Annotate | Download | only in Inline
      1 ; RUN: opt -inline -S %s | FileCheck %s
      2 ; RUN: opt -passes='cgscc(inline)' -S %s | FileCheck %s
      3 
      4 declare 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(
     10 define void @test1() personality void ()* @g {
     11 entry:
     12 ; CHECK-NEXT: entry:
     13   invoke void @test1_inlinee()
     14     to label %exit unwind label %cleanup
     15 cleanup:
     16   %pad = cleanuppad within none []
     17   call void @g() [ "funclet"(token %pad) ]
     18   cleanupret from %pad unwind to caller
     19 exit:
     20   ret void
     21 }
     22 
     23 define void @test1_inlinee() alwaysinline personality void ()* @g {
     24 entry:
     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 
     30 cleanup.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 
     41 cleanup.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 
     52 exit:
     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(
     62 define void @test2() personality void ()* @g {
     63 entry:
     64 ; CHECK-NEXT: entry:
     65   invoke void @test2_inlinee()
     66     to label %exit unwind label %cleanup
     67 cleanup:
     68   %pad = cleanuppad within none []
     69   call void @g() [ "funclet"(token %pad) ]
     70   cleanupret from %pad unwind to caller
     71 exit:
     72   ret void
     73 }
     74 
     75 define void @test2_inlinee() alwaysinline personality void ()* @g {
     76 entry:
     77   invoke void @g()
     78     to label %exit unwind label %cleanup1
     79 ; CHECK-NEXT:   invoke void @g()
     80 ; CHECK-NEXT:     unwind label %[[cleanup1:.+]]
     81 
     82 cleanup1:
     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 
     91 catchswitch:
     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 
     98 catch:
     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 
    107 ret1:
    108   cleanupret from %outer unwind label %cleanup2
    109 ; CHECK: cleanupret from %[[outer]] unwind label %[[cleanup2:.+]]
    110 
    111 cleanup2:
    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 
    119 exit:
    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(
    127 define void @test3() personality void ()* @g {
    128 entry:
    129 ; CHECK-NEXT: entry:
    130   invoke void @test3_inlinee()
    131     to label %exit unwind label %cleanup
    132 cleanup:
    133   %pad = cleanuppad within none []
    134   call void @g() [ "funclet"(token %pad) ]
    135   cleanupret from %pad unwind to caller
    136 exit:
    137   ret void
    138 }
    139 
    140 define void @test3_inlinee() alwaysinline personality void ()* @g {
    141 entry:
    142   invoke void @g()
    143     to label %exit unwind label %cleanup
    144 ; CHECK-NEXT:  invoke void @g()
    145 ; CHECK-NEXT:    unwind label %[[cleanup:.+]]
    146 
    147 cleanup:
    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 
    158 exit:
    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(
    167 define void @test4() personality void ()* @g {
    168 entry:
    169 ; CHECK-NEXT: entry:
    170   invoke void @test4_inlinee()
    171     to label %exit unwind label %cleanup
    172 cleanup:
    173   %pad = cleanuppad within none []
    174   call void @g() [ "funclet"(token %pad) ]
    175   cleanupret from %pad unwind to caller
    176 exit:
    177   ret void
    178 }
    179 
    180 define void @test4_inlinee() alwaysinline personality void ()* @g {
    181 entry:
    182   invoke void @g()
    183     to label %exit unwind label %cleanup
    184 ; CHECK-NEXT: invoke void @g()
    185 ; CHECK-NEXT:   unwind label %[[cleanup:.+]]
    186 
    187 cleanup:
    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 
    196 dispatch:
    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 
    203 catch:
    204   catchpad within %cs []
    205   br label %unreachable
    206 unreachable:
    207   unreachable
    208 exit:
    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(
    217 define void @test5() personality void ()* @g {
    218 entry:
    219 ; CHECK-NEXT: entry:
    220   invoke void @test5_inlinee()
    221     to label %exit unwind label %cleanup
    222 cleanup:
    223   %pad = cleanuppad within none []
    224   call void @g() [ "funclet"(token %pad) ]
    225   cleanupret from %pad unwind to caller
    226 exit:
    227   ret void
    228 }
    229 
    230 define void @test5_inlinee() alwaysinline personality void ()* @g {
    231 entry:
    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 
    237 noinfo.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 
    253 noinfo.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 
    262 noinfo.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 
    267 noinfo.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 
    276 noinfo.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 
    283 noinfo.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 
    288 noinfo.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 
    297 noinfo.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 
    306 cont:
    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 
    313 implicit.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 
    328 implicit.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 
    337 implicit.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 
    342 implicit.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 
    350 implicit.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 
    357 implicit.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 
    363 implicit.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 
    372 implicit.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 
    381 implicit.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 
    386 implicit.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 
    394 internal:
    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 
    407 unreachable:
    408   unreachable
    409 exit:
    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(
    419 define void @test6() personality void()* @ProcessCLRException {
    420 entry:
    421 ; CHECK-NEXT: entry:
    422   invoke void @test6_inlinee()
    423     to label %exit unwind label %cleanup
    424 cleanup:
    425   %pad = cleanuppad within none []
    426   call void @g() [ "funclet"(token %pad) ]
    427   cleanupret from %pad unwind to caller
    428 exit:
    429   ret void
    430 }
    431 
    432 define void @test6_inlinee() alwaysinline personality void ()* @ProcessCLRException {
    433 entry:
    434   invoke void @g()
    435     to label %exit unwind label %root
    436     ; CHECK-NEXT:  invoke void @g()
    437     ; CHECK-NEXT:    unwind label %[[root:.+]]
    438 root:
    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 
    447 left:
    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 
    452 left.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 
    465 left.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 
    472 left.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 
    479 root.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 
    490 right:
    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 
    499 right.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 
    508 far_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 
    513 fr.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 
    520 unreach:
    521   unreachable
    522 ; CHECK: [[unreach]]:
    523 ; CHECK-NEXT: unreachable
    524 
    525 exit:
    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(
    534 define void @test7() personality void()* @ProcessCLRException {
    535 entry:
    536 ; CHECK-NEXT: entry:
    537   invoke void @test7_inlinee()
    538     to label %exit unwind label %cleanup
    539 cleanup:
    540   %pad = cleanuppad within none []
    541   call void @g() [ "funclet"(token %pad) ]
    542   cleanupret from %pad unwind to caller
    543 exit:
    544   ret void
    545 }
    546 
    547 define void @test7_inlinee() alwaysinline personality void ()* @ProcessCLRException {
    548 entry:
    549   invoke void @g()
    550     to label %exit unwind label %root
    551 ; CHECK-NEXT:  invoke void @g()
    552 ; CHECK-NEXT:    unwind label %[[root:.+]]
    553 
    554 root:
    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 
    563 root.cont:
    564   cleanupret from %root.cp unwind to caller
    565 ; CHECK: [[root_cont]]:
    566 ; CHECK-NEXT: cleanupret from %[[root_cp]] unwind label %cleanup
    567 
    568 child:
    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 
    577 left:
    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 
    586 left.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 
    593 left.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 
    600 left.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 
    607 child.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 
    614 right:
    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 
    625 unreach:
    626   unreachable
    627 ; CHECK: [[unreach]]:
    628 ; CHECK-NEXT: unreachable
    629 
    630 exit:
    631   ret void
    632 }
    633 
    634 declare 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(
    639 define void @test8() personality void ()* @ProcessCLRException {
    640 entry:
    641   invoke void @g()
    642     to label %exit unwind label %callsite_parent
    643 callsite_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
    648 ret:
    649   cleanupret from %callsite_parent.pad unwind label %cleanup
    650 cleanup:
    651   %pad = cleanuppad within none []
    652   call void @g() [ "funclet"(token %pad) ]
    653   cleanupret from %pad unwind to caller
    654 exit:
    655   ret void
    656 }
    657 
    658 define void @test8_inlinee() alwaysinline personality void ()* @ProcessCLRException {
    659 entry:
    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 
    665 inlinee_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 
    674 exit:
    675   ret void
    676 }
    677