1; RUN: opt < %s -passes='cgscc(coro-split),simplifycfg,early-cse' -S | FileCheck %s 2 3declare ptr @malloc(i64) 4declare void @free(ptr) 5declare void @usePointer(ptr) 6declare void @usePointer2(ptr) 7 8declare token @llvm.coro.id(i32, ptr readnone, ptr nocapture readonly, ptr) 9declare i64 @llvm.coro.size.i64() 10declare ptr @llvm.coro.begin(token, ptr writeonly) 11declare i8 @llvm.coro.suspend(token, i1) 12declare i1 @llvm.coro.end(ptr, i1, token) 13declare ptr @llvm.coro.free(token, ptr nocapture readonly) 14declare token @llvm.coro.save(ptr) 15 16define void @foo() presplitcoroutine { 17entry: 18 %a0 = alloca [0 x i8] 19 %a1 = alloca i32 20 %a2 = alloca [0 x i8] 21 %a3 = alloca [0 x i8] 22 %a4 = alloca i16 23 %a5 = alloca [0 x i8] 24 %coro.id = call token @llvm.coro.id(i32 0, ptr null, ptr null, ptr null) 25 %coro.size = call i64 @llvm.coro.size.i64() 26 %coro.alloc = call ptr @malloc(i64 %coro.size) 27 %coro.state = call ptr @llvm.coro.begin(token %coro.id, ptr %coro.alloc) 28 %coro.save = call token @llvm.coro.save(ptr %coro.state) 29 %call.suspend = call i8 @llvm.coro.suspend(token %coro.save, i1 false) 30 switch i8 %call.suspend, label %suspend [ 31 i8 0, label %wakeup 32 i8 1, label %cleanup 33 ] 34 35wakeup: ; preds = %entry 36 call void @usePointer(ptr %a0) 37 call void @usePointer(ptr %a1) 38 call void @usePointer(ptr %a2) 39 call void @usePointer(ptr %a3) 40 call void @usePointer(ptr %a4) 41 call void @usePointer2(ptr %a5) 42 br label %cleanup 43 44suspend: ; preds = %cleanup, %entry 45 %unused = call i1 @llvm.coro.end(ptr %coro.state, i1 false, token none) 46 ret void 47 48cleanup: ; preds = %wakeup, %entry 49 %coro.memFree = call ptr @llvm.coro.free(token %coro.id, ptr %coro.state) 50 call void @free(ptr %coro.memFree) 51 br label %suspend 52} 53 54; CHECK: %foo.Frame = type { ptr, ptr, i32, i16, i1 } 55 56; CHECK-LABEL: @foo.resume( 57; CHECK-NEXT: entry.resume: 58; CHECK-NEXT: [[A1_RELOAD_ADDR:%.*]] = getelementptr inbounds [[FOO_FRAME:%foo.Frame]], ptr [[FRAMEPTR:%.*]], i32 0, i32 2 59; CHECK-NEXT: [[A4_RELOAD_ADDR:%.*]] = getelementptr inbounds [[FOO_FRAME]], ptr [[FRAMEPTR]], i32 0, i32 3 60; CHECK-NEXT: call void @usePointer(ptr [[FRAMEPTR]]) 61; CHECK-NEXT: call void @usePointer(ptr [[A1_RELOAD_ADDR]]) 62; CHECK-NEXT: call void @usePointer(ptr [[FRAMEPTR]]) 63; CHECK-NEXT: call void @usePointer(ptr [[FRAMEPTR]]) 64; CHECK-NEXT: call void @usePointer(ptr [[A4_RELOAD_ADDR]]) 65; CHECK-NEXT: call void @usePointer2(ptr [[FRAMEPTR]]) 66; CHECK-NEXT: call void @free(ptr [[FRAMEPTR]]) 67; CHECK-NEXT: ret void 68