////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Pablo Halpern 2009. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // ////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Ion Gaztanaga 2011-2011. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // // See http://www.boost.org/libs/container for documentation. // ////////////////////////////////////////////////////////////////////////////// #ifndef BOOST_CONTAINER_ALLOCATOR_SCOPED_ALLOCATOR_HPP #define BOOST_CONTAINER_ALLOCATOR_SCOPED_ALLOCATOR_HPP #if (defined _MSC_VER) && (_MSC_VER >= 1200) # pragma once #endif #include #include #include #include #include namespace boost { namespace container { template class scoped_allocator_adaptor; template scoped_allocator_adaptor make_scoped(); template class scoped_allocator_adaptor_base : public OuterAlloc { typedef allocator_traits OuterTraits; public: // Workaround for inability of gcc-4.4.1 to expand InnerAllocs... // typedef scoped_allocator_adaptor inner_allocator_type; typedef decltype(make_scoped()) inner_allocator_type; scoped_allocator_adaptor_base(); template scoped_allocator_adaptor_base(OuterA2&& outerAlloc, const InnerAllocs&... innerAllocs); template scoped_allocator_adaptor_base(const scoped_allocator_adaptor& other); template scoped_allocator_adaptor_base(scoped_allocator_adaptor&& other); inner_allocator_type& inner_allocator() { return _M_inner_allocs; } inner_allocator_type const& inner_allocator() const { return _M_inner_allocs; } // Allocator propagation functions. scoped_allocator_adaptor select_on_container_copy_construction() const; typedef std::integral_constant< bool, OuterTraits::propagate_on_container_copy_assignment::value || inner_allocator_type::propagate_on_container_copy_assignment::value > propagate_on_container_copy_assignment; typedef std::integral_constant< bool, OuterTraits::propagate_on_container_move_assignment::value || inner_allocator_type::propagate_on_container_move_assignment::value > propagate_on_container_move_assignment; typedef std::integral_constant< bool, OuterTraits::propagate_on_container_swap::value || inner_allocator_type::propagate_on_container_swap::value > propagate_on_container_swap; private: inner_allocator_type _M_inner_allocs; }; // Specialization with only one parameter. template class scoped_allocator_adaptor_base : public OuterAlloc { typedef allocator_traits OuterTraits; public: typedef scoped_allocator_adaptor inner_allocator_type; scoped_allocator_adaptor_base(); template scoped_allocator_adaptor_base(OuterA2&& outerAlloc); template scoped_allocator_adaptor_base(const scoped_allocator_adaptor& other); template scoped_allocator_adaptor_base(scoped_allocator_adaptor&& other); inner_allocator_type& inner_allocator() { return static_cast(*this); } inner_allocator_type const& inner_allocator() const { return static_cast(*this); } // Allocator propagation functions. scoped_allocator_adaptor select_on_container_copy_construction() const; typedef typename OuterTraits::propagate_on_container_copy_assignment propagate_on_container_copy_assignment; typedef typename OuterTraits::propagate_on_container_move_assignment propagate_on_container_move_assignment; typedef typename OuterTraits::propagate_on_container_swap propagate_on_container_swap; }; template class scoped_allocator_adaptor : public scoped_allocator_adaptor_base { typedef scoped_allocator_adaptor_base _Base; typedef allocator_traits _Traits; public: typedef OuterAlloc outer_allocator_type; typedef typename _Base::inner_allocator_type inner_allocator_type; typedef typename allocator_traits::size_type size_type; typedef typename allocator_traits::difference_type difference_type; typedef typename allocator_traits::pointer pointer; typedef typename allocator_traits::const_pointer const_pointer; typedef typename allocator_traits::void_pointer void_pointer; typedef typename allocator_traits::const_void_pointer const_void_pointer; typedef typename allocator_traits::value_type value_type; template struct rebind { typedef typename allocator_traits::template rebind_traits rebound_traits; typedef typename rebound_traits::allocator_type rebound_outer; // exposition only typedef scoped_allocator_adaptor other; }; scoped_allocator_adaptor(); scoped_allocator_adaptor(const scoped_allocator_adaptor& other); template scoped_allocator_adaptor(const scoped_allocator_adaptor& other); template scoped_allocator_adaptor(scoped_allocator_adaptor&& other); template scoped_allocator_adaptor(OuterA2&& outerAlloc, const InnerAllocs&... innerAllocs); ~scoped_allocator_adaptor(); inner_allocator_type & inner_allocator() { return _Base::inner_allocator(); } inner_allocator_type const& inner_allocator() const { return _Base::inner_allocator(); } outer_allocator_type & outer_allocator() { return *this; } outer_allocator_type const& outer_allocator() const { return *this; } pointer allocate(size_type n); pointer allocate(size_type n, const_void_pointer hint); void deallocate(pointer p, size_type n); size_type max_size() const; template void construct(T* p, Args&&... args); // Specializations to pass inner_allocator to pair::first and pair::second template void construct(std::pair* p); template void construct(std::pair* p, U&& x, V&& y); template void construct(std::pair* p, const std::pair& pr); template void construct(std::pair* p, std::pair&& pr); template void destroy(T* p); }; template inline bool operator==(const scoped_allocator_adaptor& a, const scoped_allocator_adaptor& b); template inline bool operator!=(const scoped_allocator_adaptor& a, const scoped_allocator_adaptor& b); /////////////////////////////////////////////////////////////////////////////// // Implementation of scoped_allocator_adaptor_base /////////////////////////////////////////////////////////////////////////////// template inline scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base() { } template template scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base(OuterA2&& outerAlloc, const InnerAllocs&... innerAllocs) : OuterAlloc(std::forward(outerAlloc)) , _M_inner_allocs(innerAllocs...) { } template template scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base( const scoped_allocator_adaptor& other) : OuterAlloc(other.outer_allocator()) , _M_inner_allocs(other.inner_allocator()) { } template template scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base( scoped_allocator_adaptor&& other) : OuterAlloc(std::move(other.outer_allocator())) , _M_inner_allocs(std::move(other.inner_allocator())) { } template inline scoped_allocator_adaptor scoped_allocator_adaptor_base:: select_on_container_copy_construction() const { return scoped_allocator_adaptor( allocator_traits::select_on_container_copy_construction( this->outer_allocator()), allocator_traits::select_on_container_copy_construction( this->inner_allocator())); } /////////////////////////////////////////////////////////////////////////////// // Implementation of scoped_allocator_adaptor_base specialization /////////////////////////////////////////////////////////////////////////////// template inline scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base() { } template template scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base(OuterA2&& outerAlloc) : OuterAlloc(std::forward(outerAlloc)) { } template template scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base( const scoped_allocator_adaptor& other) : OuterAlloc(other.outer_allocator()) { } template template scoped_allocator_adaptor_base:: scoped_allocator_adaptor_base( scoped_allocator_adaptor&& other) : OuterAlloc(std::move(other.outer_allocator())) { } // template // inline // scoped_allocator_adaptor& // scoped_allocator_adaptor_base::inner_allocator() // { // return *this; // } // template // inline // scoped_allocator_adaptor const& // scoped_allocator_adaptor_base::inner_allocator() cosnt // { // return *this; // } template inline scoped_allocator_adaptor scoped_allocator_adaptor_base:: select_on_container_copy_construction() const { return allocator_traits::select_on_container_copy_construction( this->outer_allocator()); } /////////////////////////////////////////////////////////////////////////////// // Implementation of scoped_allocator_adaptor details /////////////////////////////////////////////////////////////////////////////// namespace __details { // Overload resolution for __has_ctor resolves to this function // when _Tp is constructible with _Args. Returns true_type(). static void* __void_p; // Declared but not defined template inline auto __has_ctor(int, _Args&&... __args) -> decltype((new (__void_p) _Tp(__args...), std::true_type())) { return std::true_type(); } // Overload resolution for __has_ctor resolves to this function // when _Tp is not constructible with _Args. Returns false_type(). template auto __has_ctor(_LowPriorityConversion, _Args&&...) -> std::false_type { return std::false_type(); } template struct __is_scoped_allocator_imp { template static char test(int, typename T::outer_allocator_type*); template static int test(_LowPriorityConversion, void*); static const bool value = (1 == sizeof(test<_Alloc>(0, 0))); }; template struct __is_scoped_allocator : std::integral_constant::value> { }; #if 0 // Called when outer_allocator_type is not a scoped allocator // (recursion stop). template inline auto __outermost_alloc(_LowPriorityConversion, _Alloc& __a) -> _Alloc& { return __a; } // Called when outer_allocator_type is a scoped allocator to // return the outermost allocator type. template inline auto __outermost_alloc(int, _Alloc& __a) -> decltype(__outermost_alloc(0,__a.outer_allocator())) { return __a.outer_allocator(); } #endif template inline void __dispatch_scoped_construct(std::false_type __uses_alloc, _Ignore __use_alloc_prefix, _OuterAlloc& __outer_alloc, _InnerAlloc& __inner_alloc, _Tp* __p, _Args&&... __args) { // _Tp doesn't use allocators. Construct without an // allocator argument. allocator_traits<_OuterAlloc>::construct(__outer_alloc, __p, std::forward<_Args>(__args)...); } template inline void __dispatch_scoped_construct(std::true_type __uses_alloc, std::true_type __use_alloc_prefix, _OuterAlloc& __outer_alloc, _InnerAlloc& __inner_alloc, _Tp* __p, _Args&&... __args) { // _Tp doesn't use allocators. Construct without an // allocator argument. allocator_traits<_OuterAlloc>::construct(__outer_alloc, __p, allocator_arg, __inner_alloc, std::forward<_Args>(__args)...); } template inline void __dispatch_scoped_construct(std::true_type __uses_alloc, std::false_type __use_alloc_prefix, _OuterAlloc& __outer_alloc, _InnerAlloc& __inner_alloc, _Tp* __p, _Args&&... __args) { // If _Tp uses an allocator compatible with _InnerAlloc, // but the specific constructor does not have a variant that // takes an allocator argument, then program is malformed. // static_assert(has_constructor<_Tp, _Args...>::value, // "Cannot pass inner allocator to this constructor"); allocator_traits<_OuterAlloc>::construct( __outer_alloc, __p, std::forward<_Args>(__args)..., __inner_alloc); } template inline void __do_scoped_construct(std::false_type __scoped_outer, _OuterAlloc& __outer_alloc, _InnerAlloc& __inner_alloc, _Tp* __p, _Args&&... __args) { // Dispatch construction to the correct __dispatch_scoped_construct() // function based on whether _Tp uses an allocator of type // _InnerAlloc and, if so, whether there exists the following // constructor: // _Tp(allocator_arg_t, _InnerAlloc, Args...). auto __uses_alloc = uses_allocator<_Tp, _InnerAlloc>(); auto __use_alloc_prefix = __has_ctor<_Tp>(0, allocator_arg, __inner_alloc, std::forward<_Args>(__args)...); __dispatch_scoped_construct(__uses_alloc, __use_alloc_prefix, __outer_alloc, __inner_alloc, __p, std::forward<_Args>(__args)...); } template void __do_scoped_construct(std::true_type __scoped_outer, _OuterAlloc& __outer_alloc, _InnerAlloc& __inner_alloc, _Tp* __p, _Args&&... __args) { // Use outermost allocator if __outer_alloc is scoped typedef typename _OuterAlloc::outer_allocator_type outerouter; __do_scoped_construct(__is_scoped_allocator(), __outer_alloc.outer_allocator(), __inner_alloc, __p, std::forward<_Args>(__args)...); } } // end namespace __details /////////////////////////////////////////////////////////////////////////////// // Implementation of scoped_allocator_adaptor /////////////////////////////////////////////////////////////////////////////// template scoped_allocator_adaptor:: scoped_allocator_adaptor() { } template scoped_allocator_adaptor:: scoped_allocator_adaptor(const scoped_allocator_adaptor& other) : _Base(other) { } template template scoped_allocator_adaptor:: scoped_allocator_adaptor(const scoped_allocator_adaptor& other) : _Base(other) { } template template scoped_allocator_adaptor:: scoped_allocator_adaptor(scoped_allocator_adaptor&& other) : _Base(std::move(other)) { } template template scoped_allocator_adaptor:: scoped_allocator_adaptor(OuterA2&& outerAlloc, const InnerAllocs&... innerAllocs) : _Base(std::forward(outerAlloc), innerAllocs...) { } template scoped_allocator_adaptor:: ~scoped_allocator_adaptor() { } template inline typename allocator_traits::pointer scoped_allocator_adaptor:: allocate(size_type n) { return allocator_traits::allocate(outer_allocator(), n); } template inline typename allocator_traits::pointer scoped_allocator_adaptor:: allocate(size_type n, const_void_pointer hint) { return allocator_traits::allocate(outer_allocator(), n, hint); } template inline void scoped_allocator_adaptor:: deallocate(pointer p, size_type n) { allocator_traits::deallocate(outer_allocator(), p, n); } template inline typename allocator_traits::size_type scoped_allocator_adaptor::max_size() const { return allocator_traits::max_size(outer_allocator()); } template template inline void scoped_allocator_adaptor:: destroy(T* p) { allocator_traits::destroy(outer_allocator(), p); } template template inline void scoped_allocator_adaptor::construct(T* p, Args&&... args) { __do_scoped_construct(__details::__is_scoped_allocator(), this->outer_allocator(), this->inner_allocator(), p, std::forward(args)...); } template template void scoped_allocator_adaptor::construct( std::pair* p) { construct(addressof(p->first)); try { construct(addressof(p->second)); } catch (...) { destroy(addressof(p->first)); throw; } } template template void scoped_allocator_adaptor::construct( std::pair* p, U&& x, V&& y) { construct(addressof(p->first), std::forward(x)); try { construct(addressof(p->second), std::forward(y)); } catch (...) { destroy(addressof(p->first)); throw; } } template template void scoped_allocator_adaptor::construct( std::pair* p, const std::pair& pr) { construct(addressof(p->first), pr.first); try { construct(addressof(p->second), pr.second); } catch (...) { destroy(addressof(p->first)); throw; } } template template void scoped_allocator_adaptor::construct( std::pair* p, std::pair&& pr) { construct(addressof(p->first), std::move(pr.first)); try { construct(addressof(p->second), std::move(pr.second)); } catch (...) { destroy(addressof(p->first)); throw; } } template inline bool operator==(const scoped_allocator_adaptor& a, const scoped_allocator_adaptor& b) { return a.outer_allocator() == b.outer_allocator() && a.inner_allocator() == b.inner_allocator(); } template inline bool operator==(const scoped_allocator_adaptor& a, const scoped_allocator_adaptor& b) { return a.outer_allocator() == b.outer_allocator(); } template inline bool operator!=(const scoped_allocator_adaptor& a, const scoped_allocator_adaptor& b) { return ! (a == b); } }} // namespace boost { namespace container { #include #endif // BOOST_CONTAINER_ALLOCATOR_SCOPED_ALLOCATOR_HPP