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: sanitizer-new-delete
10
11 // <memory>
12
13 // template <class Y, class D> shared_ptr(unique_ptr<Y, D>&&r);
14
15 #include <memory>
16 #include <new>
17 #include <cstdlib>
18 #include <cassert>
19 #include <utility>
20
21 #include "test_macros.h"
22 #include "count_new.h"
23
24 struct B
25 {
26 static int count;
27
BB28 B() {++count;}
BB29 B(const B&) {++count;}
~BB30 virtual ~B() {--count;}
31 };
32
33 int B::count = 0;
34
35 struct A
36 : public B
37 {
38 static int count;
39
AA40 A() {++count;}
AA41 A(const A& other) : B(other) {++count;}
~AA42 ~A() {--count;}
43 };
44
45 int A::count = 0;
46
fn(const std::shared_ptr<int> &)47 void fn ( const std::shared_ptr<int> &) {}
fn(const std::shared_ptr<B> &)48 void fn ( const std::shared_ptr<B> &) { assert (false); }
49
50 template <typename T>
assert_deleter(T *)51 void assert_deleter ( T * ) { assert(false); }
52
53 namespace adl {
54 struct D {
operator ()adl::D55 void operator()(int *) const {}
56 };
57 void ref(D);
58 }
59
60 template <class T>
61 struct StatefulDeleter {
62 int state = 0;
63
StatefulDeleterStatefulDeleter64 StatefulDeleter(int val = 0) : state(val) {}
StatefulDeleterStatefulDeleter65 StatefulDeleter(StatefulDeleter const&) { assert(false); }
66
operator ()StatefulDeleter67 void operator()(T* ptr) {
68 assert(state == 42);
69 delete ptr;
70 }
71 };
72
73 template <class T>
74 struct StatefulArrayDeleter {
75 int state = 0;
76
StatefulArrayDeleterStatefulArrayDeleter77 StatefulArrayDeleter(int val = 0) : state(val) {}
StatefulArrayDeleterStatefulArrayDeleter78 StatefulArrayDeleter(StatefulArrayDeleter const&) { assert(false); }
79
operator ()StatefulArrayDeleter80 void operator()(T* ptr) {
81 assert(state == 42);
82 delete []ptr;
83 }
84 };
85
86 struct MovingDeleter {
MovingDeleterMovingDeleter87 explicit MovingDeleter(int *moves) : moves_(moves) {}
MovingDeleterMovingDeleter88 MovingDeleter(MovingDeleter&& rhs) : moves_(rhs.moves_) { *moves_ += 1; }
operator ()MovingDeleter89 void operator()(int*) const {}
90 int *moves_;
91 };
92
93 // https://llvm.org/PR53368
94 // Bogus unique_ptr-to-shared_ptr conversions should be forbidden
95 #if TEST_STD_VER >= 17
96 static_assert( std::is_constructible<std::shared_ptr<A>, std::unique_ptr<A>&&>::value, "");
97 static_assert( std::is_constructible<std::shared_ptr<A[]>, std::unique_ptr<A[]>&&>::value, "");
98 static_assert(!std::is_constructible<std::shared_ptr<A>, std::unique_ptr<A[]>&&>::value, "");
99 static_assert(!std::is_constructible<std::shared_ptr<B[]>, std::unique_ptr<A[]>&&>::value, "");
100 static_assert(!std::is_constructible<std::shared_ptr<B>, std::unique_ptr<A[]>&&>::value, "");
101 #endif
102
main(int,char **)103 int main(int, char**)
104 {
105 {
106 std::unique_ptr<A> ptr(new A);
107 A* raw_ptr = ptr.get();
108 std::shared_ptr<B> p(std::move(ptr));
109 assert(A::count == 1);
110 assert(B::count == 1);
111 assert(p.use_count() == 1);
112 assert(p.get() == raw_ptr);
113 assert(ptr.get() == 0);
114 }
115
116 {
117 std::unique_ptr<A const> ptr(new A);
118 A const* raw_ptr = ptr.get();
119 std::shared_ptr<B const> p(std::move(ptr));
120 assert(A::count == 1);
121 assert(B::count == 1);
122 assert(p.use_count() == 1);
123 assert(p.get() == raw_ptr);
124 assert(ptr.get() == 0);
125 }
126
127 #ifndef TEST_HAS_NO_EXCEPTIONS
128 assert(A::count == 0);
129 {
130 std::unique_ptr<A> ptr(new A);
131 A* raw_ptr = ptr.get();
132 globalMemCounter.throw_after = 0;
133 try
134 {
135 std::shared_ptr<B> p(std::move(ptr));
136 assert(false);
137 }
138 catch (...)
139 {
140 assert(A::count == 1);
141 assert(B::count == 1);
142 assert(ptr.get() == raw_ptr);
143 }
144 }
145 #endif
146
147 #if TEST_STD_VER > 14
148 {
149 std::unique_ptr<int> ptr;
150 std::shared_ptr<int> p(std::move(ptr));
151 assert(p.get() == 0);
152 assert(p.use_count() == 0);
153 }
154 #endif
155
156 {
157 StatefulDeleter<A> d;
158 std::unique_ptr<A, StatefulDeleter<A>&> u(new A, d);
159 std::shared_ptr<A> p(std::move(u));
160 d.state = 42;
161 assert(A::count == 1);
162 }
163 assert(A::count == 0);
164
165 { // LWG 2399
166 fn(std::unique_ptr<int>(new int));
167 }
168 #if TEST_STD_VER >= 14
169 { // LWG 2415
170 std::unique_ptr<int, void (*)(int*)> p(nullptr, assert_deleter<int>);
171 std::shared_ptr<int> p2(std::move(p)); // should not call deleter when going out of scope
172 }
173 #endif
174
175 {
176 adl::D d;
177 std::unique_ptr<int, adl::D&> u(nullptr, d);
178 std::shared_ptr<int> s = std::move(u);
179 }
180
181 assert(A::count == 0);
182
183 #if TEST_STD_VER > 14
184 {
185 StatefulArrayDeleter<A> d;
186 std::unique_ptr<A[], StatefulArrayDeleter<A>&> u(new A[4], d);
187 std::shared_ptr<A[]> p(std::move(u));
188 d.state = 42;
189 assert(A::count == 4);
190 }
191 assert(A::count == 0);
192 assert(B::count == 0);
193
194 {
195 std::unique_ptr<A[]> ptr(new A[8]);
196 A* raw_ptr = ptr.get();
197 std::shared_ptr<A[]> p(std::move(ptr));
198 assert(A::count == 8);
199 assert(p.use_count() == 1);
200 assert(p.get() == raw_ptr);
201 assert(ptr.get() == 0);
202 }
203 assert(A::count == 0);
204
205 {
206 int *p = new int[8];
207 std::unique_ptr<int[]> u(p);
208 std::shared_ptr<int[]> s(std::move(u));
209 assert(u == nullptr);
210 assert(s.get() == p);
211 }
212 #endif // TEST_STD_VER > 14
213
214 { // LWG 3548
215 {
216 int moves = 0;
217 int i = 42;
218 std::unique_ptr<int, MovingDeleter> u(&i, MovingDeleter(&moves));
219 assert(moves == 1);
220 std::shared_ptr<int> s(std::move(u));
221 assert(moves >= 2);
222 assert(u == nullptr);
223 assert(s.get() == &i);
224 }
225
226 #if TEST_STD_VER > 14
227 {
228 int moves = 0;
229 int a[8];
230 std::unique_ptr<int[], MovingDeleter> u(a, MovingDeleter(&moves));
231 assert(moves == 1);
232 std::shared_ptr<int[]> s = std::move(u);
233 assert(moves >= 2);
234 assert(u == nullptr);
235 assert(s.get() == a);
236 }
237 #endif // TEST_STD_VER > 14
238 }
239
240 return 0;
241 }
242