hashed_index.hpp 50 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725
  1. /* Copyright 2003-2015 Joaquin M Lopez Munoz.
  2. * Distributed under the Boost Software License, Version 1.0.
  3. * (See accompanying file LICENSE_1_0.txt or copy at
  4. * http://www.boost.org/LICENSE_1_0.txt)
  5. *
  6. * See http://www.boost.org/libs/multi_index for library home page.
  7. */
  8. #ifndef BOOST_MULTI_INDEX_HASHED_INDEX_HPP
  9. #define BOOST_MULTI_INDEX_HASHED_INDEX_HPP
  10. #if defined(_MSC_VER)
  11. #pragma once
  12. #endif
  13. #include <boost/config.hpp> /* keep it first to prevent nasty warns in MSVC */
  14. #include <algorithm>
  15. #include <boost/call_traits.hpp>
  16. #include <boost/detail/allocator_utilities.hpp>
  17. #include <boost/detail/no_exceptions_support.hpp>
  18. #include <boost/detail/workaround.hpp>
  19. #include <boost/foreach_fwd.hpp>
  20. #include <boost/limits.hpp>
  21. #include <boost/move/core.hpp>
  22. #include <boost/mpl/bool.hpp>
  23. #include <boost/mpl/if.hpp>
  24. #include <boost/mpl/push_front.hpp>
  25. #include <boost/multi_index/detail/access_specifier.hpp>
  26. #include <boost/multi_index/detail/auto_space.hpp>
  27. #include <boost/multi_index/detail/bucket_array.hpp>
  28. #include <boost/multi_index/detail/do_not_copy_elements_tag.hpp>
  29. #include <boost/multi_index/detail/hash_index_iterator.hpp>
  30. #include <boost/multi_index/detail/index_node_base.hpp>
  31. #include <boost/multi_index/detail/modify_key_adaptor.hpp>
  32. #include <boost/multi_index/detail/promotes_arg.hpp>
  33. #include <boost/multi_index/detail/safe_mode.hpp>
  34. #include <boost/multi_index/detail/scope_guard.hpp>
  35. #include <boost/multi_index/detail/vartempl_support.hpp>
  36. #include <boost/multi_index/hashed_index_fwd.hpp>
  37. #include <boost/tuple/tuple.hpp>
  38. #include <boost/type_traits/is_same.hpp>
  39. #include <cmath>
  40. #include <cstddef>
  41. #include <functional>
  42. #include <iterator>
  43. #include <utility>
  44. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  45. #include <initializer_list>
  46. #endif
  47. #if !defined(BOOST_MULTI_INDEX_DISABLE_SERIALIZATION)
  48. #include <boost/serialization/nvp.hpp>
  49. #endif
  50. #if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)
  51. #define BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT_OF(x) \
  52. detail::scope_guard BOOST_JOIN(check_invariant_,__LINE__)= \
  53. detail::make_obj_guard(x,&hashed_index::check_invariant_); \
  54. BOOST_JOIN(check_invariant_,__LINE__).touch();
  55. #define BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT \
  56. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT_OF(*this)
  57. #else
  58. #define BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT_OF(x)
  59. #define BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT
  60. #endif
  61. namespace boost{
  62. namespace multi_index{
  63. namespace detail{
  64. /* hashed_index adds a layer of hashed indexing to a given Super */
  65. /* Most of the implementation of unique and non-unique indices is
  66. * shared. We tell from one another on instantiation time by using
  67. * Category tags defined in hash_index_node.hpp.
  68. */
  69. template<
  70. typename KeyFromValue,typename Hash,typename Pred,
  71. typename SuperMeta,typename TagList,typename Category
  72. >
  73. class hashed_index:
  74. BOOST_MULTI_INDEX_PROTECTED_IF_MEMBER_TEMPLATE_FRIENDS SuperMeta::type
  75. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  76. ,public safe_mode::safe_container<
  77. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category> >
  78. #endif
  79. {
  80. #if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)&&\
  81. BOOST_WORKAROUND(__MWERKS__,<=0x3003)
  82. /* The "ISO C++ Template Parser" option in CW8.3 has a problem with the
  83. * lifetime of const references bound to temporaries --precisely what
  84. * scopeguards are.
  85. */
  86. #pragma parse_mfunc_templ off
  87. #endif
  88. typedef typename SuperMeta::type super;
  89. protected:
  90. typedef hashed_index_node<
  91. typename super::node_type,Category> node_type;
  92. private:
  93. typedef typename node_type::node_alg node_alg;
  94. typedef typename node_type::impl_type node_impl_type;
  95. typedef typename node_impl_type::pointer node_impl_pointer;
  96. typedef typename node_impl_type::base_pointer node_impl_base_pointer;
  97. typedef bucket_array<
  98. typename super::final_allocator_type> bucket_array_type;
  99. public:
  100. /* types */
  101. typedef typename KeyFromValue::result_type key_type;
  102. typedef typename node_type::value_type value_type;
  103. typedef KeyFromValue key_from_value;
  104. typedef Hash hasher;
  105. typedef Pred key_equal;
  106. typedef tuple<std::size_t,
  107. key_from_value,hasher,key_equal> ctor_args;
  108. typedef typename super::final_allocator_type allocator_type;
  109. typedef typename allocator_type::pointer pointer;
  110. typedef typename allocator_type::const_pointer const_pointer;
  111. typedef typename allocator_type::reference reference;
  112. typedef typename allocator_type::const_reference const_reference;
  113. typedef std::size_t size_type;
  114. typedef std::ptrdiff_t difference_type;
  115. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  116. typedef safe_mode::safe_iterator<
  117. hashed_index_iterator<
  118. node_type,bucket_array_type,
  119. hashed_index_global_iterator_tag>,
  120. hashed_index> iterator;
  121. #else
  122. typedef hashed_index_iterator<
  123. node_type,bucket_array_type,
  124. hashed_index_global_iterator_tag> iterator;
  125. #endif
  126. typedef iterator const_iterator;
  127. typedef hashed_index_iterator<
  128. node_type,bucket_array_type,
  129. hashed_index_local_iterator_tag> local_iterator;
  130. typedef local_iterator const_local_iterator;
  131. typedef TagList tag_list;
  132. protected:
  133. typedef typename super::final_node_type final_node_type;
  134. typedef tuples::cons<
  135. ctor_args,
  136. typename super::ctor_args_list> ctor_args_list;
  137. typedef typename mpl::push_front<
  138. typename super::index_type_list,
  139. hashed_index>::type index_type_list;
  140. typedef typename mpl::push_front<
  141. typename super::iterator_type_list,
  142. iterator>::type iterator_type_list;
  143. typedef typename mpl::push_front<
  144. typename super::const_iterator_type_list,
  145. const_iterator>::type const_iterator_type_list;
  146. typedef typename super::copy_map_type copy_map_type;
  147. #if !defined(BOOST_MULTI_INDEX_DISABLE_SERIALIZATION)
  148. typedef typename super::index_saver_type index_saver_type;
  149. typedef typename super::index_loader_type index_loader_type;
  150. #endif
  151. private:
  152. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  153. typedef safe_mode::safe_container<
  154. hashed_index> safe_super;
  155. #endif
  156. typedef typename call_traits<value_type>::param_type value_param_type;
  157. typedef typename call_traits<
  158. key_type>::param_type key_param_type;
  159. /* Needed to avoid commas in BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL
  160. * expansion.
  161. */
  162. typedef std::pair<iterator,bool> emplace_return_type;
  163. public:
  164. /* construct/destroy/copy
  165. * Default and copy ctors are in the protected section as indices are
  166. * not supposed to be created on their own. No range ctor either.
  167. */
  168. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& operator=(
  169. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x)
  170. {
  171. this->final()=x.final();
  172. return *this;
  173. }
  174. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  175. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& operator=(
  176. std::initializer_list<value_type> list)
  177. {
  178. this->final()=list;
  179. return *this;
  180. }
  181. #endif
  182. allocator_type get_allocator()const BOOST_NOEXCEPT
  183. {
  184. return this->final().get_allocator();
  185. }
  186. /* size and capacity */
  187. bool empty()const BOOST_NOEXCEPT{return this->final_empty_();}
  188. size_type size()const BOOST_NOEXCEPT{return this->final_size_();}
  189. size_type max_size()const BOOST_NOEXCEPT{return this->final_max_size_();}
  190. /* iterators */
  191. iterator begin()BOOST_NOEXCEPT
  192. {return make_iterator(node_type::from_impl(header()->next()->prior()));}
  193. const_iterator begin()const BOOST_NOEXCEPT
  194. {return make_iterator(node_type::from_impl(header()->next()->prior()));}
  195. iterator end()BOOST_NOEXCEPT{return make_iterator(header());}
  196. const_iterator end()const BOOST_NOEXCEPT{return make_iterator(header());}
  197. const_iterator cbegin()const BOOST_NOEXCEPT{return begin();}
  198. const_iterator cend()const BOOST_NOEXCEPT{return end();}
  199. iterator iterator_to(const value_type& x)
  200. {
  201. return make_iterator(node_from_value<node_type>(&x));
  202. }
  203. const_iterator iterator_to(const value_type& x)const
  204. {
  205. return make_iterator(node_from_value<node_type>(&x));
  206. }
  207. /* modifiers */
  208. BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL(
  209. emplace_return_type,emplace,emplace_impl)
  210. BOOST_MULTI_INDEX_OVERLOADS_TO_VARTEMPL_EXTRA_ARG(
  211. iterator,emplace_hint,emplace_hint_impl,iterator,position)
  212. std::pair<iterator,bool> insert(const value_type& x)
  213. {
  214. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  215. std::pair<final_node_type*,bool> p=this->final_insert_(x);
  216. return std::pair<iterator,bool>(make_iterator(p.first),p.second);
  217. }
  218. std::pair<iterator,bool> insert(BOOST_RV_REF(value_type) x)
  219. {
  220. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  221. std::pair<final_node_type*,bool> p=this->final_insert_rv_(x);
  222. return std::pair<iterator,bool>(make_iterator(p.first),p.second);
  223. }
  224. iterator insert(iterator position,const value_type& x)
  225. {
  226. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  227. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  228. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  229. std::pair<final_node_type*,bool> p=this->final_insert_(
  230. x,static_cast<final_node_type*>(position.get_node()));
  231. return make_iterator(p.first);
  232. }
  233. iterator insert(iterator position,BOOST_RV_REF(value_type) x)
  234. {
  235. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  236. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  237. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  238. std::pair<final_node_type*,bool> p=this->final_insert_rv_(
  239. x,static_cast<final_node_type*>(position.get_node()));
  240. return make_iterator(p.first);
  241. }
  242. template<typename InputIterator>
  243. void insert(InputIterator first,InputIterator last)
  244. {
  245. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  246. for(;first!=last;++first)this->final_insert_ref_(*first);
  247. }
  248. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  249. void insert(std::initializer_list<value_type> list)
  250. {
  251. insert(list.begin(),list.end());
  252. }
  253. #endif
  254. iterator erase(iterator position)
  255. {
  256. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  257. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  258. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  259. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  260. this->final_erase_(static_cast<final_node_type*>(position++.get_node()));
  261. return position;
  262. }
  263. size_type erase(key_param_type k)
  264. {
  265. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  266. std::size_t buc=buckets.position(hash_(k));
  267. for(node_impl_pointer x=buckets.at(buc)->prior();
  268. x!=node_impl_pointer(0);x=node_alg::next_to_inspect(x)){
  269. if(eq_(k,key(node_type::from_impl(x)->value()))){
  270. node_impl_pointer y=end_of_range(x);
  271. size_type s=0;
  272. do{
  273. node_impl_pointer z=node_alg::after(x);
  274. this->final_erase_(
  275. static_cast<final_node_type*>(node_type::from_impl(x)));
  276. x=z;
  277. ++s;
  278. }while(x!=y);
  279. return s;
  280. }
  281. }
  282. return 0;
  283. }
  284. iterator erase(iterator first,iterator last)
  285. {
  286. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(first);
  287. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(last);
  288. BOOST_MULTI_INDEX_CHECK_IS_OWNER(first,*this);
  289. BOOST_MULTI_INDEX_CHECK_IS_OWNER(last,*this);
  290. BOOST_MULTI_INDEX_CHECK_VALID_RANGE(first,last);
  291. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  292. while(first!=last){
  293. first=erase(first);
  294. }
  295. return first;
  296. }
  297. bool replace(iterator position,const value_type& x)
  298. {
  299. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  300. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  301. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  302. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  303. return this->final_replace_(
  304. x,static_cast<final_node_type*>(position.get_node()));
  305. }
  306. bool replace(iterator position,BOOST_RV_REF(value_type) x)
  307. {
  308. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  309. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  310. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  311. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  312. return this->final_replace_rv_(
  313. x,static_cast<final_node_type*>(position.get_node()));
  314. }
  315. template<typename Modifier>
  316. bool modify(iterator position,Modifier mod)
  317. {
  318. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  319. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  320. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  321. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  322. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  323. /* MSVC++ 6.0 optimizer on safe mode code chokes if this
  324. * this is not added. Left it for all compilers as it does no
  325. * harm.
  326. */
  327. position.detach();
  328. #endif
  329. return this->final_modify_(
  330. mod,static_cast<final_node_type*>(position.get_node()));
  331. }
  332. template<typename Modifier,typename Rollback>
  333. bool modify(iterator position,Modifier mod,Rollback back_)
  334. {
  335. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  336. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  337. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  338. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  339. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  340. /* MSVC++ 6.0 optimizer on safe mode code chokes if this
  341. * this is not added. Left it for all compilers as it does no
  342. * harm.
  343. */
  344. position.detach();
  345. #endif
  346. return this->final_modify_(
  347. mod,back_,static_cast<final_node_type*>(position.get_node()));
  348. }
  349. template<typename Modifier>
  350. bool modify_key(iterator position,Modifier mod)
  351. {
  352. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  353. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  354. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  355. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  356. return modify(
  357. position,modify_key_adaptor<Modifier,value_type,KeyFromValue>(mod,key));
  358. }
  359. template<typename Modifier,typename Rollback>
  360. bool modify_key(iterator position,Modifier mod,Rollback back_)
  361. {
  362. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  363. BOOST_MULTI_INDEX_CHECK_DEREFERENCEABLE_ITERATOR(position);
  364. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  365. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  366. return modify(
  367. position,
  368. modify_key_adaptor<Modifier,value_type,KeyFromValue>(mod,key),
  369. modify_key_adaptor<Rollback,value_type,KeyFromValue>(back_,key));
  370. }
  371. void clear()BOOST_NOEXCEPT
  372. {
  373. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  374. this->final_clear_();
  375. }
  376. void swap(hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x)
  377. {
  378. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  379. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT_OF(x);
  380. this->final_swap_(x.final());
  381. }
  382. /* observers */
  383. key_from_value key_extractor()const{return key;}
  384. hasher hash_function()const{return hash_;}
  385. key_equal key_eq()const{return eq_;}
  386. /* lookup */
  387. /* Internally, these ops rely on const_iterator being the same
  388. * type as iterator.
  389. */
  390. /* Implementation note: When CompatibleKey is consistently promoted to
  391. * KeyFromValue::result_type for equality comparison, the promotion is made
  392. * once in advance to increase efficiency.
  393. */
  394. template<typename CompatibleKey>
  395. iterator find(const CompatibleKey& k)const
  396. {
  397. return find(k,hash_,eq_);
  398. }
  399. template<
  400. typename CompatibleKey,typename CompatibleHash,typename CompatiblePred
  401. >
  402. iterator find(
  403. const CompatibleKey& k,
  404. const CompatibleHash& hash,const CompatiblePred& eq)const
  405. {
  406. return find(
  407. k,hash,eq,promotes_1st_arg<CompatiblePred,CompatibleKey,key_type>());
  408. }
  409. template<typename CompatibleKey>
  410. size_type count(const CompatibleKey& k)const
  411. {
  412. return count(k,hash_,eq_);
  413. }
  414. template<
  415. typename CompatibleKey,typename CompatibleHash,typename CompatiblePred
  416. >
  417. size_type count(
  418. const CompatibleKey& k,
  419. const CompatibleHash& hash,const CompatiblePred& eq)const
  420. {
  421. return count(
  422. k,hash,eq,promotes_1st_arg<CompatiblePred,CompatibleKey,key_type>());
  423. }
  424. template<typename CompatibleKey>
  425. std::pair<iterator,iterator> equal_range(const CompatibleKey& k)const
  426. {
  427. return equal_range(k,hash_,eq_);
  428. }
  429. template<
  430. typename CompatibleKey,typename CompatibleHash,typename CompatiblePred
  431. >
  432. std::pair<iterator,iterator> equal_range(
  433. const CompatibleKey& k,
  434. const CompatibleHash& hash,const CompatiblePred& eq)const
  435. {
  436. return equal_range(
  437. k,hash,eq,promotes_1st_arg<CompatiblePred,CompatibleKey,key_type>());
  438. }
  439. /* bucket interface */
  440. size_type bucket_count()const BOOST_NOEXCEPT{return buckets.size();}
  441. size_type max_bucket_count()const BOOST_NOEXCEPT{return static_cast<size_type>(-1);}
  442. size_type bucket_size(size_type n)const
  443. {
  444. size_type res=0;
  445. for(node_impl_pointer x=buckets.at(n)->prior();
  446. x!=node_impl_pointer(0);x=node_alg::after_local(x)){
  447. ++res;
  448. }
  449. return res;
  450. }
  451. size_type bucket(key_param_type k)const
  452. {
  453. return buckets.position(hash_(k));
  454. }
  455. local_iterator begin(size_type n)
  456. {
  457. return const_cast<const hashed_index*>(this)->begin(n);
  458. }
  459. const_local_iterator begin(size_type n)const
  460. {
  461. node_impl_pointer x=buckets.at(n)->prior();
  462. if(x==node_impl_pointer(0))return end(n);
  463. return make_local_iterator(node_type::from_impl(x));
  464. }
  465. local_iterator end(size_type n)
  466. {
  467. return const_cast<const hashed_index*>(this)->end(n);
  468. }
  469. const_local_iterator end(size_type)const
  470. {
  471. return make_local_iterator(0);
  472. }
  473. const_local_iterator cbegin(size_type n)const{return begin(n);}
  474. const_local_iterator cend(size_type n)const{return end(n);}
  475. local_iterator local_iterator_to(const value_type& x)
  476. {
  477. return make_local_iterator(node_from_value<node_type>(&x));
  478. }
  479. const_local_iterator local_iterator_to(const value_type& x)const
  480. {
  481. return make_local_iterator(node_from_value<node_type>(&x));
  482. }
  483. /* hash policy */
  484. float load_factor()const BOOST_NOEXCEPT
  485. {return static_cast<float>(size())/bucket_count();}
  486. float max_load_factor()const BOOST_NOEXCEPT{return mlf;}
  487. void max_load_factor(float z){mlf=z;calculate_max_load();}
  488. void rehash(size_type n)
  489. {
  490. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  491. if(size()<=max_load&&n<=bucket_count())return;
  492. size_type bc =(std::numeric_limits<size_type>::max)();
  493. float fbc=static_cast<float>(1+size()/mlf);
  494. if(bc>fbc){
  495. bc=static_cast<size_type>(fbc);
  496. if(bc<n)bc=n;
  497. }
  498. unchecked_rehash(bc);
  499. }
  500. void reserve(size_type n)
  501. {
  502. rehash(static_cast<size_type>(std::ceil(static_cast<double>(n)/mlf)));
  503. }
  504. BOOST_MULTI_INDEX_PROTECTED_IF_MEMBER_TEMPLATE_FRIENDS:
  505. hashed_index(const ctor_args_list& args_list,const allocator_type& al):
  506. super(args_list.get_tail(),al),
  507. key(tuples::get<1>(args_list.get_head())),
  508. hash_(tuples::get<2>(args_list.get_head())),
  509. eq_(tuples::get<3>(args_list.get_head())),
  510. buckets(al,header()->impl(),tuples::get<0>(args_list.get_head())),
  511. mlf(1.0f)
  512. {
  513. calculate_max_load();
  514. }
  515. hashed_index(
  516. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x):
  517. super(x),
  518. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  519. safe_super(),
  520. #endif
  521. key(x.key),
  522. hash_(x.hash_),
  523. eq_(x.eq_),
  524. buckets(x.get_allocator(),header()->impl(),x.buckets.size()),
  525. mlf(x.mlf),
  526. max_load(x.max_load)
  527. {
  528. /* Copy ctor just takes the internal configuration objects from x. The rest
  529. * is done in subsequent call to copy_().
  530. */
  531. }
  532. hashed_index(
  533. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  534. do_not_copy_elements_tag):
  535. super(x,do_not_copy_elements_tag()),
  536. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  537. safe_super(),
  538. #endif
  539. key(x.key),
  540. hash_(x.hash_),
  541. eq_(x.eq_),
  542. buckets(x.get_allocator(),header()->impl(),0),
  543. mlf(1.0f)
  544. {
  545. calculate_max_load();
  546. }
  547. ~hashed_index()
  548. {
  549. /* the container is guaranteed to be empty by now */
  550. }
  551. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  552. iterator make_iterator(node_type* node)
  553. {
  554. return iterator(node,this);
  555. }
  556. const_iterator make_iterator(node_type* node)const
  557. {
  558. return const_iterator(node,const_cast<hashed_index*>(this));
  559. }
  560. #else
  561. iterator make_iterator(node_type* node)
  562. {
  563. return iterator(node);
  564. }
  565. const_iterator make_iterator(node_type* node)const
  566. {
  567. return const_iterator(node);
  568. }
  569. #endif
  570. local_iterator make_local_iterator(node_type* node)
  571. {
  572. return local_iterator(node);
  573. }
  574. const_local_iterator make_local_iterator(node_type* node)const
  575. {
  576. return const_local_iterator(node);
  577. }
  578. void copy_(
  579. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  580. const copy_map_type& map)
  581. {
  582. copy_(x,map,Category());
  583. }
  584. void copy_(
  585. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  586. const copy_map_type& map,hashed_unique_tag)
  587. {
  588. if(x.size()!=0){
  589. node_impl_pointer end_org=x.header()->impl(),
  590. org=end_org,
  591. cpy=header()->impl();
  592. do{
  593. node_impl_pointer prev_org=org->prior(),
  594. prev_cpy=
  595. static_cast<node_type*>(map.find(static_cast<final_node_type*>(
  596. node_type::from_impl(prev_org))))->impl();
  597. cpy->prior()=prev_cpy;
  598. if(node_alg::is_first_of_bucket(org)){
  599. node_impl_base_pointer buc_org=prev_org->next(),
  600. buc_cpy=
  601. buckets.begin()+(buc_org-x.buckets.begin());
  602. prev_cpy->next()=buc_cpy;
  603. buc_cpy->prior()=cpy;
  604. }
  605. else{
  606. prev_cpy->next()=node_impl_type::base_pointer_from(cpy);
  607. }
  608. org=prev_org;
  609. cpy=prev_cpy;
  610. }while(org!=end_org);
  611. }
  612. super::copy_(x,map);
  613. }
  614. void copy_(
  615. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  616. const copy_map_type& map,hashed_non_unique_tag)
  617. {
  618. if(x.size()!=0){
  619. node_impl_pointer end_org=x.header()->impl(),
  620. org=end_org,
  621. cpy=header()->impl();
  622. do{
  623. node_impl_pointer next_org=node_alg::after(org),
  624. next_cpy=
  625. static_cast<node_type*>(map.find(static_cast<final_node_type*>(
  626. node_type::from_impl(next_org))))->impl();
  627. if(node_alg::is_first_of_bucket(next_org)){
  628. node_impl_base_pointer buc_org=org->next(),
  629. buc_cpy=
  630. buckets.begin()+(buc_org-x.buckets.begin());
  631. cpy->next()=buc_cpy;
  632. buc_cpy->prior()=next_cpy;
  633. next_cpy->prior()=cpy;
  634. }
  635. else{
  636. if(org->next()==node_impl_type::base_pointer_from(next_org)){
  637. cpy->next()=node_impl_type::base_pointer_from(next_cpy);
  638. }
  639. else{
  640. cpy->next()=
  641. node_impl_type::base_pointer_from(
  642. static_cast<node_type*>(map.find(static_cast<final_node_type*>(
  643. node_type::from_impl(
  644. node_impl_type::pointer_from(org->next())))))->impl());
  645. }
  646. if(next_org->prior()!=org){
  647. next_cpy->prior()=
  648. static_cast<node_type*>(map.find(static_cast<final_node_type*>(
  649. node_type::from_impl(next_org->prior()))))->impl();
  650. }
  651. else{
  652. next_cpy->prior()=cpy;
  653. }
  654. }
  655. org=next_org;
  656. cpy=next_cpy;
  657. }while(org!=end_org);
  658. }
  659. super::copy_(x,map);
  660. }
  661. template<typename Variant>
  662. final_node_type* insert_(
  663. value_param_type v,final_node_type*& x,Variant variant)
  664. {
  665. reserve_for_insert(size()+1);
  666. std::size_t buc=find_bucket(v);
  667. link_info pos(buckets.at(buc));
  668. if(!link_point(v,pos)){
  669. return static_cast<final_node_type*>(
  670. node_type::from_impl(node_impl_type::pointer_from(pos)));
  671. }
  672. final_node_type* res=super::insert_(v,x,variant);
  673. if(res==x)link(static_cast<node_type*>(x),pos);
  674. return res;
  675. }
  676. template<typename Variant>
  677. final_node_type* insert_(
  678. value_param_type v,node_type* position,final_node_type*& x,Variant variant)
  679. {
  680. reserve_for_insert(size()+1);
  681. std::size_t buc=find_bucket(v);
  682. link_info pos(buckets.at(buc));
  683. if(!link_point(v,pos)){
  684. return static_cast<final_node_type*>(
  685. node_type::from_impl(node_impl_type::pointer_from(pos)));
  686. }
  687. final_node_type* res=super::insert_(v,position,x,variant);
  688. if(res==x)link(static_cast<node_type*>(x),pos);
  689. return res;
  690. }
  691. void erase_(node_type* x)
  692. {
  693. unlink(x);
  694. super::erase_(x);
  695. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  696. detach_iterators(x);
  697. #endif
  698. }
  699. void delete_all_nodes_()
  700. {
  701. delete_all_nodes_(Category());
  702. }
  703. void delete_all_nodes_(hashed_unique_tag)
  704. {
  705. for(node_impl_pointer x_end=header()->impl(),x=x_end->prior();x!=x_end;){
  706. node_impl_pointer y=x->prior();
  707. this->final_delete_node_(
  708. static_cast<final_node_type*>(node_type::from_impl(x)));
  709. x=y;
  710. }
  711. }
  712. void delete_all_nodes_(hashed_non_unique_tag)
  713. {
  714. for(node_impl_pointer x_end=header()->impl(),x=x_end->prior();x!=x_end;){
  715. node_impl_pointer y=x->prior();
  716. if(y->next()!=node_impl_type::base_pointer_from(x)&&
  717. y->next()->prior()!=x){ /* n-1 of group */
  718. /* Make the second node prior() pointer back-linked so that it won't
  719. * refer to a deleted node when the time for its own destruction comes.
  720. */
  721. node_impl_pointer first=node_impl_type::pointer_from(y->next());
  722. first->next()->prior()=first;
  723. }
  724. this->final_delete_node_(
  725. static_cast<final_node_type*>(node_type::from_impl(x)));
  726. x=y;
  727. }
  728. }
  729. void clear_()
  730. {
  731. super::clear_();
  732. buckets.clear(header()->impl());
  733. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  734. safe_super::detach_dereferenceable_iterators();
  735. #endif
  736. }
  737. void swap_(
  738. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x)
  739. {
  740. std::swap(key,x.key);
  741. std::swap(hash_,x.hash_);
  742. std::swap(eq_,x.eq_);
  743. buckets.swap(x.buckets);
  744. std::swap(mlf,x.mlf);
  745. std::swap(max_load,x.max_load);
  746. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  747. safe_super::swap(x);
  748. #endif
  749. super::swap_(x);
  750. }
  751. void swap_elements_(
  752. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x)
  753. {
  754. buckets.swap(x.buckets);
  755. std::swap(mlf,x.mlf);
  756. std::swap(max_load,x.max_load);
  757. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  758. safe_super::swap(x);
  759. #endif
  760. super::swap_elements_(x);
  761. }
  762. template<typename Variant>
  763. bool replace_(value_param_type v,node_type* x,Variant variant)
  764. {
  765. if(eq_(key(v),key(x->value()))){
  766. return super::replace_(v,x,variant);
  767. }
  768. unlink_undo undo;
  769. unlink(x,undo);
  770. BOOST_TRY{
  771. std::size_t buc=find_bucket(v);
  772. link_info pos(buckets.at(buc));
  773. if(link_point(v,pos)&&super::replace_(v,x,variant)){
  774. link(x,pos);
  775. return true;
  776. }
  777. undo();
  778. return false;
  779. }
  780. BOOST_CATCH(...){
  781. undo();
  782. BOOST_RETHROW;
  783. }
  784. BOOST_CATCH_END
  785. }
  786. bool modify_(node_type* x)
  787. {
  788. std::size_t buc;
  789. bool b;
  790. BOOST_TRY{
  791. buc=find_bucket(x->value());
  792. b=in_place(x->impl(),key(x->value()),buc);
  793. }
  794. BOOST_CATCH(...){
  795. erase_(x);
  796. BOOST_RETHROW;
  797. }
  798. BOOST_CATCH_END
  799. if(!b){
  800. unlink(x);
  801. BOOST_TRY{
  802. link_info pos(buckets.at(buc));
  803. if(!link_point(x->value(),pos)){
  804. super::erase_(x);
  805. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  806. detach_iterators(x);
  807. #endif
  808. return false;
  809. }
  810. link(x,pos);
  811. }
  812. BOOST_CATCH(...){
  813. super::erase_(x);
  814. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  815. detach_iterators(x);
  816. #endif
  817. BOOST_RETHROW;
  818. }
  819. BOOST_CATCH_END
  820. }
  821. BOOST_TRY{
  822. if(!super::modify_(x)){
  823. unlink(x);
  824. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  825. detach_iterators(x);
  826. #endif
  827. return false;
  828. }
  829. else return true;
  830. }
  831. BOOST_CATCH(...){
  832. unlink(x);
  833. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  834. detach_iterators(x);
  835. #endif
  836. BOOST_RETHROW;
  837. }
  838. BOOST_CATCH_END
  839. }
  840. bool modify_rollback_(node_type* x)
  841. {
  842. std::size_t buc=find_bucket(x->value());
  843. if(in_place(x->impl(),key(x->value()),buc)){
  844. return super::modify_rollback_(x);
  845. }
  846. unlink_undo undo;
  847. unlink(x,undo);
  848. BOOST_TRY{
  849. link_info pos(buckets.at(buc));
  850. if(link_point(x->value(),pos)&&super::modify_rollback_(x)){
  851. link(x,pos);
  852. return true;
  853. }
  854. undo();
  855. return false;
  856. }
  857. BOOST_CATCH(...){
  858. undo();
  859. BOOST_RETHROW;
  860. }
  861. BOOST_CATCH_END
  862. }
  863. /* comparison */
  864. #if !defined(BOOST_NO_MEMBER_TEMPLATE_FRIENDS)
  865. /* defect macro refers to class, not function, templates, but anyway */
  866. template<typename K,typename H,typename P,typename S,typename T,typename C>
  867. friend bool operator==(
  868. const hashed_index<K,H,P,S,T,C>&,const hashed_index<K,H,P,S,T,C>& y);
  869. #endif
  870. bool equals(const hashed_index& x)const{return equals(x,Category());}
  871. bool equals(const hashed_index& x,hashed_unique_tag)const
  872. {
  873. if(size()!=x.size())return false;
  874. for(const_iterator it=begin(),it_end=end(),it2_end=x.end();
  875. it!=it_end;++it){
  876. const_iterator it2=x.find(key(*it));
  877. if(it2==it2_end||!(*it==*it2))return false;
  878. }
  879. return true;
  880. }
  881. bool equals(const hashed_index& x,hashed_non_unique_tag)const
  882. {
  883. if(size()!=x.size())return false;
  884. for(const_iterator it=begin(),it_end=end();it!=it_end;){
  885. const_iterator it2,it2_last;
  886. boost::tie(it2,it2_last)=x.equal_range(key(*it));
  887. if(it2==it2_last)return false;
  888. const_iterator it_last=make_iterator(
  889. node_type::from_impl(end_of_range(it.get_node()->impl())));
  890. if(std::distance(it,it_last)!=std::distance(it2,it2_last))return false;
  891. /* From is_permutation code in
  892. * http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2010/n3068.pdf
  893. */
  894. for(;it!=it_last;++it,++it2){
  895. if(!(*it==*it2))break;
  896. }
  897. if(it!=it_last){
  898. for(const_iterator scan=it;scan!=it_last;++scan){
  899. if(std::find(it,scan,*scan)!=scan)continue;
  900. std::ptrdiff_t matches=std::count(it2,it2_last,*scan);
  901. if(matches==0||matches!=std::count(scan,it_last,*scan))return false;
  902. }
  903. it=it_last;
  904. }
  905. }
  906. return true;
  907. }
  908. #if !defined(BOOST_MULTI_INDEX_DISABLE_SERIALIZATION)
  909. /* serialization */
  910. template<typename Archive>
  911. void save_(
  912. Archive& ar,const unsigned int version,const index_saver_type& sm)const
  913. {
  914. ar<<serialization::make_nvp("position",buckets);
  915. super::save_(ar,version,sm);
  916. }
  917. template<typename Archive>
  918. void load_(Archive& ar,const unsigned int version,const index_loader_type& lm)
  919. {
  920. ar>>serialization::make_nvp("position",buckets);
  921. super::load_(ar,version,lm);
  922. }
  923. #endif
  924. #if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)
  925. /* invariant stuff */
  926. bool invariant_()const
  927. {
  928. if(size()==0||begin()==end()){
  929. if(size()!=0||begin()!=end())return false;
  930. }
  931. else{
  932. size_type s0=0;
  933. for(const_iterator it=begin(),it_end=end();it!=it_end;++it,++s0){}
  934. if(s0!=size())return false;
  935. size_type s1=0;
  936. for(size_type buc=0;buc<bucket_count();++buc){
  937. size_type ss1=0;
  938. for(const_local_iterator it=begin(buc),it_end=end(buc);
  939. it!=it_end;++it,++ss1){
  940. if(find_bucket(*it)!=buc)return false;
  941. }
  942. if(ss1!=bucket_size(buc))return false;
  943. s1+=ss1;
  944. }
  945. if(s1!=size())return false;
  946. }
  947. return super::invariant_();
  948. }
  949. /* This forwarding function eases things for the boost::mem_fn construct
  950. * in BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT. Actually,
  951. * final_check_invariant is already an inherited member function of index.
  952. */
  953. void check_invariant_()const{this->final_check_invariant_();}
  954. #endif
  955. private:
  956. node_type* header()const{return this->final_header();}
  957. std::size_t find_bucket(value_param_type v)const
  958. {
  959. return bucket(key(v));
  960. }
  961. struct link_info_non_unique
  962. {
  963. link_info_non_unique(node_impl_base_pointer pos):
  964. first(pos),last(node_impl_base_pointer(0)){}
  965. operator const node_impl_base_pointer&()const{return this->first;}
  966. node_impl_base_pointer first,last;
  967. };
  968. typedef typename mpl::if_<
  969. is_same<Category,hashed_unique_tag>,
  970. node_impl_base_pointer,
  971. link_info_non_unique
  972. >::type link_info;
  973. bool link_point(value_param_type v,link_info& pos)
  974. {
  975. return link_point(v,pos,Category());
  976. }
  977. bool link_point(
  978. value_param_type v,node_impl_base_pointer& pos,hashed_unique_tag)
  979. {
  980. for(node_impl_pointer x=pos->prior();x!=node_impl_pointer(0);
  981. x=node_alg::after_local(x)){
  982. if(eq_(key(v),key(node_type::from_impl(x)->value()))){
  983. pos=node_impl_type::base_pointer_from(x);
  984. return false;
  985. }
  986. }
  987. return true;
  988. }
  989. bool link_point(
  990. value_param_type v,link_info_non_unique& pos,hashed_non_unique_tag)
  991. {
  992. for(node_impl_pointer x=pos.first->prior();x!=node_impl_pointer(0);
  993. x=node_alg::next_to_inspect(x)){
  994. if(eq_(key(v),key(node_type::from_impl(x)->value()))){
  995. pos.first=node_impl_type::base_pointer_from(x);
  996. pos.last=node_impl_type::base_pointer_from(last_of_range(x));
  997. return true;
  998. }
  999. }
  1000. return true;
  1001. }
  1002. node_impl_pointer last_of_range(node_impl_pointer x)const
  1003. {
  1004. return last_of_range(x,Category());
  1005. }
  1006. node_impl_pointer last_of_range(node_impl_pointer x,hashed_unique_tag)const
  1007. {
  1008. return x;
  1009. }
  1010. node_impl_pointer last_of_range(
  1011. node_impl_pointer x,hashed_non_unique_tag)const
  1012. {
  1013. node_impl_base_pointer y=x->next();
  1014. node_impl_pointer z=y->prior();
  1015. if(z==x){ /* range of size 1 or 2 */
  1016. node_impl_pointer yy=node_impl_type::pointer_from(y);
  1017. return
  1018. eq_(
  1019. key(node_type::from_impl(x)->value()),
  1020. key(node_type::from_impl(yy)->value()))?yy:x;
  1021. }
  1022. else if(z->prior()==x) /* last of bucket */
  1023. return x;
  1024. else /* group of size>2 */
  1025. return z;
  1026. }
  1027. node_impl_pointer end_of_range(node_impl_pointer x)const
  1028. {
  1029. return end_of_range(x,Category());
  1030. }
  1031. node_impl_pointer end_of_range(node_impl_pointer x,hashed_unique_tag)const
  1032. {
  1033. return node_alg::after(last_of_range(x));
  1034. }
  1035. node_impl_pointer end_of_range(
  1036. node_impl_pointer x,hashed_non_unique_tag)const
  1037. {
  1038. node_impl_base_pointer y=x->next();
  1039. node_impl_pointer z=y->prior();
  1040. if(z==x){ /* range of size 1 or 2 */
  1041. node_impl_pointer yy=node_impl_type::pointer_from(y);
  1042. if(!eq_(
  1043. key(node_type::from_impl(x)->value()),
  1044. key(node_type::from_impl(yy)->value())))yy=x;
  1045. return yy->next()->prior()==yy?
  1046. node_impl_type::pointer_from(yy->next()):
  1047. yy->next()->prior();
  1048. }
  1049. else if(z->prior()==x) /* last of bucket */
  1050. return z;
  1051. else /* group of size>2 */
  1052. return z->next()->prior()==z?
  1053. node_impl_type::pointer_from(z->next()):
  1054. z->next()->prior();
  1055. }
  1056. void link(node_type* x,const link_info& pos)
  1057. {
  1058. link(x,pos,Category());
  1059. }
  1060. void link(node_type* x,node_impl_base_pointer pos,hashed_unique_tag)
  1061. {
  1062. node_alg::link(x->impl(),pos,header()->impl());
  1063. }
  1064. void link(node_type* x,const link_info_non_unique& pos,hashed_non_unique_tag)
  1065. {
  1066. if(pos.last==node_impl_base_pointer(0)){
  1067. node_alg::link(x->impl(),pos.first,header()->impl());
  1068. }
  1069. else{
  1070. node_alg::link(
  1071. x->impl(),
  1072. node_impl_type::pointer_from(pos.first),
  1073. node_impl_type::pointer_from(pos.last));
  1074. }
  1075. }
  1076. void unlink(node_type* x)
  1077. {
  1078. node_alg::unlink(x->impl());
  1079. }
  1080. typedef typename node_alg::unlink_undo unlink_undo;
  1081. void unlink(node_type* x,unlink_undo& undo)
  1082. {
  1083. node_alg::unlink(x->impl(),undo);
  1084. }
  1085. void calculate_max_load()
  1086. {
  1087. float fml=static_cast<float>(mlf*static_cast<float>(bucket_count()));
  1088. max_load=(std::numeric_limits<size_type>::max)();
  1089. if(max_load>fml)max_load=static_cast<size_type>(fml);
  1090. }
  1091. void reserve_for_insert(size_type n)
  1092. {
  1093. if(n>max_load){
  1094. size_type bc =(std::numeric_limits<size_type>::max)();
  1095. float fbc=static_cast<float>(1+static_cast<double>(n)/mlf);
  1096. if(bc>fbc)bc =static_cast<size_type>(fbc);
  1097. unchecked_rehash(bc);
  1098. }
  1099. }
  1100. void unchecked_rehash(size_type n){unchecked_rehash(n,Category());}
  1101. void unchecked_rehash(size_type n,hashed_unique_tag)
  1102. {
  1103. node_impl_type cpy_end_node;
  1104. node_impl_pointer cpy_end=node_impl_pointer(&cpy_end_node),
  1105. end_=header()->impl();
  1106. bucket_array_type buckets_cpy(get_allocator(),cpy_end,n);
  1107. if(size()!=0){
  1108. auto_space<
  1109. std::size_t,allocator_type> hashes(get_allocator(),size());
  1110. auto_space<
  1111. node_impl_pointer,allocator_type> node_ptrs(get_allocator(),size());
  1112. std::size_t i=0,size_=size();
  1113. bool within_bucket=false;
  1114. BOOST_TRY{
  1115. for(;i!=size_;++i){
  1116. node_impl_pointer x=end_->prior();
  1117. /* only this can possibly throw */
  1118. std::size_t h=hash_(key(node_type::from_impl(x)->value()));
  1119. hashes.data()[i]=h;
  1120. node_ptrs.data()[i]=x;
  1121. within_bucket=!node_alg::unlink_last(end_);
  1122. node_alg::link(x,buckets_cpy.at(buckets_cpy.position(h)),cpy_end);
  1123. }
  1124. }
  1125. BOOST_CATCH(...){
  1126. if(i!=0){
  1127. std::size_t prev_buc=buckets.position(hashes.data()[i-1]);
  1128. if(!within_bucket)prev_buc=~prev_buc;
  1129. for(std::size_t j=i;j--;){
  1130. std::size_t buc=buckets.position(hashes.data()[j]);
  1131. node_impl_pointer x=node_ptrs.data()[j];
  1132. if(buc==prev_buc)node_alg::append(x,end_);
  1133. else node_alg::link(x,buckets.at(buc),end_);
  1134. prev_buc=buc;
  1135. }
  1136. }
  1137. BOOST_RETHROW;
  1138. }
  1139. BOOST_CATCH_END
  1140. }
  1141. end_->prior()=cpy_end->prior()!=cpy_end?cpy_end->prior():end_;
  1142. end_->next()=cpy_end->next();
  1143. end_->prior()->next()->prior()=end_->next()->prior()->prior()=end_;
  1144. buckets.swap(buckets_cpy);
  1145. calculate_max_load();
  1146. }
  1147. void unchecked_rehash(size_type n,hashed_non_unique_tag)
  1148. {
  1149. node_impl_type cpy_end_node;
  1150. node_impl_pointer cpy_end=node_impl_pointer(&cpy_end_node),
  1151. end_=header()->impl();
  1152. bucket_array_type buckets_cpy(get_allocator(),cpy_end,n);
  1153. if(size()!=0){
  1154. auto_space<
  1155. std::size_t,allocator_type> hashes(get_allocator(),size());
  1156. auto_space<
  1157. node_impl_pointer,allocator_type> node_ptrs(get_allocator(),size());
  1158. std::size_t i=0;
  1159. bool within_bucket=false;
  1160. BOOST_TRY{
  1161. for(;;++i){
  1162. node_impl_pointer x=end_->prior();
  1163. if(x==end_)break;
  1164. /* only this can possibly throw */
  1165. std::size_t h=hash_(key(node_type::from_impl(x)->value()));
  1166. hashes.data()[i]=h;
  1167. node_ptrs.data()[i]=x;
  1168. std::pair<node_impl_pointer,bool> p=
  1169. node_alg::unlink_last_group(end_);
  1170. node_alg::link_range(
  1171. p.first,x,buckets_cpy.at(buckets_cpy.position(h)),cpy_end);
  1172. within_bucket=!(p.second);
  1173. }
  1174. }
  1175. BOOST_CATCH(...){
  1176. if(i!=0){
  1177. std::size_t prev_buc=buckets.position(hashes.data()[i-1]);
  1178. if(!within_bucket)prev_buc=~prev_buc;
  1179. for(std::size_t j=i;j--;){
  1180. std::size_t buc=buckets.position(hashes.data()[j]);
  1181. node_impl_pointer x=node_ptrs.data()[j],
  1182. y=
  1183. x->prior()->next()!=node_impl_type::base_pointer_from(x)&&
  1184. x->prior()->next()->prior()!=x?
  1185. node_impl_type::pointer_from(x->prior()->next()):x;
  1186. node_alg::unlink_range(y,x);
  1187. if(buc==prev_buc)node_alg::append_range(y,x,end_);
  1188. else node_alg::link_range(y,x,buckets.at(buc),end_);
  1189. prev_buc=buc;
  1190. }
  1191. }
  1192. BOOST_RETHROW;
  1193. }
  1194. BOOST_CATCH_END
  1195. }
  1196. end_->prior()=cpy_end->prior()!=cpy_end?cpy_end->prior():end_;
  1197. end_->next()=cpy_end->next();
  1198. end_->prior()->next()->prior()=end_->next()->prior()->prior()=end_;
  1199. buckets.swap(buckets_cpy);
  1200. calculate_max_load();
  1201. }
  1202. bool in_place(node_impl_pointer x,key_param_type k,std::size_t buc)const
  1203. {
  1204. return in_place(x,k,buc,Category());
  1205. }
  1206. bool in_place(
  1207. node_impl_pointer x,key_param_type k,std::size_t buc,
  1208. hashed_unique_tag)const
  1209. {
  1210. bool found=false;
  1211. for(node_impl_pointer y=buckets.at(buc)->prior();
  1212. y!=node_impl_pointer(0);y=node_alg::after_local(y)){
  1213. if(y==x)found=true;
  1214. else if(eq_(k,key(node_type::from_impl(y)->value())))return false;
  1215. }
  1216. return found;
  1217. }
  1218. bool in_place(
  1219. node_impl_pointer x,key_param_type k,std::size_t buc,
  1220. hashed_non_unique_tag)const
  1221. {
  1222. bool found=false;
  1223. int range_size=0;
  1224. for(node_impl_pointer y=buckets.at(buc)->prior();y!=node_impl_pointer(0);){
  1225. if(node_alg::is_first_of_group(y)){ /* group of 3 or more */
  1226. if(y==x){
  1227. /* in place <-> equal to some other member of the group */
  1228. return eq_(
  1229. k,
  1230. key(node_type::from_impl(
  1231. node_impl_type::pointer_from(y->next()))->value()));
  1232. }
  1233. else{
  1234. node_impl_pointer z=
  1235. node_alg::after_local(y->next()->prior()); /* end of range */
  1236. if(eq_(k,key(node_type::from_impl(y)->value()))){
  1237. if(found)return false; /* x lies outside */
  1238. do{
  1239. if(y==x)return true;
  1240. y=node_alg::after_local(y);
  1241. }while(y!=z);
  1242. return false; /* x not found */
  1243. }
  1244. else{
  1245. if(range_size==1&&!found)return false;
  1246. if(range_size==2)return found;
  1247. range_size=0;
  1248. y=z; /* skip range (and potentially x, too, which is fine) */
  1249. }
  1250. }
  1251. }
  1252. else{ /* group of 1 or 2 */
  1253. if(y==x){
  1254. if(range_size==1)return true;
  1255. range_size=1;
  1256. found=true;
  1257. }
  1258. else if(eq_(k,key(node_type::from_impl(y)->value()))){
  1259. if(range_size==0&&found)return false;
  1260. if(range_size==1&&!found)return false;
  1261. if(range_size==2)return false;
  1262. ++range_size;
  1263. }
  1264. else{
  1265. if(range_size==1&&!found)return false;
  1266. if(range_size==2)return found;
  1267. range_size=0;
  1268. }
  1269. y=node_alg::after_local(y);
  1270. }
  1271. }
  1272. return found;
  1273. }
  1274. #if defined(BOOST_MULTI_INDEX_ENABLE_SAFE_MODE)
  1275. void detach_iterators(node_type* x)
  1276. {
  1277. iterator it=make_iterator(x);
  1278. safe_mode::detach_equivalent_iterators(it);
  1279. }
  1280. #endif
  1281. template<BOOST_MULTI_INDEX_TEMPLATE_PARAM_PACK>
  1282. std::pair<iterator,bool> emplace_impl(BOOST_MULTI_INDEX_FUNCTION_PARAM_PACK)
  1283. {
  1284. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  1285. std::pair<final_node_type*,bool>p=
  1286. this->final_emplace_(BOOST_MULTI_INDEX_FORWARD_PARAM_PACK);
  1287. return std::pair<iterator,bool>(make_iterator(p.first),p.second);
  1288. }
  1289. template<BOOST_MULTI_INDEX_TEMPLATE_PARAM_PACK>
  1290. iterator emplace_hint_impl(
  1291. iterator position,BOOST_MULTI_INDEX_FUNCTION_PARAM_PACK)
  1292. {
  1293. BOOST_MULTI_INDEX_CHECK_VALID_ITERATOR(position);
  1294. BOOST_MULTI_INDEX_CHECK_IS_OWNER(position,*this);
  1295. BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT;
  1296. std::pair<final_node_type*,bool>p=
  1297. this->final_emplace_hint_(
  1298. static_cast<final_node_type*>(position.get_node()),
  1299. BOOST_MULTI_INDEX_FORWARD_PARAM_PACK);
  1300. return make_iterator(p.first);
  1301. }
  1302. template<
  1303. typename CompatibleHash,typename CompatiblePred
  1304. >
  1305. iterator find(
  1306. const key_type& k,
  1307. const CompatibleHash& hash,const CompatiblePred& eq,mpl::true_)const
  1308. {
  1309. return find(k,hash,eq,mpl::false_());
  1310. }
  1311. template<
  1312. typename CompatibleKey,typename CompatibleHash,typename CompatiblePred
  1313. >
  1314. iterator find(
  1315. const CompatibleKey& k,
  1316. const CompatibleHash& hash,const CompatiblePred& eq,mpl::false_)const
  1317. {
  1318. std::size_t buc=buckets.position(hash(k));
  1319. for(node_impl_pointer x=buckets.at(buc)->prior();
  1320. x!=node_impl_pointer(0);x=node_alg::next_to_inspect(x)){
  1321. if(eq(k,key(node_type::from_impl(x)->value()))){
  1322. return make_iterator(node_type::from_impl(x));
  1323. }
  1324. }
  1325. return end();
  1326. }
  1327. template<
  1328. typename CompatibleHash,typename CompatiblePred
  1329. >
  1330. size_type count(
  1331. const key_type& k,
  1332. const CompatibleHash& hash,const CompatiblePred& eq,mpl::true_)const
  1333. {
  1334. return count(k,hash,eq,mpl::false_());
  1335. }
  1336. template<
  1337. typename CompatibleKey,typename CompatibleHash,typename CompatiblePred
  1338. >
  1339. size_type count(
  1340. const CompatibleKey& k,
  1341. const CompatibleHash& hash,const CompatiblePred& eq,mpl::false_)const
  1342. {
  1343. std::size_t buc=buckets.position(hash(k));
  1344. for(node_impl_pointer x=buckets.at(buc)->prior();
  1345. x!=node_impl_pointer(0);x=node_alg::next_to_inspect(x)){
  1346. if(eq(k,key(node_type::from_impl(x)->value()))){
  1347. size_type res=0;
  1348. node_impl_pointer y=end_of_range(x);
  1349. do{
  1350. ++res;
  1351. x=node_alg::after(x);
  1352. }while(x!=y);
  1353. return res;
  1354. }
  1355. }
  1356. return 0;
  1357. }
  1358. template<
  1359. typename CompatibleHash,typename CompatiblePred
  1360. >
  1361. std::pair<iterator,iterator> equal_range(
  1362. const key_type& k,
  1363. const CompatibleHash& hash,const CompatiblePred& eq,mpl::true_)const
  1364. {
  1365. return equal_range(k,hash,eq,mpl::false_());
  1366. }
  1367. template<
  1368. typename CompatibleKey,typename CompatibleHash,typename CompatiblePred
  1369. >
  1370. std::pair<iterator,iterator> equal_range(
  1371. const CompatibleKey& k,
  1372. const CompatibleHash& hash,const CompatiblePred& eq,mpl::false_)const
  1373. {
  1374. std::size_t buc=buckets.position(hash(k));
  1375. for(node_impl_pointer x=buckets.at(buc)->prior();
  1376. x!=node_impl_pointer(0);x=node_alg::next_to_inspect(x)){
  1377. if(eq(k,key(node_type::from_impl(x)->value()))){
  1378. return std::pair<iterator,iterator>(
  1379. make_iterator(node_type::from_impl(x)),
  1380. make_iterator(node_type::from_impl(end_of_range(x))));
  1381. }
  1382. }
  1383. return std::pair<iterator,iterator>(end(),end());
  1384. }
  1385. key_from_value key;
  1386. hasher hash_;
  1387. key_equal eq_;
  1388. bucket_array_type buckets;
  1389. float mlf;
  1390. size_type max_load;
  1391. #if defined(BOOST_MULTI_INDEX_ENABLE_INVARIANT_CHECKING)&&\
  1392. BOOST_WORKAROUND(__MWERKS__,<=0x3003)
  1393. #pragma parse_mfunc_templ reset
  1394. #endif
  1395. };
  1396. /* comparison */
  1397. template<
  1398. typename KeyFromValue,typename Hash,typename Pred,
  1399. typename SuperMeta,typename TagList,typename Category
  1400. >
  1401. bool operator==(
  1402. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  1403. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& y)
  1404. {
  1405. return x.equals(y);
  1406. }
  1407. template<
  1408. typename KeyFromValue,typename Hash,typename Pred,
  1409. typename SuperMeta,typename TagList,typename Category
  1410. >
  1411. bool operator!=(
  1412. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  1413. const hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& y)
  1414. {
  1415. return !(x==y);
  1416. }
  1417. /* specialized algorithms */
  1418. template<
  1419. typename KeyFromValue,typename Hash,typename Pred,
  1420. typename SuperMeta,typename TagList,typename Category
  1421. >
  1422. void swap(
  1423. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& x,
  1424. hashed_index<KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>& y)
  1425. {
  1426. x.swap(y);
  1427. }
  1428. } /* namespace multi_index::detail */
  1429. /* hashed index specifiers */
  1430. template<typename Arg1,typename Arg2,typename Arg3,typename Arg4>
  1431. struct hashed_unique
  1432. {
  1433. typedef typename detail::hashed_index_args<
  1434. Arg1,Arg2,Arg3,Arg4> index_args;
  1435. typedef typename index_args::tag_list_type::type tag_list_type;
  1436. typedef typename index_args::key_from_value_type key_from_value_type;
  1437. typedef typename index_args::hash_type hash_type;
  1438. typedef typename index_args::pred_type pred_type;
  1439. template<typename Super>
  1440. struct node_class
  1441. {
  1442. typedef detail::hashed_index_node<Super,detail::hashed_unique_tag> type;
  1443. };
  1444. template<typename SuperMeta>
  1445. struct index_class
  1446. {
  1447. typedef detail::hashed_index<
  1448. key_from_value_type,hash_type,pred_type,
  1449. SuperMeta,tag_list_type,detail::hashed_unique_tag> type;
  1450. };
  1451. };
  1452. template<typename Arg1,typename Arg2,typename Arg3,typename Arg4>
  1453. struct hashed_non_unique
  1454. {
  1455. typedef typename detail::hashed_index_args<
  1456. Arg1,Arg2,Arg3,Arg4> index_args;
  1457. typedef typename index_args::tag_list_type::type tag_list_type;
  1458. typedef typename index_args::key_from_value_type key_from_value_type;
  1459. typedef typename index_args::hash_type hash_type;
  1460. typedef typename index_args::pred_type pred_type;
  1461. template<typename Super>
  1462. struct node_class
  1463. {
  1464. typedef detail::hashed_index_node<
  1465. Super,detail::hashed_non_unique_tag> type;
  1466. };
  1467. template<typename SuperMeta>
  1468. struct index_class
  1469. {
  1470. typedef detail::hashed_index<
  1471. key_from_value_type,hash_type,pred_type,
  1472. SuperMeta,tag_list_type,detail::hashed_non_unique_tag> type;
  1473. };
  1474. };
  1475. } /* namespace multi_index */
  1476. } /* namespace boost */
  1477. /* Boost.Foreach compatibility */
  1478. template<
  1479. typename KeyFromValue,typename Hash,typename Pred,
  1480. typename SuperMeta,typename TagList,typename Category
  1481. >
  1482. inline boost::mpl::true_* boost_foreach_is_noncopyable(
  1483. boost::multi_index::detail::hashed_index<
  1484. KeyFromValue,Hash,Pred,SuperMeta,TagList,Category>*&,
  1485. boost_foreach_argument_dependent_lookup_hack)
  1486. {
  1487. return 0;
  1488. }
  1489. #undef BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT
  1490. #undef BOOST_MULTI_INDEX_HASHED_INDEX_CHECK_INVARIANT_OF
  1491. #endif