xref: /llvm-project/libcxx/test/std/utilities/utility/pairs/pairs.pair/assign.pair_like_rv_const.pass.cpp (revision 0065d75099cd12b0c4a4b42a41bd36d8920b7537)
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
10 
11 // <utility>
12 
13 // template <class T1, class T2> struct pair
14 
15 //  template <pair-like P> constexpr const pair& operator=(P&&) const;  // since C++23
16 
17 #include <array>
18 #include <cassert>
19 #include <concepts>
20 #include <ranges>
21 #include <string>
22 #include <tuple>
23 #include <type_traits>
24 #include <utility>
25 
test()26 constexpr bool test() {
27   // Make sure assignment works from array and tuple
28   {
29     // Check from std::array
30     {
31       int x = 91, y = 92;
32       std::array<int, 2> a = {1, 2};
33       std::pair<int&, int&> const p = {x, y};
34       std::same_as<std::pair<int&, int&> const&> decltype(auto) result = (p = a);
35       assert(&result == &p);
36       assert(x == 1);
37       assert(y == 2);
38       static_assert(!std::is_assignable_v<std::pair<int&, int&> const&, std::array<int, 1>>); // too small
39       static_assert( std::is_assignable_v<std::pair<int&, int&> const&, std::array<int, 2>>); // works (test the test)
40       static_assert(!std::is_assignable_v<std::pair<int&, int&> const&, std::array<int, 3>>); // too large
41     }
42 
43     // Check from std::tuple
44     {
45       int x = 91, y = 92;
46       std::tuple<int, int> a = {1, 2};
47       std::pair<int&, int&> const p = {x, y};
48       std::same_as<std::pair<int&, int&> const&> decltype(auto) result = (p = a);
49       assert(&result == &p);
50       assert(x == 1);
51       assert(y == 2);
52       static_assert(!std::is_assignable_v<std::pair<int&, int&> const&, std::tuple<int>>); // too small
53       static_assert( std::is_assignable_v<std::pair<int&, int&> const&, std::tuple<int, int>>); // works (test the test)
54       static_assert(!std::is_assignable_v<std::pair<int&, int&> const&, std::tuple<int, int, int>>); // too large
55     }
56 
57     // Make sure it works for ranges::subrange. This actually deserves an explanation: even though
58     // the assignment operator explicitly excludes ranges::subrange specializations, such assignments
59     // end up working because of ranges::subrange's implicit conversion to pair-like types.
60     // This test ensures that the interoperability works as intended.
61     {
62       struct ConstAssignable {
63         mutable int* ptr = nullptr;
64         ConstAssignable() = default;
65         constexpr ConstAssignable(int* p) : ptr(p) { } // enable `subrange::operator pair-like`
66         constexpr ConstAssignable const& operator=(ConstAssignable const& other) const { ptr = other.ptr; return *this; }
67 
68         constexpr ConstAssignable(ConstAssignable const&) = default; // defeat -Wdeprecated-copy
69         constexpr ConstAssignable& operator=(ConstAssignable const&) = default; // defeat -Wdeprecated-copy
70       };
71       int data[] = {1, 2, 3, 4, 5};
72       std::ranges::subrange<int*> a(data);
73       std::pair<ConstAssignable, ConstAssignable> const p;
74       std::same_as<std::pair<ConstAssignable, ConstAssignable> const&> decltype(auto) result = (p = a);
75       assert(&result == &p);
76       assert(p.first.ptr == data);
77       assert(p.second.ptr == data + 5);
78     }
79   }
80 
81   // Make sure we allow element conversion from a pair-like
82   {
83     struct ConstAssignable {
84       mutable int val = 0;
85       ConstAssignable() = default;
86       constexpr ConstAssignable const& operator=(int v) const { val = v; return *this; }
87     };
88     std::tuple<int, int> a = {1, 2};
89     std::pair<ConstAssignable, ConstAssignable> const p;
90     std::same_as<std::pair<ConstAssignable, ConstAssignable> const&> decltype(auto) result = (p = a);
91     assert(&result == &p);
92     assert(p.first.val == 1);
93     assert(p.second.val == 2);
94     static_assert(!std::is_assignable_v<std::pair<ConstAssignable, ConstAssignable> const&, std::tuple<void*, int>>); // first not convertible
95     static_assert(!std::is_assignable_v<std::pair<ConstAssignable, ConstAssignable> const&, std::tuple<int, void*>>); // second not convertible
96     static_assert( std::is_assignable_v<std::pair<ConstAssignable, ConstAssignable> const&, std::tuple<int, int>>); // works (test the test)
97   }
98 
99   // Make sure we forward the pair-like elements
100   {
101     struct NoCopy {
102       NoCopy() = default;
103       NoCopy(NoCopy const&) = delete;
104       NoCopy(NoCopy&&) = default;
105       NoCopy& operator=(NoCopy const&) = delete;
106       constexpr NoCopy const& operator=(NoCopy&&) const { return *this; }
107     };
108     std::tuple<NoCopy, NoCopy> a;
109     std::pair<NoCopy, NoCopy> const p;
110     p = std::move(a);
111   }
112 
113   return true;
114 }
115 
main(int,char **)116 int main(int, char**) {
117   test();
118   static_assert(test());
119 
120   return 0;
121 }
122