table.hpp 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873
  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_ALL_HPP_INCLUDED
  6. #define BOOST_UNORDERED_DETAIL_ALL_HPP_INCLUDED
  7. #include <boost/config.hpp>
  8. #if defined(BOOST_HAS_PRAGMA_ONCE)
  9. #pragma once
  10. #endif
  11. #include <boost/unordered/detail/buckets.hpp>
  12. #include <boost/unordered/detail/util.hpp>
  13. #include <boost/type_traits/aligned_storage.hpp>
  14. #include <boost/type_traits/alignment_of.hpp>
  15. #include <cmath>
  16. #if defined(BOOST_MSVC)
  17. #pragma warning(push)
  18. #pragma warning(disable:4127) // conditional expression is constant
  19. #endif
  20. #if defined(BOOST_UNORDERED_DEPRECATED_EQUALITY)
  21. #if defined(__EDG__)
  22. #elif defined(_MSC_VER) || defined(__BORLANDC__) || defined(__DMC__)
  23. #pragma message("Warning: BOOST_UNORDERED_DEPRECATED_EQUALITY is no longer supported.")
  24. #elif defined(__GNUC__) || defined(__HP_aCC) || \
  25. defined(__SUNPRO_CC) || defined(__IBMCPP__)
  26. #warning "BOOST_UNORDERED_DEPRECATED_EQUALITY is no longer supported."
  27. #endif
  28. #endif
  29. namespace boost { namespace unordered { namespace detail {
  30. ////////////////////////////////////////////////////////////////////////////
  31. // convert double to std::size_t
  32. inline std::size_t double_to_size(double f)
  33. {
  34. return f >= static_cast<double>(
  35. (std::numeric_limits<std::size_t>::max)()) ?
  36. (std::numeric_limits<std::size_t>::max)() :
  37. static_cast<std::size_t>(f);
  38. }
  39. // The space used to store values in a node.
  40. template <typename ValueType>
  41. struct value_base
  42. {
  43. typedef ValueType value_type;
  44. typename boost::aligned_storage<
  45. sizeof(value_type),
  46. boost::alignment_of<value_type>::value>::type data_;
  47. value_base() :
  48. data_()
  49. {}
  50. void* address() {
  51. return this;
  52. }
  53. value_type& value() {
  54. return *(ValueType*) this;
  55. }
  56. value_type* value_ptr() {
  57. return (ValueType*) this;
  58. }
  59. private:
  60. value_base& operator=(value_base const&);
  61. };
  62. template <typename NodeAlloc>
  63. struct copy_nodes
  64. {
  65. typedef boost::unordered::detail::allocator_traits<NodeAlloc>
  66. node_allocator_traits;
  67. node_constructor<NodeAlloc> constructor;
  68. explicit copy_nodes(NodeAlloc& a) : constructor(a) {}
  69. typename node_allocator_traits::pointer create(
  70. typename node_allocator_traits::value_type::value_type const& v)
  71. {
  72. constructor.construct_with_value2(v);
  73. return constructor.release();
  74. }
  75. };
  76. template <typename NodeAlloc>
  77. struct move_nodes
  78. {
  79. typedef boost::unordered::detail::allocator_traits<NodeAlloc>
  80. node_allocator_traits;
  81. node_constructor<NodeAlloc> constructor;
  82. explicit move_nodes(NodeAlloc& a) : constructor(a) {}
  83. typename node_allocator_traits::pointer create(
  84. typename node_allocator_traits::value_type::value_type& v)
  85. {
  86. constructor.construct_with_value2(boost::move(v));
  87. return constructor.release();
  88. }
  89. };
  90. template <typename Buckets>
  91. struct assign_nodes
  92. {
  93. node_holder<typename Buckets::node_allocator> holder;
  94. explicit assign_nodes(Buckets& b) : holder(b) {}
  95. typename Buckets::node_pointer create(
  96. typename Buckets::value_type const& v)
  97. {
  98. return holder.copy_of(v);
  99. }
  100. };
  101. template <typename Buckets>
  102. struct move_assign_nodes
  103. {
  104. node_holder<typename Buckets::node_allocator> holder;
  105. explicit move_assign_nodes(Buckets& b) : holder(b) {}
  106. typename Buckets::node_pointer create(
  107. typename Buckets::value_type& v)
  108. {
  109. return holder.move_copy_of(v);
  110. }
  111. };
  112. template <typename Types>
  113. struct table :
  114. boost::unordered::detail::functions<
  115. typename Types::hasher,
  116. typename Types::key_equal>
  117. {
  118. private:
  119. table(table const&);
  120. table& operator=(table const&);
  121. public:
  122. typedef typename Types::node node;
  123. typedef typename Types::bucket bucket;
  124. typedef typename Types::hasher hasher;
  125. typedef typename Types::key_equal key_equal;
  126. typedef typename Types::key_type key_type;
  127. typedef typename Types::extractor extractor;
  128. typedef typename Types::value_type value_type;
  129. typedef typename Types::table table_impl;
  130. typedef typename Types::link_pointer link_pointer;
  131. typedef typename Types::policy policy;
  132. typedef boost::unordered::detail::functions<
  133. typename Types::hasher,
  134. typename Types::key_equal> functions;
  135. typedef typename functions::set_hash_functions set_hash_functions;
  136. typedef typename Types::allocator allocator;
  137. typedef typename boost::unordered::detail::
  138. rebind_wrap<allocator, node>::type node_allocator;
  139. typedef typename boost::unordered::detail::
  140. rebind_wrap<allocator, bucket>::type bucket_allocator;
  141. typedef boost::unordered::detail::allocator_traits<node_allocator>
  142. node_allocator_traits;
  143. typedef boost::unordered::detail::allocator_traits<bucket_allocator>
  144. bucket_allocator_traits;
  145. typedef typename node_allocator_traits::pointer
  146. node_pointer;
  147. typedef typename node_allocator_traits::const_pointer
  148. const_node_pointer;
  149. typedef typename bucket_allocator_traits::pointer
  150. bucket_pointer;
  151. typedef boost::unordered::detail::node_constructor<node_allocator>
  152. node_constructor;
  153. typedef boost::unordered::iterator_detail::
  154. iterator<node> iterator;
  155. typedef boost::unordered::iterator_detail::
  156. c_iterator<node> c_iterator;
  157. typedef boost::unordered::iterator_detail::
  158. l_iterator<node, policy> l_iterator;
  159. typedef boost::unordered::iterator_detail::
  160. cl_iterator<node, policy> cl_iterator;
  161. ////////////////////////////////////////////////////////////////////////
  162. // Members
  163. boost::unordered::detail::compressed<bucket_allocator, node_allocator>
  164. allocators_;
  165. std::size_t bucket_count_;
  166. std::size_t size_;
  167. float mlf_;
  168. std::size_t max_load_;
  169. bucket_pointer buckets_;
  170. ////////////////////////////////////////////////////////////////////////
  171. // Data access
  172. bucket_allocator const& bucket_alloc() const
  173. {
  174. return allocators_.first();
  175. }
  176. node_allocator const& node_alloc() const
  177. {
  178. return allocators_.second();
  179. }
  180. bucket_allocator& bucket_alloc()
  181. {
  182. return allocators_.first();
  183. }
  184. node_allocator& node_alloc()
  185. {
  186. return allocators_.second();
  187. }
  188. std::size_t max_bucket_count() const
  189. {
  190. // -1 to account for the start bucket.
  191. return policy::prev_bucket_count(
  192. bucket_allocator_traits::max_size(bucket_alloc()) - 1);
  193. }
  194. bucket_pointer get_bucket(std::size_t bucket_index) const
  195. {
  196. BOOST_ASSERT(buckets_);
  197. return buckets_ + static_cast<std::ptrdiff_t>(bucket_index);
  198. }
  199. link_pointer get_previous_start() const
  200. {
  201. return get_bucket(bucket_count_)->first_from_start();
  202. }
  203. link_pointer get_previous_start(std::size_t bucket_index) const
  204. {
  205. return get_bucket(bucket_index)->next_;
  206. }
  207. iterator begin() const
  208. {
  209. return size_ ? iterator(get_previous_start()->next_) : iterator();
  210. }
  211. iterator begin(std::size_t bucket_index) const
  212. {
  213. if (!size_) return iterator();
  214. link_pointer prev = get_previous_start(bucket_index);
  215. return prev ? iterator(prev->next_) : iterator();
  216. }
  217. std::size_t hash_to_bucket(std::size_t hash_value) const
  218. {
  219. return policy::to_bucket(bucket_count_, hash_value);
  220. }
  221. float load_factor() const
  222. {
  223. BOOST_ASSERT(bucket_count_ != 0);
  224. return static_cast<float>(size_)
  225. / static_cast<float>(bucket_count_);
  226. }
  227. std::size_t bucket_size(std::size_t index) const
  228. {
  229. iterator it = begin(index);
  230. if (!it.node_) return 0;
  231. std::size_t count = 0;
  232. while(it.node_ && hash_to_bucket(it.node_->hash_) == index)
  233. {
  234. ++count;
  235. ++it;
  236. }
  237. return count;
  238. }
  239. ////////////////////////////////////////////////////////////////////////
  240. // Load methods
  241. std::size_t max_size() const
  242. {
  243. using namespace std;
  244. // size < mlf_ * count
  245. return boost::unordered::detail::double_to_size(ceil(
  246. static_cast<double>(mlf_) *
  247. static_cast<double>(max_bucket_count())
  248. )) - 1;
  249. }
  250. void recalculate_max_load()
  251. {
  252. using namespace std;
  253. // From 6.3.1/13:
  254. // Only resize when size >= mlf_ * count
  255. max_load_ = buckets_ ? boost::unordered::detail::double_to_size(ceil(
  256. static_cast<double>(mlf_) *
  257. static_cast<double>(bucket_count_)
  258. )) : 0;
  259. }
  260. void max_load_factor(float z)
  261. {
  262. BOOST_ASSERT(z > 0);
  263. mlf_ = (std::max)(z, minimum_max_load_factor);
  264. recalculate_max_load();
  265. }
  266. std::size_t min_buckets_for_size(std::size_t size) const
  267. {
  268. BOOST_ASSERT(mlf_ >= minimum_max_load_factor);
  269. using namespace std;
  270. // From 6.3.1/13:
  271. // size < mlf_ * count
  272. // => count > size / mlf_
  273. //
  274. // Or from rehash post-condition:
  275. // count > size / mlf_
  276. return policy::new_bucket_count(
  277. boost::unordered::detail::double_to_size(floor(
  278. static_cast<double>(size) /
  279. static_cast<double>(mlf_)) + 1));
  280. }
  281. ////////////////////////////////////////////////////////////////////////
  282. // Constructors
  283. table(std::size_t num_buckets,
  284. hasher const& hf,
  285. key_equal const& eq,
  286. node_allocator const& a) :
  287. functions(hf, eq),
  288. allocators_(a,a),
  289. bucket_count_(policy::new_bucket_count(num_buckets)),
  290. size_(0),
  291. mlf_(1.0f),
  292. max_load_(0),
  293. buckets_()
  294. {}
  295. table(table const& x, node_allocator const& a) :
  296. functions(x),
  297. allocators_(a,a),
  298. bucket_count_(x.min_buckets_for_size(x.size_)),
  299. size_(0),
  300. mlf_(x.mlf_),
  301. max_load_(0),
  302. buckets_()
  303. {}
  304. table(table& x, boost::unordered::detail::move_tag m) :
  305. functions(x, m),
  306. allocators_(x.allocators_, m),
  307. bucket_count_(x.bucket_count_),
  308. size_(x.size_),
  309. mlf_(x.mlf_),
  310. max_load_(x.max_load_),
  311. buckets_(x.buckets_)
  312. {
  313. x.buckets_ = bucket_pointer();
  314. x.size_ = 0;
  315. x.max_load_ = 0;
  316. }
  317. table(table& x, node_allocator const& a,
  318. boost::unordered::detail::move_tag m) :
  319. functions(x, m),
  320. allocators_(a, a),
  321. bucket_count_(x.bucket_count_),
  322. size_(0),
  323. mlf_(x.mlf_),
  324. max_load_(x.max_load_),
  325. buckets_()
  326. {}
  327. ////////////////////////////////////////////////////////////////////////
  328. // Initialisation.
  329. void init(table const& x)
  330. {
  331. if (x.size_) {
  332. create_buckets(bucket_count_);
  333. copy_nodes<node_allocator> node_creator(node_alloc());
  334. table_impl::fill_buckets(x.begin(), *this, node_creator);
  335. }
  336. }
  337. void move_init(table& x)
  338. {
  339. if(node_alloc() == x.node_alloc()) {
  340. move_buckets_from(x);
  341. }
  342. else if(x.size_) {
  343. // TODO: Could pick new bucket size?
  344. create_buckets(bucket_count_);
  345. move_nodes<node_allocator> node_creator(node_alloc());
  346. node_holder<node_allocator> nodes(x);
  347. table_impl::fill_buckets(nodes.begin(), *this, node_creator);
  348. }
  349. }
  350. ////////////////////////////////////////////////////////////////////////
  351. // Create buckets
  352. void create_buckets(std::size_t new_count)
  353. {
  354. boost::unordered::detail::array_constructor<bucket_allocator>
  355. constructor(bucket_alloc());
  356. // Creates an extra bucket to act as the start node.
  357. constructor.construct(bucket(), new_count + 1);
  358. if (buckets_)
  359. {
  360. // Copy the nodes to the new buckets, including the dummy
  361. // node if there is one.
  362. (constructor.get() +
  363. static_cast<std::ptrdiff_t>(new_count))->next_ =
  364. (buckets_ + static_cast<std::ptrdiff_t>(
  365. bucket_count_))->next_;
  366. destroy_buckets();
  367. }
  368. else if (bucket::extra_node)
  369. {
  370. node_constructor a(node_alloc());
  371. a.construct();
  372. (constructor.get() +
  373. static_cast<std::ptrdiff_t>(new_count))->next_ =
  374. a.release();
  375. }
  376. bucket_count_ = new_count;
  377. buckets_ = constructor.release();
  378. recalculate_max_load();
  379. }
  380. ////////////////////////////////////////////////////////////////////////
  381. // Swap and Move
  382. void swap_allocators(table& other, false_type)
  383. {
  384. boost::unordered::detail::func::ignore_unused_variable_warning(other);
  385. // According to 23.2.1.8, if propagate_on_container_swap is
  386. // false the behaviour is undefined unless the allocators
  387. // are equal.
  388. BOOST_ASSERT(node_alloc() == other.node_alloc());
  389. }
  390. void swap_allocators(table& other, true_type)
  391. {
  392. allocators_.swap(other.allocators_);
  393. }
  394. // Only swaps the allocators if propagate_on_container_swap
  395. void swap(table& x)
  396. {
  397. set_hash_functions op1(*this, x);
  398. set_hash_functions op2(x, *this);
  399. // I think swap can throw if Propagate::value,
  400. // since the allocators' swap can throw. Not sure though.
  401. swap_allocators(x,
  402. boost::unordered::detail::integral_constant<bool,
  403. allocator_traits<node_allocator>::
  404. propagate_on_container_swap::value>());
  405. boost::swap(buckets_, x.buckets_);
  406. boost::swap(bucket_count_, x.bucket_count_);
  407. boost::swap(size_, x.size_);
  408. std::swap(mlf_, x.mlf_);
  409. std::swap(max_load_, x.max_load_);
  410. op1.commit();
  411. op2.commit();
  412. }
  413. // Only call with nodes allocated with the currect allocator, or
  414. // one that is equal to it. (Can't assert because other's
  415. // allocators might have already been moved).
  416. void move_buckets_from(table& other)
  417. {
  418. BOOST_ASSERT(!buckets_);
  419. buckets_ = other.buckets_;
  420. bucket_count_ = other.bucket_count_;
  421. size_ = other.size_;
  422. other.buckets_ = bucket_pointer();
  423. other.size_ = 0;
  424. other.max_load_ = 0;
  425. }
  426. ////////////////////////////////////////////////////////////////////////
  427. // Delete/destruct
  428. ~table()
  429. {
  430. delete_buckets();
  431. }
  432. void delete_node(link_pointer prev)
  433. {
  434. node_pointer n = static_cast<node_pointer>(prev->next_);
  435. prev->next_ = n->next_;
  436. boost::unordered::detail::func::destroy_value_impl(node_alloc(),
  437. n->value_ptr());
  438. boost::unordered::detail::func::destroy(boost::addressof(*n));
  439. node_allocator_traits::deallocate(node_alloc(), n, 1);
  440. --size_;
  441. }
  442. std::size_t delete_nodes(link_pointer prev, link_pointer end)
  443. {
  444. BOOST_ASSERT(prev->next_ != end);
  445. std::size_t count = 0;
  446. do {
  447. delete_node(prev);
  448. ++count;
  449. } while (prev->next_ != end);
  450. return count;
  451. }
  452. void delete_buckets()
  453. {
  454. if(buckets_) {
  455. if (size_) delete_nodes(get_previous_start(), link_pointer());
  456. if (bucket::extra_node) {
  457. node_pointer n = static_cast<node_pointer>(
  458. get_bucket(bucket_count_)->next_);
  459. boost::unordered::detail::func::destroy(
  460. boost::addressof(*n));
  461. node_allocator_traits::deallocate(node_alloc(), n, 1);
  462. }
  463. destroy_buckets();
  464. buckets_ = bucket_pointer();
  465. max_load_ = 0;
  466. }
  467. BOOST_ASSERT(!size_);
  468. }
  469. void clear()
  470. {
  471. if (!size_) return;
  472. delete_nodes(get_previous_start(), link_pointer());
  473. clear_buckets();
  474. BOOST_ASSERT(!size_);
  475. }
  476. void clear_buckets()
  477. {
  478. bucket_pointer end = get_bucket(bucket_count_);
  479. for(bucket_pointer it = buckets_; it != end; ++it)
  480. {
  481. it->next_ = node_pointer();
  482. }
  483. }
  484. void destroy_buckets()
  485. {
  486. bucket_pointer end = get_bucket(bucket_count_ + 1);
  487. for(bucket_pointer it = buckets_; it != end; ++it)
  488. {
  489. boost::unordered::detail::func::destroy(
  490. boost::addressof(*it));
  491. }
  492. bucket_allocator_traits::deallocate(bucket_alloc(),
  493. buckets_, bucket_count_ + 1);
  494. }
  495. ////////////////////////////////////////////////////////////////////////
  496. // Fix buckets after delete
  497. //
  498. std::size_t fix_bucket(std::size_t bucket_index, link_pointer prev)
  499. {
  500. link_pointer end = prev->next_;
  501. std::size_t bucket_index2 = bucket_index;
  502. if (end)
  503. {
  504. bucket_index2 = hash_to_bucket(
  505. static_cast<node_pointer>(end)->hash_);
  506. // If begin and end are in the same bucket, then
  507. // there's nothing to do.
  508. if (bucket_index == bucket_index2) return bucket_index2;
  509. // Update the bucket containing end.
  510. get_bucket(bucket_index2)->next_ = prev;
  511. }
  512. // Check if this bucket is now empty.
  513. bucket_pointer this_bucket = get_bucket(bucket_index);
  514. if (this_bucket->next_ == prev)
  515. this_bucket->next_ = link_pointer();
  516. return bucket_index2;
  517. }
  518. ////////////////////////////////////////////////////////////////////////
  519. // Assignment
  520. void assign(table const& x)
  521. {
  522. if (this != boost::addressof(x))
  523. {
  524. assign(x,
  525. boost::unordered::detail::integral_constant<bool,
  526. allocator_traits<node_allocator>::
  527. propagate_on_container_copy_assignment::value>());
  528. }
  529. }
  530. void assign(table const& x, false_type)
  531. {
  532. // Strong exception safety.
  533. set_hash_functions new_func_this(*this, x);
  534. new_func_this.commit();
  535. mlf_ = x.mlf_;
  536. recalculate_max_load();
  537. if (!size_ && !x.size_) return;
  538. if (x.size_ >= max_load_) {
  539. create_buckets(min_buckets_for_size(x.size_));
  540. }
  541. else {
  542. clear_buckets();
  543. }
  544. // assign_nodes takes ownership of the container's elements,
  545. // assigning to them if possible, and deleting any that are
  546. // left over.
  547. assign_nodes<table> node_creator(*this);
  548. table_impl::fill_buckets(x.begin(), *this, node_creator);
  549. }
  550. void assign(table const& x, true_type)
  551. {
  552. if (node_alloc() == x.node_alloc()) {
  553. allocators_.assign(x.allocators_);
  554. assign(x, false_type());
  555. }
  556. else {
  557. set_hash_functions new_func_this(*this, x);
  558. // Delete everything with current allocators before assigning
  559. // the new ones.
  560. delete_buckets();
  561. allocators_.assign(x.allocators_);
  562. // Copy over other data, all no throw.
  563. new_func_this.commit();
  564. mlf_ = x.mlf_;
  565. bucket_count_ = min_buckets_for_size(x.size_);
  566. max_load_ = 0;
  567. // Finally copy the elements.
  568. if (x.size_) {
  569. create_buckets(bucket_count_);
  570. copy_nodes<node_allocator> node_creator(node_alloc());
  571. table_impl::fill_buckets(x.begin(), *this, node_creator);
  572. }
  573. }
  574. }
  575. void move_assign(table& x)
  576. {
  577. if (this != boost::addressof(x))
  578. {
  579. move_assign(x,
  580. boost::unordered::detail::integral_constant<bool,
  581. allocator_traits<node_allocator>::
  582. propagate_on_container_move_assignment::value>());
  583. }
  584. }
  585. void move_assign(table& x, true_type)
  586. {
  587. delete_buckets();
  588. set_hash_functions new_func_this(*this, x);
  589. allocators_.move_assign(x.allocators_);
  590. // No throw from here.
  591. mlf_ = x.mlf_;
  592. max_load_ = x.max_load_;
  593. move_buckets_from(x);
  594. new_func_this.commit();
  595. }
  596. void move_assign(table& x, false_type)
  597. {
  598. if (node_alloc() == x.node_alloc()) {
  599. delete_buckets();
  600. set_hash_functions new_func_this(*this, x);
  601. // No throw from here.
  602. mlf_ = x.mlf_;
  603. max_load_ = x.max_load_;
  604. move_buckets_from(x);
  605. new_func_this.commit();
  606. }
  607. else {
  608. set_hash_functions new_func_this(*this, x);
  609. new_func_this.commit();
  610. mlf_ = x.mlf_;
  611. recalculate_max_load();
  612. if (!size_ && !x.size_) return;
  613. if (x.size_ >= max_load_) {
  614. create_buckets(min_buckets_for_size(x.size_));
  615. }
  616. else {
  617. clear_buckets();
  618. }
  619. // move_assign_nodes takes ownership of the container's
  620. // elements, assigning to them if possible, and deleting
  621. // any that are left over.
  622. move_assign_nodes<table> node_creator(*this);
  623. node_holder<node_allocator> nodes(x);
  624. table_impl::fill_buckets(nodes.begin(), *this, node_creator);
  625. }
  626. }
  627. // Accessors
  628. key_type const& get_key(value_type const& x) const
  629. {
  630. return extractor::extract(x);
  631. }
  632. std::size_t hash(key_type const& k) const
  633. {
  634. return policy::apply_hash(this->hash_function(), k);
  635. }
  636. // Find Node
  637. template <typename Key, typename Hash, typename Pred>
  638. iterator generic_find_node(
  639. Key const& k,
  640. Hash const& hf,
  641. Pred const& eq) const
  642. {
  643. return static_cast<table_impl const*>(this)->
  644. find_node_impl(policy::apply_hash(hf, k), k, eq);
  645. }
  646. iterator find_node(
  647. std::size_t key_hash,
  648. key_type const& k) const
  649. {
  650. return static_cast<table_impl const*>(this)->
  651. find_node_impl(key_hash, k, this->key_eq());
  652. }
  653. iterator find_node(key_type const& k) const
  654. {
  655. return static_cast<table_impl const*>(this)->
  656. find_node_impl(hash(k), k, this->key_eq());
  657. }
  658. iterator find_matching_node(iterator n) const
  659. {
  660. // TODO: Does this apply to C++11?
  661. //
  662. // For some stupid reason, I decided to support equality comparison
  663. // when different hash functions are used. So I can't use the hash
  664. // value from the node here.
  665. return find_node(get_key(*n));
  666. }
  667. // Reserve and rehash
  668. void reserve_for_insert(std::size_t);
  669. void rehash(std::size_t);
  670. void reserve(std::size_t);
  671. };
  672. ////////////////////////////////////////////////////////////////////////////
  673. // Reserve & Rehash
  674. // basic exception safety
  675. template <typename Types>
  676. inline void table<Types>::reserve_for_insert(std::size_t size)
  677. {
  678. if (!buckets_) {
  679. create_buckets((std::max)(bucket_count_,
  680. min_buckets_for_size(size)));
  681. }
  682. // According to the standard this should be 'size >= max_load_',
  683. // but I think this is better, defect report filed.
  684. else if(size > max_load_) {
  685. std::size_t num_buckets
  686. = min_buckets_for_size((std::max)(size,
  687. size_ + (size_ >> 1)));
  688. if (num_buckets != bucket_count_)
  689. static_cast<table_impl*>(this)->rehash_impl(num_buckets);
  690. }
  691. }
  692. // if hash function throws, basic exception safety
  693. // strong otherwise.
  694. template <typename Types>
  695. inline void table<Types>::rehash(std::size_t min_buckets)
  696. {
  697. using namespace std;
  698. if(!size_) {
  699. delete_buckets();
  700. bucket_count_ = policy::new_bucket_count(min_buckets);
  701. }
  702. else {
  703. min_buckets = policy::new_bucket_count((std::max)(min_buckets,
  704. boost::unordered::detail::double_to_size(floor(
  705. static_cast<double>(size_) /
  706. static_cast<double>(mlf_))) + 1));
  707. if(min_buckets != bucket_count_)
  708. static_cast<table_impl*>(this)->rehash_impl(min_buckets);
  709. }
  710. }
  711. template <typename Types>
  712. inline void table<Types>::reserve(std::size_t num_elements)
  713. {
  714. rehash(static_cast<std::size_t>(
  715. std::ceil(static_cast<double>(num_elements) / mlf_)));
  716. }
  717. }}}
  718. #if defined(BOOST_MSVC)
  719. #pragma warning(pop)
  720. #endif
  721. #endif