// { dg-do run }
// Test co-await in while condition.
#include "../coro.h"
// boiler-plate for tests of codegen
#include "../coro1-ret-int-yield-int.h"
/* An awaiter that suspends always and returns an int as the
   await_resume output.  */
struct IntAwaiter {
  int v;
  IntAwaiter (int _v) : v(_v) {}
  bool await_ready () { return false; }
  void await_suspend (coro::coroutine_handle<>) {}
  int await_resume () { return v; }
};
/* An awaiter that suspends always and returns a boolean as the
   await_resume output.  The boolean toggles on each call.  */
struct BoolAwaiter {
  bool v;
  BoolAwaiter (bool _v) : v(_v) {}
  bool await_ready () { return false; }
  void await_suspend (coro::coroutine_handle<>) {}
  bool await_resume () { v = !v; return v; }
};
/* We will be able to establish that the second part of the conditional
   expression is not evaluated (if it was, then we'd need an additional
   resume to complete the coroutine).  */
struct coro1
my_coro (int t)
{
  bool x = co_await IntAwaiter (t) == 5 || co_await BoolAwaiter (false);
  if (x)
    co_return 6174;
  co_return 42;
}
int main ()
{
  PRINT ("main: create coro");
  struct coro1 x = my_coro (5);
  if (x.handle.done())
    {
      PRINT ("main: apparently done when we should not be...");
      abort ();
    }
  PRINT ("main: resume initial suspend");
  x.handle.resume();
  PRINT ("main: resume IntAwaiter");
  x.handle.resume();
  // The evaluation of 'co_await IntAwaiter (t) == 5' should be true, thus
  // the second co_await in the expression will be unexecuted. 
  
  int y = x.handle.promise().get_value();
  if ( y != 6174 )
    {
      PRINTF ("main: apparently wrong value : %d\n", y);
      abort ();
    }
  if (!x.handle.done())
    {
      PRINT ("main: apparently not done...");
      abort ();
    }
  PRINT ("main: returning");
  return 0;
}