123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300 |
- #ifndef BOOST_POLY_COLLECTION_DETAIL_ALLOCATOR_ADAPTOR_HPP
- #define BOOST_POLY_COLLECTION_DETAIL_ALLOCATOR_ADAPTOR_HPP
- #if defined(_MSC_VER)
- #pragma once
- #endif
- #include <boost/mp11/function.hpp>
- #include <boost/mp11/integer_sequence.hpp>
- #include <boost/poly_collection/detail/is_constructible.hpp>
- #include <new>
- #include <memory>
- #include <tuple>
- #include <type_traits>
- #include <utility>
- namespace boost{
- namespace poly_collection{
- namespace detail{
- template<typename T>
- class value_holder_base;
- template<typename T>
- class value_holder;
- template<typename T,typename Allocator,typename... Args>
- struct uses_alloc_ctor_impl
- {
- using RawAllocator=typename std::remove_cv<
- typename std::remove_reference<Allocator>::type
- >::type;
- static const bool ua=std::uses_allocator<T,RawAllocator>::value;
- static const int ic=is_constructible<
- T,std::allocator_arg_t,Allocator,Args...>::value?1:0;
- static const int value=ua?2-ic:0;
- };
- template<typename T,typename Allocator,typename... Args>
- struct uses_alloc_ctor:
- std::integral_constant<int,uses_alloc_ctor_impl<T,Allocator,Args...>::value>
- {};
- template<typename Allocator,typename=void>
- struct allocator_is_always_equal:std::is_empty<Allocator>{};
- template<typename Allocator>
- struct allocator_is_always_equal<
- Allocator,
- mp11::mp_void<
- typename std::allocator_traits<Allocator>::is_always_equal
- >
- >:std::allocator_traits<Allocator>::is_always_equal{};
- template<typename Allocator>
- struct allocator_adaptor:Allocator
- {
- using traits=std::allocator_traits<Allocator>;
- using value_type=typename traits::value_type;
- using size_type=typename traits::size_type;
- using difference_type=typename traits::difference_type;
- using pointer=typename traits::pointer;
- using const_pointer=typename traits::const_pointer;
- using void_pointer=typename traits::void_pointer;
- using const_void_pointer=typename traits::const_void_pointer;
- using propagate_on_container_copy_assignment=
- typename traits::propagate_on_container_copy_assignment;
- using propagate_on_container_move_assignment=
- typename traits::propagate_on_container_move_assignment;
- using propagate_on_container_swap=
- typename traits::propagate_on_container_swap;
- using is_always_equal=typename allocator_is_always_equal<Allocator>::type;
- template<typename U>
- struct rebind
- {
- using other=allocator_adaptor<typename traits::template rebind_alloc<U>>;
- };
- allocator_adaptor()=default;
- allocator_adaptor(const allocator_adaptor&)=default;
- template<
- typename Allocator2,
- typename std::enable_if<
- is_constructible<Allocator,const Allocator2&>::value
- >::type* =nullptr
- >
- allocator_adaptor(const Allocator2& x)noexcept:Allocator{x}{}
- template<
- typename Allocator2,
- typename std::enable_if<
- is_constructible<Allocator,const Allocator2&>::value
- >::type* =nullptr
- >
- allocator_adaptor(const allocator_adaptor<Allocator2>& x)noexcept:
- Allocator{x.allocator()}{}
- allocator_adaptor& operator=(const allocator_adaptor&)=default;
- Allocator& allocator()noexcept{return *this;}
- const Allocator& allocator()const noexcept{return *this;}
- template<typename T,typename... Args>
- void construct(T* p,Args&&... args)
- {
- construct_(
- uses_alloc_ctor<T,allocator_adaptor&,Args...>{},
- p,std::forward<Args>(args)...);
- }
- template<typename T,typename... Args>
- void construct(value_holder<T>* p,Args&&... args)
- {
- ::new ((void*)p) value_holder<T>(allocator(),std::forward<Args>(args)...);
- }
- template<typename T1,typename T2,typename... Args1,typename... Args2>
- void construct(
- std::pair<T1,T2>* p,std::piecewise_construct_t,
- std::tuple<Args1...> x,std::tuple<Args2...> y)
- {
- ::new ((void*)p) std::pair<T1,T2>(
- std::piecewise_construct,
- transform_tuple(
- uses_alloc_ctor<T1,allocator_adaptor&,Args1...>{},
- std::move(x),
- mp11::make_index_sequence<sizeof...(Args1)>{}),
- transform_tuple(
- uses_alloc_ctor<T2,allocator_adaptor&,Args2...>{},
- std::move(y),
- mp11::make_index_sequence<sizeof...(Args2)>{})
- );
- }
- template<typename T1,typename T2>
- void construct(std::pair<T1,T2>* p)
- {
- construct(p,std::piecewise_construct,std::tuple<>{},std::tuple<>{});
- }
- template<typename T1,typename T2,typename U,typename V>
- void construct(std::pair<T1,T2>* p,U&& x,V&& y)
- {
- construct(
- p,std::piecewise_construct,
- std::forward_as_tuple(std::forward<U>(x)),
- std::forward_as_tuple(std::forward<V>(y)));
- }
- template<typename T1,typename T2,typename U,typename V>
- void construct(std::pair<T1,T2>* p,const std::pair<U,V>& x)
- {
- construct(
- p,std::piecewise_construct,
- std::forward_as_tuple(x.first),std::forward_as_tuple(x.second));
- }
- template<typename T1,typename T2,typename U,typename V>
- void construct(std::pair<T1,T2>* p,std::pair<U,V>&& x)
- {
- construct(
- p,std::piecewise_construct,
- std::forward_as_tuple(std::forward<U>(x.first)),
- std::forward_as_tuple(std::forward<V>(x.second)));
- }
- template<typename T>
- void destroy(T* p)
- {
- p->~T();
- }
- template<typename T>
- void destroy(value_holder<T>* p)
- {
- traits::destroy(
- allocator(),
- reinterpret_cast<T*>(static_cast<value_holder_base<T>*>(p)));
- }
- allocator_adaptor
- select_on_container_copy_construction()const noexcept
- {
- return traits::select_on_container_copy_construction(allocator());
- }
- private:
- template<typename T,typename... Args>
- void construct_(
- std::integral_constant<int,0>,
- T* p,Args&&... args)
- {
- ::new ((void*)p) T(std::forward<Args>(args)...);
- }
- template<typename T,typename... Args>
- void construct_(
- std::integral_constant<int,1>,
- T* p,Args&&... args)
- {
- ::new ((void*)p) T(std::allocator_arg,*this,std::forward<Args>(args)...);
- }
- template<typename T,typename... Args>
- void construct_(
- std::integral_constant<int,2>,
- T* p,Args&&... args)
- {
- ::new ((void*)p) T(std::forward<Args>(args)...,*this);
- }
- template<typename... Args,std::size_t... I>
- std::tuple<Args&&...> transform_tuple(
- std::integral_constant<int,0>,
- std::tuple<Args...>&& t,mp11::index_sequence<I...>)
- {
- return std::tuple<Args&&...>(std::get<I>(std::move(t))...);
- }
- template<typename... Args,std::size_t... I>
- std::tuple<std::allocator_arg_t,allocator_adaptor&,Args&&...>
- transform_tuple(
- std::integral_constant<int,1>,
- std::tuple<Args...>&& t,mp11::index_sequence<I...>)
- {
- return std::tuple<
- std::allocator_arg_t,allocator_adaptor&,Args&&...>(
- std::allocator_arg,*this,std::get<I>(std::move(t))...);
- }
- template<typename... Args,std::size_t... I>
- std::tuple<Args&&...,allocator_adaptor&>
- transform_tuple(
- std::integral_constant<int,2>,
- std::tuple<Args...>&& t,mp11::index_sequence<I...>)
- {
- return std::tuple<Args&&...,allocator_adaptor&>(
- std::get<I>(std::move(t))...,*this);
- }
- };
- template<typename Allocator1,typename Allocator2>
- bool operator==(
- const allocator_adaptor<Allocator1>& x,
- const allocator_adaptor<Allocator2>& y)noexcept
- {
- return x.allocator()==y.allocator();
- }
- template<typename Allocator1,typename Allocator2>
- bool operator!=(
- const allocator_adaptor<Allocator1>& x,
- const allocator_adaptor<Allocator2>& y)noexcept
- {
- return !(x==y);
- }
- }
- }
- }
- #endif
|