xref: /freebsd/contrib/llvm-project/libcxx/include/__memory/unique_ptr.h (revision 152382e6613d7998fe6f5233767df54d3fdec329)
1 // -*- C++ -*-
2 //===----------------------------------------------------------------------===//
3 //
4 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5 // See https://llvm.org/LICENSE.txt for license information.
6 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #ifndef _LIBCPP___MEMORY_UNIQUE_PTR_H
11 #define _LIBCPP___MEMORY_UNIQUE_PTR_H
12 
13 #include <__compare/compare_three_way.h>
14 #include <__compare/compare_three_way_result.h>
15 #include <__compare/three_way_comparable.h>
16 #include <__config>
17 #include <__functional/hash.h>
18 #include <__functional/operations.h>
19 #include <__memory/allocator_traits.h> // __pointer
20 #include <__memory/auto_ptr.h>
21 #include <__memory/compressed_pair.h>
22 #include <__type_traits/add_lvalue_reference.h>
23 #include <__type_traits/common_type.h>
24 #include <__type_traits/conditional.h>
25 #include <__type_traits/dependent_type.h>
26 #include <__type_traits/integral_constant.h>
27 #include <__type_traits/is_array.h>
28 #include <__type_traits/is_assignable.h>
29 #include <__type_traits/is_constructible.h>
30 #include <__type_traits/is_convertible.h>
31 #include <__type_traits/is_function.h>
32 #include <__type_traits/is_pointer.h>
33 #include <__type_traits/is_reference.h>
34 #include <__type_traits/is_same.h>
35 #include <__type_traits/is_swappable.h>
36 #include <__type_traits/is_trivially_relocatable.h>
37 #include <__type_traits/is_void.h>
38 #include <__type_traits/remove_extent.h>
39 #include <__type_traits/remove_pointer.h>
40 #include <__type_traits/type_identity.h>
41 #include <__utility/declval.h>
42 #include <__utility/forward.h>
43 #include <__utility/move.h>
44 #include <cstddef>
45 
46 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
47 #  pragma GCC system_header
48 #endif
49 
50 _LIBCPP_PUSH_MACROS
51 #include <__undef_macros>
52 
53 _LIBCPP_BEGIN_NAMESPACE_STD
54 
55 #ifndef _LIBCPP_CXX03_LANG
56 
57 template <class _Ptr>
58 struct __is_noexcept_deref_or_void {
59   static constexpr bool value = noexcept(*std::declval<_Ptr>());
60 };
61 
62 template <>
63 struct __is_noexcept_deref_or_void<void*> : true_type {};
64 #endif
65 
66 template <class _Tp>
67 struct _LIBCPP_TEMPLATE_VIS default_delete {
68   static_assert(!is_function<_Tp>::value, "default_delete cannot be instantiated for function types");
69 #ifndef _LIBCPP_CXX03_LANG
70   _LIBCPP_HIDE_FROM_ABI constexpr default_delete() _NOEXCEPT = default;
71 #else
72   _LIBCPP_HIDE_FROM_ABI default_delete() {}
73 #endif
74   template <class _Up, __enable_if_t<is_convertible<_Up*, _Tp*>::value, int> = 0>
75   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 default_delete(const default_delete<_Up>&) _NOEXCEPT {}
76 
77   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void operator()(_Tp* __ptr) const _NOEXCEPT {
78     static_assert(sizeof(_Tp) >= 0, "cannot delete an incomplete type");
79     static_assert(!is_void<_Tp>::value, "cannot delete an incomplete type");
80     delete __ptr;
81   }
82 };
83 
84 template <class _Tp>
85 struct _LIBCPP_TEMPLATE_VIS default_delete<_Tp[]> {
86 private:
87   template <class _Up>
88   struct _EnableIfConvertible : enable_if<is_convertible<_Up (*)[], _Tp (*)[]>::value> {};
89 
90 public:
91 #ifndef _LIBCPP_CXX03_LANG
92   _LIBCPP_HIDE_FROM_ABI constexpr default_delete() _NOEXCEPT = default;
93 #else
94   _LIBCPP_HIDE_FROM_ABI default_delete() {}
95 #endif
96 
97   template <class _Up>
98   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23
99   default_delete(const default_delete<_Up[]>&, typename _EnableIfConvertible<_Up>::type* = 0) _NOEXCEPT {}
100 
101   template <class _Up>
102   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 typename _EnableIfConvertible<_Up>::type
103   operator()(_Up* __ptr) const _NOEXCEPT {
104     static_assert(sizeof(_Up) >= 0, "cannot delete an incomplete type");
105     delete[] __ptr;
106   }
107 };
108 
109 template <class _Deleter>
110 struct __unique_ptr_deleter_sfinae {
111   static_assert(!is_reference<_Deleter>::value, "incorrect specialization");
112   typedef const _Deleter& __lval_ref_type;
113   typedef _Deleter&& __good_rval_ref_type;
114   typedef true_type __enable_rval_overload;
115 };
116 
117 template <class _Deleter>
118 struct __unique_ptr_deleter_sfinae<_Deleter const&> {
119   typedef const _Deleter& __lval_ref_type;
120   typedef const _Deleter&& __bad_rval_ref_type;
121   typedef false_type __enable_rval_overload;
122 };
123 
124 template <class _Deleter>
125 struct __unique_ptr_deleter_sfinae<_Deleter&> {
126   typedef _Deleter& __lval_ref_type;
127   typedef _Deleter&& __bad_rval_ref_type;
128   typedef false_type __enable_rval_overload;
129 };
130 
131 #if defined(_LIBCPP_ABI_ENABLE_UNIQUE_PTR_TRIVIAL_ABI)
132 #  define _LIBCPP_UNIQUE_PTR_TRIVIAL_ABI __attribute__((__trivial_abi__))
133 #else
134 #  define _LIBCPP_UNIQUE_PTR_TRIVIAL_ABI
135 #endif
136 
137 template <class _Tp, class _Dp = default_delete<_Tp> >
138 class _LIBCPP_UNIQUE_PTR_TRIVIAL_ABI _LIBCPP_TEMPLATE_VIS unique_ptr {
139 public:
140   typedef _Tp element_type;
141   typedef _Dp deleter_type;
142   typedef _LIBCPP_NODEBUG typename __pointer<_Tp, deleter_type>::type pointer;
143 
144   static_assert(!is_rvalue_reference<deleter_type>::value, "the specified deleter type cannot be an rvalue reference");
145 
146   // A unique_ptr contains the following members which may be trivially relocatable:
147   // - pointer : this may be trivially relocatable, so it's checked
148   // - deleter_type: this may be trivially relocatable, so it's checked
149   //
150   // This unique_ptr implementation only contains a pointer to the unique object and a deleter, so there are no
151   // references to itself. This means that the entire structure is trivially relocatable if its members are.
152   using __trivially_relocatable = __conditional_t<
153       __libcpp_is_trivially_relocatable<pointer>::value && __libcpp_is_trivially_relocatable<deleter_type>::value,
154       unique_ptr,
155       void>;
156 
157 private:
158   __compressed_pair<pointer, deleter_type> __ptr_;
159 
160   typedef _LIBCPP_NODEBUG __unique_ptr_deleter_sfinae<_Dp> _DeleterSFINAE;
161 
162   template <bool _Dummy>
163   using _LValRefType _LIBCPP_NODEBUG = typename __dependent_type<_DeleterSFINAE, _Dummy>::__lval_ref_type;
164 
165   template <bool _Dummy>
166   using _GoodRValRefType _LIBCPP_NODEBUG = typename __dependent_type<_DeleterSFINAE, _Dummy>::__good_rval_ref_type;
167 
168   template <bool _Dummy>
169   using _BadRValRefType _LIBCPP_NODEBUG = typename __dependent_type<_DeleterSFINAE, _Dummy>::__bad_rval_ref_type;
170 
171   template <bool _Dummy, class _Deleter = typename __dependent_type< __type_identity<deleter_type>, _Dummy>::type>
172   using _EnableIfDeleterDefaultConstructible _LIBCPP_NODEBUG =
173       __enable_if_t<is_default_constructible<_Deleter>::value && !is_pointer<_Deleter>::value>;
174 
175   template <class _ArgType>
176   using _EnableIfDeleterConstructible _LIBCPP_NODEBUG = __enable_if_t<is_constructible<deleter_type, _ArgType>::value>;
177 
178   template <class _UPtr, class _Up>
179   using _EnableIfMoveConvertible _LIBCPP_NODEBUG =
180       __enable_if_t< is_convertible<typename _UPtr::pointer, pointer>::value && !is_array<_Up>::value >;
181 
182   template <class _UDel>
183   using _EnableIfDeleterConvertible _LIBCPP_NODEBUG =
184       __enable_if_t< (is_reference<_Dp>::value && is_same<_Dp, _UDel>::value) ||
185                      (!is_reference<_Dp>::value && is_convertible<_UDel, _Dp>::value) >;
186 
187   template <class _UDel>
188   using _EnableIfDeleterAssignable = __enable_if_t< is_assignable<_Dp&, _UDel&&>::value >;
189 
190 public:
191   template <bool _Dummy = true, class = _EnableIfDeleterDefaultConstructible<_Dummy> >
192   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR unique_ptr() _NOEXCEPT : __ptr_(__value_init_tag(), __value_init_tag()) {}
193 
194   template <bool _Dummy = true, class = _EnableIfDeleterDefaultConstructible<_Dummy> >
195   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR unique_ptr(nullptr_t) _NOEXCEPT
196       : __ptr_(__value_init_tag(), __value_init_tag()) {}
197 
198   template <bool _Dummy = true, class = _EnableIfDeleterDefaultConstructible<_Dummy> >
199   _LIBCPP_HIDE_FROM_ABI
200   _LIBCPP_CONSTEXPR_SINCE_CXX23 explicit unique_ptr(pointer __p) _NOEXCEPT : __ptr_(__p, __value_init_tag()) {}
201 
202   template <bool _Dummy = true, class = _EnableIfDeleterConstructible<_LValRefType<_Dummy> > >
203   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(pointer __p, _LValRefType<_Dummy> __d) _NOEXCEPT
204       : __ptr_(__p, __d) {}
205 
206   template <bool _Dummy = true, class = _EnableIfDeleterConstructible<_GoodRValRefType<_Dummy> > >
207   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(pointer __p, _GoodRValRefType<_Dummy> __d) _NOEXCEPT
208       : __ptr_(__p, std::move(__d)) {
209     static_assert(!is_reference<deleter_type>::value, "rvalue deleter bound to reference");
210   }
211 
212   template <bool _Dummy = true, class = _EnableIfDeleterConstructible<_BadRValRefType<_Dummy> > >
213   _LIBCPP_HIDE_FROM_ABI unique_ptr(pointer __p, _BadRValRefType<_Dummy> __d) = delete;
214 
215   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(unique_ptr&& __u) _NOEXCEPT
216       : __ptr_(__u.release(), std::forward<deleter_type>(__u.get_deleter())) {}
217 
218   template <class _Up,
219             class _Ep,
220             class = _EnableIfMoveConvertible<unique_ptr<_Up, _Ep>, _Up>,
221             class = _EnableIfDeleterConvertible<_Ep> >
222   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(unique_ptr<_Up, _Ep>&& __u) _NOEXCEPT
223       : __ptr_(__u.release(), std::forward<_Ep>(__u.get_deleter())) {}
224 
225 #if _LIBCPP_STD_VER <= 14 || defined(_LIBCPP_ENABLE_CXX17_REMOVED_AUTO_PTR)
226   template <class _Up,
227             __enable_if_t<is_convertible<_Up*, _Tp*>::value && is_same<_Dp, default_delete<_Tp> >::value, int> = 0>
228   _LIBCPP_HIDE_FROM_ABI unique_ptr(auto_ptr<_Up>&& __p) _NOEXCEPT : __ptr_(__p.release(), __value_init_tag()) {}
229 #endif
230 
231   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr& operator=(unique_ptr&& __u) _NOEXCEPT {
232     reset(__u.release());
233     __ptr_.second() = std::forward<deleter_type>(__u.get_deleter());
234     return *this;
235   }
236 
237   template <class _Up,
238             class _Ep,
239             class = _EnableIfMoveConvertible<unique_ptr<_Up, _Ep>, _Up>,
240             class = _EnableIfDeleterAssignable<_Ep> >
241   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr& operator=(unique_ptr<_Up, _Ep>&& __u) _NOEXCEPT {
242     reset(__u.release());
243     __ptr_.second() = std::forward<_Ep>(__u.get_deleter());
244     return *this;
245   }
246 
247 #if _LIBCPP_STD_VER <= 14 || defined(_LIBCPP_ENABLE_CXX17_REMOVED_AUTO_PTR)
248   template <class _Up,
249             __enable_if_t<is_convertible<_Up*, _Tp*>::value && is_same<_Dp, default_delete<_Tp> >::value, int> = 0>
250   _LIBCPP_HIDE_FROM_ABI unique_ptr& operator=(auto_ptr<_Up> __p) {
251     reset(__p.release());
252     return *this;
253   }
254 #endif
255 
256 #ifdef _LIBCPP_CXX03_LANG
257   unique_ptr(unique_ptr const&)            = delete;
258   unique_ptr& operator=(unique_ptr const&) = delete;
259 #endif
260 
261   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 ~unique_ptr() { reset(); }
262 
263   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr& operator=(nullptr_t) _NOEXCEPT {
264     reset();
265     return *this;
266   }
267 
268   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 __add_lvalue_reference_t<_Tp> operator*() const
269       _NOEXCEPT_(__is_noexcept_deref_or_void<pointer>::value) {
270     return *__ptr_.first();
271   }
272   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 pointer operator->() const _NOEXCEPT { return __ptr_.first(); }
273   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 pointer get() const _NOEXCEPT { return __ptr_.first(); }
274   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 deleter_type& get_deleter() _NOEXCEPT { return __ptr_.second(); }
275   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 const deleter_type& get_deleter() const _NOEXCEPT {
276     return __ptr_.second();
277   }
278   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 explicit operator bool() const _NOEXCEPT {
279     return __ptr_.first() != nullptr;
280   }
281 
282   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 pointer release() _NOEXCEPT {
283     pointer __t    = __ptr_.first();
284     __ptr_.first() = pointer();
285     return __t;
286   }
287 
288   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void reset(pointer __p = pointer()) _NOEXCEPT {
289     pointer __tmp  = __ptr_.first();
290     __ptr_.first() = __p;
291     if (__tmp)
292       __ptr_.second()(__tmp);
293   }
294 
295   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void swap(unique_ptr& __u) _NOEXCEPT { __ptr_.swap(__u.__ptr_); }
296 };
297 
298 template <class _Tp, class _Dp>
299 class _LIBCPP_UNIQUE_PTR_TRIVIAL_ABI _LIBCPP_TEMPLATE_VIS unique_ptr<_Tp[], _Dp> {
300 public:
301   typedef _Tp element_type;
302   typedef _Dp deleter_type;
303   typedef typename __pointer<_Tp, deleter_type>::type pointer;
304 
305   // A unique_ptr contains the following members which may be trivially relocatable:
306   // - pointer : this may be trivially relocatable, so it's checked
307   // - deleter_type: this may be trivially relocatable, so it's checked
308   //
309   // This unique_ptr implementation only contains a pointer to the unique object and a deleter, so there are no
310   // references to itself. This means that the entire structure is trivially relocatable if its members are.
311   using __trivially_relocatable = __conditional_t<
312       __libcpp_is_trivially_relocatable<pointer>::value && __libcpp_is_trivially_relocatable<deleter_type>::value,
313       unique_ptr,
314       void>;
315 
316 private:
317   __compressed_pair<pointer, deleter_type> __ptr_;
318 
319   template <class _From>
320   struct _CheckArrayPointerConversion : is_same<_From, pointer> {};
321 
322   template <class _FromElem>
323   struct _CheckArrayPointerConversion<_FromElem*>
324       : integral_constant<bool,
325                           is_same<_FromElem*, pointer>::value ||
326                               (is_same<pointer, element_type*>::value &&
327                                is_convertible<_FromElem (*)[], element_type (*)[]>::value) > {};
328 
329   typedef __unique_ptr_deleter_sfinae<_Dp> _DeleterSFINAE;
330 
331   template <bool _Dummy>
332   using _LValRefType _LIBCPP_NODEBUG = typename __dependent_type<_DeleterSFINAE, _Dummy>::__lval_ref_type;
333 
334   template <bool _Dummy>
335   using _GoodRValRefType _LIBCPP_NODEBUG = typename __dependent_type<_DeleterSFINAE, _Dummy>::__good_rval_ref_type;
336 
337   template <bool _Dummy>
338   using _BadRValRefType _LIBCPP_NODEBUG = typename __dependent_type<_DeleterSFINAE, _Dummy>::__bad_rval_ref_type;
339 
340   template <bool _Dummy, class _Deleter = typename __dependent_type< __type_identity<deleter_type>, _Dummy>::type>
341   using _EnableIfDeleterDefaultConstructible _LIBCPP_NODEBUG =
342       __enable_if_t<is_default_constructible<_Deleter>::value && !is_pointer<_Deleter>::value>;
343 
344   template <class _ArgType>
345   using _EnableIfDeleterConstructible _LIBCPP_NODEBUG = __enable_if_t<is_constructible<deleter_type, _ArgType>::value>;
346 
347   template <class _Pp>
348   using _EnableIfPointerConvertible _LIBCPP_NODEBUG = __enable_if_t< _CheckArrayPointerConversion<_Pp>::value >;
349 
350   template <class _UPtr, class _Up, class _ElemT = typename _UPtr::element_type>
351   using _EnableIfMoveConvertible _LIBCPP_NODEBUG =
352       __enable_if_t< is_array<_Up>::value && is_same<pointer, element_type*>::value &&
353                      is_same<typename _UPtr::pointer, _ElemT*>::value &&
354                      is_convertible<_ElemT (*)[], element_type (*)[]>::value >;
355 
356   template <class _UDel>
357   using _EnableIfDeleterConvertible _LIBCPP_NODEBUG =
358       __enable_if_t< (is_reference<_Dp>::value && is_same<_Dp, _UDel>::value) ||
359                      (!is_reference<_Dp>::value && is_convertible<_UDel, _Dp>::value) >;
360 
361   template <class _UDel>
362   using _EnableIfDeleterAssignable _LIBCPP_NODEBUG = __enable_if_t< is_assignable<_Dp&, _UDel&&>::value >;
363 
364 public:
365   template <bool _Dummy = true, class = _EnableIfDeleterDefaultConstructible<_Dummy> >
366   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR unique_ptr() _NOEXCEPT : __ptr_(__value_init_tag(), __value_init_tag()) {}
367 
368   template <bool _Dummy = true, class = _EnableIfDeleterDefaultConstructible<_Dummy> >
369   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR unique_ptr(nullptr_t) _NOEXCEPT
370       : __ptr_(__value_init_tag(), __value_init_tag()) {}
371 
372   template <class _Pp,
373             bool _Dummy = true,
374             class       = _EnableIfDeleterDefaultConstructible<_Dummy>,
375             class       = _EnableIfPointerConvertible<_Pp> >
376   _LIBCPP_HIDE_FROM_ABI
377   _LIBCPP_CONSTEXPR_SINCE_CXX23 explicit unique_ptr(_Pp __p) _NOEXCEPT : __ptr_(__p, __value_init_tag()) {}
378 
379   template <class _Pp,
380             bool _Dummy = true,
381             class       = _EnableIfDeleterConstructible<_LValRefType<_Dummy> >,
382             class       = _EnableIfPointerConvertible<_Pp> >
383   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(_Pp __p, _LValRefType<_Dummy> __d) _NOEXCEPT
384       : __ptr_(__p, __d) {}
385 
386   template <bool _Dummy = true, class = _EnableIfDeleterConstructible<_LValRefType<_Dummy> > >
387   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(nullptr_t, _LValRefType<_Dummy> __d) _NOEXCEPT
388       : __ptr_(nullptr, __d) {}
389 
390   template <class _Pp,
391             bool _Dummy = true,
392             class       = _EnableIfDeleterConstructible<_GoodRValRefType<_Dummy> >,
393             class       = _EnableIfPointerConvertible<_Pp> >
394   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(_Pp __p, _GoodRValRefType<_Dummy> __d) _NOEXCEPT
395       : __ptr_(__p, std::move(__d)) {
396     static_assert(!is_reference<deleter_type>::value, "rvalue deleter bound to reference");
397   }
398 
399   template <bool _Dummy = true, class = _EnableIfDeleterConstructible<_GoodRValRefType<_Dummy> > >
400   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(nullptr_t, _GoodRValRefType<_Dummy> __d) _NOEXCEPT
401       : __ptr_(nullptr, std::move(__d)) {
402     static_assert(!is_reference<deleter_type>::value, "rvalue deleter bound to reference");
403   }
404 
405   template <class _Pp,
406             bool _Dummy = true,
407             class       = _EnableIfDeleterConstructible<_BadRValRefType<_Dummy> >,
408             class       = _EnableIfPointerConvertible<_Pp> >
409   _LIBCPP_HIDE_FROM_ABI unique_ptr(_Pp __p, _BadRValRefType<_Dummy> __d) = delete;
410 
411   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(unique_ptr&& __u) _NOEXCEPT
412       : __ptr_(__u.release(), std::forward<deleter_type>(__u.get_deleter())) {}
413 
414   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr& operator=(unique_ptr&& __u) _NOEXCEPT {
415     reset(__u.release());
416     __ptr_.second() = std::forward<deleter_type>(__u.get_deleter());
417     return *this;
418   }
419 
420   template <class _Up,
421             class _Ep,
422             class = _EnableIfMoveConvertible<unique_ptr<_Up, _Ep>, _Up>,
423             class = _EnableIfDeleterConvertible<_Ep> >
424   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr(unique_ptr<_Up, _Ep>&& __u) _NOEXCEPT
425       : __ptr_(__u.release(), std::forward<_Ep>(__u.get_deleter())) {}
426 
427   template <class _Up,
428             class _Ep,
429             class = _EnableIfMoveConvertible<unique_ptr<_Up, _Ep>, _Up>,
430             class = _EnableIfDeleterAssignable<_Ep> >
431   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr& operator=(unique_ptr<_Up, _Ep>&& __u) _NOEXCEPT {
432     reset(__u.release());
433     __ptr_.second() = std::forward<_Ep>(__u.get_deleter());
434     return *this;
435   }
436 
437 #ifdef _LIBCPP_CXX03_LANG
438   unique_ptr(unique_ptr const&)            = delete;
439   unique_ptr& operator=(unique_ptr const&) = delete;
440 #endif
441 
442 public:
443   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 ~unique_ptr() { reset(); }
444 
445   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 unique_ptr& operator=(nullptr_t) _NOEXCEPT {
446     reset();
447     return *this;
448   }
449 
450   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 __add_lvalue_reference_t<_Tp> operator[](size_t __i) const {
451     return __ptr_.first()[__i];
452   }
453   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 pointer get() const _NOEXCEPT { return __ptr_.first(); }
454 
455   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 deleter_type& get_deleter() _NOEXCEPT { return __ptr_.second(); }
456 
457   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 const deleter_type& get_deleter() const _NOEXCEPT {
458     return __ptr_.second();
459   }
460   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 explicit operator bool() const _NOEXCEPT {
461     return __ptr_.first() != nullptr;
462   }
463 
464   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 pointer release() _NOEXCEPT {
465     pointer __t    = __ptr_.first();
466     __ptr_.first() = pointer();
467     return __t;
468   }
469 
470   template <class _Pp, __enable_if_t<_CheckArrayPointerConversion<_Pp>::value, int> = 0>
471   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void reset(_Pp __p) _NOEXCEPT {
472     pointer __tmp  = __ptr_.first();
473     __ptr_.first() = __p;
474     if (__tmp)
475       __ptr_.second()(__tmp);
476   }
477 
478   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void reset(nullptr_t = nullptr) _NOEXCEPT {
479     pointer __tmp  = __ptr_.first();
480     __ptr_.first() = nullptr;
481     if (__tmp)
482       __ptr_.second()(__tmp);
483   }
484 
485   _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void swap(unique_ptr& __u) _NOEXCEPT { __ptr_.swap(__u.__ptr_); }
486 };
487 
488 template <class _Tp, class _Dp, __enable_if_t<__is_swappable_v<_Dp>, int> = 0>
489 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 void
490 swap(unique_ptr<_Tp, _Dp>& __x, unique_ptr<_Tp, _Dp>& __y) _NOEXCEPT {
491   __x.swap(__y);
492 }
493 
494 template <class _T1, class _D1, class _T2, class _D2>
495 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool
496 operator==(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
497   return __x.get() == __y.get();
498 }
499 
500 #if _LIBCPP_STD_VER <= 17
501 template <class _T1, class _D1, class _T2, class _D2>
502 inline _LIBCPP_HIDE_FROM_ABI bool operator!=(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
503   return !(__x == __y);
504 }
505 #endif
506 
507 template <class _T1, class _D1, class _T2, class _D2>
508 inline _LIBCPP_HIDE_FROM_ABI bool operator<(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
509   typedef typename unique_ptr<_T1, _D1>::pointer _P1;
510   typedef typename unique_ptr<_T2, _D2>::pointer _P2;
511   typedef typename common_type<_P1, _P2>::type _Vp;
512   return less<_Vp>()(__x.get(), __y.get());
513 }
514 
515 template <class _T1, class _D1, class _T2, class _D2>
516 inline _LIBCPP_HIDE_FROM_ABI bool operator>(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
517   return __y < __x;
518 }
519 
520 template <class _T1, class _D1, class _T2, class _D2>
521 inline _LIBCPP_HIDE_FROM_ABI bool operator<=(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
522   return !(__y < __x);
523 }
524 
525 template <class _T1, class _D1, class _T2, class _D2>
526 inline _LIBCPP_HIDE_FROM_ABI bool operator>=(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
527   return !(__x < __y);
528 }
529 
530 #if _LIBCPP_STD_VER >= 20
531 template <class _T1, class _D1, class _T2, class _D2>
532   requires three_way_comparable_with<typename unique_ptr<_T1, _D1>::pointer, typename unique_ptr<_T2, _D2>::pointer>
533 _LIBCPP_HIDE_FROM_ABI
534 compare_three_way_result_t<typename unique_ptr<_T1, _D1>::pointer, typename unique_ptr<_T2, _D2>::pointer>
535 operator<=>(const unique_ptr<_T1, _D1>& __x, const unique_ptr<_T2, _D2>& __y) {
536   return compare_three_way()(__x.get(), __y.get());
537 }
538 #endif
539 
540 template <class _T1, class _D1>
541 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool
542 operator==(const unique_ptr<_T1, _D1>& __x, nullptr_t) _NOEXCEPT {
543   return !__x;
544 }
545 
546 #if _LIBCPP_STD_VER <= 17
547 template <class _T1, class _D1>
548 inline _LIBCPP_HIDE_FROM_ABI bool operator==(nullptr_t, const unique_ptr<_T1, _D1>& __x) _NOEXCEPT {
549   return !__x;
550 }
551 
552 template <class _T1, class _D1>
553 inline _LIBCPP_HIDE_FROM_ABI bool operator!=(const unique_ptr<_T1, _D1>& __x, nullptr_t) _NOEXCEPT {
554   return static_cast<bool>(__x);
555 }
556 
557 template <class _T1, class _D1>
558 inline _LIBCPP_HIDE_FROM_ABI bool operator!=(nullptr_t, const unique_ptr<_T1, _D1>& __x) _NOEXCEPT {
559   return static_cast<bool>(__x);
560 }
561 #endif // _LIBCPP_STD_VER <= 17
562 
563 template <class _T1, class _D1>
564 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator<(const unique_ptr<_T1, _D1>& __x, nullptr_t) {
565   typedef typename unique_ptr<_T1, _D1>::pointer _P1;
566   return less<_P1>()(__x.get(), nullptr);
567 }
568 
569 template <class _T1, class _D1>
570 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator<(nullptr_t, const unique_ptr<_T1, _D1>& __x) {
571   typedef typename unique_ptr<_T1, _D1>::pointer _P1;
572   return less<_P1>()(nullptr, __x.get());
573 }
574 
575 template <class _T1, class _D1>
576 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator>(const unique_ptr<_T1, _D1>& __x, nullptr_t) {
577   return nullptr < __x;
578 }
579 
580 template <class _T1, class _D1>
581 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator>(nullptr_t, const unique_ptr<_T1, _D1>& __x) {
582   return __x < nullptr;
583 }
584 
585 template <class _T1, class _D1>
586 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator<=(const unique_ptr<_T1, _D1>& __x, nullptr_t) {
587   return !(nullptr < __x);
588 }
589 
590 template <class _T1, class _D1>
591 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator<=(nullptr_t, const unique_ptr<_T1, _D1>& __x) {
592   return !(__x < nullptr);
593 }
594 
595 template <class _T1, class _D1>
596 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator>=(const unique_ptr<_T1, _D1>& __x, nullptr_t) {
597   return !(__x < nullptr);
598 }
599 
600 template <class _T1, class _D1>
601 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 bool operator>=(nullptr_t, const unique_ptr<_T1, _D1>& __x) {
602   return !(nullptr < __x);
603 }
604 
605 #if _LIBCPP_STD_VER >= 20
606 template <class _T1, class _D1>
607   requires three_way_comparable< typename unique_ptr<_T1, _D1>::pointer>
608 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 compare_three_way_result_t<typename unique_ptr<_T1, _D1>::pointer>
609 operator<=>(const unique_ptr<_T1, _D1>& __x, nullptr_t) {
610   return compare_three_way()(__x.get(), static_cast<typename unique_ptr<_T1, _D1>::pointer>(nullptr));
611 }
612 #endif
613 
614 #if _LIBCPP_STD_VER >= 14
615 
616 template <class _Tp>
617 struct __unique_if {
618   typedef unique_ptr<_Tp> __unique_single;
619 };
620 
621 template <class _Tp>
622 struct __unique_if<_Tp[]> {
623   typedef unique_ptr<_Tp[]> __unique_array_unknown_bound;
624 };
625 
626 template <class _Tp, size_t _Np>
627 struct __unique_if<_Tp[_Np]> {
628   typedef void __unique_array_known_bound;
629 };
630 
631 template <class _Tp, class... _Args>
632 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 typename __unique_if<_Tp>::__unique_single
633 make_unique(_Args&&... __args) {
634   return unique_ptr<_Tp>(new _Tp(std::forward<_Args>(__args)...));
635 }
636 
637 template <class _Tp>
638 inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 typename __unique_if<_Tp>::__unique_array_unknown_bound
639 make_unique(size_t __n) {
640   typedef __remove_extent_t<_Tp> _Up;
641   return unique_ptr<_Tp>(new _Up[__n]());
642 }
643 
644 template <class _Tp, class... _Args>
645 typename __unique_if<_Tp>::__unique_array_known_bound make_unique(_Args&&...) = delete;
646 
647 #endif // _LIBCPP_STD_VER >= 14
648 
649 #if _LIBCPP_STD_VER >= 20
650 
651 template <class _Tp>
652 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 typename __unique_if<_Tp>::__unique_single
653 make_unique_for_overwrite() {
654   return unique_ptr<_Tp>(new _Tp);
655 }
656 
657 template <class _Tp>
658 _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX23 typename __unique_if<_Tp>::__unique_array_unknown_bound
659 make_unique_for_overwrite(size_t __n) {
660   return unique_ptr<_Tp>(new __remove_extent_t<_Tp>[__n]);
661 }
662 
663 template <class _Tp, class... _Args>
664 typename __unique_if<_Tp>::__unique_array_known_bound make_unique_for_overwrite(_Args&&...) = delete;
665 
666 #endif // _LIBCPP_STD_VER >= 20
667 
668 template <class _Tp>
669 struct _LIBCPP_TEMPLATE_VIS hash;
670 
671 template <class _Tp, class _Dp>
672 #ifdef _LIBCPP_CXX03_LANG
673 struct _LIBCPP_TEMPLATE_VIS hash<unique_ptr<_Tp, _Dp> >
674 #else
675 struct _LIBCPP_TEMPLATE_VIS hash<__enable_hash_helper< unique_ptr<_Tp, _Dp>, typename unique_ptr<_Tp, _Dp>::pointer> >
676 #endif
677 {
678 #if _LIBCPP_STD_VER <= 17 || defined(_LIBCPP_ENABLE_CXX20_REMOVED_BINDER_TYPEDEFS)
679   _LIBCPP_DEPRECATED_IN_CXX17 typedef unique_ptr<_Tp, _Dp> argument_type;
680   _LIBCPP_DEPRECATED_IN_CXX17 typedef size_t result_type;
681 #endif
682 
683   _LIBCPP_HIDE_FROM_ABI size_t operator()(const unique_ptr<_Tp, _Dp>& __ptr) const {
684     typedef typename unique_ptr<_Tp, _Dp>::pointer pointer;
685     return hash<pointer>()(__ptr.get());
686   }
687 };
688 
689 _LIBCPP_END_NAMESPACE_STD
690 
691 _LIBCPP_POP_MACROS
692 
693 #endif // _LIBCPP___MEMORY_UNIQUE_PTR_H
694