coroutines: Improve error recovery [PR94817, PR94829].
authorIain Sandoe <iain@sandoe.co.uk>
Thu, 7 May 2020 18:48:31 +0000 (19:48 +0100)
committerIain Sandoe <iain@sandoe.co.uk>
Thu, 7 May 2020 20:06:43 +0000 (21:06 +0100)
When we have completely missing key information (e.g. the
coroutine_traits) or a partially transformed function body, we
need to try and balance returning useful information about
failures with the possibility that some part of the diagnostics
machinery or following code will not be able to handle the
state.

The PRs (and revised testcase) point to cases where that processing
has failed.

This revises the process to avoid special handling for the
ramp, and falls back on the same code used for regular function
fails.

There are test-cases (in addition to the ones for the PRs) that now
cover all early exit points [where the transforms are considered
to have failed in a manner that does not allow compilation to
continue].

gcc/cp/ChangeLog:

2020-05-07  Iain Sandoe  <iain@sandoe.co.uk>

PR c++/94817
PR c++/94829
* coroutines.cc (morph_fn_to_coro): Set unformed outline
functions to error_mark_node.  For early error returns suppress
warnings about missing ramp return values.  Fix reinstatement
of the function body on pre-existing initial error.
* decl.c (finish_function): Use the normal error path for fails
in the ramp function, do not try to compile the helpers if the
transform fails.

gcc/testsuite/ChangeLog:

2020-05-07  Iain Sandoe  <iain@sandoe.co.uk>

PR c++/94817
PR c++/94829
* g++.dg/coroutines/coro-missing-final-suspend.C: New test.
* g++.dg/coroutines/coro-missing-initial-suspend.C: New test.
* g++.dg/coroutines/coro-missing-promise-yield.C: Check for
continuation of compilation.
* g++.dg/coroutines/coro-missing-promise.C: Likewise.
* g++.dg/coroutines/coro-missing-ret-value.C: Likewise
* g++.dg/coroutines/coro-missing-ret-void.C: Likewise
* g++.dg/coroutines/coro-missing-ueh-3.C: Likewise
* g++.dg/coroutines/pr94817.C: New test.
* g++.dg/coroutines/pr94829.C: New test.

14 files changed:
gcc/cp/ChangeLog
gcc/cp/coroutines.cc
gcc/cp/decl.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/coroutines/coro-missing-final-suspend.C [new file with mode: 0644]
gcc/testsuite/g++.dg/coroutines/coro-missing-initial-suspend.C [new file with mode: 0644]
gcc/testsuite/g++.dg/coroutines/coro-missing-promise-yield.C
gcc/testsuite/g++.dg/coroutines/coro-missing-promise.C
gcc/testsuite/g++.dg/coroutines/coro-missing-ret-value.C
gcc/testsuite/g++.dg/coroutines/coro-missing-ret-void.C
gcc/testsuite/g++.dg/coroutines/coro-missing-ueh-3.C
gcc/testsuite/g++.dg/coroutines/coro1-ret-int-yield-int.h
gcc/testsuite/g++.dg/coroutines/pr94817.C [new file with mode: 0644]
gcc/testsuite/g++.dg/coroutines/pr94829.C [new file with mode: 0644]

index 031741e2264d3df1903ee585a6032c933c76f9fd..0c19a6fe47c6857f2b5f0e1d747651c33035cba6 100644 (file)
@@ -1,3 +1,15 @@
+2020-05-07  Iain Sandoe  <iain@sandoe.co.uk>
+
+       PR c++/94817
+       PR c++/94829
+       * coroutines.cc (morph_fn_to_coro): Set unformed outline
+       functions to error_mark_node.  For early error returns suppress
+       warnings about missing ramp return values.  Fix reinstatement
+       of the function body on pre-existing initial error.
+       * decl.c (finish_function): Use the normal error path for fails
+       in the ramp function, do not try to compile the helpers if the
+       transform fails.
+
 2020-05-07  Marek Polacek  <polacek@redhat.com>
 
        PR c++/94590 - Detect long double -> double narrowing.
index fc319a58cc258b5379fb4652a33f1dfb71d171e2..8f0f2d5960d510cf43fd99d4c4b11606ccc097e6 100644 (file)
@@ -3540,14 +3540,21 @@ morph_fn_to_coro (tree orig, tree *resumer, tree *destroyer)
 {
   gcc_checking_assert (orig && TREE_CODE (orig) == FUNCTION_DECL);
 
+  *resumer = error_mark_node;
+  *destroyer = error_mark_node;
   if (!coro_function_valid_p (orig))
-    return false;
+    {
+      /* For early errors, we do not want a diagnostic about the missing
+        ramp return value, since the user cannot fix this - a 'return' is
+        not allowed in a coroutine.  */
+      TREE_NO_WARNING (orig) = true;
+      return false;
+    }
 
   /* 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));
-  if (fnbody == NULL_TREE)
-    return false;
+  gcc_checking_assert (fnbody != NULL_TREE);
 
   /* We don't have the locus of the opening brace - it's filled in later (and
      there doesn't really seem to be any easy way to get at it).
@@ -3561,7 +3568,9 @@ morph_fn_to_coro (tree orig, tree *resumer, tree *destroyer)
   if (body_start == NULL_TREE || body_start == error_mark_node)
     {
       DECL_SAVED_TREE (orig) = push_stmt_list ();
-      append_to_statement_list (DECL_SAVED_TREE (orig), &fnbody);
+      append_to_statement_list (fnbody, &DECL_SAVED_TREE (orig));
+      /* Suppress warnings about the missing return value.  */
+      TREE_NO_WARNING (orig) = true;
       return false;
     }
 
@@ -3630,13 +3639,21 @@ morph_fn_to_coro (tree orig, tree *resumer, tree *destroyer)
      the coroutine promise.  */
   tree initial_await = build_init_or_final_await (fn_start, false);
   if (initial_await == error_mark_node)
-    return false;
+    {
+      /* Suppress warnings about the missing return value.  */
+      TREE_NO_WARNING (orig) = true;
+      return false;
+    }
   /* The type of the frame var for this is the type of its temp proxy.  */
   tree initial_suspend_type = TREE_TYPE (TREE_OPERAND (initial_await, 1));
 
   tree final_await = build_init_or_final_await (fn_start, true);
   if (final_await == error_mark_node)
-    return false;
+    {
+      /* Suppress warnings about the missing return value.  */
+      TREE_NO_WARNING (orig) = true;
+      return false;
+    }
 
   /* The type of the frame var for this is the type of its temp proxy.  */
   tree final_suspend_type = TREE_TYPE (TREE_OPERAND (final_await, 1));
@@ -4219,6 +4236,8 @@ morph_fn_to_coro (tree orig, tree *resumer, tree *destroyer)
     {
       BIND_EXPR_BODY (ramp_bind) = pop_stmt_list (ramp_body);
       DECL_SAVED_TREE (orig) = newbody;
+      /* Suppress warnings about the missing return value.  */
+      TREE_NO_WARNING (orig) = true;
       return false;
     }
 
index cc12e0051543b0af22bb240a8bb32e8168941105..73a06a60786e73b81fad14535c6b57fab8b00a77 100644 (file)
@@ -16886,6 +16886,7 @@ finish_function (bool inline_p)
   bool coro_p = flag_coroutines
                && !processing_template_decl
                && DECL_COROUTINE_P (fndecl);
+  bool coro_emit_helpers = false;
 
   /* When we get some parse errors, we can end up without a
      current_function_decl, so cope.  */
@@ -16914,18 +16915,16 @@ finish_function (bool inline_p)
 
   if (coro_p)
     {
-      if (!morph_fn_to_coro (fndecl, &resumer, &destroyer))
-       {
-         DECL_SAVED_TREE (fndecl) = pop_stmt_list (DECL_SAVED_TREE (fndecl));
-         poplevel (1, 0, 1);
-         DECL_SAVED_TREE (fndecl) = error_mark_node;
-         return fndecl;
-       }
+      /* Only try to emit the coroutine outlined helper functions if the
+        transforms succeeded.  Otherwise, treat errors in the same way as
+        a regular function.  */
+      coro_emit_helpers = morph_fn_to_coro (fndecl, &resumer, &destroyer);
 
       /* We should handle coroutine IFNs in middle end lowering.  */
       cfun->coroutine_component = true;
 
-      if (use_eh_spec_block (fndecl))
+      /* Do not try to process the ramp's EH unless outlining succeeded.  */
+      if (coro_emit_helpers && use_eh_spec_block (fndecl))
        finish_eh_spec_block (TYPE_RAISES_EXCEPTIONS
                              (TREE_TYPE (fndecl)),
                              current_eh_spec_block);
@@ -17174,8 +17173,9 @@ finish_function (bool inline_p)
       && !DECL_OMP_DECLARE_REDUCTION_P (fndecl))
     cp_genericize (fndecl);
 
-  /* Emit the resumer and destroyer functions now.  */
-  if (coro_p)
+  /* Emit the resumer and destroyer functions now, providing that we have
+     not encountered some fatal error.  */
+  if (coro_emit_helpers)
     {
       emit_coro_helper (resumer);
       emit_coro_helper (destroyer);
index 6712c1dfbb27af23dadb10e3cc3457b294a4a068..61670c97729124e1cb631fbfb2a442eca5869557 100644 (file)
@@ -1,3 +1,18 @@
+2020-05-07  Iain Sandoe  <iain@sandoe.co.uk>
+
+       PR c++/94817
+       PR c++/94829
+       * g++.dg/coroutines/coro-missing-final-suspend.C: New test.
+       * g++.dg/coroutines/coro-missing-initial-suspend.C: New test.
+       * g++.dg/coroutines/coro-missing-promise-yield.C: Check for
+       continuation of compilation.
+       * g++.dg/coroutines/coro-missing-promise.C: Likewise.
+       * g++.dg/coroutines/coro-missing-ret-value.C: Likewise
+       * g++.dg/coroutines/coro-missing-ret-void.C: Likewise
+       * g++.dg/coroutines/coro-missing-ueh-3.C: Likewise
+       * g++.dg/coroutines/pr94817.C: New test.
+       * g++.dg/coroutines/pr94829.C: New test.
+
 2020-05-07  Marek Polacek  <polacek@redhat.com>
 
        PR c++/94590 - Detect long double -> double narrowing.
diff --git a/gcc/testsuite/g++.dg/coroutines/coro-missing-final-suspend.C b/gcc/testsuite/g++.dg/coroutines/coro-missing-final-suspend.C
new file mode 100644 (file)
index 0000000..6a0878c
--- /dev/null
@@ -0,0 +1,19 @@
+//  { dg-additional-options "-fsyntax-only -w" }
+#include "coro.h"
+
+// Check diagnostic return from missing promise initial suspend entry.
+
+#define MISSING_FINAL_SUSPEND
+#include "coro1-ret-int-yield-int.h"
+
+coro1
+my_coro () // { dg-error {no member named 'final_suspend' in} }
+{
+  co_return 0;
+}
+
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
diff --git a/gcc/testsuite/g++.dg/coroutines/coro-missing-initial-suspend.C b/gcc/testsuite/g++.dg/coroutines/coro-missing-initial-suspend.C
new file mode 100644 (file)
index 0000000..c5b88ff
--- /dev/null
@@ -0,0 +1,19 @@
+//  { dg-additional-options "-fsyntax-only -w" }
+#include "coro.h"
+
+// Check diagnostic return from missing promise initial suspend entry.
+
+#define MISSING_INITIAL_SUSPEND
+#include "coro1-ret-int-yield-int.h"
+
+coro1
+my_coro () // { dg-error {no member named 'initial_suspend' in} }
+{
+  co_return 0;
+}
+
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
index d489c3953ac9d22179109f0f0e2f8f2230606f1b..a85a4d54bbe9d6be649666c758ac9efb9e3771cd 100644 (file)
@@ -27,6 +27,12 @@ bar ()
   co_return 0;
 }
 
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
+
 int main (int ac, char *av[]) {
   MissingPromiseYield x = bar ();
   return 0;
index e75f2002db0c9ef717edebf495b8e8c8c5141f0b..834cf1209cd5b0017305454b51a7da400b4758ee 100644 (file)
@@ -16,3 +16,9 @@ bar ()
   co_yield 22; // { dg-error {unable to find the promise type for this coroutine} }
   co_return 0;
 }
+
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
index f238c4b9a3543ad80fc4eff64f4d99792b5cacb9..f745fa9c852e6a633198d10397bfeebb8adc0c83 100644 (file)
@@ -28,6 +28,12 @@ bar ()
   co_return 6174; // { dg-error {no member named 'return_value' in} }
 }
 
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
+
 int main (int ac, char *av[]) {
   MissingRetValue x = bar ();
   return 0;
index c9f84e59020b2e9115ec88da2c12c080a83896f9..8639a9d4f769f9b43235142b9a1dbfd8cfcf95c3 100644 (file)
@@ -28,6 +28,12 @@ bar ()
   co_return; // { dg-error "no member named .return_void. in" }
 }
 
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
+
 int main (int ac, char *av[]) {
   MissingRetVoid x = bar ();
   return 0;
index d775d8a630e6767458b4a13193a0062ad8dd138f..36bd7da499187e034f0eca4edc9c9aa2ffe97c70 100644 (file)
@@ -12,6 +12,12 @@ bar ()
   co_return;
 }
 
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
+
 int main (int ac, char *av[]) {
   MissingUEH x = bar ();
   return 0;
index 67ac197fee449b33718fade6133bb4d036dcd1e3..7cf5052592453e161311b65a57763c66afdad3bd 100644 (file)
@@ -99,14 +99,20 @@ struct coro1 {
     return handle_type::from_promise (*this);
   }
 
+#ifdef MISSING_INITIAL_SUSPEND
+#else
   auto initial_suspend () {
     PRINT ("get initial_suspend (always)");
     return suspend_always_prt{};
   }
+#endif
+#ifdef MISSING_FINAL_SUSPEND
+#else
   auto final_suspend () {
     PRINT ("get final_suspend (always)");
     return suspend_always_prt{};
   }
+#endif
 
 #ifdef USE_AWAIT_TRANSFORM
 
diff --git a/gcc/testsuite/g++.dg/coroutines/pr94817.C b/gcc/testsuite/g++.dg/coroutines/pr94817.C
new file mode 100644 (file)
index 0000000..8f667bb
--- /dev/null
@@ -0,0 +1,10 @@
+
+void no_coroutine_traits() {
+  co_await 4; // { dg-error {coroutines require a traits template\; cannot find 'std::coroutine_traits'} }
+}
+
+// check we have not messed up continuation of the compilation.
+template <class... Args>
+struct void_t_imp {
+  using type = void;
+};
diff --git a/gcc/testsuite/g++.dg/coroutines/pr94829.C b/gcc/testsuite/g++.dg/coroutines/pr94829.C
new file mode 100644 (file)
index 0000000..289c280
--- /dev/null
@@ -0,0 +1,49 @@
+
+namespace std::experimental {
+template <typename R, typename... T> struct coroutine_traits {
+  using promise_type = typename R::promise_type;
+};
+
+template <class Promise = void> struct coroutine_handle;
+template <> struct coroutine_handle<void> {
+  static coroutine_handle from_address(void *) noexcept;
+  coroutine_handle() = default;
+  template <class PromiseType>
+  coroutine_handle(coroutine_handle<PromiseType>) noexcept;
+};
+template <class Promise> struct coroutine_handle : coroutine_handle<void> {
+  coroutine_handle() = default;
+  static coroutine_handle from_address(void *) noexcept;
+};
+}
+
+struct suspend_always {
+  bool await_ready() noexcept;
+  void await_suspend(std::experimental::coroutine_handle<>) noexcept;
+  void await_resume() noexcept;
+};
+
+struct Task {
+  struct promise_type {
+    Task get_return_object();
+    void return_void() {}
+    suspend_always initial_suspend() noexcept;
+    suspend_always final_suspend() noexcept;
+    void unhandled_exception() noexcept;
+  };
+};
+
+template <typename _AwrT> auto SyncAwait(_AwrT &&A) {
+  if (!A.await_ready()) {
+    auto AwaitAsync = [&]() -> Task {
+      try { (void)(co_await A); } catch (...) {} // { dg-error {coroutines require a traits template; cannot find 'std::coroutine_traits'} }
+    };
+    Task t = AwaitAsync();
+  }
+  return A.await_resume();
+}
+
+void f() {
+  suspend_always test;
+  SyncAwait(test);
+}