unique.hpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628
  1. // Copyright (C) 2003-2004 Jeremy B. Maitin-Shepard.
  2. // Copyright (C) 2005-2011 Daniel James
  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. #ifndef BOOST_UNORDERED_DETAIL_UNIQUE_HPP_INCLUDED
  6. #define BOOST_UNORDERED_DETAIL_UNIQUE_HPP_INCLUDED
  7. #include <boost/config.hpp>
  8. #if defined(BOOST_HAS_PRAGMA_ONCE)
  9. #pragma once
  10. #endif
  11. #include <boost/unordered/detail/table.hpp>
  12. #include <boost/unordered/detail/extract_key.hpp>
  13. #include <boost/throw_exception.hpp>
  14. #include <stdexcept>
  15. namespace boost { namespace unordered { namespace detail {
  16. template <typename A, typename T> struct unique_node;
  17. template <typename T> struct ptr_node;
  18. template <typename Types> struct table_impl;
  19. template <typename A, typename T>
  20. struct unique_node :
  21. boost::unordered::detail::value_base<T>
  22. {
  23. typedef typename ::boost::unordered::detail::rebind_wrap<
  24. A, unique_node<A, T> >::type allocator;
  25. typedef typename ::boost::unordered::detail::
  26. allocator_traits<allocator>::pointer node_pointer;
  27. typedef node_pointer link_pointer;
  28. link_pointer next_;
  29. std::size_t hash_;
  30. unique_node() :
  31. next_(),
  32. hash_(0)
  33. {}
  34. void init(node_pointer)
  35. {
  36. }
  37. private:
  38. unique_node& operator=(unique_node const&);
  39. };
  40. template <typename T>
  41. struct ptr_node :
  42. boost::unordered::detail::ptr_bucket
  43. {
  44. typedef T value_type;
  45. typedef boost::unordered::detail::ptr_bucket bucket_base;
  46. typedef ptr_node<T>* node_pointer;
  47. typedef ptr_bucket* link_pointer;
  48. std::size_t hash_;
  49. boost::unordered::detail::value_base<T> value_base_;
  50. ptr_node() :
  51. bucket_base(),
  52. hash_(0)
  53. {}
  54. void init(node_pointer)
  55. {
  56. }
  57. void* address() { return value_base_.address(); }
  58. value_type& value() { return value_base_.value(); }
  59. value_type* value_ptr() { return value_base_.value_ptr(); }
  60. private:
  61. ptr_node& operator=(ptr_node const&);
  62. };
  63. // If the allocator uses raw pointers use ptr_node
  64. // Otherwise use node.
  65. template <typename A, typename T, typename NodePtr, typename BucketPtr>
  66. struct pick_node2
  67. {
  68. typedef boost::unordered::detail::unique_node<A, T> node;
  69. typedef typename boost::unordered::detail::allocator_traits<
  70. typename boost::unordered::detail::rebind_wrap<A, node>::type
  71. >::pointer node_pointer;
  72. typedef boost::unordered::detail::bucket<node_pointer> bucket;
  73. typedef node_pointer link_pointer;
  74. };
  75. template <typename A, typename T>
  76. struct pick_node2<A, T,
  77. boost::unordered::detail::ptr_node<T>*,
  78. boost::unordered::detail::ptr_bucket*>
  79. {
  80. typedef boost::unordered::detail::ptr_node<T> node;
  81. typedef boost::unordered::detail::ptr_bucket bucket;
  82. typedef bucket* link_pointer;
  83. };
  84. template <typename A, typename T>
  85. struct pick_node
  86. {
  87. typedef boost::unordered::detail::allocator_traits<
  88. typename boost::unordered::detail::rebind_wrap<A,
  89. boost::unordered::detail::ptr_node<T> >::type
  90. > tentative_node_traits;
  91. typedef boost::unordered::detail::allocator_traits<
  92. typename boost::unordered::detail::rebind_wrap<A,
  93. boost::unordered::detail::ptr_bucket >::type
  94. > tentative_bucket_traits;
  95. typedef pick_node2<A, T,
  96. typename tentative_node_traits::pointer,
  97. typename tentative_bucket_traits::pointer> pick;
  98. typedef typename pick::node node;
  99. typedef typename pick::bucket bucket;
  100. typedef typename pick::link_pointer link_pointer;
  101. };
  102. template <typename A, typename T, typename H, typename P>
  103. struct set
  104. {
  105. typedef boost::unordered::detail::set<A, T, H, P> types;
  106. typedef A allocator;
  107. typedef T value_type;
  108. typedef H hasher;
  109. typedef P key_equal;
  110. typedef T key_type;
  111. typedef boost::unordered::detail::allocator_traits<allocator> traits;
  112. typedef boost::unordered::detail::pick_node<allocator, value_type> pick;
  113. typedef typename pick::node node;
  114. typedef typename pick::bucket bucket;
  115. typedef typename pick::link_pointer link_pointer;
  116. typedef boost::unordered::detail::table_impl<types> table;
  117. typedef boost::unordered::detail::set_extractor<value_type> extractor;
  118. typedef typename boost::unordered::detail::pick_policy<T>::type policy;
  119. };
  120. template <typename A, typename K, typename M, typename H, typename P>
  121. struct map
  122. {
  123. typedef boost::unordered::detail::map<A, K, M, H, P> types;
  124. typedef A allocator;
  125. typedef std::pair<K const, M> value_type;
  126. typedef H hasher;
  127. typedef P key_equal;
  128. typedef K key_type;
  129. typedef boost::unordered::detail::allocator_traits<allocator>
  130. traits;
  131. typedef boost::unordered::detail::pick_node<allocator, value_type> pick;
  132. typedef typename pick::node node;
  133. typedef typename pick::bucket bucket;
  134. typedef typename pick::link_pointer link_pointer;
  135. typedef boost::unordered::detail::table_impl<types> table;
  136. typedef boost::unordered::detail::map_extractor<key_type, value_type>
  137. extractor;
  138. typedef typename boost::unordered::detail::pick_policy<K>::type policy;
  139. };
  140. template <typename Types>
  141. struct table_impl : boost::unordered::detail::table<Types>
  142. {
  143. typedef boost::unordered::detail::table<Types> table;
  144. typedef typename table::value_type value_type;
  145. typedef typename table::bucket bucket;
  146. typedef typename table::policy policy;
  147. typedef typename table::node_pointer node_pointer;
  148. typedef typename table::node_allocator node_allocator;
  149. typedef typename table::node_allocator_traits node_allocator_traits;
  150. typedef typename table::bucket_pointer bucket_pointer;
  151. typedef typename table::link_pointer link_pointer;
  152. typedef typename table::hasher hasher;
  153. typedef typename table::key_equal key_equal;
  154. typedef typename table::key_type key_type;
  155. typedef typename table::node_constructor node_constructor;
  156. typedef typename table::extractor extractor;
  157. typedef typename table::iterator iterator;
  158. typedef typename table::c_iterator c_iterator;
  159. typedef std::pair<iterator, bool> emplace_return;
  160. // Constructors
  161. table_impl(std::size_t n,
  162. hasher const& hf,
  163. key_equal const& eq,
  164. node_allocator const& a)
  165. : table(n, hf, eq, a)
  166. {}
  167. table_impl(table_impl const& x)
  168. : table(x, node_allocator_traits::
  169. select_on_container_copy_construction(x.node_alloc()))
  170. {
  171. this->init(x);
  172. }
  173. table_impl(table_impl const& x,
  174. node_allocator const& a)
  175. : table(x, a)
  176. {
  177. this->init(x);
  178. }
  179. table_impl(table_impl& x,
  180. boost::unordered::detail::move_tag m)
  181. : table(x, m)
  182. {}
  183. table_impl(table_impl& x,
  184. node_allocator const& a,
  185. boost::unordered::detail::move_tag m)
  186. : table(x, a, m)
  187. {
  188. this->move_init(x);
  189. }
  190. // Accessors
  191. template <class Key, class Pred>
  192. iterator find_node_impl(
  193. std::size_t key_hash,
  194. Key const& k,
  195. Pred const& eq) const
  196. {
  197. std::size_t bucket_index = this->hash_to_bucket(key_hash);
  198. iterator n = this->begin(bucket_index);
  199. for (;;)
  200. {
  201. if (!n.node_) return n;
  202. std::size_t node_hash = n.node_->hash_;
  203. if (key_hash == node_hash)
  204. {
  205. if (eq(k, this->get_key(*n)))
  206. return n;
  207. }
  208. else
  209. {
  210. if (this->hash_to_bucket(node_hash) != bucket_index)
  211. return iterator();
  212. }
  213. ++n;
  214. }
  215. }
  216. std::size_t count(key_type const& k) const
  217. {
  218. return this->find_node(k).node_ ? 1 : 0;
  219. }
  220. value_type& at(key_type const& k) const
  221. {
  222. if (this->size_) {
  223. iterator it = this->find_node(k);
  224. if (it.node_) return *it;
  225. }
  226. boost::throw_exception(
  227. std::out_of_range("Unable to find key in unordered_map."));
  228. }
  229. std::pair<iterator, iterator>
  230. equal_range(key_type const& k) const
  231. {
  232. iterator n = this->find_node(k);
  233. iterator n2 = n;
  234. if (n2.node_) ++n2;
  235. return std::make_pair(n, n2);
  236. }
  237. // equals
  238. bool equals(table_impl const& other) const
  239. {
  240. if(this->size_ != other.size_) return false;
  241. for(iterator n1 = this->begin(); n1.node_; ++n1)
  242. {
  243. iterator n2 = other.find_matching_node(n1);
  244. if (!n2.node_ || *n1 != *n2)
  245. return false;
  246. }
  247. return true;
  248. }
  249. // Emplace/Insert
  250. inline iterator add_node(
  251. node_constructor& a,
  252. std::size_t key_hash)
  253. {
  254. node_pointer n = a.release();
  255. n->hash_ = key_hash;
  256. bucket_pointer b = this->get_bucket(this->hash_to_bucket(key_hash));
  257. if (!b->next_)
  258. {
  259. link_pointer start_node = this->get_previous_start();
  260. if (start_node->next_) {
  261. this->get_bucket(this->hash_to_bucket(
  262. static_cast<node_pointer>(start_node->next_)->hash_)
  263. )->next_ = n;
  264. }
  265. b->next_ = start_node;
  266. n->next_ = start_node->next_;
  267. start_node->next_ = n;
  268. }
  269. else
  270. {
  271. n->next_ = b->next_->next_;
  272. b->next_->next_ = n;
  273. }
  274. ++this->size_;
  275. return iterator(n);
  276. }
  277. value_type& operator[](key_type const& k)
  278. {
  279. std::size_t key_hash = this->hash(k);
  280. iterator pos = this->find_node(key_hash, k);
  281. if (pos.node_) return *pos;
  282. // Create the node before rehashing in case it throws an
  283. // exception (need strong safety in such a case).
  284. node_constructor a(this->node_alloc());
  285. a.construct_with_value(BOOST_UNORDERED_EMPLACE_ARGS3(
  286. boost::unordered::piecewise_construct,
  287. boost::make_tuple(k),
  288. boost::make_tuple()));
  289. this->reserve_for_insert(this->size_ + 1);
  290. return *add_node(a, key_hash);
  291. }
  292. #if defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  293. # if defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  294. emplace_return emplace(boost::unordered::detail::emplace_args1<
  295. boost::unordered::detail::please_ignore_this_overload> const&)
  296. {
  297. BOOST_ASSERT(false);
  298. return emplace_return(this->begin(), false);
  299. }
  300. # else
  301. emplace_return emplace(
  302. boost::unordered::detail::please_ignore_this_overload const&)
  303. {
  304. BOOST_ASSERT(false);
  305. return emplace_return(this->begin(), false);
  306. }
  307. # endif
  308. #endif
  309. template <BOOST_UNORDERED_EMPLACE_TEMPLATE>
  310. emplace_return emplace(BOOST_UNORDERED_EMPLACE_ARGS)
  311. {
  312. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  313. return emplace_impl(
  314. extractor::extract(BOOST_UNORDERED_EMPLACE_FORWARD),
  315. BOOST_UNORDERED_EMPLACE_FORWARD);
  316. #else
  317. return emplace_impl(
  318. extractor::extract(args.a0, args.a1),
  319. BOOST_UNORDERED_EMPLACE_FORWARD);
  320. #endif
  321. }
  322. #if defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  323. template <typename A0>
  324. emplace_return emplace(
  325. boost::unordered::detail::emplace_args1<A0> const& args)
  326. {
  327. return emplace_impl(extractor::extract(args.a0), args);
  328. }
  329. #endif
  330. template <BOOST_UNORDERED_EMPLACE_TEMPLATE>
  331. emplace_return emplace_impl(key_type const& k,
  332. BOOST_UNORDERED_EMPLACE_ARGS)
  333. {
  334. std::size_t key_hash = this->hash(k);
  335. iterator pos = this->find_node(key_hash, k);
  336. if (pos.node_) return emplace_return(pos, false);
  337. // Create the node before rehashing in case it throws an
  338. // exception (need strong safety in such a case).
  339. node_constructor a(this->node_alloc());
  340. a.construct_with_value(BOOST_UNORDERED_EMPLACE_FORWARD);
  341. // reserve has basic exception safety if the hash function
  342. // throws, strong otherwise.
  343. this->reserve_for_insert(this->size_ + 1);
  344. return emplace_return(this->add_node(a, key_hash), true);
  345. }
  346. emplace_return emplace_impl_with_node(node_constructor& a)
  347. {
  348. key_type const& k = this->get_key(a.value());
  349. std::size_t key_hash = this->hash(k);
  350. iterator pos = this->find_node(key_hash, k);
  351. if (pos.node_) return emplace_return(pos, false);
  352. // reserve has basic exception safety if the hash function
  353. // throws, strong otherwise.
  354. this->reserve_for_insert(this->size_ + 1);
  355. return emplace_return(this->add_node(a, key_hash), true);
  356. }
  357. template <BOOST_UNORDERED_EMPLACE_TEMPLATE>
  358. emplace_return emplace_impl(no_key, BOOST_UNORDERED_EMPLACE_ARGS)
  359. {
  360. // Don't have a key, so construct the node first in order
  361. // to be able to lookup the position.
  362. node_constructor a(this->node_alloc());
  363. a.construct_with_value(BOOST_UNORDERED_EMPLACE_FORWARD);
  364. return emplace_impl_with_node(a);
  365. }
  366. ////////////////////////////////////////////////////////////////////////
  367. // Insert range methods
  368. //
  369. // if hash function throws, or inserting > 1 element, basic exception
  370. // safety strong otherwise
  371. template <class InputIt>
  372. void insert_range(InputIt i, InputIt j)
  373. {
  374. if(i != j)
  375. return insert_range_impl(extractor::extract(*i), i, j);
  376. }
  377. template <class InputIt>
  378. void insert_range_impl(key_type const& k, InputIt i, InputIt j)
  379. {
  380. node_constructor a(this->node_alloc());
  381. insert_range_impl2(a, k, i, j);
  382. while(++i != j) {
  383. // Note: can't use get_key as '*i' might not be value_type - it
  384. // could be a pair with first_types as key_type without const or
  385. // a different second_type.
  386. //
  387. // TODO: Might be worth storing the value_type instead of the
  388. // key here. Could be more efficient if '*i' is expensive. Could
  389. // be less efficient if copying the full value_type is
  390. // expensive.
  391. insert_range_impl2(a, extractor::extract(*i), i, j);
  392. }
  393. }
  394. template <class InputIt>
  395. void insert_range_impl2(node_constructor& a, key_type const& k,
  396. InputIt i, InputIt j)
  397. {
  398. // No side effects in this initial code
  399. std::size_t key_hash = this->hash(k);
  400. iterator pos = this->find_node(key_hash, k);
  401. if (!pos.node_) {
  402. a.construct_with_value2(*i);
  403. if(this->size_ + 1 > this->max_load_)
  404. this->reserve_for_insert(this->size_ +
  405. boost::unordered::detail::insert_size(i, j));
  406. // Nothing after this point can throw.
  407. this->add_node(a, key_hash);
  408. }
  409. }
  410. template <class InputIt>
  411. void insert_range_impl(no_key, InputIt i, InputIt j)
  412. {
  413. node_constructor a(this->node_alloc());
  414. do {
  415. a.construct_with_value2(*i);
  416. emplace_impl_with_node(a);
  417. } while(++i != j);
  418. }
  419. ////////////////////////////////////////////////////////////////////////
  420. // Erase
  421. //
  422. // no throw
  423. std::size_t erase_key(key_type const& k)
  424. {
  425. if(!this->size_) return 0;
  426. std::size_t key_hash = this->hash(k);
  427. std::size_t bucket_index = this->hash_to_bucket(key_hash);
  428. link_pointer prev = this->get_previous_start(bucket_index);
  429. if (!prev) return 0;
  430. for (;;)
  431. {
  432. if (!prev->next_) return 0;
  433. std::size_t node_hash =
  434. static_cast<node_pointer>(prev->next_)->hash_;
  435. if (this->hash_to_bucket(node_hash) != bucket_index)
  436. return 0;
  437. if (node_hash == key_hash &&
  438. this->key_eq()(k, this->get_key(
  439. static_cast<node_pointer>(prev->next_)->value())))
  440. break;
  441. prev = prev->next_;
  442. }
  443. link_pointer end = static_cast<node_pointer>(prev->next_)->next_;
  444. std::size_t deleted_count = this->delete_nodes(prev, end);
  445. this->fix_bucket(bucket_index, prev);
  446. return deleted_count;
  447. }
  448. iterator erase(c_iterator r)
  449. {
  450. BOOST_ASSERT(r.node_);
  451. iterator next(r.node_);
  452. ++next;
  453. erase_nodes(r.node_, next.node_);
  454. return next;
  455. }
  456. iterator erase_range(c_iterator r1, c_iterator r2)
  457. {
  458. if (r1 == r2) return iterator(r2.node_);
  459. erase_nodes(r1.node_, r2.node_);
  460. return iterator(r2.node_);
  461. }
  462. void erase_nodes(node_pointer i, node_pointer j)
  463. {
  464. std::size_t bucket_index = this->hash_to_bucket(i->hash_);
  465. // Find the node before i.
  466. link_pointer prev = this->get_previous_start(bucket_index);
  467. while(prev->next_ != i) prev = prev->next_;
  468. // Delete the nodes.
  469. do {
  470. this->delete_node(prev);
  471. bucket_index = this->fix_bucket(bucket_index, prev);
  472. } while (prev->next_ != j);
  473. }
  474. ////////////////////////////////////////////////////////////////////////
  475. // fill_buckets
  476. template <class NodeCreator>
  477. static void fill_buckets(iterator n, table& dst,
  478. NodeCreator& creator)
  479. {
  480. link_pointer prev = dst.get_previous_start();
  481. while (n.node_) {
  482. node_pointer node = creator.create(*n);
  483. node->hash_ = n.node_->hash_;
  484. prev->next_ = node;
  485. ++dst.size_;
  486. ++n;
  487. prev = place_in_bucket(dst, prev);
  488. }
  489. }
  490. // strong otherwise exception safety
  491. void rehash_impl(std::size_t num_buckets)
  492. {
  493. BOOST_ASSERT(this->buckets_);
  494. this->create_buckets(num_buckets);
  495. link_pointer prev = this->get_previous_start();
  496. while (prev->next_)
  497. prev = place_in_bucket(*this, prev);
  498. }
  499. // Iterate through the nodes placing them in the correct buckets.
  500. // pre: prev->next_ is not null.
  501. static link_pointer place_in_bucket(table& dst, link_pointer prev)
  502. {
  503. node_pointer n = static_cast<node_pointer>(prev->next_);
  504. bucket_pointer b = dst.get_bucket(dst.hash_to_bucket(n->hash_));
  505. if (!b->next_) {
  506. b->next_ = prev;
  507. return n;
  508. }
  509. else {
  510. prev->next_ = n->next_;
  511. n->next_ = b->next_->next_;
  512. b->next_->next_ = n;
  513. return prev;
  514. }
  515. }
  516. };
  517. }}}
  518. #endif