Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / CodeGenCoroutines / coro-symmetric-transfer-02.cpp
blob890d55e41de95380c7ad53265d19af2a1de79f4a
1 // RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -std=c++20 -O1 -emit-llvm %s -o - -disable-llvm-passes | FileCheck %s
3 #include "Inputs/coroutine.h"
5 struct Task {
6 struct promise_type {
7 Task get_return_object() noexcept {
8 return Task{std::coroutine_handle<promise_type>::from_promise(*this)};
11 void return_void() noexcept {}
13 struct final_awaiter {
14 bool await_ready() noexcept { return false; }
15 std::coroutine_handle<> await_suspend(std::coroutine_handle<promise_type> h) noexcept {
16 h.destroy();
17 return {};
19 void await_resume() noexcept {}
22 void unhandled_exception() noexcept {}
24 final_awaiter final_suspend() noexcept { return {}; }
26 std::suspend_always initial_suspend() noexcept { return {}; }
28 template <typename Awaitable>
29 auto await_transform(Awaitable &&awaitable) {
30 return awaitable.co_viaIfAsync();
34 using handle_t = std::coroutine_handle<promise_type>;
36 class Awaiter {
37 public:
38 explicit Awaiter(handle_t coro) noexcept;
39 Awaiter(Awaiter &&other) noexcept;
40 Awaiter(const Awaiter &) = delete;
41 ~Awaiter();
43 bool await_ready() noexcept { return false; }
44 handle_t await_suspend(std::coroutine_handle<> continuation) noexcept;
45 void await_resume();
47 private:
48 handle_t coro_;
51 Task(handle_t coro) noexcept : coro_(coro) {}
53 handle_t coro_;
55 Task(const Task &t) = delete;
56 Task(Task &&t) noexcept;
57 ~Task();
58 Task &operator=(Task t) noexcept;
60 Awaiter co_viaIfAsync();
63 static Task foo() {
64 co_return;
67 Task bar() {
68 auto mode = 2;
69 switch (mode) {
70 case 1:
71 co_await foo();
72 break;
73 case 2:
74 co_await foo();
75 break;
76 default:
77 break;
81 // CHECK-LABEL: define{{.*}} void @_Z3barv
82 // CHECK: %[[MODE:.+]] = load i32, ptr %mode
83 // CHECK-NEXT: switch i32 %[[MODE]], label %{{.+}} [
84 // CHECK-NEXT: i32 1, label %[[CASE1:.+]]
85 // CHECK-NEXT: i32 2, label %[[CASE2:.+]]
86 // CHECK-NEXT: ]
88 // CHECK: [[CASE1]]:
89 // CHECK: br i1 %{{.+}}, label %[[CASE1_AWAIT_READY:.+]], label %[[CASE1_AWAIT_SUSPEND:.+]]
90 // CHECK: [[CASE1_AWAIT_SUSPEND]]:
91 // CHECK-NEXT: %{{.+}} = call token @llvm.coro.save(ptr null)
92 // CHECK-NEXT: call void @llvm.lifetime.start.p0(i64 8, ptr %[[TMP1:.+]])
94 // CHECK: call void @llvm.lifetime.end.p0(i64 8, ptr %[[TMP1]])
95 // CHECK-NEXT: call void @llvm.coro.resume
96 // CHECK-NEXT: %{{.+}} = call i8 @llvm.coro.suspend
97 // CHECK-NEXT: switch i8 %{{.+}}, label %coro.ret [
98 // CHECK-NEXT: i8 0, label %[[CASE1_AWAIT_READY]]
99 // CHECK-NEXT: i8 1, label %[[CASE1_AWAIT_CLEANUP:.+]]
100 // CHECK-NEXT: ]
101 // CHECK: [[CASE1_AWAIT_CLEANUP]]:
102 // make sure that the awaiter eventually gets cleaned up.
103 // CHECK: call void @{{.+Awaiter.+}}
105 // CHECK: [[CASE2]]:
106 // CHECK: br i1 %{{.+}}, label %[[CASE2_AWAIT_READY:.+]], label %[[CASE2_AWAIT_SUSPEND:.+]]
107 // CHECK: [[CASE2_AWAIT_SUSPEND]]:
108 // CHECK-NEXT: %{{.+}} = call token @llvm.coro.save(ptr null)
109 // CHECK-NEXT: call void @llvm.lifetime.start.p0(i64 8, ptr %[[TMP2:.+]])
111 // CHECK: call void @llvm.lifetime.end.p0(i64 8, ptr %[[TMP2]])
112 // CHECK-NEXT: call void @llvm.coro.resume
113 // CHECK-NEXT: %{{.+}} = call i8 @llvm.coro.suspend
114 // CHECK-NEXT: switch i8 %{{.+}}, label %coro.ret [
115 // CHECK-NEXT: i8 0, label %[[CASE2_AWAIT_READY]]
116 // CHECK-NEXT: i8 1, label %[[CASE2_AWAIT_CLEANUP:.+]]
117 // CHECK-NEXT: ]
118 // CHECK: [[CASE2_AWAIT_CLEANUP]]:
119 // make sure that the awaiter eventually gets cleaned up.
120 // CHECK: call void @{{.+Awaiter.+}}