30#ifndef _ALLOC_TRAITS_H
31#define _ALLOC_TRAITS_H 1
35#if __cplusplus >= 201103L
46namespace std _GLIBCXX_VISIBILITY(default)
48_GLIBCXX_BEGIN_NAMESPACE_VERSION
50#if __cplusplus >= 201103L
52#pragma GCC diagnostic push
53#pragma GCC diagnostic ignored "-Wc++14-extensions"
54#pragma GCC diagnostic ignored "-Wc++17-extensions"
57 struct __allocator_traits_base
60 template<
typename _Tp,
typename _Up>
62 template<
typename _Tp,
typename _Up,
typename =
void>
64 struct __rebind : __replace_first_arg<_Tp, _Up>
66 static_assert(is_same<
67 typename __replace_first_arg<_Tp, typename _Tp::value_type>::type,
69 "allocator_traits<A>::rebind_alloc<A::value_type> must be A");
72 template<
typename _Tp,
typename _Up>
74 requires requires {
typename _Tp::template rebind<_Up>::other; }
75 struct __rebind<_Tp, _Up>
77 struct __rebind<_Tp, _Up,
78 __void_t<typename _Tp::template rebind<_Up>::other>>
81 using type =
typename _Tp::template rebind<_Up>::other;
83 static_assert(is_same<
84 typename _Tp::template rebind<typename _Tp::value_type>::other,
86 "allocator_traits<A>::rebind_alloc<A::value_type> must be A");
90 template<
typename _Tp>
91 using __pointer =
typename _Tp::pointer;
92 template<
typename _Tp>
93 using __c_pointer =
typename _Tp::const_pointer;
94 template<
typename _Tp>
95 using __v_pointer =
typename _Tp::void_pointer;
96 template<
typename _Tp>
97 using __cv_pointer =
typename _Tp::const_void_pointer;
98 template<
typename _Tp>
99 using __pocca =
typename _Tp::propagate_on_container_copy_assignment;
100 template<
typename _Tp>
101 using __pocma =
typename _Tp::propagate_on_container_move_assignment;
102 template<
typename _Tp>
103 using __pocs =
typename _Tp::propagate_on_container_swap;
104 template<
typename _Tp>
105 using __equal = __type_identity<typename _Tp::is_always_equal>;
109 template<
typename _Alloc,
typename _Sz,
typename _Vp>
110 static constexpr bool __has_allocate_hint
111 =
requires (_Alloc& __a, _Sz __n, _Vp __hint) {
112 __a.allocate(__n, __hint);
115 template<
typename _Alloc,
typename _Sz,
typename _Vp>
116 using __allocate_hint_t
119 template<
typename _Alloc,
typename _Sz,
typename _Vp,
typename =
void>
120 static constexpr bool __has_allocate_hint =
false;
121 template<
typename _Alloc,
typename _Sz,
typename _Vp>
122 static constexpr bool
123 __has_allocate_hint<_Alloc, _Sz, _Vp,
124 __void_t<__allocate_hint_t<_Alloc, _Sz, _Vp>>>
133 template<
typename _Alloc,
typename _Tp,
typename... _Args>
134 static constexpr bool __has_construct
135 =
requires (_Alloc& __a, _Tp* __p, _Args&&... __args) {
138 template<
typename _Tp,
typename... _Args>
139 static constexpr bool __can_construct_at
140 =
requires (_Tp* __p, _Args&&... __args) {
141#if __cpp_constexpr_dynamic_alloc
147 template<
typename _Alloc,
typename _Tp,
typename... _Args>
148 static constexpr bool __can_construct
149 = __has_construct<_Alloc, _Tp, _Args...>
150 || __can_construct_at<_Tp, _Args...>;
152 template<
typename _Alloc,
typename _Tp,
typename... _Args>
156 template<
typename _Alloc,
typename _Tp,
typename,
typename... _Args>
157 static constexpr bool __has_construct_impl =
false;
158 template<
typename _Alloc,
typename _Tp,
typename... _Args>
159 static constexpr bool
160 __has_construct_impl<_Alloc, _Tp,
161 __void_t<__construct_t<_Alloc, _Tp, _Args...>>,
164 template<
typename _Alloc,
typename _Tp,
typename... _Args>
165 static constexpr bool __has_construct
166 = __has_construct_impl<_Alloc, _Tp, void, _Args...>;
167 template<
typename _Tp,
typename... _Args>
170 template<
typename _Tp,
typename,
typename... _Args>
171 static constexpr bool __has_new_expr =
false;
172 template<
typename _Tp,
typename... _Args>
173 static constexpr bool
174 __has_new_expr<_Tp, __void_t<__new_expr_t<_Tp, _Args...>>, _Args...>
176 template<
typename _Alloc,
typename _Tp,
typename... _Args>
177 static constexpr bool __can_construct
178 = __has_construct<_Alloc, _Tp, _Args...>
179 || __has_new_expr<_Tp, void, _Args...>;
184 template<
typename _Alloc,
typename _Tp>
185 static constexpr bool __has_destroy =
requires (_Alloc& __a, _Tp* __p) {
189 template<
typename _Alloc,
typename _Tp>
192 template<
typename _Alloc,
typename _Tp,
typename =
void>
193 static constexpr bool __has_destroy =
false;
194 template<
typename _Alloc,
typename _Tp>
195 static constexpr bool __has_destroy<_Alloc, _Tp,
196 __void_t<__destroy_t<_Alloc, _Tp>>>
202 template<
typename _Alloc>
203 static constexpr bool __has_max_size =
requires (
const _Alloc& __a) {
207 template<
typename _Alloc>
209 template<
typename _Alloc,
typename =
void>
210 static constexpr bool __has_max_size =
false;
211 template<
typename _Alloc>
212 static constexpr bool __has_max_size<_Alloc,
213 __void_t<__max_size_t<_Alloc>>>
220 template<
typename _Alloc>
221 static constexpr bool __has_soccc =
requires (
const _Alloc& __a) {
222 __a.select_on_container_copy_construction();
225 template<
typename _Alloc>
228 .select_on_container_copy_construction());
229 template<
typename _Alloc,
typename =
void>
230 static constexpr bool __has_soccc =
false;
231 template<
typename _Alloc>
232 static constexpr bool __has_soccc<_Alloc, __void_t<__soccc_t<_Alloc>>>
237 template<
typename _Alloc,
typename _Up>
239 =
typename __allocator_traits_base::template __rebind<_Alloc, _Up>::type;
248 template<
typename _Alloc>
261 using pointer = __detected_or_t<value_type*, __pointer, _Alloc>;
265 template<
template<
typename>
class _Func,
typename _Tp,
typename =
void>
271 template<
template<
typename>
class _Func,
typename _Tp>
272 struct _Ptr<_Func, _Tp, __void_t<_Func<_Alloc>>>
274 using type = _Func<_Alloc>;
278 template<
typename _A2,
typename _PtrT,
typename =
void>
280 {
using type =
typename pointer_traits<_PtrT>::difference_type; };
282 template<
typename _A2,
typename _PtrT>
284 {
using type =
typename _A2::difference_type; };
287 template<
typename _A2,
typename _DiffT,
typename =
void>
288 struct _Size : make_unsigned<_DiffT> { };
290 template<
typename _A2,
typename _DiffT>
291 struct _Size<_A2, _DiffT, __void_t<typename _A2::
size_type>>
292 {
using type =
typename _A2::size_type; };
333 using size_type =
typename _Size<_Alloc, difference_type>::type;
342 = __detected_or_t<false_type, __pocca, _Alloc>;
351 = __detected_or_t<false_type, __pocma, _Alloc>;
360 = __detected_or_t<false_type, __pocs, _Alloc>;
369 =
typename __detected_or_t<is_empty<_Alloc>, __equal, _Alloc>::type;
371 template<
typename _Tp>
372 using rebind_alloc = __alloc_rebind<_Alloc, _Tp>;
373 template<
typename _Tp>
383 _GLIBCXX_NODISCARD
static _GLIBCXX20_CONSTEXPR
pointer
385 {
return __a.allocate(__n); }
398 _GLIBCXX_NODISCARD
static _GLIBCXX20_CONSTEXPR
pointer
401 if constexpr (__has_allocate_hint<_Alloc, size_type, const_void_pointer>)
402 return __a.allocate(__n, __hint);
404 return __a.allocate(__n);
407#ifdef __glibcxx_allocate_at_least
420 [[nodiscard]]
static constexpr allocation_result<pointer, size_type>
423 if constexpr (
requires { __a.allocate_at_least(__n); })
424 return __a.allocate_at_least(__n);
426 return { __a.allocate(__n), __n };
438 static _GLIBCXX20_CONSTEXPR
void
440 { __a.deallocate(__p, __n); }
453 template<
typename _Tp,
typename... _Args>
454#if __cpp_concepts && __cpp_constexpr_dynamic_alloc
455 requires __can_construct<_Alloc, _Tp, _Args...>
456 static constexpr void
458 static __enable_if_t<__can_construct<_Alloc, _Tp, _Args...>>
460 construct(_Alloc& __a, _Tp* __p, _Args&&... __args)
461 noexcept(_S_nothrow_construct<_Tp, _Args...>())
463 if constexpr (__has_construct<_Alloc, _Tp, _Args...>)
477 template<
typename _Tp>
478 static _GLIBCXX20_CONSTEXPR
void
480 noexcept(_S_nothrow_destroy<_Tp>())
482 if constexpr (__has_destroy<_Alloc, _Tp>)
497 max_size(
const _Alloc& __a)
noexcept
499 if constexpr (__has_max_size<_Alloc>)
500 return __a.max_size();
504 return __gnu_cxx::__numeric_traits<size_type>::__max
516 static _GLIBCXX20_CONSTEXPR _Alloc
519 if constexpr (__has_soccc<_Alloc>)
520 return __rhs.select_on_container_copy_construction();
526#if __cpp_constexpr >= 201304
527 template<
typename _Tp,
typename... _Args>
528 static constexpr bool
529 _S_nothrow_construct(_Alloc* __a =
nullptr, _Tp* __p =
nullptr)
531 if constexpr (__has_construct<_Alloc, _Tp, _Args...>)
534 return __is_nothrow_new_constructible<_Tp, _Args...>;
537 template<
typename _Tp>
538 static constexpr bool
539 _S_nothrow_destroy(_Alloc* __a =
nullptr, _Tp* __p =
nullptr)
541 if constexpr (__has_destroy<_Alloc, _Tp>)
542 return noexcept(__a->destroy(__p));
544 return is_nothrow_destructible<_Tp>::value;
547 template<
typename _Tp,
typename... _Args>
549 __enable_if_t<__has_construct<_Alloc, _Tp, _Args...>,
bool>
550 _S_nothrow_construct(_Alloc* __a =
nullptr, _Tp* __p =
nullptr)
553 template<
typename _Tp,
typename... _Args>
555 __enable_if_t<!__has_construct<_Alloc, _Tp, _Args...>,
bool>
556 _S_nothrow_construct(_Alloc* =
nullptr, _Tp* __p =
nullptr)
557 {
return __is_nothrow_new_constructible<_Tp, _Args...>; }
559 template<
typename _Tp>
561 __enable_if_t<__has_destroy<_Alloc, _Tp>,
bool>
562 _S_nothrow_destroy(_Alloc* __a =
nullptr, _Tp* __p =
nullptr)
563 {
return noexcept(__a->destroy(__p)); }
565 template<
typename _Tp>
567 __enable_if_t<!__has_destroy<_Alloc, _Tp>,
bool>
568 _S_nothrow_destroy(_Alloc* =
nullptr, _Tp* __p =
nullptr)
569 {
return is_nothrow_destructible<_Tp>::value; }
572#pragma GCC diagnostic pop
582 template<
typename _Tp>
621 template<
typename _Up>
624 template<
typename _Up>
634 [[__nodiscard__,__gnu__::__always_inline__]]
635 static _GLIBCXX20_CONSTEXPR
pointer
637 {
return __a.allocate(__n); }
649 [[__nodiscard__,__gnu__::__always_inline__]]
650 static _GLIBCXX20_CONSTEXPR pointer
654#if __cplusplus <= 201703L
655 return __a.allocate(__n, __hint);
657 return __a.allocate(__n);
661#ifdef __glibcxx_allocate_at_least
671 [[nodiscard,__gnu__::__always_inline__]]
672 static constexpr allocation_result<pointer, size_type>
674 {
return __a.allocate_at_least(__n); }
685 [[__gnu__::__always_inline__]]
686 static _GLIBCXX20_CONSTEXPR
void
688 { __a.deallocate(__p, __n); }
701 template<
typename _Up,
typename... _Args>
702 [[__gnu__::__always_inline__]]
703 static _GLIBCXX20_CONSTEXPR
void
705 _Up* __p, _Args&&... __args)
706#if __cplusplus <= 201703L
709 noexcept(__is_nothrow_new_constructible<_Up, _Args...>)
712#if __cplusplus <= 201703L
714#elif __cpp_constexpr_dynamic_alloc
728 template<
typename _Up>
729 [[__gnu__::__always_inline__]]
730 static _GLIBCXX20_CONSTEXPR
void
734#if __cplusplus <= 201703L
737 std::destroy_at(__p);
746 [[__gnu__::__always_inline__]]
750#if __cplusplus <= 201703L
751 return __a.max_size();
762 [[__gnu__::__always_inline__]]
763 static _GLIBCXX20_CONSTEXPR allocator_type
814 template<
typename _Up>
817 template<
typename _Up>
824#ifdef __glibcxx_allocate_at_least
825 static allocation_result<void*, size_type>
844 template<
typename _Up,
typename... _Args>
845 [[__gnu__::__always_inline__]]
846 static _GLIBCXX20_CONSTEXPR
void
848 noexcept(__is_nothrow_new_constructible<_Up, _Args...>)
858 template<
typename _Up>
859 [[__gnu__::__always_inline__]]
860 static _GLIBCXX20_CONSTEXPR
void
874 [[__gnu__::__always_inline__]]
882#pragma GCC diagnostic push
883#pragma GCC diagnostic ignored "-Wc++17-extensions"
884 template<
typename _Alloc>
885 [[__gnu__::__always_inline__]]
886 _GLIBCXX14_CONSTEXPR
inline void
887 __alloc_on_copy(_Alloc& __one,
const _Alloc& __two)
889 using __traits = allocator_traits<_Alloc>;
891 typename __traits::propagate_on_container_copy_assignment::type;
892 if constexpr (__pocca::value)
896 template<
typename _Alloc>
897 [[__gnu__::__always_inline__]]
899 __alloc_on_copy(
const _Alloc& __a)
902 return __traits::select_on_container_copy_construction(__a);
905 template<
typename _Alloc>
906 [[__gnu__::__always_inline__]]
907 _GLIBCXX14_CONSTEXPR
inline void
908 __alloc_on_move(_Alloc& __one, _Alloc& __two)
912 =
typename __traits::propagate_on_container_move_assignment::type;
913 if constexpr (__pocma::value)
917 template<
typename _Alloc>
918 [[__gnu__::__always_inline__]]
919 _GLIBCXX14_CONSTEXPR
inline void
920 __alloc_on_swap(_Alloc& __one, _Alloc& __two)
923 using __pocs =
typename __traits::propagate_on_container_swap::type;
924 if constexpr (__pocs::value)
930#pragma GCC diagnostic pop
932 template<
typename _Alloc,
typename _Tp,
933 typename _ValueT = __remove_cvref_t<typename _Alloc::value_type>,
935 struct __is_alloc_insertable_impl
939 template<
typename _Alloc,
typename _Tp,
typename _ValueT>
940 struct __is_alloc_insertable_impl<_Alloc, _Tp, _ValueT,
942 std::declval<_Alloc&>(), std::declval<_ValueT*>(),
943 std::declval<_Tp>()))>>
950 template<
typename _Alloc>
951 struct __is_copy_insertable
952 : __is_alloc_insertable_impl<_Alloc,
953 typename _Alloc::value_type const&>::type
958 template<
typename _Tp>
959 struct __is_copy_insertable<
allocator<_Tp>>
967 template<
typename _Alloc>
968 struct __is_move_insertable
969 : __is_alloc_insertable_impl<_Alloc, typename _Alloc::value_type>::type
974 template<
typename _Tp>
975 struct __is_move_insertable<
allocator<_Tp>>
981 template<
typename _Alloc,
typename =
void>
984 template<
typename _Alloc>
985 struct __is_allocator<_Alloc,
987 decltype(std::declval<_Alloc&>().allocate(size_t{}))>>
990 template<
typename _Alloc>
991 using _RequireAllocator
994 template<
typename _Alloc>
995 using _RequireNotAllocator
998#if __cpp_concepts >= 201907L
999 template<
typename _Alloc>
1000 concept __allocator_like =
requires (_Alloc& __a) {
1001 typename _Alloc::value_type;
1002 __a.deallocate(__a.allocate(1u), 1u);
1005 template<
typename _Alloc>
1006 concept __not_allocator_like = !__allocator_like<_Alloc>;
1014 template<
typename _Alloc,
bool = __is_empty(_Alloc)>
1016 {
static void _S_do_it(_Alloc&, _Alloc&) _GLIBCXX_NOEXCEPT { } };
1018 template<
typename _Alloc>
1019 struct __alloc_swap<_Alloc, false>
1022 _S_do_it(_Alloc& __one, _Alloc& __two) _GLIBCXX_NOEXCEPT
1030#if __cplusplus >= 201103L
1031 template<
typename _Tp,
bool
1032 = __or_<is_copy_constructible<typename _Tp::value_type>,
1034 struct __shrink_to_fit_aux
1035 {
static bool _S_do_it(_Tp&)
noexcept {
return false; } };
1037 template<
typename _Tp>
1038 struct __shrink_to_fit_aux<_Tp, true>
1040 _GLIBCXX20_CONSTEXPR
1042 _S_do_it(_Tp& __c)
noexcept
1047 _Tp(__make_move_if_noexcept_iterator(__c.begin()),
1048 __make_move_if_noexcept_iterator(__c.end()),
1049 __c.get_allocator()).swap(__c);
1067 template<
typename _ForwardIterator,
typename _Allocator>
1068 _GLIBCXX20_CONSTEXPR
1070 _Destroy(_ForwardIterator __first, _ForwardIterator __last,
1071 _Allocator& __alloc)
1073 for (; __first != __last; ++__first)
1074#
if __cplusplus < 201103L
1083 template<
typename _ForwardIterator,
typename _Tp>
1084 __attribute__((__always_inline__)) _GLIBCXX20_CONSTEXPR
1086 _Destroy(_ForwardIterator __first, _ForwardIterator __last,
1095_GLIBCXX_END_NAMESPACE_VERSION
__bool_constant< true > true_type
The type used as a compile-time boolean with true value.
__bool_constant< false > false_type
The type used as a compile-time boolean with false value.
auto declval() noexcept -> decltype(__declval< _Tp >(0))
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
constexpr _Tp * __addressof(_Tp &__r) noexcept
Same as C++11 std::addressof.
constexpr _Tp && forward(typename std::remove_reference< _Tp >::type &__t) noexcept
Forward an lvalue.
ISO C++ entities toplevel namespace is std.
constexpr void _Construct(_Tp *__p, _Args &&... __args)
constexpr void _Destroy(_ForwardIterator __first, _ForwardIterator __last)
Define a member typedef type only if a boolean constant is true.
is_nothrow_move_constructible
Uniform interface to all allocator types.
__detected_or_t< false_type, __pocma, _OuterAlloc > propagate_on_container_move_assignment
typename _Ptr< __v_pointer, void >::type void_pointer
__detected_or_t< value_type *, __pointer, _OuterAlloc > pointer
static constexpr void construct(_Alloc &__a, _Tp *__p, _Args &&... __args) noexcept(_S_nothrow_construct< _Tp, _Args... >())
Construct an object of type _Tp.
static constexpr pointer allocate(_Alloc &__a, size_type __n)
Allocate memory.
typename _Size< _Alloc, difference_type >::type size_type
The allocator's size type.
typename _Ptr< __cv_pointer, const void >::type const_void_pointer
static constexpr allocation_result< pointer, size_type > allocate_at_least(_Alloc &__a, size_type __n)
Allocate memory, generously.
static constexpr void destroy(_Alloc &__a, _Tp *__p) noexcept(_S_nothrow_destroy< _Tp >())
Destroy an object of type _Tp.
typename _Diff< _Alloc, pointer >::type difference_type
The allocator's difference type.
typename _Ptr< __c_pointer, const value_type >::type const_pointer
_OuterAlloc::value_type value_type
static constexpr void deallocate(_Alloc &__a, pointer __p, size_type __n)
Deallocate memory.
typename __detected_or_t< is_empty< _OuterAlloc >, __equal, _OuterAlloc >::type is_always_equal
static constexpr size_type max_size(const _Alloc &__a) noexcept
The maximum supported allocation size.
__detected_or_t< false_type, __pocca, _OuterAlloc > propagate_on_container_copy_assignment
static constexpr _Alloc select_on_container_copy_construction(const _Alloc &__rhs)
Obtain an allocator to use when copying a container.
__detected_or_t< false_type, __pocs, _OuterAlloc > propagate_on_container_swap
_OuterAlloc allocator_type
allocator< _Tp > allocator_type
The allocator type.
static constexpr void construct(allocator_type &__a, _Up *__p, _Args &&... __args) noexcept(__is_nothrow_new_constructible< _Up, _Args... >)
Construct an object of type _Up.
void * void_pointer
The allocator's void pointer type.
_Tp * pointer
The allocator's pointer type.
false_type propagate_on_container_swap
How the allocator is propagated on swap.
static constexpr pointer allocate(allocator_type &__a, size_type __n)
Allocate memory.
_Tp value_type
The allocated type.
static constexpr pointer allocate(allocator_type &__a, size_type __n, const_void_pointer __hint)
Allocate memory.
std::ptrdiff_t difference_type
The allocator's difference type.
true_type is_always_equal
Whether all instances of the allocator type compare equal.
const _Tp * const_pointer
The allocator's const pointer type.
static constexpr allocation_result< pointer, size_type > allocate_at_least(allocator_type &__a, size_type __n)
Allocate memory, generously.
const void * const_void_pointer
The allocator's const void pointer type.
true_type propagate_on_container_move_assignment
How the allocator is propagated on move assignment.
static constexpr void deallocate(allocator_type &__a, pointer __p, size_type __n)
Deallocate memory.
static constexpr size_type max_size(const allocator_type &__a) noexcept
The maximum supported allocation size.
static constexpr allocator_type select_on_container_copy_construction(const allocator_type &__rhs)
Obtain an allocator to use when copying a container.
static constexpr void destroy(allocator_type &__a, _Up *__p) noexcept(is_nothrow_destructible< _Up >::value)
Destroy an object of type _Up.
false_type propagate_on_container_copy_assignment
How the allocator is propagated on copy assignment.
std::size_t size_type
The allocator's size type.
false_type propagate_on_container_copy_assignment
How the allocator is propagated on copy assignment.
void * pointer
The allocator's pointer type.
void * void_pointer
The allocator's void pointer type.
static void deallocate(allocator_type &, void *, size_type)=delete
deallocate is ill-formed for allocator<void>
allocator< void > allocator_type
The allocator type.
true_type is_always_equal
Whether all instances of the allocator type compare equal.
void value_type
The allocated type.
static size_type max_size(const allocator_type &)=delete
max_size is ill-formed for allocator<void>
std::size_t size_type
The allocator's size type.
true_type propagate_on_container_move_assignment
How the allocator is propagated on move assignment.
const void * const_pointer
The allocator's const pointer type.
std::ptrdiff_t difference_type
The allocator's difference type.
static void * allocate(allocator_type &, size_type, const void *=nullptr)=delete
allocate is ill-formed for allocator<void>
static constexpr allocator_type select_on_container_copy_construction(const allocator_type &__rhs)
Obtain an allocator to use when copying a container.
static constexpr void construct(allocator_type &, _Up *__p, _Args &&... __args) noexcept(__is_nothrow_new_constructible< _Up, _Args... >)
Construct an object of type _Up.
const void * const_void_pointer
The allocator's const void pointer type.
static constexpr void destroy(allocator_type &, _Up *__p) noexcept(is_nothrow_destructible< _Up >::value)
Destroy an object of type _Up.
false_type propagate_on_container_swap
How the allocator is propagated on swap.
The standard allocator, as per C++03 [20.4.1].
Uniform interface to all pointer-like types.