mirror of
https://github.com/boostorg/container.git
synced 2025-08-01 21:44:27 +02:00
Fixes #218 ("small_vector static capacity is too small when not a multiple of 8 bytes")
This commit is contained in:
@@ -1341,6 +1341,7 @@ use [*Boost.Container]? There are several reasons for that:
|
|||||||
[section:release_notes_boost_1_80_00 Boost 1.80 Release]
|
[section:release_notes_boost_1_80_00 Boost 1.80 Release]
|
||||||
|
|
||||||
* Fixed bugs/issues:
|
* Fixed bugs/issues:
|
||||||
|
* [@https://github.com/boostorg/container/issues/221 GitHub #218: ['"small_vector static capacity is too small when not a multiple of 8 bytes"]].
|
||||||
* [@https://github.com/boostorg/container/issues/221 GitHub #221: ['"flat_set and friends should offer a const sequence_type& sequence() const method (...)"]].
|
* [@https://github.com/boostorg/container/issues/221 GitHub #221: ['"flat_set and friends should offer a const sequence_type& sequence() const method (...)"]].
|
||||||
* [@https://github.com/boostorg/container/pull/222 GitHub #222: ['"Fix incomplete type error when using list with pair"]].
|
* [@https://github.com/boostorg/container/pull/222 GitHub #222: ['"Fix incomplete type error when using list with pair"]].
|
||||||
* [@https://github.com/boostorg/container/issues/223 GitHub #223: ['"Possible copypaste typo"]].
|
* [@https://github.com/boostorg/container/issues/223 GitHub #223: ['"Possible copypaste typo"]].
|
||||||
|
@@ -77,14 +77,14 @@ struct get_vopt_from_svopt<void>
|
|||||||
typedef void type;
|
typedef void type;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <class T, class SecondaryAllocator, class Options>
|
template <class T, class SecAlloc, class Options>
|
||||||
struct vector_for_small_vector
|
struct vector_for_small_vector
|
||||||
{
|
{
|
||||||
typedef vector
|
typedef vector
|
||||||
< T
|
< T
|
||||||
, small_vector_allocator
|
, small_vector_allocator
|
||||||
< T
|
< T
|
||||||
, typename allocator_traits<typename real_allocator<T, SecondaryAllocator>::type>::template portable_rebind_alloc<void>::type
|
, typename allocator_traits<typename real_allocator<T, SecAlloc>::type>::template portable_rebind_alloc<void>::type
|
||||||
, Options>
|
, Options>
|
||||||
, typename dtl::get_vopt_from_svopt<Options>::type
|
, typename dtl::get_vopt_from_svopt<Options>::type
|
||||||
> type;
|
> type;
|
||||||
@@ -118,15 +118,15 @@ struct vector_for_small_vector
|
|||||||
//! `boost::container::vector< T, small_vector_allocator<T, Allocator> >`
|
//! `boost::container::vector< T, small_vector_allocator<T, Allocator> >`
|
||||||
//! and internal storage can be obtained downcasting that vector
|
//! and internal storage can be obtained downcasting that vector
|
||||||
//! to `small_vector_base<T>`.
|
//! to `small_vector_base<T>`.
|
||||||
template<class T, class VoidAllocator BOOST_CONTAINER_DOCONLY(= void), class Options BOOST_CONTAINER_DOCONLY(= void)>
|
template<class T, class VoidAlloc BOOST_CONTAINER_DOCONLY(= void), class Options BOOST_CONTAINER_DOCONLY(= void)>
|
||||||
class small_vector_allocator
|
class small_vector_allocator
|
||||||
: public allocator_traits<typename real_allocator<T, VoidAllocator>::type>::template portable_rebind_alloc<T>::type
|
: public allocator_traits<VoidAlloc>::template portable_rebind_alloc<T>::type
|
||||||
{
|
{
|
||||||
typedef unsigned int allocation_type;
|
typedef unsigned int allocation_type;
|
||||||
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
private:
|
private:
|
||||||
|
|
||||||
typedef typename allocator_traits<typename real_allocator<T, VoidAllocator>::type>::template portable_rebind_alloc<T>::type allocator_type;
|
typedef typename allocator_traits<VoidAlloc>::template portable_rebind_alloc<T>::type allocator_type;
|
||||||
|
|
||||||
BOOST_COPYABLE_AND_MOVABLE(small_vector_allocator)
|
BOOST_COPYABLE_AND_MOVABLE(small_vector_allocator)
|
||||||
|
|
||||||
@@ -282,65 +282,36 @@ class small_vector_allocator
|
|||||||
{ return !(l == r); }
|
{ return !(l == r); }
|
||||||
|
|
||||||
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
/*
|
|
||||||
//!An advanced function that offers in-place expansion shrink to fit and new allocation
|
|
||||||
//!capabilities. Memory allocated with this function can only be deallocated with deallocate()
|
|
||||||
//!or deallocate_many().
|
|
||||||
//!This function is available only with Version == 2
|
|
||||||
pointer allocation_command(allocation_type command,
|
|
||||||
size_type limit_size,
|
|
||||||
size_type &prefer_in_recvd_out_size,
|
|
||||||
pointer &reuse)
|
|
||||||
{ return allocator_traits_type::allocation_command(command, limit_size, prefer_in_recvd_out_size, reuse); }
|
|
||||||
|
|
||||||
//!Returns maximum the number of objects the previously allocated memory
|
|
||||||
//!pointed by p can hold.
|
|
||||||
//!Memory must not have been allocated with
|
|
||||||
//!allocate_one or allocate_individual.
|
|
||||||
//!This function is available only with Version == 2
|
|
||||||
size_type size(pointer p) const BOOST_NOEXCEPT_OR_NOTHROW
|
|
||||||
{ return allocator_traits_type::size(p); }
|
|
||||||
*/
|
|
||||||
private:
|
private:
|
||||||
/*
|
|
||||||
//!Allocates just one object. Memory allocated with this function
|
|
||||||
//!must be deallocated only with deallocate_one().
|
|
||||||
//!Throws bad_alloc if there is no enough memory
|
|
||||||
//!This function is available only with Version == 2
|
|
||||||
using allocator_type::allocate_one;
|
|
||||||
using allocator_type::allocate_individual;
|
|
||||||
using allocator_type::deallocate_one;
|
|
||||||
using allocator_type::deallocate_individual;
|
|
||||||
using allocator_type::allocate_many;
|
|
||||||
using allocator_type::deallocate_many;*/
|
|
||||||
|
|
||||||
typedef vector_alloc_holder< small_vector_allocator, size_type > vector_alloc_holder_t;
|
|
||||||
typedef typename dtl::vector_for_small_vector<T, allocator_type, Options>::type vector_base;
|
|
||||||
typedef small_vector_base<value_type, allocator_type, Options> derived_type;
|
typedef small_vector_base<value_type, allocator_type, Options> derived_type;
|
||||||
|
typedef typename dtl::vector_for_small_vector
|
||||||
|
<value_type, allocator_type, Options>::type vector_type;
|
||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE bool is_internal_storage(const_pointer p) const
|
BOOST_CONTAINER_FORCEINLINE bool is_internal_storage(const_pointer p) const
|
||||||
{ return this->internal_storage() == p; }
|
{ return this->internal_storage() == p; }
|
||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE
|
public:
|
||||||
const_pointer internal_storage() const
|
BOOST_CONTAINER_FORCEINLINE const_pointer internal_storage() const BOOST_NOEXCEPT_OR_NOTHROW;
|
||||||
{
|
BOOST_CONTAINER_FORCEINLINE pointer internal_storage() BOOST_NOEXCEPT_OR_NOTHROW;
|
||||||
const vector_alloc_holder_t &v_holder = static_cast<const vector_alloc_holder_t &>(*this);
|
|
||||||
const vector_base &v_base = *move_detail::force_ptr<const vector_base *>(&v_holder);
|
|
||||||
const derived_type &d_base = static_cast<const derived_type &>(v_base);
|
|
||||||
return d_base.internal_storage();
|
|
||||||
}
|
|
||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE
|
|
||||||
pointer internal_storage()
|
|
||||||
{
|
|
||||||
vector_alloc_holder_t &v_holder = static_cast<vector_alloc_holder_t &>(*this);
|
|
||||||
vector_base &v_base = *move_detail::force_ptr<vector_base *>(&v_holder);
|
|
||||||
derived_type &d_base = static_cast<derived_type &>(v_base);
|
|
||||||
return d_base.internal_storage();
|
|
||||||
}
|
|
||||||
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<class T, std::size_t N, std::size_t Alignment>
|
||||||
|
struct small_vector_storage
|
||||||
|
{
|
||||||
|
typedef typename dtl::aligned_storage
|
||||||
|
<sizeof(T)*N, Alignment>::type storage_type;
|
||||||
|
storage_type m_storage;
|
||||||
|
static const std::size_t sms_size = sizeof(storage_type)/sizeof(T);
|
||||||
|
};
|
||||||
|
|
||||||
|
template<class T, std::size_t Alignment>
|
||||||
|
struct small_vector_storage<T, 0u, Alignment>
|
||||||
|
{
|
||||||
|
static const std::size_t sms_size = 0u;
|
||||||
|
};
|
||||||
|
|
||||||
//! This class consists of common code from all small_vector<T, N> types that don't depend on the
|
//! This class consists of common code from all small_vector<T, N> types that don't depend on the
|
||||||
//! "N" template parameter. This class is non-copyable and non-destructible, so this class typically
|
//! "N" template parameter. This class is non-copyable and non-destructible, so this class typically
|
||||||
//! used as reference argument to functions that read or write small vectors. Since `small_vector<T, N>`
|
//! used as reference argument to functions that read or write small vectors. Since `small_vector<T, N>`
|
||||||
@@ -366,20 +337,20 @@ class small_vector_allocator
|
|||||||
//!
|
//!
|
||||||
//! All `boost::container:vector` member functions are inherited. See `vector` documentation for details.
|
//! All `boost::container:vector` member functions are inherited. See `vector` documentation for details.
|
||||||
//!
|
//!
|
||||||
template <class T, class SecondaryAllocator, class Options>
|
template <class T, class SecAlloc, class Options>
|
||||||
class small_vector_base
|
class small_vector_base
|
||||||
: public dtl::vector_for_small_vector<T, SecondaryAllocator, Options>::type
|
: public dtl::vector_for_small_vector<T, SecAlloc, Options>::type
|
||||||
{
|
{
|
||||||
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKEDVECTOR
|
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKEDVECTOR
|
||||||
public:
|
public:
|
||||||
//Make it public as it will be inherited by small_vector and container
|
//Make it public as it will be inherited by small_vector and container
|
||||||
//must have this public member
|
//must have this public member
|
||||||
typedef typename real_allocator<T, SecondaryAllocator>::type secondary_allocator_t;
|
typedef typename real_allocator<T, SecAlloc>::type secondary_allocator_t;
|
||||||
typedef typename allocator_traits<secondary_allocator_t>::
|
typedef typename allocator_traits<secondary_allocator_t>::
|
||||||
template portable_rebind_alloc<void>::type void_allocator_t;
|
template portable_rebind_alloc<void>::type void_allocator_t;
|
||||||
typedef typename dtl::get_small_vector_opt<Options>::type options_t;
|
typedef typename dtl::get_small_vector_opt<Options>::type options_t;
|
||||||
typedef typename dtl::vector_for_small_vector
|
typedef typename dtl::vector_for_small_vector
|
||||||
<T, SecondaryAllocator, Options>::type base_type;
|
<T, SecAlloc, Options>::type base_type;
|
||||||
typedef typename allocator_traits<secondary_allocator_t>::pointer pointer;
|
typedef typename allocator_traits<secondary_allocator_t>::pointer pointer;
|
||||||
typedef typename allocator_traits<secondary_allocator_t>::const_pointer const_pointer;
|
typedef typename allocator_traits<secondary_allocator_t>::const_pointer const_pointer;
|
||||||
typedef typename allocator_traits<secondary_allocator_t>::void_pointer void_pointer;
|
typedef typename allocator_traits<secondary_allocator_t>::void_pointer void_pointer;
|
||||||
@@ -393,34 +364,18 @@ class small_vector_base
|
|||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE
|
BOOST_CONTAINER_FORCEINLINE
|
||||||
const_pointer internal_storage() const BOOST_NOEXCEPT_OR_NOTHROW
|
const_pointer internal_storage() const BOOST_NOEXCEPT_OR_NOTHROW
|
||||||
{
|
{ return this->base_type::get_stored_allocator().internal_storage(); }
|
||||||
typedef typename boost::intrusive::pointer_traits<const_pointer>::template
|
|
||||||
rebind_pointer<const unsigned char>::type const_char_pointer;
|
|
||||||
const_void_pointer void_p = boost::intrusive::pointer_traits<const_char_pointer>::
|
|
||||||
pointer_to(*m_storage_start.data);
|
|
||||||
return boost::intrusive::pointer_traits<const_pointer>::static_cast_from(void_p);
|
|
||||||
}
|
|
||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE
|
BOOST_CONTAINER_FORCEINLINE
|
||||||
pointer internal_storage() BOOST_NOEXCEPT_OR_NOTHROW
|
pointer internal_storage() BOOST_NOEXCEPT_OR_NOTHROW
|
||||||
{
|
{ return this->base_type::get_stored_allocator().internal_storage(); }
|
||||||
typedef typename boost::intrusive::pointer_traits<pointer>::template
|
|
||||||
rebind_pointer<unsigned char>::type char_pointer;
|
|
||||||
void_pointer void_p = boost::intrusive::pointer_traits<char_pointer>::
|
|
||||||
pointer_to(*m_storage_start.data);
|
|
||||||
return boost::intrusive::pointer_traits<pointer>::static_cast_from(void_p);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
private:
|
||||||
base_type &as_base() { return static_cast<base_type&>(*this); }
|
base_type &as_base() { return static_cast<base_type&>(*this); }
|
||||||
const base_type &as_base() const { return static_cast<const base_type&>(*this); }
|
const base_type &as_base() const { return static_cast<const base_type&>(*this); }
|
||||||
|
|
||||||
static const std::size_t final_alignment =
|
|
||||||
options_t::inplace_alignment ? options_t::inplace_alignment : dtl::alignment_of<T>::value;
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
typedef typename dtl::aligned_storage
|
|
||||||
<sizeof(T), final_alignment>::type storage_type;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE explicit small_vector_base(initial_capacity_t, std::size_t initial_capacity)
|
BOOST_CONTAINER_FORCEINLINE explicit small_vector_base(initial_capacity_t, std::size_t initial_capacity)
|
||||||
@@ -433,11 +388,6 @@ class small_vector_base
|
|||||||
{}
|
{}
|
||||||
|
|
||||||
//~small_vector_base(){}
|
//~small_vector_base(){}
|
||||||
|
|
||||||
private:
|
|
||||||
//The only member
|
|
||||||
storage_type m_storage_start;
|
|
||||||
|
|
||||||
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@@ -469,67 +419,74 @@ class small_vector_base
|
|||||||
|
|
||||||
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
|
|
||||||
/////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// small_vector_storage_calculator
|
|
||||||
//
|
|
||||||
/////////////////////////////////////////////////////
|
|
||||||
template<std::size_t Needed, std::size_t Hdr, std::size_t SSize, bool NeedsZero = (0u == Needed || Needed <= Hdr)>
|
|
||||||
struct small_vector_storage_calculator_helper
|
|
||||||
{
|
|
||||||
static const std::size_t value = (Needed - Hdr - 1u)/SSize + 1u;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<std::size_t Needed, std::size_t Hdr, std::size_t SSize>
|
|
||||||
struct small_vector_storage_calculator_helper<Needed, Hdr, SSize, true>
|
|
||||||
{
|
|
||||||
static const std::size_t value = 0u;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<class Storage, class Allocator, class T, std::size_t N, class Options>
|
|
||||||
struct small_vector_storage_calculator
|
|
||||||
{
|
|
||||||
typedef small_vector_base<T, Allocator, Options> svh_type;
|
|
||||||
typedef typename real_allocator<T, Allocator>::type value_allocator_t;
|
|
||||||
typedef typename allocator_traits<value_allocator_t>::template portable_rebind_alloc<void>::type void_allocator_t;
|
|
||||||
typedef typename dtl::vector_for_small_vector<T, void_allocator_t, Options>::type svhb_type;
|
|
||||||
|
|
||||||
static const std::size_t s_align = dtl::alignment_of<Storage>::value;
|
|
||||||
static const std::size_t s_size = sizeof(Storage);
|
|
||||||
static const std::size_t svh_sizeof = sizeof(svh_type);
|
|
||||||
static const std::size_t svhb_sizeof = sizeof(svhb_type);
|
|
||||||
static const std::size_t s_start = ((svhb_sizeof-1)/s_align+1)*s_align;
|
|
||||||
static const std::size_t header_bytes = svh_sizeof-s_start;
|
|
||||||
static const std::size_t needed_bytes = sizeof(T)*N;
|
|
||||||
static const std::size_t needed_extra_storages =
|
|
||||||
small_vector_storage_calculator_helper<needed_bytes, header_bytes, s_size>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
/////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////
|
||||||
//
|
//
|
||||||
// small_vector_storage_definer
|
// small_vector_storage_definer
|
||||||
//
|
//
|
||||||
/////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////
|
||||||
template<class Storage, std::size_t N>
|
template<class T, std::size_t N, class Options>
|
||||||
struct small_vector_storage
|
|
||||||
{
|
|
||||||
Storage m_rest_of_storage[N];
|
|
||||||
};
|
|
||||||
|
|
||||||
template<class Storage>
|
|
||||||
struct small_vector_storage<Storage, 0>
|
|
||||||
{};
|
|
||||||
|
|
||||||
template<class T, class Allocator, std::size_t N, class Options>
|
|
||||||
struct small_vector_storage_definer
|
struct small_vector_storage_definer
|
||||||
{
|
{
|
||||||
typedef T value_type;
|
typedef typename dtl::get_small_vector_opt<Options>::type options_t;
|
||||||
typedef typename small_vector_base<value_type, Allocator, Options>::storage_type storage_type;
|
static const std::size_t final_alignment =
|
||||||
static const std::size_t needed_extra_storages =
|
options_t::inplace_alignment ? options_t::inplace_alignment : dtl::alignment_of<T>::value;
|
||||||
small_vector_storage_calculator<storage_type, Allocator, value_type, N, Options>::needed_extra_storages;
|
typedef small_vector_storage<T, N, final_alignment> type;
|
||||||
typedef small_vector_storage<storage_type, needed_extra_storages> type;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template <class T, class SecAlloc, class Options>
|
||||||
|
struct small_vector_storage_strawman
|
||||||
|
: public small_vector_base<T, SecAlloc, Options>
|
||||||
|
, public small_vector_storage_definer<T, 1, Options>::type
|
||||||
|
{
|
||||||
|
typedef typename small_vector_storage_definer<T, 1, Options>::type sm_storage_t;
|
||||||
|
};
|
||||||
|
|
||||||
|
//Internal storage hack
|
||||||
|
template<class T, class VoidAlloc, class Options>
|
||||||
|
BOOST_CONTAINER_FORCEINLINE typename small_vector_allocator<T, VoidAlloc, Options>::const_pointer
|
||||||
|
small_vector_allocator<T, VoidAlloc, Options>::internal_storage() const BOOST_NOEXCEPT_OR_NOTHROW
|
||||||
|
{
|
||||||
|
typedef small_vector_storage_strawman<T, allocator_type, Options> strawman_t;
|
||||||
|
typedef typename strawman_t::sm_storage_t sm_storage_t;
|
||||||
|
|
||||||
|
#if defined(BOOST_GCC) && (BOOST_GCC >= 40600)
|
||||||
|
#pragma GCC diagnostic push
|
||||||
|
#pragma GCC diagnostic ignored "-Wcast-align"
|
||||||
|
#endif
|
||||||
|
const vector_type& v = reinterpret_cast<const vector_type&>(*this);
|
||||||
|
BOOST_ASSERT((std::size_t(this) % dtl::alignment_of<strawman_t>::value) == 0);
|
||||||
|
#if defined(BOOST_GCC) && (BOOST_GCC >= 40600)
|
||||||
|
#pragma GCC diagnostic pop
|
||||||
|
#endif
|
||||||
|
|
||||||
|
const strawman_t &straw = static_cast<const strawman_t&>(v);
|
||||||
|
const sm_storage_t& stor = static_cast<const sm_storage_t&>(straw);
|
||||||
|
return boost::intrusive::pointer_traits<pointer>::pointer_to(*((T*)&stor.m_storage));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class T, class VoidAlloc, class Options>
|
||||||
|
BOOST_CONTAINER_FORCEINLINE typename small_vector_allocator<T, VoidAlloc, Options>::pointer
|
||||||
|
small_vector_allocator<T, VoidAlloc, Options>::internal_storage() BOOST_NOEXCEPT_OR_NOTHROW
|
||||||
|
{
|
||||||
|
typedef small_vector_storage_strawman<T, allocator_type, Options> strawman_t;
|
||||||
|
typedef typename strawman_t::sm_storage_t sm_storage_t;
|
||||||
|
|
||||||
|
#if defined(BOOST_GCC) && (BOOST_GCC >= 40600)
|
||||||
|
#pragma GCC diagnostic push
|
||||||
|
#pragma GCC diagnostic ignored "-Wcast-align"
|
||||||
|
#endif
|
||||||
|
vector_type& v = reinterpret_cast<vector_type&>(*this);
|
||||||
|
BOOST_ASSERT((std::size_t(this) % dtl::alignment_of<strawman_t>::value) == 0);
|
||||||
|
#if defined(BOOST_GCC) && (BOOST_GCC >= 40600)
|
||||||
|
#pragma GCC diagnostic pop
|
||||||
|
#endif
|
||||||
|
|
||||||
|
strawman_t &straw = static_cast<strawman_t&>(v);
|
||||||
|
sm_storage_t& stor = static_cast<sm_storage_t&>(straw);
|
||||||
|
return boost::intrusive::pointer_traits<pointer>::pointer_to(*((T*)&stor.m_storage));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
|
|
||||||
//! small_vector is a vector-like container optimized for the case when it contains few elements.
|
//! small_vector is a vector-like container optimized for the case when it contains few elements.
|
||||||
@@ -547,41 +504,31 @@ struct small_vector_storage_definer
|
|||||||
//! for the default allocator
|
//! for the default allocator
|
||||||
//! |tparam Options A type produced from \c boost::container::small_vector_options.
|
//! |tparam Options A type produced from \c boost::container::small_vector_options.
|
||||||
template <class T, std::size_t N, class Allocator BOOST_CONTAINER_DOCONLY(= void), class Options BOOST_CONTAINER_DOCONLY(= void) >
|
template <class T, std::size_t N, class Allocator BOOST_CONTAINER_DOCONLY(= void), class Options BOOST_CONTAINER_DOCONLY(= void) >
|
||||||
class small_vector : public small_vector_base<T, Allocator, Options>
|
class small_vector
|
||||||
|
: public small_vector_base<T, Allocator, Options>
|
||||||
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
, private small_vector_storage_definer<T, Allocator, N, Options>::type
|
, private small_vector_storage_definer<T, N, Options>::type
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
typedef small_vector_base<T, Allocator, Options> base_type;
|
|
||||||
typedef typename small_vector_storage_definer
|
|
||||||
<T, Allocator, N, Options>::type remaining_storage_holder;
|
|
||||||
|
|
||||||
BOOST_COPYABLE_AND_MOVABLE(small_vector)
|
BOOST_COPYABLE_AND_MOVABLE(small_vector)
|
||||||
|
|
||||||
typedef allocator_traits<typename base_type::allocator_type> allocator_traits_type;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
typedef small_vector_storage_calculator
|
typedef small_vector_base<T, Allocator, Options> base_type;
|
||||||
< typename small_vector_base<T, Allocator, Options>::storage_type
|
|
||||||
, Allocator, T, N, Options> storage_test;
|
|
||||||
|
|
||||||
static const std::size_t needed_extra_storages = storage_test::needed_extra_storages;
|
|
||||||
static const std::size_t needed_bytes = storage_test::needed_bytes;
|
|
||||||
static const std::size_t header_bytes = storage_test::header_bytes;
|
|
||||||
static const std::size_t s_start = storage_test::s_start;
|
|
||||||
|
|
||||||
typedef typename base_type::allocator_type allocator_type;
|
typedef typename base_type::allocator_type allocator_type;
|
||||||
typedef typename base_type::size_type size_type;
|
typedef typename base_type::size_type size_type;
|
||||||
typedef typename base_type::value_type value_type;
|
typedef typename base_type::value_type value_type;
|
||||||
|
|
||||||
BOOST_CONTAINER_FORCEINLINE static std::size_t internal_capacity()
|
BOOST_CONTAINER_FORCEINLINE static std::size_t internal_capacity()
|
||||||
{ return (sizeof(small_vector) - storage_test::s_start)/sizeof(T); }
|
{ return static_capacity; }
|
||||||
|
|
||||||
|
typedef allocator_traits<typename base_type::allocator_type> allocator_traits_type;
|
||||||
|
|
||||||
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
|
||||||
|
|
||||||
//! @brief The capacity/max size of the container
|
//! @brief The capacity/max size of the container
|
||||||
static const size_type static_capacity = N;
|
static const size_type static_capacity = small_vector_storage_definer<T, N, Options>::type::sms_size;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
BOOST_CONTAINER_FORCEINLINE small_vector()
|
BOOST_CONTAINER_FORCEINLINE small_vector()
|
||||||
|
Reference in New Issue
Block a user