Fix false warning messages about missing return in coroutine.

The patch sets current_function_returns_value flag in templates for
all co_return/co_yield/co_await cases, as well as for ramp function.

gcc/cp/ChangeLog
    * coroutines.cc (finish_co_await_expr): Set return value flag.
    (finish_co_yield_expr, morph_fn_to_coro): Ditto.

gcc/testsuite/ChangeLog
    * g++.dg/coroutines/co-return-warning-1.C: New test.
This commit is contained in:
Bin Cheng 2020-01-21 12:16:16 +08:00
parent e0a5b313c1
commit b313d3c49c
4 changed files with 64 additions and 0 deletions

View File

@ -1,3 +1,8 @@
2020-01-21 Bin Cheng <bin.cheng@linux.alibaba.com>
* coroutines.cc (finish_co_await_expr): Set return value flag.
(finish_co_yield_expr, morph_fn_to_coro): Ditto.
2020-01-19 Jason Merrill <jason@redhat.com>
PR c++/33799 - destroy return value, take 2.

View File

@ -753,6 +753,8 @@ finish_co_await_expr (location_t kw, tree expr)
if (processing_template_decl)
{
current_function_returns_value = 1;
if (check_for_bare_parameter_packs (expr))
return error_mark_node;
@ -826,6 +828,8 @@ finish_co_yield_expr (location_t kw, tree expr)
if (processing_template_decl)
{
current_function_returns_value = 1;
if (check_for_bare_parameter_packs (expr))
return error_mark_node;
@ -2870,6 +2874,9 @@ morph_fn_to_coro (tree orig, tree *resumer, tree *destroyer)
if (!coro_function_valid_p (orig))
return false;
/* The ramp function does return a value. */
current_function_returns_value = 1;
/* We can't validly get here with an empty statement list, since there's no
way for the FE to decide it's a coroutine in the absence of any code. */
tree fnbody = pop_stmt_list (DECL_SAVED_TREE (orig));

View File

@ -1,3 +1,7 @@
2020-01-20 Bin Cheng <bin.cheng@linux.alibaba.com>
* g++.dg/coroutines/co-return-warning-1.C: New test.
2020-01-21 Kito Cheng <kito.cheng@sifive.com>
PR target/93304

View File

@ -0,0 +1,48 @@
// { dg-additional-options "-std=c++17 -w" }
#include <coroutine>
class resumable {
public:
struct promise_type;
using coro_handle = std::coroutine_handle<promise_type>;
resumable(coro_handle handle) : handle_(handle) {}
resumable(resumable&) = delete;
resumable(resumable&&) = delete;
bool resume() {
if (not handle_.done())
handle_.resume();
return not handle_.done();
}
int recent_val();
~resumable() { handle_.destroy(); }
private:
coro_handle handle_;
};
struct resumable::promise_type {
int value_;
using coro_handle = std::coroutine_handle<promise_type>;
auto get_return_object() {
return coro_handle::from_promise(*this);
}
auto initial_suspend() { return std::suspend_always(); }
auto final_suspend() { return std::suspend_always(); }
void return_value(int v) { value_ = v; }
std::suspend_always yield_value(int v) {
value_ = v;
return std::suspend_always();
}
void unhandled_exception() {}
};
int resumable::recent_val(){return handle_.promise().value_;}
resumable foo(int n){
int x = 1;
co_await std::suspend_always();
int y = 2;
co_yield n + x + y;
}