Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / SemaCXX / coroutine-rvo.cpp
blob6bf1dee67557ccec9d87132d35e8950398ad8c34
1 // RUN: %clang_cc1 -verify -std=c++20 -fsyntax-only %s
3 namespace std {
4 template <class Promise = void> struct coroutine_handle {
5 coroutine_handle() = default;
6 static coroutine_handle from_address(void *) noexcept;
7 };
9 template <> struct coroutine_handle<void> {
10 static coroutine_handle from_address(void *) noexcept;
11 coroutine_handle() = default;
12 template <class PromiseType>
13 coroutine_handle(coroutine_handle<PromiseType>) noexcept;
16 template <class... Args>
17 struct void_t_imp {
18 using type = void;
20 template <class... Args>
21 using void_t = typename void_t_imp<Args...>::type;
23 template <class T, class = void>
24 struct traits_sfinae_base {};
26 template <class T>
27 struct traits_sfinae_base<T, void_t<typename T::promise_type>> {
28 using promise_type = typename T::promise_type;
31 template <class Ret, class... Args>
32 struct coroutine_traits : public traits_sfinae_base<Ret> {};
33 } // namespace std
35 struct suspend_never {
36 bool await_ready() noexcept;
37 void await_suspend(std::coroutine_handle<>) noexcept;
38 void await_resume() noexcept;
41 struct MoveOnly {
42 MoveOnly() = default;
43 MoveOnly(const MoveOnly&) = delete;
44 MoveOnly(MoveOnly &&) = default;
47 struct NoCopyNoMove {
48 NoCopyNoMove() = default;
49 NoCopyNoMove(const NoCopyNoMove &) = delete;
52 template <typename T>
53 struct task {
54 struct promise_type {
55 auto initial_suspend() { return suspend_never{}; }
56 auto final_suspend() noexcept { return suspend_never{}; }
57 auto get_return_object() { return task{}; }
58 static void unhandled_exception() {}
59 void return_value(T &&value) {} // expected-note 4{{passing argument}}
63 task<NoCopyNoMove> local2val() {
64 NoCopyNoMove value;
65 co_return value;
68 task<NoCopyNoMove &> local2ref() {
69 NoCopyNoMove value;
70 co_return value; // expected-error {{non-const lvalue reference to type 'NoCopyNoMove' cannot bind to a temporary of type 'NoCopyNoMove'}}
73 // We need the move constructor for construction of the coroutine.
74 task<MoveOnly> param2val(MoveOnly value) {
75 co_return value;
78 task<NoCopyNoMove> lvalue2val(NoCopyNoMove &value) {
79 co_return value; // expected-error {{rvalue reference to type 'NoCopyNoMove' cannot bind to lvalue of type 'NoCopyNoMove'}}
82 task<NoCopyNoMove> rvalue2val(NoCopyNoMove &&value) {
83 co_return value;
86 task<NoCopyNoMove &> lvalue2ref(NoCopyNoMove &value) {
87 co_return value;
90 task<NoCopyNoMove &> rvalue2ref(NoCopyNoMove &&value) {
91 co_return value; // expected-error {{non-const lvalue reference to type 'NoCopyNoMove' cannot bind to a temporary of type 'NoCopyNoMove'}}
94 struct To {
95 operator MoveOnly() &&;
97 task<MoveOnly> conversion_operator() {
98 To t;
99 co_return t;
102 struct Construct {
103 Construct(MoveOnly);
105 task<Construct> converting_constructor() {
106 MoveOnly w;
107 co_return w;
110 struct Derived : MoveOnly {};
111 task<MoveOnly> derived2base() {
112 Derived result;
113 co_return result;
116 struct RetThis {
117 task<RetThis> foo() && {
118 co_return *this; // expected-error {{rvalue reference to type 'RetThis' cannot bind to lvalue of type 'RetThis'}}
122 template <typename, typename>
123 struct is_same { static constexpr bool value = false; };
125 template <typename T>
126 struct is_same<T, T> { static constexpr bool value = true; };
128 template <typename T>
129 struct generic_task {
130 struct promise_type {
131 auto initial_suspend() { return suspend_never{}; }
132 auto final_suspend() noexcept { return suspend_never{}; }
133 auto get_return_object() { return generic_task{}; }
134 static void unhandled_exception();
135 template <typename U>
136 void return_value(U &&value) {
137 static_assert(is_same<T, U>::value);
142 generic_task<MoveOnly> param2template(MoveOnly value) {
143 co_return value; // We should deduce U = MoveOnly.
146 generic_task<NoCopyNoMove &> lvalue2template(NoCopyNoMove &value) {
147 co_return value; // We should deduce U = NoCopyNoMove&.