]> git.proxmox.com Git - ceph.git/blame - ceph/src/boost/boost/container/small_vector.hpp
update sources to ceph Nautilus 14.2.1
[ceph.git] / ceph / src / boost / boost / container / small_vector.hpp
CommitLineData
7c673cae
FG
1//////////////////////////////////////////////////////////////////////////////
2//
3// (C) Copyright Ion Gaztanaga 2015-2015. Distributed under the Boost
4// Software License, Version 1.0. (See accompanying file
5// LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6//
7// See http://www.boost.org/libs/container for documentation.
8//
9//////////////////////////////////////////////////////////////////////////////
10
11#ifndef BOOST_CONTAINER_CONTAINER_SMALL_VECTOR_HPP
12#define BOOST_CONTAINER_CONTAINER_SMALL_VECTOR_HPP
13
14#ifndef BOOST_CONFIG_HPP
15# include <boost/config.hpp>
16#endif
17
18#if defined(BOOST_HAS_PRAGMA_ONCE)
19# pragma once
20#endif
21
22#include <boost/container/detail/config_begin.hpp>
23#include <boost/container/detail/workaround.hpp>
24
25// container
26#include <boost/container/container_fwd.hpp>
27#include <boost/container/vector.hpp>
28#include <boost/container/allocator_traits.hpp>
29#include <boost/container/new_allocator.hpp> //new_allocator
30// container/detail
31#include <boost/container/detail/type_traits.hpp>
32#include <boost/container/detail/version_type.hpp>
33
34//move
35#include <boost/move/adl_move_swap.hpp>
36#include <boost/move/iterator.hpp>
37
38//move/detail
39#if defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
40#include <boost/move/detail/fwd_macros.hpp>
41#endif
42
43//std
44#if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
45#include <initializer_list> //for std::initializer_list
46#endif
47
48namespace boost {
49namespace container {
50
51#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
52
53template <class T, class Allocator = new_allocator<T> >
54class small_vector_base;
55
56#endif
57
58//! A non-standard allocator used to implement `small_vector`.
59//! Users should never use it directly. It is described here
60//! for documentation purposes.
61//!
62//! This allocator inherits from a standard-conforming allocator
63//! and forwards member functions to the standard allocator except
64//! when internal storage is being used as memory source.
65//!
66//! This allocator is a "partially_propagable" allocator and
67//! defines `is_partially_propagable` as true_type.
68//!
69//! A partially propagable allocator means that not all storage
70//! allocatod by an instance of `small_vector_allocator` can be
71//! deallocated by another instance of this type, even if both
72//! instances compare equal or an instance is propagated to another
73//! one using the copy/move constructor or assignment. The storage that
74//! can never be propagated is identified by `storage_is_unpropagable(p)`.
75//!
76//! `boost::container::vector` supports partially propagable allocators
77//! fallbacking to deep copy/swap/move operations when internal storage
78//! is being used to store vector elements.
79//!
80//! `small_vector_allocator` assumes that will be instantiated as
81//! `boost::container::vector< T, small_vector_allocator<Allocator> >`
82//! and internal storage can be obtained downcasting that vector
83//! to `small_vector_base<T>`.
84template<class Allocator>
85class small_vector_allocator
86 : public Allocator
87{
88 typedef unsigned int allocation_type;
89 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
90 private:
91
92 BOOST_COPYABLE_AND_MOVABLE(small_vector_allocator)
93
94 BOOST_CONTAINER_FORCEINLINE const Allocator &as_base() const
95 { return static_cast<const Allocator&>(*this); }
96
97 BOOST_CONTAINER_FORCEINLINE Allocator &as_base()
98 { return static_cast<Allocator&>(*this); }
99
100 #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
101
102 public:
103 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
104 typedef allocator_traits<Allocator> allocator_traits_type;
105 #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
106
107 typedef typename allocator_traits<Allocator>::value_type value_type;
108 typedef typename allocator_traits<Allocator>::pointer pointer;
109 typedef typename allocator_traits<Allocator>::const_pointer const_pointer;
110 typedef typename allocator_traits<Allocator>::reference reference;
111 typedef typename allocator_traits<Allocator>::const_reference const_reference;
112 typedef typename allocator_traits<Allocator>::size_type size_type;
113 typedef typename allocator_traits<Allocator>::difference_type difference_type;
114 typedef typename allocator_traits<Allocator>::void_pointer void_pointer;
115 typedef typename allocator_traits<Allocator>::const_void_pointer const_void_pointer;
116
117 typedef typename allocator_traits<Allocator>::propagate_on_container_copy_assignment propagate_on_container_copy_assignment;
118 typedef typename allocator_traits<Allocator>::propagate_on_container_move_assignment propagate_on_container_move_assignment;
119 typedef typename allocator_traits<Allocator>::propagate_on_container_swap propagate_on_container_swap;
120 //! An integral constant with member `value == false`
11fdf7f2 121 typedef BOOST_CONTAINER_IMPDEF(dtl::bool_<false>) is_always_equal;
7c673cae 122 //! An integral constant with member `value == true`
11fdf7f2 123 typedef BOOST_CONTAINER_IMPDEF(dtl::bool_<true>) is_partially_propagable;
7c673cae 124
11fdf7f2 125 BOOST_CONTAINER_DOCIGN(typedef dtl::version_type<small_vector_allocator BOOST_CONTAINER_I 1> version;)
7c673cae
FG
126
127 //!Obtains an small_vector_allocator that allocates
128 //!objects of type T2
129 template<class T2>
130 struct rebind
131 {
132 typedef typename allocator_traits<Allocator>::template rebind_alloc<T2>::type other;
133 };
134
135 #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) || defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
136 //!Constructor from arbitrary arguments
137 template<class ...Args>
138 BOOST_CONTAINER_FORCEINLINE explicit small_vector_allocator(BOOST_FWD_REF(Args) ...args)
139 : Allocator(::boost::forward<Args>(args)...)
140 {}
141 #else
142 #define BOOST_CONTAINER_SMALL_VECTOR_ALLOCATOR_CTOR_CODE(N) \
143 BOOST_MOVE_TMPL_LT##N BOOST_MOVE_CLASS##N BOOST_MOVE_GT##N \
144 BOOST_CONTAINER_FORCEINLINE explicit small_vector_allocator(BOOST_MOVE_UREF##N)\
145 : Allocator(BOOST_MOVE_FWD##N)\
146 {}\
147 //
148 BOOST_MOVE_ITERATE_0TO9(BOOST_CONTAINER_SMALL_VECTOR_ALLOCATOR_CTOR_CODE)
149 #undef BOOST_CONTAINER_SMALL_VECTOR_ALLOCATOR_CTOR_CODE
150 #endif
151
152 //!Constructor from other small_vector_allocator.
153 //!Never throws
154 BOOST_CONTAINER_FORCEINLINE small_vector_allocator
155 (const small_vector_allocator &other) BOOST_NOEXCEPT_OR_NOTHROW
156 : Allocator(other.as_base())
157 {}
158
159 //!Move constructor from small_vector_allocator.
160 //!Never throws
161 BOOST_CONTAINER_FORCEINLINE small_vector_allocator
162 (BOOST_RV_REF(small_vector_allocator) other) BOOST_NOEXCEPT_OR_NOTHROW
163 : Allocator(::boost::move(other.as_base()))
164 {}
165
166 //!Constructor from related small_vector_allocator.
167 //!Never throws
168 template<class OtherAllocator>
169 BOOST_CONTAINER_FORCEINLINE small_vector_allocator
170 (const small_vector_allocator<OtherAllocator> &other) BOOST_NOEXCEPT_OR_NOTHROW
171 : Allocator(other.as_base())
172 {}
173
174 //!Move constructor from related small_vector_allocator.
175 //!Never throws
176 template<class OtherAllocator>
177 BOOST_CONTAINER_FORCEINLINE small_vector_allocator
178 (BOOST_RV_REF(small_vector_allocator<OtherAllocator>) other) BOOST_NOEXCEPT_OR_NOTHROW
179 : Allocator(::boost::move(other.as_base()))
180 {}
181
182 //!Assignment from other small_vector_allocator.
183 //!Never throws
184 BOOST_CONTAINER_FORCEINLINE small_vector_allocator &
185 operator=(BOOST_COPY_ASSIGN_REF(small_vector_allocator) other) BOOST_NOEXCEPT_OR_NOTHROW
186 { return static_cast<small_vector_allocator&>(this->Allocator::operator=(other.as_base())); }
187
188 //!Move constructor from other small_vector_allocator.
189 //!Never throws
190 BOOST_CONTAINER_FORCEINLINE small_vector_allocator &
191 operator=(BOOST_RV_REF(small_vector_allocator) other) BOOST_NOEXCEPT_OR_NOTHROW
192 { return static_cast<small_vector_allocator&>(this->Allocator::operator=(::boost::move(other.as_base()))); }
193
194 //!Assignment from related small_vector_allocator.
195 //!Never throws
196 template<class OtherAllocator>
197 BOOST_CONTAINER_FORCEINLINE small_vector_allocator &
198 operator=(BOOST_COPY_ASSIGN_REF(small_vector_allocator<OtherAllocator>) other) BOOST_NOEXCEPT_OR_NOTHROW
199 { return static_cast<small_vector_allocator&>(this->Allocator::operator=(other.as_base())); }
200
201 //!Move assignment from related small_vector_allocator.
202 //!Never throws
203 template<class OtherAllocator>
204 BOOST_CONTAINER_FORCEINLINE small_vector_allocator &
205 operator=(BOOST_RV_REF(small_vector_allocator<OtherAllocator>) other) BOOST_NOEXCEPT_OR_NOTHROW
206 { return static_cast<small_vector_allocator&>(this->Allocator::operator=(::boost::move(other.as_base()))); }
207
208 //!Allocates storage from the standard-conforming allocator
209 BOOST_CONTAINER_FORCEINLINE pointer allocate(size_type count, const_void_pointer hint = const_void_pointer())
210 { return allocator_traits_type::allocate(this->as_base(), count, hint); }
211
212 //!Deallocates previously allocated memory.
213 //!Never throws
214 void deallocate(pointer ptr, size_type n) BOOST_NOEXCEPT_OR_NOTHROW
215 {
216 if(!this->is_internal_storage(ptr))
217 allocator_traits_type::deallocate(this->as_base(), ptr, n);
218 }
219
220 //!Returns the maximum number of elements that could be allocated.
221 //!Never throws
222 BOOST_CONTAINER_FORCEINLINE size_type max_size() const BOOST_NOEXCEPT_OR_NOTHROW
223 { return allocator_traits_type::max_size(this->as_base()); }
224
225 small_vector_allocator select_on_container_copy_construction() const
226 { return small_vector_allocator(allocator_traits_type::select_on_container_copy_construction(this->as_base())); }
227
228 bool storage_is_unpropagable(pointer p) const
229 { return this->is_internal_storage(p) || allocator_traits_type::storage_is_unpropagable(this->as_base(), p); }
230
231 //!Swaps two allocators, does nothing
232 //!because this small_vector_allocator is stateless
233 BOOST_CONTAINER_FORCEINLINE friend void swap(small_vector_allocator &l, small_vector_allocator &r) BOOST_NOEXCEPT_OR_NOTHROW
234 { boost::adl_move_swap(l.as_base(), r.as_base()); }
235
236 //!An small_vector_allocator always compares to true, as memory allocated with one
237 //!instance can be deallocated by another instance (except for unpropagable storage)
238 BOOST_CONTAINER_FORCEINLINE friend bool operator==(const small_vector_allocator &l, const small_vector_allocator &r) BOOST_NOEXCEPT_OR_NOTHROW
239 { return allocator_traits_type::equal(l.as_base(), r.as_base()); }
240
241 //!An small_vector_allocator always compares to false, as memory allocated with one
242 //!instance can be deallocated by another instance
243 BOOST_CONTAINER_FORCEINLINE friend bool operator!=(const small_vector_allocator &l, const small_vector_allocator &r) BOOST_NOEXCEPT_OR_NOTHROW
244 { return !(l == r); }
245
246 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
247 /*
248 //!An advanced function that offers in-place expansion shrink to fit and new allocation
249 //!capabilities. Memory allocated with this function can only be deallocated with deallocate()
250 //!or deallocate_many().
251 //!This function is available only with Version == 2
252 pointer allocation_command(allocation_type command,
253 size_type limit_size,
254 size_type &prefer_in_recvd_out_size,
255 pointer &reuse)
256 { return allocator_traits_type::allocation_command(command, limit_size, prefer_in_recvd_out_size, reuse); }
257
258 //!Returns maximum the number of objects the previously allocated memory
259 //!pointed by p can hold.
260 //!Memory must not have been allocated with
261 //!allocate_one or allocate_individual.
262 //!This function is available only with Version == 2
263 size_type size(pointer p) const BOOST_NOEXCEPT_OR_NOTHROW
264 { return allocator_traits_type::size(p); }
265 */
266 private:
267 /*
268 //!Allocates just one object. Memory allocated with this function
269 //!must be deallocated only with deallocate_one().
270 //!Throws bad_alloc if there is no enough memory
271 //!This function is available only with Version == 2
272 using Allocator::allocate_one;
273 using Allocator::allocate_individual;
274 using Allocator::deallocate_one;
275 using Allocator::deallocate_individual;
276 using Allocator::allocate_many;
277 using Allocator::deallocate_many;*/
278
279 BOOST_CONTAINER_FORCEINLINE bool is_internal_storage(pointer p) const
280 { return this->internal_storage() == p; }
281
282 pointer internal_storage() const
283 {
284 typedef typename Allocator::value_type value_type;
11fdf7f2
TL
285 typedef typename allocator_traits_type::size_type size_type;
286 typedef vector_alloc_holder< small_vector_allocator<Allocator>, size_type > vector_alloc_holder_t;
7c673cae
FG
287 typedef vector<value_type, small_vector_allocator<Allocator> > vector_base;
288 typedef small_vector_base<value_type, Allocator> derived_type;
289 //
290 const vector_alloc_holder_t &v_holder = static_cast<const vector_alloc_holder_t &>(*this);
291 const vector_base &v_base = reinterpret_cast<const vector_base &>(v_holder);
292 const derived_type &d_base = static_cast<const derived_type &>(v_base);
293 return d_base.internal_storage();
294 }
295 #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
296};
297
298//! This class consists of common code from all small_vector<T, N> types that don't depend on the
299//! "N" template parameter. This class is non-copyable and non-destructible, so this class typically
300//! used as reference argument to functions that read or write small vectors. Since `small_vector<T, N>`
301//! derives from `small_vector_base<T>`, the conversion to `small_vector_base` is implicit
302//! <pre>
303//!
304//! //Clients can pass any small_vector<Foo, N>.
305//! void read_any_small_vector_of_foo(const small_vector_base<Foo> &in_parameter);
306//!
307//! void modify_any_small_vector_of_foo(small_vector_base<Foo> &in_out_parameter);
308//!
309//! void some_function()
310//! {
311//!
312//! small_vector<Foo, 8> myvector;
313//!
314//! read_any_small_vector_of_foo(myvector); // Reads myvector
315//!
316//! modify_any_small_vector_of_foo(myvector); // Modifies myvector
317//!
318//! }
319//! </pre>
320//!
321//! All `boost::container:vector` member functions are inherited. See `vector` documentation for details.
322//!
323template <class T, class SecondaryAllocator>
324class small_vector_base
325 : public vector<T, small_vector_allocator<SecondaryAllocator> >
326{
327 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
328 public:
329 //Make it public as it will be inherited by small_vector and container
330 //must have this public member
331 typedef typename allocator_traits<SecondaryAllocator>::pointer pointer;
332
333 private:
334 BOOST_COPYABLE_AND_MOVABLE(small_vector_base)
335
336 friend class small_vector_allocator<SecondaryAllocator>;
337
338 pointer internal_storage() const BOOST_NOEXCEPT_OR_NOTHROW
339 {
340 return boost::intrusive::pointer_traits<pointer>::pointer_to
11fdf7f2 341 (*const_cast<T*>(static_cast<const T*>(static_cast<const void*>(m_storage_start.data))));
7c673cae
FG
342 }
343
344 typedef vector<T, small_vector_allocator<SecondaryAllocator> > base_type;
345 base_type &as_base() { return static_cast<base_type&>(*this); }
346 const base_type &as_base() const { return static_cast<const base_type&>(*this); }
347
348 public:
11fdf7f2
TL
349 typedef typename dtl::aligned_storage
350 <sizeof(T), dtl::alignment_of<T>::value>::type storage_type;
7c673cae
FG
351 typedef small_vector_allocator<SecondaryAllocator> allocator_type;
352
353 protected:
7c673cae
FG
354
355 BOOST_CONTAINER_FORCEINLINE explicit small_vector_base(initial_capacity_t, std::size_t initial_capacity)
356 : base_type(initial_capacity_t(), this->internal_storage(), initial_capacity)
357 {}
358
359 template<class AllocFwd>
360 BOOST_CONTAINER_FORCEINLINE explicit small_vector_base(initial_capacity_t, std::size_t capacity, BOOST_FWD_REF(AllocFwd) a)
361 : base_type(initial_capacity_t(), this->internal_storage(), capacity, ::boost::forward<AllocFwd>(a))
362 {}
363
364 //~small_vector_base(){}
365
366 private:
367 //The only member
368 storage_type m_storage_start;
369
370 #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
371
372 public:
373 BOOST_CONTAINER_FORCEINLINE small_vector_base& operator=(BOOST_COPY_ASSIGN_REF(small_vector_base) other)
374 { return static_cast<small_vector_base&>(this->base_type::operator=(static_cast<base_type const&>(other))); }
375
376 BOOST_CONTAINER_FORCEINLINE small_vector_base& operator=(BOOST_RV_REF(small_vector_base) other)
377 { return static_cast<small_vector_base&>(this->base_type::operator=(BOOST_MOVE_BASE(base_type, other))); }
378
379 BOOST_CONTAINER_FORCEINLINE void swap(small_vector_base &other)
380 { return this->base_type::swap(other); }
381
382 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
383 protected:
384 void move_construct_impl(base_type &x, const allocator_type &a)
385 {
386 if(base_type::is_propagable_from(x.get_stored_allocator(), x.data(), a, true)){
387 this->steal_resources(x);
388 }
389 else{
b32b8144
FG
390 this->assign( boost::make_move_iterator(boost::movelib::iterator_to_raw_pointer(x.begin()))
391 , boost::make_move_iterator(boost::movelib::iterator_to_raw_pointer(x.end ()))
7c673cae
FG
392 );
393 }
394 }
395 #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
396};
397
398#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
399
400/////////////////////////////////////////////////////
401//
402// small_vector_storage_calculator
403//
404/////////////////////////////////////////////////////
405template<std::size_t Needed, std::size_t Hdr, std::size_t SSize, bool NeedsZero = (0u == Needed || Needed <= Hdr)>
406struct small_vector_storage_calculator_helper
407{
408 static const std::size_t value = (Needed - Hdr - 1u)/SSize + 1u;
409};
410
411template<std::size_t Needed, std::size_t Hdr, std::size_t SSize>
412struct small_vector_storage_calculator_helper<Needed, Hdr, SSize, true>
413{
414 static const std::size_t value = 0u;
415};
416
417template<class Storage, class Allocator, class T, std::size_t N>
418struct small_vector_storage_calculator
419{
420 typedef small_vector_base<T, Allocator> svh_type;
421 typedef vector<T, small_vector_allocator<Allocator> > svhb_type;
11fdf7f2 422 static const std::size_t s_align = dtl::alignment_of<Storage>::value;
7c673cae
FG
423 static const std::size_t s_size = sizeof(Storage);
424 static const std::size_t svh_sizeof = sizeof(svh_type);
425 static const std::size_t svhb_sizeof = sizeof(svhb_type);
426 static const std::size_t s_start = ((svhb_sizeof-1)/s_align+1)*s_align;
427 static const std::size_t header_bytes = svh_sizeof-s_start;
428 static const std::size_t needed_bytes = sizeof(T)*N;
429 static const std::size_t needed_extra_storages =
430 small_vector_storage_calculator_helper<needed_bytes, header_bytes, s_size>::value;
431};
432
433/////////////////////////////////////////////////////
434//
435// small_vector_storage_definer
436//
437/////////////////////////////////////////////////////
438template<class Storage, std::size_t N>
439struct small_vector_storage
440{
441 Storage m_rest_of_storage[N];
442};
443
444template<class Storage>
445struct small_vector_storage<Storage, 0>
446{};
447
448template<class Allocator, std::size_t N>
449struct small_vector_storage_definer
450{
451 typedef typename Allocator::value_type value_type;
452 typedef typename small_vector_base<value_type, Allocator>::storage_type storage_type;
453 static const std::size_t needed_extra_storages =
454 small_vector_storage_calculator<storage_type, Allocator, value_type, N>::needed_extra_storages;
455 typedef small_vector_storage<storage_type, needed_extra_storages> type;
456};
457
458#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
459
460//! small_vector is a vector-like container optimized for the case when it contains few elements.
461//! It contains some preallocated elements in-place, which can avoid the use of dynamic storage allocation
462//! when the actual number of elements is below that preallocated threshold.
463//!
464//! `small_vector<T, N, Allocator>` is convertible to `small_vector_base<T, Allocator>` that is independent
465//! from the preallocated element capacity, so client code does not need to be templated on that N argument.
466//!
467//! All `boost::container::vector` member functions are inherited. See `vector` documentation for details.
468//!
469//! \tparam T The type of object that is stored in the small_vector
470//! \tparam N The number of preallocated elements stored inside small_vector. It shall be less than Allocator::max_size();
471//! \tparam Allocator The allocator used for memory management when the number of elements exceeds N.
472template <class T, std::size_t N, class Allocator BOOST_CONTAINER_DOCONLY(= new_allocator<T>) >
473class small_vector : public small_vector_base<T, Allocator>
474 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
475 , private small_vector_storage_definer<Allocator, N>::type
476 #endif
477{
478 #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
479 typedef small_vector_base<T, Allocator> base_type;
480 typedef typename small_vector_storage_definer<Allocator, N>::type remaining_storage_holder;
481
482 BOOST_COPYABLE_AND_MOVABLE(small_vector)
483
7c673cae
FG
484 typedef allocator_traits<typename base_type::allocator_type> allocator_traits_type;
485
486 public:
487 typedef small_vector_storage_calculator< typename small_vector_base<T, Allocator>
488 ::storage_type, Allocator, T, N> storage_test;
489
490 static const std::size_t needed_extra_storages = storage_test::needed_extra_storages;
491 static const std::size_t needed_bytes = storage_test::needed_bytes;
492 static const std::size_t header_bytes = storage_test::header_bytes;
493 static const std::size_t s_start = storage_test::s_start;
494
495 typedef typename base_type::allocator_type allocator_type;
496 typedef typename base_type::size_type size_type;
497 typedef typename base_type::value_type value_type;
498
499 BOOST_CONTAINER_FORCEINLINE static std::size_t internal_capacity()
500 { return (sizeof(small_vector) - storage_test::s_start)/sizeof(T); }
501
502 #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
503
504 //! @brief The capacity/max size of the container
505 static const size_type static_capacity = N;
506
507 public:
508 BOOST_CONTAINER_FORCEINLINE small_vector()
11fdf7f2 509 BOOST_NOEXCEPT_IF(dtl::is_nothrow_default_constructible<Allocator>::value)
7c673cae
FG
510 : base_type(initial_capacity_t(), internal_capacity())
511 {}
512
513 BOOST_CONTAINER_FORCEINLINE explicit small_vector(const allocator_type &a)
514 : base_type(initial_capacity_t(), internal_capacity(), a)
515 {}
516
517 BOOST_CONTAINER_FORCEINLINE explicit small_vector(size_type n)
518 : base_type(initial_capacity_t(), internal_capacity())
519 { this->resize(n); }
520
521 BOOST_CONTAINER_FORCEINLINE small_vector(size_type n, const allocator_type &a)
522 : base_type(initial_capacity_t(), internal_capacity(), a)
523 { this->resize(n); }
524
525 BOOST_CONTAINER_FORCEINLINE small_vector(size_type n, default_init_t)
526 : base_type(initial_capacity_t(), internal_capacity())
527 { this->resize(n, default_init_t()); }
528
529 BOOST_CONTAINER_FORCEINLINE small_vector(size_type n, default_init_t, const allocator_type &a)
530 : base_type(initial_capacity_t(), internal_capacity(), a)
531 { this->resize(n, default_init_t()); }
532
533 BOOST_CONTAINER_FORCEINLINE small_vector(size_type n, const value_type &v)
534 : base_type(initial_capacity_t(), internal_capacity())
535 { this->resize(n, v); }
536
537 BOOST_CONTAINER_FORCEINLINE small_vector(size_type n, const value_type &v, const allocator_type &a)
538 : base_type(initial_capacity_t(), internal_capacity(), a)
539 { this->resize(n, v); }
540
541 template <class InIt>
542 BOOST_CONTAINER_FORCEINLINE small_vector(InIt first, InIt last
11fdf7f2
TL
543 BOOST_CONTAINER_DOCIGN(BOOST_MOVE_I typename dtl::disable_if_c
544 < dtl::is_convertible<InIt BOOST_MOVE_I size_type>::value
545 BOOST_MOVE_I dtl::nat >::type * = 0)
7c673cae
FG
546 )
547 : base_type(initial_capacity_t(), internal_capacity())
548 { this->assign(first, last); }
549
550 template <class InIt>
551 BOOST_CONTAINER_FORCEINLINE small_vector(InIt first, InIt last, const allocator_type& a
11fdf7f2
TL
552 BOOST_CONTAINER_DOCIGN(BOOST_MOVE_I typename dtl::disable_if_c
553 < dtl::is_convertible<InIt BOOST_MOVE_I size_type>::value
554 BOOST_MOVE_I dtl::nat >::type * = 0)
7c673cae
FG
555 )
556 : base_type(initial_capacity_t(), internal_capacity(), a)
557 { this->assign(first, last); }
558
559 BOOST_CONTAINER_FORCEINLINE small_vector(const small_vector &other)
560 : base_type( initial_capacity_t(), internal_capacity()
561 , allocator_traits_type::select_on_container_copy_construction(other.get_stored_allocator()))
562 { this->assign(other.cbegin(), other.cend()); }
563
564 BOOST_CONTAINER_FORCEINLINE small_vector(const small_vector &other, const allocator_type &a)
565 : base_type(initial_capacity_t(), internal_capacity(), a)
566 { this->assign(other.cbegin(), other.cend()); }
567
568 BOOST_CONTAINER_FORCEINLINE explicit small_vector(const base_type &other)
569 : base_type( initial_capacity_t(), internal_capacity()
570 , allocator_traits_type::select_on_container_copy_construction(other.get_stored_allocator()))
571 { this->assign(other.cbegin(), other.cend()); }
572
573 BOOST_CONTAINER_FORCEINLINE explicit small_vector(BOOST_RV_REF(base_type) other)
574 : base_type(initial_capacity_t(), internal_capacity(), ::boost::move(other.get_stored_allocator()))
575 { this->move_construct_impl(other, other.get_stored_allocator()); }
576
577 BOOST_CONTAINER_FORCEINLINE small_vector(BOOST_RV_REF(small_vector) other)
578 : base_type(initial_capacity_t(), internal_capacity(), ::boost::move(other.get_stored_allocator()))
579 { this->move_construct_impl(other, other.get_stored_allocator()); }
580
581 BOOST_CONTAINER_FORCEINLINE small_vector(BOOST_RV_REF(small_vector) other, const allocator_type &a)
582 : base_type(initial_capacity_t(), internal_capacity(), a)
583 { this->move_construct_impl(other, a); }
584
585 #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
586 BOOST_CONTAINER_FORCEINLINE small_vector(std::initializer_list<value_type> il, const allocator_type& a = allocator_type())
587 : base_type(initial_capacity_t(), internal_capacity(), a)
588 {
589 this->assign(il.begin(), il.end());
590 }
591 #endif
592
593 BOOST_CONTAINER_FORCEINLINE small_vector& operator=(BOOST_COPY_ASSIGN_REF(small_vector) other)
594 { return static_cast<small_vector&>(this->base_type::operator=(static_cast<base_type const&>(other))); }
595
596 BOOST_CONTAINER_FORCEINLINE small_vector& operator=(BOOST_RV_REF(small_vector) other)
597 { return static_cast<small_vector&>(this->base_type::operator=(BOOST_MOVE_BASE(base_type, other))); }
598
599 BOOST_CONTAINER_FORCEINLINE small_vector& operator=(const base_type &other)
600 { return static_cast<small_vector&>(this->base_type::operator=(other)); }
601
602 BOOST_CONTAINER_FORCEINLINE small_vector& operator=(BOOST_RV_REF(base_type) other)
603 { return static_cast<small_vector&>(this->base_type::operator=(boost::move(other))); }
604
605 BOOST_CONTAINER_FORCEINLINE void swap(small_vector &other)
606 { return this->base_type::swap(other); }
607};
608
609}}
610
611#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
612/*
613namespace boost {
614
615//!has_trivial_destructor_after_move<> == true_type
616//!specialization for optimizations
617template <class T, class Allocator>
618struct has_trivial_destructor_after_move<boost::container::vector<T, Allocator> >
619{
620 typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
621 static const bool value = ::boost::has_trivial_destructor_after_move<Allocator>::value &&
622 ::boost::has_trivial_destructor_after_move<pointer>::value;
623};
624
625}
626*/
627#endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
628
629#include <boost/container/detail/config_end.hpp>
630
631#endif // #ifndef BOOST_CONTAINER_CONTAINER_SMALL_VECTOR_HPP