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 // <tuple>
10
11 // template <class... Types> class tuple;
12
13 // tuple& operator=(tuple&& u);
14
15 // UNSUPPORTED: c++03
16
17 #include <memory>
18 #include <tuple>
19 #include <utility>
20 #include <cassert>
21
22 #include "test_macros.h"
23 #include "MoveOnly.h"
24
25 struct NonAssignable {
26 NonAssignable& operator=(NonAssignable const&) = delete;
27 NonAssignable& operator=(NonAssignable&&) = delete;
28 };
29 struct CopyAssignable {
30 CopyAssignable& operator=(CopyAssignable const&) = default;
31 CopyAssignable& operator=(CopyAssignable&&) = delete;
32 };
33 static_assert(std::is_copy_assignable<CopyAssignable>::value, "");
34 struct MoveAssignable {
35 MoveAssignable& operator=(MoveAssignable const&) = delete;
36 MoveAssignable& operator=(MoveAssignable&&) = default;
37 };
38
39
40 struct CountAssign {
41 static int copied;
42 static int moved;
resetCountAssign43 static void reset() { copied = moved = 0; }
44 CountAssign() = default;
operator =CountAssign45 CountAssign& operator=(CountAssign const&) { ++copied; return *this; }
operator =CountAssign46 CountAssign& operator=(CountAssign&&) { ++moved; return *this; }
47 };
48 int CountAssign::copied = 0;
49 int CountAssign::moved = 0;
50
51
main(int,char **)52 int main(int, char**)
53 {
54 {
55 typedef std::tuple<> T;
56 T t0;
57 T t;
58 t = std::move(t0);
59 }
60 {
61 typedef std::tuple<MoveOnly> T;
62 T t0(MoveOnly(0));
63 T t;
64 t = std::move(t0);
65 assert(std::get<0>(t) == 0);
66 }
67 {
68 typedef std::tuple<MoveOnly, MoveOnly> T;
69 T t0(MoveOnly(0), MoveOnly(1));
70 T t;
71 t = std::move(t0);
72 assert(std::get<0>(t) == 0);
73 assert(std::get<1>(t) == 1);
74 }
75 {
76 typedef std::tuple<MoveOnly, MoveOnly, MoveOnly> T;
77 T t0(MoveOnly(0), MoveOnly(1), MoveOnly(2));
78 T t;
79 t = std::move(t0);
80 assert(std::get<0>(t) == 0);
81 assert(std::get<1>(t) == 1);
82 assert(std::get<2>(t) == 2);
83 }
84 {
85 // test reference assignment.
86 using T = std::tuple<int&, int&&>;
87 int x = 42;
88 int y = 100;
89 int x2 = -1;
90 int y2 = 500;
91 T t(x, std::move(y));
92 T t2(x2, std::move(y2));
93 t = std::move(t2);
94 assert(std::get<0>(t) == x2);
95 assert(&std::get<0>(t) == &x);
96 assert(std::get<1>(t) == y2);
97 assert(&std::get<1>(t) == &y);
98 }
99 {
100 // test that the implicitly generated move assignment operator
101 // is properly deleted
102 using T = std::tuple<std::unique_ptr<int>>;
103 static_assert(std::is_move_assignable<T>::value, "");
104 static_assert(!std::is_copy_assignable<T>::value, "");
105
106 }
107 {
108 using T = std::tuple<int, NonAssignable>;
109 static_assert(!std::is_move_assignable<T>::value, "");
110 }
111 {
112 using T = std::tuple<int, MoveAssignable>;
113 static_assert(std::is_move_assignable<T>::value, "");
114 }
115 {
116 // The move should decay to a copy.
117 CountAssign::reset();
118 using T = std::tuple<CountAssign, CopyAssignable>;
119 static_assert(std::is_move_assignable<T>::value, "");
120 T t1;
121 T t2;
122 t1 = std::move(t2);
123 assert(CountAssign::copied == 1);
124 assert(CountAssign::moved == 0);
125 }
126
127 return 0;
128 }
129