(root)/
gcc-13.2.0/
libstdc++-v3/
testsuite/
20_util/
expected/
unexpected.cc
// { dg-options "-std=gnu++23" }
// { dg-do run { target c++23 } }

#include <expected>
#include <testsuite_hooks.h>

static_assert( sizeof(std::unexpected<char>) == 1 );

constexpr bool
test()
{
  std::unexpected<int> u1(1);
  VERIFY( u1.error() == 1 );

  std::unexpected<int> u2(std::in_place, 2);
  VERIFY( u2.error() == 2 );

  struct X
  {
    constexpr X(int i, int j) : n(i+j) { }
    constexpr X(std::initializer_list<int> l, void*) : n(l.size()) { }

    constexpr X(const X&) = default;
    constexpr X(X&& x) :n(x.n) { x.n = -1; }

    constexpr X& operator=(const X&) = default;
    constexpr X& operator=(X&& x) { n = x.n; x.n = -1; return *this; }

    constexpr bool operator==(const X&) const = default;
    constexpr bool operator==(int i) const { return n == i; }

    int n;
  };

  std::unexpected<X> u3(std::in_place, 2, 1);
  VERIFY( u3.error() == 3 );

  std::unexpected<X> u4(std::in_place, {1,2,3,4}, nullptr);
  VERIFY( u4.error() == 4 );

  std::unexpected<X> u5(u4);
  VERIFY( u5.error() == 4 );
  VERIFY( u4.error() == 4 );

  std::unexpected<X> u6(std::move(u4));
  VERIFY( u6.error() == 4 );
  VERIFY( u4.error() == -1 );

  u6 = u3;
  VERIFY( u6.error() == 3 );
  VERIFY( u3.error() == 3 );

  u5 = std::move(u3);
  VERIFY( u5.error() == 3 );
  VERIFY( u3.error() == -1 );

  u5.swap(u3);
  VERIFY( u3.error() == 3 );
  VERIFY( u5.error() == -1 );

  swap(u5, u3);
  VERIFY( u5.error() == 3 );
  VERIFY( u3.error() == -1 );

  VERIFY( u1 == u1 );
  VERIFY( u1 != u2 );
  VERIFY( u3 == u4 );

  // CTAD
  std::unexpected u7(1L);
  static_assert( std::is_same_v<decltype(u7), std::unexpected<long>> );

  return true;
}

static_assert( std::is_swappable_v<std::unexpected<int>> );
struct A { A& operator=(A&&) = delete; };
static_assert( ! std::is_swappable_v<std::unexpected<A>> );

int main()
{
  static_assert( test() );
  test();
}