diff options
author | Gor Nishanov <GorNishanov@gmail.com> | 2017-05-17 03:09:22 +0000 |
---|---|---|
committer | Gor Nishanov <GorNishanov@gmail.com> | 2017-05-17 03:09:22 +0000 |
commit | db38485588fdba5b02225c7973c4cc1c74edf544 (patch) | |
tree | b60de95f96d7209cae2840d4daf352c60985a9b1 /llvm/test/Transforms/Coroutines | |
parent | 58bb7bd18acc63b90e6a98d4c1a337e76412e0a6 (diff) | |
download | bcm5719-llvm-db38485588fdba5b02225c7973c4cc1c74edf544.tar.gz bcm5719-llvm-db38485588fdba5b02225c7973c4cc1c74edf544.zip |
[coroutines] Handle spills before catchswitch
If we need to spill the result of the PHI instruction, we insert the spill after
all of the PHIs and EHPads, however, in a catchswitch block there is no
room to insert the spill. Make room by splitting away catchswitch into a separate
block.
Before the fix:
catch.dispatch:
%val = phi i32 [ 1, %if.then ], [ 2, %if.else ]
%switch = catchswitch within none [label %catch] unwind label %cleanuppad
After:
catch.dispatch:
%val = phi i32 [ 1, %if.then ], [ 2, %if.else ]
%tok = cleanuppad within none []
; spill goes here
cleanupret from %tok unwind label %catch.dispatch.switch
catch.dispatch.switch:
%switch = catchswitch within none [label %catch] unwind label %cleanuppad
https://reviews.llvm.org/D31846
llvm-svn: 303232
Diffstat (limited to 'llvm/test/Transforms/Coroutines')
-rw-r--r-- | llvm/test/Transforms/Coroutines/coro-catchswitch.ll | 88 |
1 files changed, 88 insertions, 0 deletions
diff --git a/llvm/test/Transforms/Coroutines/coro-catchswitch.ll b/llvm/test/Transforms/Coroutines/coro-catchswitch.ll new file mode 100644 index 00000000000..dd06f1280ca --- /dev/null +++ b/llvm/test/Transforms/Coroutines/coro-catchswitch.ll @@ -0,0 +1,88 @@ +; Verifies that we can insert the spill for a PHI preceding the catchswitch +; RUN: opt < %s -coro-split -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) "coroutine.presplit"="1" personality i32 0 { +entry: + %id = call token @llvm.coro.id(i32 8, i8* null, i8* null, i8* null) + %size = call i32 @llvm.coro.size.i32() + %alloc = call i8* @malloc(i32 %size) + %hdl = call i8* @llvm.coro.begin(token %id, i8* %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, %f.Frame* %FramePtr, i32 0, i32 4 +; CHECK: store i32 %val, i32* %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 [i8* null, i32 64, i8* 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(i8* %hdl, i1 0) + ret void + +cleanuppad: + %cpad = cleanuppad within none [] + cleanupret from %cpad unwind to caller +} + +; Function Attrs: argmemonly nounwind readonly +declare token @llvm.coro.id(i32, i8* readnone, i8* nocapture readonly, i8*) #1 + +; Function Attrs: nounwind +declare i1 @llvm.coro.alloc(token) #2 + +; Function Attrs: nobuiltin +declare i32 @llvm.coro.size.i32() #4 +declare i8* @llvm.coro.begin(token, i8* writeonly) #2 +declare token @llvm.coro.save(i8*) +declare i8 @llvm.coro.suspend(token, i1) + +declare void @may_throw1() +declare void @may_throw2() +declare void @print(i32) +declare noalias i8* @malloc(i32) +declare void @free(i8*) + +declare i1 @llvm.coro.end(i8*, i1) #2 + +; Function Attrs: nobuiltin nounwind + +; Function Attrs: argmemonly nounwind readonly +declare i8* @llvm.coro.free(token, i8* nocapture readonly) #1 |