1e356f681SHui Xie //===----------------------------------------------------------------------===//
2*6a54dfbfSLouis Dionne //
3e356f681SHui Xie // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e356f681SHui Xie // See https://llvm.org/LICENSE.txt for license information.
5e356f681SHui Xie // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e356f681SHui Xie //
7e356f681SHui Xie //===----------------------------------------------------------------------===//
8e356f681SHui Xie 
9e356f681SHui Xie // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20
10e356f681SHui Xie 
11e356f681SHui Xie // template<class T2, class E2> requires (!is_void_v<T2>)
12e356f681SHui Xie //   friend constexpr bool operator==(const expected& x, const expected<T2, E2>& y);
13e356f681SHui Xie 
14e356f681SHui Xie #include <cassert>
15e356f681SHui Xie #include <concepts>
16e356f681SHui Xie #include <expected>
17e356f681SHui Xie #include <type_traits>
18e356f681SHui Xie #include <utility>
19e356f681SHui Xie 
20e356f681SHui Xie #include "test_macros.h"
21e356f681SHui Xie 
22e356f681SHui Xie // Test constraint
23e356f681SHui Xie template <class T1, class T2>
24e356f681SHui Xie concept CanCompare = requires(T1 t1, T2 t2) { t1 == t2; };
25e356f681SHui Xie 
26e356f681SHui Xie struct Foo{};
27e356f681SHui Xie static_assert(!CanCompare<Foo, Foo>);
28e356f681SHui Xie 
29e356f681SHui Xie static_assert(CanCompare<std::expected<int, int>, std::expected<int, int>>);
30e356f681SHui Xie static_assert(CanCompare<std::expected<int, int>, std::expected<short, short>>);
31e356f681SHui Xie 
32e356f681SHui Xie // Note this is true because other overloads are unconstrained
33e356f681SHui Xie static_assert(CanCompare<std::expected<int, int>, std::expected<void, int>>);
34e356f681SHui Xie 
35e356f681SHui Xie constexpr bool test() {
36e356f681SHui Xie   // x.has_value() && y.has_value()
37e356f681SHui Xie   {
38e356f681SHui Xie     const std::expected<int, int> e1(5);
39e356f681SHui Xie     const std::expected<int, int> e2(10);
40e356f681SHui Xie     const std::expected<int, int> e3(5);
41e356f681SHui Xie     assert(e1 != e2);
42e356f681SHui Xie     assert(e1 == e3);
43e356f681SHui Xie   }
44e356f681SHui Xie 
45e356f681SHui Xie   // !x.has_value() && y.has_value()
46e356f681SHui Xie   {
47e356f681SHui Xie     const std::expected<int, int> e1(std::unexpect, 5);
48e356f681SHui Xie     const std::expected<int, int> e2(10);
49e356f681SHui Xie     const std::expected<int, int> e3(5);
50e356f681SHui Xie     assert(e1 != e2);
51e356f681SHui Xie     assert(e1 != e3);
52e356f681SHui Xie   }
53e356f681SHui Xie 
54e356f681SHui Xie   // x.has_value() && !y.has_value()
55e356f681SHui Xie   {
56e356f681SHui Xie     const std::expected<int, int> e1(5);
57e356f681SHui Xie     const std::expected<int, int> e2(std::unexpect, 10);
58e356f681SHui Xie     const std::expected<int, int> e3(std::unexpect, 5);
59e356f681SHui Xie     assert(e1 != e2);
60e356f681SHui Xie     assert(e1 != e3);
61e356f681SHui Xie   }
62e356f681SHui Xie 
63e356f681SHui Xie   // !x.has_value() && !y.has_value()
64e356f681SHui Xie   {
65e356f681SHui Xie     const std::expected<int, int> e1(std::unexpect, 5);
66e356f681SHui Xie     const std::expected<int, int> e2(std::unexpect, 10);
67e356f681SHui Xie     const std::expected<int, int> e3(std::unexpect, 5);
68e356f681SHui Xie     assert(e1 != e2);
69e356f681SHui Xie     assert(e1 == e3);
70e356f681SHui Xie   }
71e356f681SHui Xie 
72e356f681SHui Xie   return true;
73e356f681SHui Xie }
74e356f681SHui Xie 
75e356f681SHui Xie int main(int, char**) {
76e356f681SHui Xie   test();
77e356f681SHui Xie   static_assert(test());
78e356f681SHui Xie   return 0;
79e356f681SHui Xie }
80