allocate.hpp 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128
  1. // Copyright 2005-2011 Daniel James.
  2. // Copyright 2009 Pablo Halpern.
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. // See http://www.boost.org/libs/unordered for documentation
  6. #ifndef BOOST_UNORDERED_ALLOCATE_HPP
  7. #define BOOST_UNORDERED_ALLOCATE_HPP
  8. #include <boost/config.hpp>
  9. #if defined(BOOST_HAS_PRAGMA_ONCE)
  10. #pragma once
  11. #endif
  12. #include <boost/unordered/detail/fwd.hpp>
  13. #include <boost/move/move.hpp>
  14. #include <boost/preprocessor/cat.hpp>
  15. #include <boost/preprocessor/inc.hpp>
  16. #include <boost/preprocessor/dec.hpp>
  17. #include <boost/preprocessor/repetition/enum.hpp>
  18. #include <boost/preprocessor/repetition/enum_params.hpp>
  19. #include <boost/preprocessor/repetition/enum_binary_params.hpp>
  20. #include <boost/preprocessor/repetition/repeat_from_to.hpp>
  21. #include <boost/type_traits/is_class.hpp>
  22. #include <boost/type_traits/add_lvalue_reference.hpp>
  23. #include <boost/tuple/tuple.hpp>
  24. #include <boost/utility/enable_if.hpp>
  25. #include <boost/utility/addressof.hpp>
  26. #include <boost/detail/select_type.hpp>
  27. #include <boost/assert.hpp>
  28. #include <utility>
  29. #if !defined(BOOST_NO_CXX11_HDR_TUPLE)
  30. #include <tuple>
  31. #endif
  32. #if defined(BOOST_MSVC)
  33. #pragma warning(push)
  34. #pragma warning(disable:4512) // assignment operator could not be generated.
  35. #pragma warning(disable:4345) // behavior change: an object of POD type
  36. // constructed with an initializer of the form ()
  37. // will be default-initialized.
  38. #endif
  39. #define BOOST_UNORDERED_EMPLACE_LIMIT 10
  40. namespace boost { namespace unordered { namespace detail {
  41. ////////////////////////////////////////////////////////////////////////////
  42. // Bits and pieces for implementing traits
  43. template <typename T> typename boost::add_lvalue_reference<T>::type make();
  44. struct choice9 { typedef char (&type)[9]; };
  45. struct choice8 : choice9 { typedef char (&type)[8]; };
  46. struct choice7 : choice8 { typedef char (&type)[7]; };
  47. struct choice6 : choice7 { typedef char (&type)[6]; };
  48. struct choice5 : choice6 { typedef char (&type)[5]; };
  49. struct choice4 : choice5 { typedef char (&type)[4]; };
  50. struct choice3 : choice4 { typedef char (&type)[3]; };
  51. struct choice2 : choice3 { typedef char (&type)[2]; };
  52. struct choice1 : choice2 { typedef char (&type)[1]; };
  53. choice1 choose();
  54. typedef choice1::type yes_type;
  55. typedef choice2::type no_type;
  56. struct private_type
  57. {
  58. private_type const &operator,(int) const;
  59. };
  60. template <typename T>
  61. no_type is_private_type(T const&);
  62. yes_type is_private_type(private_type const&);
  63. struct convert_from_anything {
  64. template <typename T>
  65. convert_from_anything(T const&);
  66. };
  67. ////////////////////////////////////////////////////////////////////////////
  68. // emplace_args
  69. //
  70. // Either forwarding variadic arguments, or storing the arguments in
  71. // emplace_args##n
  72. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  73. #define BOOST_UNORDERED_EMPLACE_TEMPLATE typename... Args
  74. #define BOOST_UNORDERED_EMPLACE_ARGS BOOST_FWD_REF(Args)... args
  75. #define BOOST_UNORDERED_EMPLACE_FORWARD boost::forward<Args>(args)...
  76. #define BOOST_UNORDERED_EMPLACE_ARGS1(a0) a0
  77. #define BOOST_UNORDERED_EMPLACE_ARGS2(a0, a1) a0, a1
  78. #define BOOST_UNORDERED_EMPLACE_ARGS3(a0, a1, a2) a0, a1, a2
  79. #else
  80. #define BOOST_UNORDERED_EMPLACE_TEMPLATE typename Args
  81. #define BOOST_UNORDERED_EMPLACE_ARGS Args const& args
  82. #define BOOST_UNORDERED_EMPLACE_FORWARD args
  83. #define BOOST_UNORDERED_FWD_PARAM(z, n, a) \
  84. BOOST_FWD_REF(BOOST_PP_CAT(A, n)) BOOST_PP_CAT(a, n)
  85. #define BOOST_UNORDERED_CALL_FORWARD(z, i, a) \
  86. boost::forward<BOOST_PP_CAT(A,i)>(BOOST_PP_CAT(a,i))
  87. #define BOOST_UNORDERED_EARGS(z, n, _) \
  88. template <BOOST_PP_ENUM_PARAMS_Z(z, n, typename A)> \
  89. struct BOOST_PP_CAT(emplace_args, n) \
  90. { \
  91. BOOST_PP_REPEAT_##z(n, BOOST_UNORDERED_EARGS_MEMBER, _) \
  92. BOOST_PP_CAT(emplace_args, n) ( \
  93. BOOST_PP_ENUM_BINARY_PARAMS_Z(z, n, Arg, b) \
  94. ) : BOOST_PP_ENUM_##z(n, BOOST_UNORDERED_EARGS_INIT, _) \
  95. {} \
  96. \
  97. }; \
  98. \
  99. template <BOOST_PP_ENUM_PARAMS_Z(z, n, typename A)> \
  100. inline BOOST_PP_CAT(emplace_args, n) < \
  101. BOOST_PP_ENUM_PARAMS_Z(z, n, A) \
  102. > create_emplace_args( \
  103. BOOST_PP_ENUM_##z(n, BOOST_UNORDERED_FWD_PARAM, b) \
  104. ) \
  105. { \
  106. BOOST_PP_CAT(emplace_args, n) < \
  107. BOOST_PP_ENUM_PARAMS_Z(z, n, A) \
  108. > e(BOOST_PP_ENUM_PARAMS_Z(z, n, b)); \
  109. return e; \
  110. }
  111. #define BOOST_UNORDERED_EMPLACE_ARGS1 create_emplace_args
  112. #define BOOST_UNORDERED_EMPLACE_ARGS2 create_emplace_args
  113. #define BOOST_UNORDERED_EMPLACE_ARGS3 create_emplace_args
  114. #if defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  115. #define BOOST_UNORDERED_EARGS_MEMBER(z, n, _) \
  116. typedef BOOST_FWD_REF(BOOST_PP_CAT(A, n)) BOOST_PP_CAT(Arg, n); \
  117. BOOST_PP_CAT(Arg, n) BOOST_PP_CAT(a, n);
  118. #define BOOST_UNORDERED_EARGS_INIT(z, n, _) \
  119. BOOST_PP_CAT(a, n)( \
  120. boost::forward<BOOST_PP_CAT(A,n)>(BOOST_PP_CAT(b, n)))
  121. #else
  122. #define BOOST_UNORDERED_EARGS_MEMBER(z, n, _) \
  123. typedef typename boost::add_lvalue_reference<BOOST_PP_CAT(A, n)>::type \
  124. BOOST_PP_CAT(Arg, n); \
  125. BOOST_PP_CAT(Arg, n) BOOST_PP_CAT(a, n);
  126. #define BOOST_UNORDERED_EARGS_INIT(z, n, _) \
  127. BOOST_PP_CAT(a, n)(BOOST_PP_CAT(b, n))
  128. #endif
  129. BOOST_PP_REPEAT_FROM_TO(1, BOOST_UNORDERED_EMPLACE_LIMIT, BOOST_UNORDERED_EARGS,
  130. _)
  131. #undef BOOST_UNORDERED_DEFINE_EMPLACE_ARGS
  132. #undef BOOST_UNORDERED_EARGS_MEMBER
  133. #undef BOOST_UNORDERED_EARGS_INIT
  134. #endif
  135. }}}
  136. ////////////////////////////////////////////////////////////////////////////////
  137. //
  138. // Pick which version of allocator_traits to use
  139. //
  140. // 0 = Own partial implementation
  141. // 1 = std::allocator_traits
  142. // 2 = boost::container::allocator_traits
  143. #if !defined(BOOST_UNORDERED_USE_ALLOCATOR_TRAITS)
  144. # if defined(__GXX_EXPERIMENTAL_CXX0X__) && \
  145. (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 7))
  146. # define BOOST_UNORDERED_USE_ALLOCATOR_TRAITS 0
  147. # elif defined(BOOST_MSVC)
  148. # if BOOST_MSVC < 1400
  149. // Use container's allocator_traits for older versions of Visual
  150. // C++ as I don't test with them.
  151. # define BOOST_UNORDERED_USE_ALLOCATOR_TRAITS 2
  152. # endif
  153. # endif
  154. #endif
  155. #if !defined(BOOST_UNORDERED_USE_ALLOCATOR_TRAITS)
  156. # define BOOST_UNORDERED_USE_ALLOCATOR_TRAITS 0
  157. #endif
  158. ////////////////////////////////////////////////////////////////////////////////
  159. //
  160. // Some utilities for implementing allocator_traits, but useful elsewhere so
  161. // they're always defined.
  162. #if !defined(BOOST_NO_CXX11_HDR_TYPE_TRAITS)
  163. # include <type_traits>
  164. #endif
  165. namespace boost { namespace unordered { namespace detail {
  166. ////////////////////////////////////////////////////////////////////////////
  167. // Integral_constrant, true_type, false_type
  168. //
  169. // Uses the standard versions if available.
  170. #if !defined(BOOST_NO_CXX11_HDR_TYPE_TRAITS)
  171. using std::integral_constant;
  172. using std::true_type;
  173. using std::false_type;
  174. #else
  175. template <typename T, T Value>
  176. struct integral_constant { enum { value = Value }; };
  177. typedef boost::unordered::detail::integral_constant<bool, true> true_type;
  178. typedef boost::unordered::detail::integral_constant<bool, false> false_type;
  179. #endif
  180. ////////////////////////////////////////////////////////////////////////////
  181. // Explicitly call a destructor
  182. #if defined(BOOST_MSVC)
  183. #pragma warning(push)
  184. #pragma warning(disable:4100) // unreferenced formal parameter
  185. #endif
  186. namespace func {
  187. template <class T>
  188. inline void destroy(T* x) {
  189. x->~T();
  190. }
  191. }
  192. #if defined(BOOST_MSVC)
  193. #pragma warning(pop)
  194. #endif
  195. ////////////////////////////////////////////////////////////////////////////
  196. // Expression test mechanism
  197. //
  198. // When SFINAE expressions are available, define
  199. // BOOST_UNORDERED_HAS_FUNCTION which can check if a function call is
  200. // supported by a class, otherwise define BOOST_UNORDERED_HAS_MEMBER which
  201. // can detect if a class has the specified member, but not that it has the
  202. // correct type, this is good enough for a passable impression of
  203. // allocator_traits.
  204. #if !defined(BOOST_NO_SFINAE_EXPR)
  205. template <typename T, unsigned int> struct expr_test;
  206. template <typename T> struct expr_test<T, sizeof(char)> : T {};
  207. # define BOOST_UNORDERED_CHECK_EXPRESSION(count, result, expression) \
  208. template <typename U> \
  209. static typename boost::unordered::detail::expr_test< \
  210. BOOST_PP_CAT(choice, result), \
  211. sizeof(for_expr_test(( \
  212. (expression), \
  213. 0)))>::type test( \
  214. BOOST_PP_CAT(choice, count))
  215. # define BOOST_UNORDERED_DEFAULT_EXPRESSION(count, result) \
  216. template <typename U> \
  217. static BOOST_PP_CAT(choice, result)::type test( \
  218. BOOST_PP_CAT(choice, count))
  219. # define BOOST_UNORDERED_HAS_FUNCTION(name, thing, args, _) \
  220. struct BOOST_PP_CAT(has_, name) \
  221. { \
  222. template <typename U> static char for_expr_test(U const&); \
  223. BOOST_UNORDERED_CHECK_EXPRESSION(1, 1, \
  224. boost::unordered::detail::make< thing >().name args); \
  225. BOOST_UNORDERED_DEFAULT_EXPRESSION(2, 2); \
  226. \
  227. enum { value = sizeof(test<T>(choose())) == sizeof(choice1::type) };\
  228. }
  229. #else
  230. template <typename T> struct identity { typedef T type; };
  231. # define BOOST_UNORDERED_CHECK_MEMBER(count, result, name, member) \
  232. \
  233. typedef typename boost::unordered::detail::identity<member>::type \
  234. BOOST_PP_CAT(check, count); \
  235. \
  236. template <BOOST_PP_CAT(check, count) e> \
  237. struct BOOST_PP_CAT(test, count) { \
  238. typedef BOOST_PP_CAT(choice, result) type; \
  239. }; \
  240. \
  241. template <class U> static typename \
  242. BOOST_PP_CAT(test, count)<&U::name>::type \
  243. test(BOOST_PP_CAT(choice, count))
  244. # define BOOST_UNORDERED_DEFAULT_MEMBER(count, result) \
  245. template <class U> static BOOST_PP_CAT(choice, result)::type \
  246. test(BOOST_PP_CAT(choice, count))
  247. # define BOOST_UNORDERED_HAS_MEMBER(name) \
  248. struct BOOST_PP_CAT(has_, name) \
  249. { \
  250. struct impl { \
  251. struct base_mixin { int name; }; \
  252. struct base : public T, public base_mixin {}; \
  253. \
  254. BOOST_UNORDERED_CHECK_MEMBER(1, 1, name, int base_mixin::*); \
  255. BOOST_UNORDERED_DEFAULT_MEMBER(2, 2); \
  256. \
  257. enum { value = sizeof(choice2::type) == \
  258. sizeof(test<base>(choose())) \
  259. }; \
  260. }; \
  261. \
  262. enum { value = impl::value }; \
  263. }
  264. #endif
  265. }}}
  266. ////////////////////////////////////////////////////////////////////////////////
  267. //
  268. // Allocator traits
  269. //
  270. // First our implementation, then later light wrappers around the alternatives
  271. #if BOOST_UNORDERED_USE_ALLOCATOR_TRAITS == 0
  272. # include <boost/limits.hpp>
  273. # include <boost/utility/enable_if.hpp>
  274. # include <boost/pointer_to_other.hpp>
  275. # if defined(BOOST_NO_SFINAE_EXPR)
  276. # include <boost/type_traits/is_same.hpp>
  277. # endif
  278. # if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && \
  279. !defined(BOOST_NO_SFINAE_EXPR)
  280. # define BOOST_UNORDERED_DETAIL_FULL_CONSTRUCT 1
  281. # else
  282. # define BOOST_UNORDERED_DETAIL_FULL_CONSTRUCT 0
  283. # endif
  284. namespace boost { namespace unordered { namespace detail {
  285. // TODO: Does this match std::allocator_traits<Alloc>::rebind_alloc<T>?
  286. template <typename Alloc, typename T>
  287. struct rebind_wrap
  288. {
  289. typedef typename Alloc::BOOST_NESTED_TEMPLATE rebind<T>::other type;
  290. };
  291. # if defined(BOOST_MSVC) && BOOST_MSVC <= 1400
  292. # define BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(tname) \
  293. template <typename Tp, typename Default> \
  294. struct default_type_ ## tname { \
  295. \
  296. template <typename X> \
  297. static choice1::type test(choice1, typename X::tname* = 0); \
  298. \
  299. template <typename X> \
  300. static choice2::type test(choice2, void* = 0); \
  301. \
  302. struct DefaultWrap { typedef Default tname; }; \
  303. \
  304. enum { value = (1 == sizeof(test<Tp>(choose()))) }; \
  305. \
  306. typedef typename boost::detail::if_true<value>:: \
  307. BOOST_NESTED_TEMPLATE then<Tp, DefaultWrap> \
  308. ::type::tname type; \
  309. }
  310. # else
  311. template <typename T, typename T2>
  312. struct sfinae : T2 {};
  313. # define BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(tname) \
  314. template <typename Tp, typename Default> \
  315. struct default_type_ ## tname { \
  316. \
  317. template <typename X> \
  318. static typename boost::unordered::detail::sfinae< \
  319. typename X::tname, choice1>::type \
  320. test(choice1); \
  321. \
  322. template <typename X> \
  323. static choice2::type test(choice2); \
  324. \
  325. struct DefaultWrap { typedef Default tname; }; \
  326. \
  327. enum { value = (1 == sizeof(test<Tp>(choose()))) }; \
  328. \
  329. typedef typename boost::detail::if_true<value>:: \
  330. BOOST_NESTED_TEMPLATE then<Tp, DefaultWrap> \
  331. ::type::tname type; \
  332. }
  333. # endif
  334. # define BOOST_UNORDERED_DEFAULT_TYPE(T,tname, arg) \
  335. typename default_type_ ## tname<T, arg>::type
  336. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(pointer);
  337. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(const_pointer);
  338. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(void_pointer);
  339. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(const_void_pointer);
  340. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(difference_type);
  341. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(size_type);
  342. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(propagate_on_container_copy_assignment);
  343. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(propagate_on_container_move_assignment);
  344. BOOST_UNORDERED_DEFAULT_TYPE_TMPLT(propagate_on_container_swap);
  345. # if !defined(BOOST_NO_SFINAE_EXPR)
  346. template <typename T>
  347. BOOST_UNORDERED_HAS_FUNCTION(
  348. select_on_container_copy_construction, U const, (), 0
  349. );
  350. template <typename T>
  351. BOOST_UNORDERED_HAS_FUNCTION(
  352. max_size, U const, (), 0
  353. );
  354. # if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  355. template <typename T, typename ValueType, typename... Args>
  356. BOOST_UNORDERED_HAS_FUNCTION(
  357. construct, U, (
  358. boost::unordered::detail::make<ValueType*>(),
  359. boost::unordered::detail::make<Args const>()...), 2
  360. );
  361. # else
  362. template <typename T, typename ValueType>
  363. BOOST_UNORDERED_HAS_FUNCTION(
  364. construct, U, (
  365. boost::unordered::detail::make<ValueType*>(),
  366. boost::unordered::detail::make<ValueType const>()), 2
  367. );
  368. # endif
  369. template <typename T, typename ValueType>
  370. BOOST_UNORDERED_HAS_FUNCTION(
  371. destroy, U, (boost::unordered::detail::make<ValueType*>()), 1
  372. );
  373. # else
  374. template <typename T>
  375. BOOST_UNORDERED_HAS_MEMBER(select_on_container_copy_construction);
  376. template <typename T>
  377. BOOST_UNORDERED_HAS_MEMBER(max_size);
  378. template <typename T, typename ValueType>
  379. BOOST_UNORDERED_HAS_MEMBER(construct);
  380. template <typename T, typename ValueType>
  381. BOOST_UNORDERED_HAS_MEMBER(destroy);
  382. # endif
  383. namespace func
  384. {
  385. template <typename Alloc>
  386. inline Alloc call_select_on_container_copy_construction(const Alloc& rhs,
  387. typename boost::enable_if_c<
  388. boost::unordered::detail::
  389. has_select_on_container_copy_construction<Alloc>::value, void*
  390. >::type = 0)
  391. {
  392. return rhs.select_on_container_copy_construction();
  393. }
  394. template <typename Alloc>
  395. inline Alloc call_select_on_container_copy_construction(const Alloc& rhs,
  396. typename boost::disable_if_c<
  397. boost::unordered::detail::
  398. has_select_on_container_copy_construction<Alloc>::value, void*
  399. >::type = 0)
  400. {
  401. return rhs;
  402. }
  403. template <typename SizeType, typename Alloc>
  404. inline SizeType call_max_size(const Alloc& a,
  405. typename boost::enable_if_c<
  406. boost::unordered::detail::has_max_size<Alloc>::value, void*
  407. >::type = 0)
  408. {
  409. return a.max_size();
  410. }
  411. template <typename SizeType, typename Alloc>
  412. inline SizeType call_max_size(const Alloc&, typename boost::disable_if_c<
  413. boost::unordered::detail::has_max_size<Alloc>::value, void*
  414. >::type = 0)
  415. {
  416. return (std::numeric_limits<SizeType>::max)();
  417. }
  418. } // namespace func.
  419. template <typename Alloc>
  420. struct allocator_traits
  421. {
  422. typedef Alloc allocator_type;
  423. typedef typename Alloc::value_type value_type;
  424. typedef BOOST_UNORDERED_DEFAULT_TYPE(Alloc, pointer, value_type*)
  425. pointer;
  426. template <typename T>
  427. struct pointer_to_other : boost::pointer_to_other<pointer, T> {};
  428. typedef BOOST_UNORDERED_DEFAULT_TYPE(Alloc, const_pointer,
  429. typename pointer_to_other<const value_type>::type)
  430. const_pointer;
  431. //typedef BOOST_UNORDERED_DEFAULT_TYPE(Alloc, void_pointer,
  432. // typename pointer_to_other<void>::type)
  433. // void_pointer;
  434. //
  435. //typedef BOOST_UNORDERED_DEFAULT_TYPE(Alloc, const_void_pointer,
  436. // typename pointer_to_other<const void>::type)
  437. // const_void_pointer;
  438. typedef BOOST_UNORDERED_DEFAULT_TYPE(Alloc, difference_type,
  439. std::ptrdiff_t) difference_type;
  440. typedef BOOST_UNORDERED_DEFAULT_TYPE(Alloc, size_type, std::size_t)
  441. size_type;
  442. // TODO: rebind_alloc and rebind_traits
  443. static pointer allocate(Alloc& a, size_type n)
  444. { return a.allocate(n); }
  445. // I never use this, so I'll just comment it out for now.
  446. //
  447. //static pointer allocate(Alloc& a, size_type n,
  448. // const_void_pointer hint)
  449. // { return DEFAULT_FUNC(allocate, pointer)(a, n, hint); }
  450. static void deallocate(Alloc& a, pointer p, size_type n)
  451. { a.deallocate(p, n); }
  452. public:
  453. # if BOOST_UNORDERED_DETAIL_FULL_CONSTRUCT
  454. template <typename T, typename... Args>
  455. static typename boost::enable_if_c<
  456. boost::unordered::detail::has_construct<Alloc, T, Args...>
  457. ::value>::type
  458. construct(Alloc& a, T* p, BOOST_FWD_REF(Args)... x)
  459. {
  460. a.construct(p, boost::forward<Args>(x)...);
  461. }
  462. template <typename T, typename... Args>
  463. static typename boost::disable_if_c<
  464. boost::unordered::detail::has_construct<Alloc, T, Args...>
  465. ::value>::type
  466. construct(Alloc&, T* p, BOOST_FWD_REF(Args)... x)
  467. {
  468. new ((void*) p) T(boost::forward<Args>(x)...);
  469. }
  470. template <typename T>
  471. static typename boost::enable_if_c<
  472. boost::unordered::detail::has_destroy<Alloc, T>::value>::type
  473. destroy(Alloc& a, T* p)
  474. {
  475. a.destroy(p);
  476. }
  477. template <typename T>
  478. static typename boost::disable_if_c<
  479. boost::unordered::detail::has_destroy<Alloc, T>::value>::type
  480. destroy(Alloc&, T* p)
  481. {
  482. boost::unordered::detail::func::destroy(p);
  483. }
  484. # elif !defined(BOOST_NO_SFINAE_EXPR)
  485. template <typename T>
  486. static typename boost::enable_if_c<
  487. boost::unordered::detail::has_construct<Alloc, T>::value>::type
  488. construct(Alloc& a, T* p, T const& x)
  489. {
  490. a.construct(p, x);
  491. }
  492. template <typename T>
  493. static typename boost::disable_if_c<
  494. boost::unordered::detail::has_construct<Alloc, T>::value>::type
  495. construct(Alloc&, T* p, T const& x)
  496. {
  497. new ((void*) p) T(x);
  498. }
  499. template <typename T>
  500. static typename boost::enable_if_c<
  501. boost::unordered::detail::has_destroy<Alloc, T>::value>::type
  502. destroy(Alloc& a, T* p)
  503. {
  504. a.destroy(p);
  505. }
  506. template <typename T>
  507. static typename boost::disable_if_c<
  508. boost::unordered::detail::has_destroy<Alloc, T>::value>::type
  509. destroy(Alloc&, T* p)
  510. {
  511. boost::unordered::detail::func::destroy(p);
  512. }
  513. # else
  514. // If we don't have SFINAE expressions, only call construct for the
  515. // copy constructor for the allocator's value_type - as that's
  516. // the only construct method that old fashioned allocators support.
  517. template <typename T>
  518. static void construct(Alloc& a, T* p, T const& x,
  519. typename boost::enable_if_c<
  520. boost::unordered::detail::has_construct<Alloc, T>::value &&
  521. boost::is_same<T, value_type>::value,
  522. void*>::type = 0)
  523. {
  524. a.construct(p, x);
  525. }
  526. template <typename T>
  527. static void construct(Alloc&, T* p, T const& x,
  528. typename boost::disable_if_c<
  529. boost::unordered::detail::has_construct<Alloc, T>::value &&
  530. boost::is_same<T, value_type>::value,
  531. void*>::type = 0)
  532. {
  533. new ((void*) p) T(x);
  534. }
  535. template <typename T>
  536. static void destroy(Alloc& a, T* p,
  537. typename boost::enable_if_c<
  538. boost::unordered::detail::has_destroy<Alloc, T>::value &&
  539. boost::is_same<T, value_type>::value,
  540. void*>::type = 0)
  541. {
  542. a.destroy(p);
  543. }
  544. template <typename T>
  545. static void destroy(Alloc&, T* p,
  546. typename boost::disable_if_c<
  547. boost::unordered::detail::has_destroy<Alloc, T>::value &&
  548. boost::is_same<T, value_type>::value,
  549. void*>::type = 0)
  550. {
  551. boost::unordered::detail::func::destroy(p);
  552. }
  553. # endif
  554. static size_type max_size(const Alloc& a)
  555. {
  556. return boost::unordered::detail::func::
  557. call_max_size<size_type>(a);
  558. }
  559. // Allocator propagation on construction
  560. static Alloc select_on_container_copy_construction(Alloc const& rhs)
  561. {
  562. return boost::unordered::detail::func::
  563. call_select_on_container_copy_construction(rhs);
  564. }
  565. // Allocator propagation on assignment and swap.
  566. // Return true if lhs is modified.
  567. typedef BOOST_UNORDERED_DEFAULT_TYPE(
  568. Alloc, propagate_on_container_copy_assignment, false_type)
  569. propagate_on_container_copy_assignment;
  570. typedef BOOST_UNORDERED_DEFAULT_TYPE(
  571. Alloc,propagate_on_container_move_assignment, false_type)
  572. propagate_on_container_move_assignment;
  573. typedef BOOST_UNORDERED_DEFAULT_TYPE(
  574. Alloc,propagate_on_container_swap,false_type)
  575. propagate_on_container_swap;
  576. };
  577. }}}
  578. # undef BOOST_UNORDERED_DEFAULT_TYPE_TMPLT
  579. # undef BOOST_UNORDERED_DEFAULT_TYPE
  580. ////////////////////////////////////////////////////////////////////////////////
  581. //
  582. // std::allocator_traits
  583. #elif BOOST_UNORDERED_USE_ALLOCATOR_TRAITS == 1
  584. # include <memory>
  585. # define BOOST_UNORDERED_DETAIL_FULL_CONSTRUCT 1
  586. namespace boost { namespace unordered { namespace detail {
  587. template <typename Alloc>
  588. struct allocator_traits : std::allocator_traits<Alloc> {};
  589. template <typename Alloc, typename T>
  590. struct rebind_wrap
  591. {
  592. typedef typename std::allocator_traits<Alloc>::
  593. template rebind_alloc<T> type;
  594. };
  595. }}}
  596. ////////////////////////////////////////////////////////////////////////////////
  597. //
  598. // boost::container::allocator_traits
  599. #elif BOOST_UNORDERED_USE_ALLOCATOR_TRAITS == 2
  600. # include <boost/container/allocator_traits.hpp>
  601. # define BOOST_UNORDERED_DETAIL_FULL_CONSTRUCT 0
  602. namespace boost { namespace unordered { namespace detail {
  603. template <typename Alloc>
  604. struct allocator_traits :
  605. boost::container::allocator_traits<Alloc> {};
  606. template <typename Alloc, typename T>
  607. struct rebind_wrap :
  608. boost::container::allocator_traits<Alloc>::
  609. template portable_rebind_alloc<T>
  610. {};
  611. }}}
  612. #else
  613. #error "Invalid BOOST_UNORDERED_USE_ALLOCATOR_TRAITS value."
  614. #endif
  615. namespace boost { namespace unordered { namespace detail { namespace func {
  616. ////////////////////////////////////////////////////////////////////////////
  617. // call_construct
  618. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  619. # if BOOST_UNORDERED_DETAIL_FULL_CONSTRUCT
  620. template <typename Alloc, typename T, typename... Args>
  621. inline void call_construct(Alloc& alloc, T* address,
  622. BOOST_FWD_REF(Args)... args)
  623. {
  624. boost::unordered::detail::allocator_traits<Alloc>::construct(alloc,
  625. address, boost::forward<Args>(args)...);
  626. }
  627. template <typename Alloc, typename T>
  628. inline void destroy_value_impl(Alloc& alloc, T* x) {
  629. boost::unordered::detail::allocator_traits<Alloc>::destroy(alloc, x);
  630. }
  631. # else
  632. template <typename Alloc, typename T, typename... Args>
  633. inline void call_construct(Alloc&, T* address,
  634. BOOST_FWD_REF(Args)... args)
  635. {
  636. new((void*) address) T(boost::forward<Args>(args)...);
  637. }
  638. template <typename Alloc, typename T>
  639. inline void destroy_value_impl(Alloc&, T* x) {
  640. boost::unordered::detail::func::destroy(x);
  641. }
  642. # endif
  643. #else
  644. template <typename Alloc, typename T>
  645. inline void destroy_value_impl(Alloc&, T* x) {
  646. boost::unordered::detail::func::destroy(x);
  647. }
  648. #endif
  649. ////////////////////////////////////////////////////////////////////////////
  650. // Construct from tuple
  651. //
  652. // Used for piecewise construction.
  653. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  654. # define BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE(n, namespace_) \
  655. template<typename Alloc, typename T> \
  656. void construct_from_tuple(Alloc& alloc, T* ptr, namespace_ tuple<>) \
  657. { \
  658. boost::unordered::detail::func::call_construct(alloc, ptr); \
  659. } \
  660. \
  661. BOOST_PP_REPEAT_FROM_TO(1, n, \
  662. BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL, namespace_)
  663. # define BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL(z, n, namespace_) \
  664. template<typename Alloc, typename T, \
  665. BOOST_PP_ENUM_PARAMS_Z(z, n, typename A)> \
  666. void construct_from_tuple(Alloc& alloc, T* ptr, \
  667. namespace_ tuple<BOOST_PP_ENUM_PARAMS_Z(z, n, A)> const& x) \
  668. { \
  669. boost::unordered::detail::func::call_construct(alloc, ptr, \
  670. BOOST_PP_ENUM_##z(n, BOOST_UNORDERED_GET_TUPLE_ARG, namespace_) \
  671. ); \
  672. }
  673. # define BOOST_UNORDERED_GET_TUPLE_ARG(z, n, namespace_) \
  674. namespace_ get<n>(x)
  675. #elif !defined(__SUNPRO_CC)
  676. # define BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE(n, namespace_) \
  677. template<typename Alloc, typename T> \
  678. void construct_from_tuple(Alloc&, T* ptr, namespace_ tuple<>) \
  679. { \
  680. new ((void*) ptr) T(); \
  681. } \
  682. \
  683. BOOST_PP_REPEAT_FROM_TO(1, n, \
  684. BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL, namespace_)
  685. # define BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL(z, n, namespace_) \
  686. template<typename Alloc, typename T, \
  687. BOOST_PP_ENUM_PARAMS_Z(z, n, typename A)> \
  688. void construct_from_tuple(Alloc&, T* ptr, \
  689. namespace_ tuple<BOOST_PP_ENUM_PARAMS_Z(z, n, A)> const& x) \
  690. { \
  691. new ((void*) ptr) T( \
  692. BOOST_PP_ENUM_##z(n, BOOST_UNORDERED_GET_TUPLE_ARG, namespace_) \
  693. ); \
  694. }
  695. # define BOOST_UNORDERED_GET_TUPLE_ARG(z, n, namespace_) \
  696. namespace_ get<n>(x)
  697. #else
  698. template <int N> struct length {};
  699. # define BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE(n, namespace_) \
  700. template<typename Alloc, typename T> \
  701. void construct_from_tuple_impl( \
  702. boost::unordered::detail::func::length<0>, Alloc&, T* ptr, \
  703. namespace_ tuple<>) \
  704. { \
  705. new ((void*) ptr) T(); \
  706. } \
  707. \
  708. BOOST_PP_REPEAT_FROM_TO(1, n, \
  709. BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL, namespace_)
  710. # define BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL(z, n, namespace_) \
  711. template<typename Alloc, typename T, \
  712. BOOST_PP_ENUM_PARAMS_Z(z, n, typename A)> \
  713. void construct_from_tuple_impl( \
  714. boost::unordered::detail::func::length<n>, Alloc&, T* ptr, \
  715. namespace_ tuple<BOOST_PP_ENUM_PARAMS_Z(z, n, A)> const& x) \
  716. { \
  717. new ((void*) ptr) T( \
  718. BOOST_PP_ENUM_##z(n, BOOST_UNORDERED_GET_TUPLE_ARG, namespace_) \
  719. ); \
  720. }
  721. # define BOOST_UNORDERED_GET_TUPLE_ARG(z, n, namespace_) \
  722. namespace_ get<n>(x)
  723. #endif
  724. BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE(10, boost::)
  725. #if !defined(__SUNPRO_CC) && !defined(BOOST_NO_CXX11_HDR_TUPLE)
  726. BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE(10, std::)
  727. #endif
  728. #undef BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE
  729. #undef BOOST_UNORDERED_CONSTRUCT_FROM_TUPLE_IMPL
  730. #undef BOOST_UNORDERED_GET_TUPLE_ARG
  731. #if defined(__SUNPRO_CC)
  732. template <typename Alloc, typename T, typename Tuple>
  733. void construct_from_tuple(Alloc& alloc, T* ptr, Tuple const& x)
  734. {
  735. construct_from_tuple_impl(
  736. boost::unordered::detail::func::length<
  737. boost::tuples::length<Tuple>::value>(),
  738. alloc, ptr, x);
  739. }
  740. #endif
  741. ////////////////////////////////////////////////////////////////////////////
  742. // Trait to check for piecewise construction.
  743. template <typename A0>
  744. struct use_piecewise {
  745. static choice1::type test(choice1,
  746. boost::unordered::piecewise_construct_t);
  747. static choice2::type test(choice2, ...);
  748. enum { value = sizeof(choice1::type) ==
  749. sizeof(test(choose(), boost::unordered::detail::make<A0>())) };
  750. };
  751. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  752. ////////////////////////////////////////////////////////////////////////////
  753. // Construct from variadic parameters
  754. // For the standard pair constructor.
  755. template <typename Alloc, typename T, typename... Args>
  756. inline void construct_value_impl(Alloc& alloc, T* address,
  757. BOOST_FWD_REF(Args)... args)
  758. {
  759. boost::unordered::detail::func::call_construct(alloc,
  760. address, boost::forward<Args>(args)...);
  761. }
  762. // Special case for piece_construct
  763. //
  764. // TODO: When possible, it might be better to use std::pair's
  765. // constructor for std::piece_construct with std::tuple.
  766. template <typename Alloc, typename A, typename B,
  767. typename A0, typename A1, typename A2>
  768. inline typename enable_if<use_piecewise<A0>, void>::type
  769. construct_value_impl(Alloc& alloc, std::pair<A, B>* address,
  770. BOOST_FWD_REF(A0), BOOST_FWD_REF(A1) a1, BOOST_FWD_REF(A2) a2)
  771. {
  772. boost::unordered::detail::func::construct_from_tuple(alloc,
  773. boost::addressof(address->first), boost::forward<A1>(a1));
  774. boost::unordered::detail::func::construct_from_tuple(alloc,
  775. boost::addressof(address->second), boost::forward<A2>(a2));
  776. }
  777. #else // BOOST_NO_CXX11_VARIADIC_TEMPLATES
  778. ////////////////////////////////////////////////////////////////////////////////
  779. // Construct from emplace_args
  780. // Explicitly write out first three overloads for the sake of sane
  781. // error messages.
  782. template <typename Alloc, typename T, typename A0>
  783. inline void construct_value_impl(Alloc&, T* address,
  784. emplace_args1<A0> const& args)
  785. {
  786. new((void*) address) T(boost::forward<A0>(args.a0));
  787. }
  788. template <typename Alloc, typename T, typename A0, typename A1>
  789. inline void construct_value_impl(Alloc&, T* address,
  790. emplace_args2<A0, A1> const& args)
  791. {
  792. new((void*) address) T(
  793. boost::forward<A0>(args.a0),
  794. boost::forward<A1>(args.a1)
  795. );
  796. }
  797. template <typename Alloc, typename T, typename A0, typename A1, typename A2>
  798. inline void construct_value_impl(Alloc&, T* address,
  799. emplace_args3<A0, A1, A2> const& args)
  800. {
  801. new((void*) address) T(
  802. boost::forward<A0>(args.a0),
  803. boost::forward<A1>(args.a1),
  804. boost::forward<A2>(args.a2)
  805. );
  806. }
  807. // Use a macro for the rest.
  808. #define BOOST_UNORDERED_CONSTRUCT_IMPL(z, num_params, _) \
  809. template < \
  810. typename Alloc, typename T, \
  811. BOOST_PP_ENUM_PARAMS_Z(z, num_params, typename A) \
  812. > \
  813. inline void construct_value_impl(Alloc&, T* address, \
  814. boost::unordered::detail::BOOST_PP_CAT(emplace_args,num_params) < \
  815. BOOST_PP_ENUM_PARAMS_Z(z, num_params, A) \
  816. > const& args) \
  817. { \
  818. new((void*) address) T( \
  819. BOOST_PP_ENUM_##z(num_params, BOOST_UNORDERED_CALL_FORWARD, \
  820. args.a)); \
  821. }
  822. BOOST_PP_REPEAT_FROM_TO(4, BOOST_UNORDERED_EMPLACE_LIMIT,
  823. BOOST_UNORDERED_CONSTRUCT_IMPL, _)
  824. #undef BOOST_UNORDERED_CONSTRUCT_IMPL
  825. // Construct with piece_construct
  826. template <typename Alloc, typename A, typename B,
  827. typename A0, typename A1, typename A2>
  828. inline void construct_value_impl(Alloc& alloc, std::pair<A, B>* address,
  829. boost::unordered::detail::emplace_args3<A0, A1, A2> const& args,
  830. typename enable_if<use_piecewise<A0>, void*>::type = 0)
  831. {
  832. boost::unordered::detail::func::construct_from_tuple(alloc,
  833. boost::addressof(address->first), args.a1);
  834. boost::unordered::detail::func::construct_from_tuple(alloc,
  835. boost::addressof(address->second), args.a2);
  836. }
  837. #endif // BOOST_NO_CXX11_VARIADIC_TEMPLATES
  838. }}}}
  839. namespace boost { namespace unordered { namespace detail {
  840. ////////////////////////////////////////////////////////////////////////////
  841. //
  842. // array_constructor
  843. //
  844. // Allocate and construct an array in an exception safe manner, and
  845. // clean up if an exception is thrown before the container takes charge
  846. // of it.
  847. template <typename Allocator>
  848. struct array_constructor
  849. {
  850. typedef boost::unordered::detail::allocator_traits<Allocator> traits;
  851. typedef typename traits::pointer pointer;
  852. Allocator& alloc_;
  853. pointer ptr_;
  854. pointer constructed_;
  855. std::size_t length_;
  856. array_constructor(Allocator& a)
  857. : alloc_(a), ptr_(), constructed_(), length_(0)
  858. {
  859. constructed_ = pointer();
  860. ptr_ = pointer();
  861. }
  862. ~array_constructor() {
  863. if (ptr_) {
  864. for(pointer p = ptr_; p != constructed_; ++p) {
  865. boost::unordered::detail::func::destroy(
  866. boost::addressof(*p));
  867. }
  868. traits::deallocate(alloc_, ptr_, length_);
  869. }
  870. }
  871. template <typename V>
  872. void construct(V const& v, std::size_t l)
  873. {
  874. BOOST_ASSERT(!ptr_);
  875. length_ = l;
  876. ptr_ = traits::allocate(alloc_, length_);
  877. pointer end = ptr_ + static_cast<std::ptrdiff_t>(length_);
  878. for(constructed_ = ptr_; constructed_ != end; ++constructed_) {
  879. new ((void*) boost::addressof(*constructed_)) V(v);
  880. }
  881. }
  882. pointer get() const
  883. {
  884. return ptr_;
  885. }
  886. pointer release()
  887. {
  888. pointer p(ptr_);
  889. ptr_ = pointer();
  890. return p;
  891. }
  892. private:
  893. array_constructor(array_constructor const&);
  894. array_constructor& operator=(array_constructor const&);
  895. };
  896. }}}
  897. #if defined(BOOST_MSVC)
  898. #pragma warning(pop)
  899. #endif
  900. #endif