small_vector.hpp 27 KB

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