forked from boostorg/container
- Added Extended Allocator support for Boost.Container. - Improved Doxygen generated reference - Fixed ticket #9338 ("VS2005 compiler errors in swap() definition after including container/memory_util.hpp")
413 lines
14 KiB
C++
413 lines
14 KiB
C++
//////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// (C) Copyright Ion Gaztanaga 2011-2013. 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.
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
#include <boost/container/detail/config_begin.hpp>
|
|
#include <cstddef>
|
|
#include <boost/container/allocator_traits.hpp>
|
|
#include <boost/static_assert.hpp>
|
|
#include <boost/type_traits/is_same.hpp>
|
|
#include <boost/type_traits/integral_constant.hpp>
|
|
#include <boost/container/detail/function_detector.hpp>
|
|
#include <boost/move/utility.hpp>
|
|
#include <memory>
|
|
|
|
template<class T>
|
|
class SimpleAllocator
|
|
{
|
|
bool allocate_called_;
|
|
bool deallocate_called_;
|
|
public:
|
|
typedef T value_type;
|
|
|
|
template <class U>
|
|
SimpleAllocator(SimpleAllocator<U>)
|
|
: allocate_called_(false)
|
|
, deallocate_called_(false)
|
|
{}
|
|
|
|
SimpleAllocator()
|
|
: allocate_called_(false)
|
|
, deallocate_called_(false)
|
|
{}
|
|
|
|
T* allocate(std::size_t)
|
|
{ allocate_called_ = true; return 0; }
|
|
|
|
void deallocate(T*, std::size_t)
|
|
{ deallocate_called_ = true; }
|
|
|
|
bool allocate_called() const
|
|
{ return allocate_called_; }
|
|
|
|
bool deallocate_called() const
|
|
{ return deallocate_called_; }
|
|
};
|
|
|
|
template<class T>
|
|
class SimpleSmartPtr
|
|
{
|
|
public:
|
|
|
|
SimpleSmartPtr()
|
|
: ptr_(0)
|
|
{}
|
|
|
|
SimpleSmartPtr(const SimpleSmartPtr &c)
|
|
{ this->ptr_ = c.ptr_; }
|
|
|
|
SimpleSmartPtr & operator=(const SimpleSmartPtr &c)
|
|
{ this->ptr_ = c.ptr_; }
|
|
|
|
typedef T* pointer;
|
|
|
|
private:
|
|
T *ptr_;
|
|
};
|
|
|
|
template<class T>
|
|
class ComplexAllocator
|
|
{
|
|
bool allocate_called_;
|
|
bool deallocate_called_;
|
|
bool allocate_hint_called_;
|
|
bool destroy_called_;
|
|
mutable bool max_size_called_;
|
|
mutable bool select_on_container_copy_construction_called_;
|
|
bool construct_called_;
|
|
|
|
public:
|
|
typedef T value_type;
|
|
typedef SimpleSmartPtr<T> pointer;
|
|
typedef SimpleSmartPtr<const T> const_pointer;
|
|
typedef typename boost::container::
|
|
container_detail::unvoid<T>::type & reference;
|
|
typedef const typename boost::container::
|
|
container_detail::unvoid<T>::type & const_reference;
|
|
typedef SimpleSmartPtr<void> void_pointer;
|
|
typedef SimpleSmartPtr<const void> const_void_pointer;
|
|
typedef signed short difference_type;
|
|
typedef unsigned short size_type;
|
|
typedef boost::true_type propagate_on_container_copy_assignment;
|
|
typedef boost::true_type propagate_on_container_move_assignment;
|
|
typedef boost::true_type propagate_on_container_swap;
|
|
|
|
ComplexAllocator()
|
|
: allocate_called_(false)
|
|
, deallocate_called_(false)
|
|
, allocate_hint_called_(false)
|
|
, destroy_called_(false)
|
|
, max_size_called_(false)
|
|
, select_on_container_copy_construction_called_(false)
|
|
, construct_called_(false)
|
|
{}
|
|
|
|
pointer allocate(size_type)
|
|
{ allocate_called_ = true; return pointer(); }
|
|
|
|
void deallocate(pointer, size_type)
|
|
{ deallocate_called_ = true; }
|
|
|
|
//optional
|
|
ComplexAllocator select_on_container_copy_construction() const
|
|
{ select_on_container_copy_construction_called_ = true; return *this; }
|
|
|
|
pointer allocate(size_type n, const const_void_pointer &)
|
|
{ allocate_hint_called_ = true; return allocate(n); }
|
|
|
|
template<class U>
|
|
void destroy(U*)
|
|
{ destroy_called_ = true; }
|
|
|
|
size_type max_size() const
|
|
{ max_size_called_ = true; return size_type(size_type(0)-1); }
|
|
|
|
#define BOOST_PP_LOCAL_MACRO(n) \
|
|
template<class U BOOST_PP_ENUM_TRAILING_PARAMS(n, class P) > \
|
|
void construct(U *p BOOST_PP_ENUM_TRAILING(n, BOOST_CONTAINER_PP_PARAM_LIST, _)) \
|
|
{ \
|
|
construct_called_ = true; \
|
|
::new (p) U (BOOST_PP_ENUM(n, BOOST_CONTAINER_PP_PARAM_FORWARD, _)); \
|
|
} \
|
|
//
|
|
#define BOOST_PP_LOCAL_LIMITS (0, BOOST_CONTAINER_MAX_CONSTRUCTOR_PARAMETERS)
|
|
#include BOOST_PP_LOCAL_ITERATE()
|
|
|
|
template<class U>
|
|
void construct(U *p, boost::container::default_init_t)
|
|
{
|
|
construct_called_ = true;
|
|
::new (p) U;
|
|
}
|
|
|
|
//getters
|
|
bool allocate_called() const
|
|
{ return allocate_called_; }
|
|
|
|
bool deallocate_called() const
|
|
{ return deallocate_called_; }
|
|
|
|
bool allocate_hint_called() const
|
|
{ return allocate_hint_called_; }
|
|
|
|
bool destroy_called() const
|
|
{ return destroy_called_; }
|
|
|
|
bool max_size_called() const
|
|
{ return max_size_called_; }
|
|
|
|
bool select_on_container_copy_construction_called() const
|
|
{ return select_on_container_copy_construction_called_; }
|
|
|
|
bool construct_called() const
|
|
{ return construct_called_; }
|
|
};
|
|
|
|
class copymovable
|
|
{
|
|
BOOST_COPYABLE_AND_MOVABLE(copymovable)
|
|
|
|
public:
|
|
|
|
bool copymoveconstructed_;
|
|
bool moved_;
|
|
|
|
copymovable(int, int, int)
|
|
: copymoveconstructed_(false), moved_(false)
|
|
{}
|
|
|
|
copymovable()
|
|
: copymoveconstructed_(false), moved_(false)
|
|
{}
|
|
|
|
copymovable(const copymovable &)
|
|
: copymoveconstructed_(true), moved_(false)
|
|
{}
|
|
|
|
copymovable(BOOST_RV_REF(copymovable))
|
|
: copymoveconstructed_(true), moved_(true)
|
|
{}
|
|
|
|
copymovable & operator=(BOOST_COPY_ASSIGN_REF(copymovable) ){ return *this; }
|
|
copymovable & operator=(BOOST_RV_REF(copymovable) ){ return *this; }
|
|
|
|
bool copymoveconstructed() const
|
|
{ return copymoveconstructed_; }
|
|
|
|
bool moved() const
|
|
{ return moved_; }
|
|
};
|
|
|
|
void test_void_allocator()
|
|
{
|
|
boost::container::allocator_traits<std::allocator<void> > stdtraits; (void)stdtraits;
|
|
boost::container::allocator_traits<SimpleAllocator<void> > simtraits; (void)simtraits;
|
|
boost::container::allocator_traits<ComplexAllocator<void> > comtraits; (void)comtraits;
|
|
}
|
|
|
|
int main()
|
|
{
|
|
test_void_allocator();
|
|
|
|
//SimpleAllocator
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::value_type, int>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::pointer, int*>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::const_pointer, const int*>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::void_pointer, void*>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::const_void_pointer, const void*>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::difference_type, std::ptrdiff_t>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::size_type, std::size_t>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::propagate_on_container_copy_assignment::value == false ));
|
|
BOOST_STATIC_ASSERT(( boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::propagate_on_container_move_assignment::value == false ));
|
|
BOOST_STATIC_ASSERT(( boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::propagate_on_container_swap::value == false ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::rebind_traits<double>::allocator_type
|
|
, SimpleAllocator<double> >::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< SimpleAllocator<int> >::rebind_alloc<double>::value_type
|
|
, double >::value ));
|
|
|
|
//ComplexAllocator
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::value_type, int>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::pointer, SimpleSmartPtr<int> >::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::const_pointer, SimpleSmartPtr<const int> >::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::void_pointer, SimpleSmartPtr<void> >::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::const_void_pointer, SimpleSmartPtr<const void> >::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::difference_type, signed short>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::size_type, unsigned short>::value ));
|
|
BOOST_STATIC_ASSERT(( boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::propagate_on_container_copy_assignment::value == true ));
|
|
BOOST_STATIC_ASSERT(( boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::propagate_on_container_move_assignment::value == true ));
|
|
BOOST_STATIC_ASSERT(( boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::propagate_on_container_swap::value == true ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::rebind_traits<double>::allocator_type
|
|
, ComplexAllocator<double> >::value ));
|
|
BOOST_STATIC_ASSERT(( boost::is_same<boost::container::allocator_traits
|
|
< ComplexAllocator<int> >::rebind_alloc<double>::value_type
|
|
, double >::value ));
|
|
|
|
typedef ComplexAllocator<int> CAlloc;
|
|
typedef SimpleAllocator<int> SAlloc;
|
|
typedef boost::container::allocator_traits<CAlloc> CAllocTraits;
|
|
typedef boost::container::allocator_traits<SAlloc> SAllocTraits;
|
|
CAlloc c_alloc;
|
|
SAlloc s_alloc;
|
|
|
|
//allocate
|
|
CAllocTraits::allocate(c_alloc, 1);
|
|
if(!c_alloc.allocate_called()){
|
|
return 1;
|
|
}
|
|
SAllocTraits::allocate(s_alloc, 1);
|
|
if(!s_alloc.allocate_called()){
|
|
return 1;
|
|
}
|
|
|
|
//deallocate
|
|
CAllocTraits::deallocate(c_alloc, CAllocTraits::pointer(), 1);
|
|
if(!c_alloc.deallocate_called()){
|
|
return 1;
|
|
}
|
|
SAllocTraits::deallocate(s_alloc, SAllocTraits::pointer(), 1);
|
|
if(!s_alloc.deallocate_called()){
|
|
return 1;
|
|
}
|
|
|
|
//allocate with hint
|
|
CAllocTraits::allocate(c_alloc, 1, CAllocTraits::const_void_pointer());
|
|
if(!c_alloc.allocate_hint_called()){
|
|
return 1;
|
|
}
|
|
SAllocTraits::allocate(s_alloc, 1, SAllocTraits::const_void_pointer());
|
|
|
|
//destroy
|
|
float dummy;
|
|
CAllocTraits::destroy(c_alloc, &dummy);
|
|
if(!c_alloc.destroy_called()){
|
|
return 1;
|
|
}
|
|
SAllocTraits::destroy(s_alloc, &dummy);
|
|
|
|
//max_size
|
|
CAllocTraits::max_size(c_alloc);
|
|
if(!c_alloc.max_size_called()){
|
|
return 1;
|
|
}
|
|
SAllocTraits::max_size(s_alloc);
|
|
|
|
//select_on_container_copy_construction
|
|
CAllocTraits::select_on_container_copy_construction(c_alloc);
|
|
if(!c_alloc.select_on_container_copy_construction_called()){
|
|
return 1;
|
|
}
|
|
SAllocTraits::select_on_container_copy_construction(s_alloc);
|
|
|
|
//construct
|
|
{
|
|
copymovable c;
|
|
c.copymoveconstructed_ = true;
|
|
c.copymoveconstructed_ = true;
|
|
CAllocTraits::construct(c_alloc, &c);
|
|
if(!c_alloc.construct_called() || c.copymoveconstructed() || c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
int i = 5;
|
|
CAllocTraits::construct(c_alloc, &i, boost::container::default_init);
|
|
if(!c_alloc.construct_called() || i != 5){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
copymovable c2;
|
|
CAllocTraits::construct(c_alloc, &c, c2);
|
|
if(!c_alloc.construct_called() || !c.copymoveconstructed() || c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
copymovable c2;
|
|
CAllocTraits::construct(c_alloc, &c, ::boost::move(c2));
|
|
if(!c_alloc.construct_called() || !c.copymoveconstructed() || !c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
c.copymoveconstructed_ = true;
|
|
c.copymoveconstructed_ = true;
|
|
SAllocTraits::construct(s_alloc, &c);
|
|
if(c.copymoveconstructed() || c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
int i = 4;
|
|
SAllocTraits::construct(s_alloc, &i, boost::container::default_init);
|
|
if(i != 4){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
copymovable c2;
|
|
SAllocTraits::construct(s_alloc, &c, c2);
|
|
if(!c.copymoveconstructed() || c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
copymovable c2;
|
|
SAllocTraits::construct(s_alloc, &c, ::boost::move(c2));
|
|
if(!c.copymoveconstructed() || !c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
CAllocTraits::construct(c_alloc, &c, 0, 1, 2);
|
|
if(!c_alloc.construct_called() || c.copymoveconstructed() || c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
{
|
|
copymovable c;
|
|
copymovable c2;
|
|
SAllocTraits::construct(s_alloc, &c, 0, 1, 2);
|
|
if(c.copymoveconstructed() || c.moved()){
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#include <boost/container/detail/config_end.hpp>
|