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