| ; Verifies that we can insert the spill for a PHI preceding the catchswitch |
| ; RUN: opt < %s -passes='cgscc(coro-split),simplifycfg,early-cse' -S | FileCheck %s |
| |
| target datalayout = "e-m:x-p:32:32-i64:64-f80:32-n8:16:32-a:0:32-S32" |
| target triple = "i686-pc-windows-msvc" |
| |
| ; CHECK-LABEL: define void @f( |
| define void @f(i1 %cond) presplitcoroutine personality i32 0 { |
| entry: |
| %id = call token @llvm.coro.id(i32 8, ptr null, ptr null, ptr null) |
| %size = call i32 @llvm.coro.size.i32() |
| %alloc = call ptr @malloc(i32 %size) |
| %hdl = call ptr @llvm.coro.begin(token %id, ptr %alloc) |
| br i1 %cond, label %if.else, label %if.then |
| |
| if.then: |
| invoke void @may_throw1() |
| to label %coro.ret unwind label %catch.dispatch |
| |
| if.else: |
| invoke void @may_throw2() |
| to label %coro.ret unwind label %catch.dispatch |
| |
| catch.dispatch: ; preds = %if.else, %if.then |
| %val = phi i32 [ 1, %if.then ], [ 2, %if.else ] |
| %switch = catchswitch within none [label %catch] unwind label %cleanuppad |
| |
| ; Verifies that we split out the PHI into a separate block |
| ; added a cleanuppad spill cleanupret unwinding into the catchswitch. |
| |
| ; CHECK: catch.dispatch: |
| ; CHECK: %val = phi i32 [ 2, %if.else ], [ 1, %if.then ] |
| ; CHECK: %[[Pad:.+]] = cleanuppad within none [] |
| ; CHECK: %val.spill.addr = getelementptr inbounds %f.Frame, ptr %hdl, i32 0, i32 2 |
| ; CHECK: store i32 %val, ptr %val.spill.addr |
| ; CHECK: cleanupret from %[[Pad]] unwind label %[[Switch:.+]] |
| |
| ; CHECK: [[Switch]]: |
| ; CHECK: %switch = catchswitch within none [label %catch] unwind to caller |
| |
| catch: ; preds = %catch.dispatch |
| %pad = catchpad within %switch [ptr null, i32 64, ptr null] |
| catchret from %pad to label %suspend |
| |
| suspend: |
| %sp = call i8 @llvm.coro.suspend(token none, i1 false) |
| switch i8 %sp, label %coro.ret [ |
| i8 0, label %resume |
| i8 1, label %coro.ret |
| ] |
| |
| resume: ; preds = %await2.suspend |
| call void @print(i32 %val) |
| br label %coro.ret |
| |
| coro.ret: |
| call i1 @llvm.coro.end(ptr %hdl, i1 0, token none) |
| ret void |
| |
| cleanuppad: |
| %cpad = cleanuppad within none [] |
| cleanupret from %cpad unwind to caller |
| } |
| |
| ; Function Attrs: argmemonly nounwind readonly |
| declare token @llvm.coro.id(i32, ptr readnone, ptr nocapture readonly, ptr) #1 |
| |
| ; Function Attrs: nounwind |
| declare i1 @llvm.coro.alloc(token) #2 |
| |
| ; Function Attrs: nobuiltin |
| declare i32 @llvm.coro.size.i32() #4 |
| declare ptr @llvm.coro.begin(token, ptr writeonly) #2 |
| declare token @llvm.coro.save(ptr) |
| declare i8 @llvm.coro.suspend(token, i1) |
| |
| declare void @may_throw1() |
| declare void @may_throw2() |
| declare void @print(i32) |
| declare noalias ptr @malloc(i32) |
| declare void @free(ptr) |
| |
| declare i1 @llvm.coro.end(ptr, i1, token) #2 |
| |
| ; Function Attrs: nobuiltin nounwind |
| |
| ; Function Attrs: argmemonly nounwind readonly |
| declare ptr @llvm.coro.free(token, ptr nocapture readonly) #1 |