1 //===----------------------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20, c++23
10 
11 // <functional>
12 
13 // class reference_wrapper
14 
15 // [refwrap.comparisons], comparisons
16 
17 // friend constexpr auto operator<=>(reference_wrapper, reference_wrapper<const T>); // Since C++26
18 
19 #include <cassert>
20 #include <concepts>
21 #include <functional>
22 
23 #include "test_comparisons.h"
24 #include "test_macros.h"
25 
26 #include "helper_concepts.h"
27 #include "helper_types.h"
28 
29 // Test SFINAE.
30 
31 static_assert(std::three_way_comparable_with<std::reference_wrapper<StrongOrder>, const StrongOrder>);
32 static_assert(std::three_way_comparable_with<std::reference_wrapper<WeakOrder>, const WeakOrder>);
33 static_assert(std::three_way_comparable_with<std::reference_wrapper<PartialOrder>, const PartialOrder>);
34 
35 static_assert(!std::three_way_comparable_with<std::reference_wrapper<StrongOrder>, const NonComparable>);
36 static_assert(!std::three_way_comparable_with<std::reference_wrapper<WeakOrder>, const NonComparable>);
37 static_assert(!std::three_way_comparable_with<std::reference_wrapper<PartialOrder>, const NonComparable>);
38 
39 // Test comparisons.
40 
41 template <typename T, typename Order>
test()42 constexpr void test() {
43   T t{47};
44 
45   T bigger{94};
46   T smaller{82};
47 
48   T unordered{std::numeric_limits<int>::min()};
49 
50   // Identical contents
51   {
52     std::reference_wrapper<T> rw1{t};
53     std::reference_wrapper<const T> rw2{t};
54     assert(testOrder(rw1, rw2, Order::equivalent));
55   }
56   // Less
57   {
58     std::reference_wrapper<T> rw1{smaller};
59     std::reference_wrapper<const T> rw2{bigger};
60     assert(testOrder(rw1, rw2, Order::less));
61   }
62   // Greater
63   {
64     std::reference_wrapper<T> rw1{bigger};
65     std::reference_wrapper<const T> rw2{smaller};
66     assert(testOrder(rw1, rw2, Order::greater));
67   }
68   // Unordered
69   if constexpr (std::same_as<T, PartialOrder>) {
70     std::reference_wrapper<T> rw1{bigger};
71     std::reference_wrapper<const T> rw2{unordered};
72     assert(testOrder(rw1, rw2, Order::unordered));
73   }
74 }
75 
test()76 constexpr bool test() {
77   test<int, std::strong_ordering>();
78   test<StrongOrder, std::strong_ordering>();
79   test<int, std::weak_ordering>();
80   test<WeakOrder, std::weak_ordering>();
81   test<int, std::partial_ordering>();
82   test<PartialOrder, std::partial_ordering>();
83 
84   // `LessAndEqComp` does not have `operator<=>`. Ordering is synthesized based on `operator<`
85   test<LessAndEqComp, std::weak_ordering>();
86 
87   return true;
88 }
89 
main(int,char **)90 int main(int, char**) {
91   test();
92   static_assert(test());
93 
94   return 0;
95 }
96