util.hpp 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  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_UTIL_HPP_INCLUDED
  6. #define BOOST_UNORDERED_DETAIL_UTIL_HPP_INCLUDED
  7. #include <boost/config.hpp>
  8. #if defined(BOOST_HAS_PRAGMA_ONCE)
  9. #pragma once
  10. #endif
  11. #include <boost/type_traits/is_convertible.hpp>
  12. #include <boost/type_traits/is_empty.hpp>
  13. #include <boost/iterator/iterator_categories.hpp>
  14. #include <boost/utility/enable_if.hpp>
  15. #include <boost/detail/select_type.hpp>
  16. #include <boost/move/move.hpp>
  17. #include <boost/preprocessor/seq/size.hpp>
  18. #include <boost/preprocessor/seq/enum.hpp>
  19. #include <boost/swap.hpp>
  20. namespace boost { namespace unordered { namespace detail {
  21. static const float minimum_max_load_factor = 1e-3f;
  22. static const std::size_t default_bucket_count = 11;
  23. struct move_tag {};
  24. struct empty_emplace {};
  25. namespace func {
  26. template <class T>
  27. inline void ignore_unused_variable_warning(T const&) {}
  28. }
  29. ////////////////////////////////////////////////////////////////////////////
  30. // iterator SFINAE
  31. template <typename I>
  32. struct is_forward :
  33. boost::is_convertible<
  34. typename boost::iterator_traversal<I>::type,
  35. boost::forward_traversal_tag>
  36. {};
  37. template <typename I, typename ReturnType>
  38. struct enable_if_forward :
  39. boost::enable_if_c<
  40. boost::unordered::detail::is_forward<I>::value,
  41. ReturnType>
  42. {};
  43. template <typename I, typename ReturnType>
  44. struct disable_if_forward :
  45. boost::disable_if_c<
  46. boost::unordered::detail::is_forward<I>::value,
  47. ReturnType>
  48. {};
  49. ////////////////////////////////////////////////////////////////////////////
  50. // primes
  51. #define BOOST_UNORDERED_PRIMES \
  52. (17ul)(29ul)(37ul)(53ul)(67ul)(79ul) \
  53. (97ul)(131ul)(193ul)(257ul)(389ul)(521ul)(769ul) \
  54. (1031ul)(1543ul)(2053ul)(3079ul)(6151ul)(12289ul)(24593ul) \
  55. (49157ul)(98317ul)(196613ul)(393241ul)(786433ul) \
  56. (1572869ul)(3145739ul)(6291469ul)(12582917ul)(25165843ul) \
  57. (50331653ul)(100663319ul)(201326611ul)(402653189ul)(805306457ul) \
  58. (1610612741ul)(3221225473ul)(4294967291ul)
  59. template<class T> struct prime_list_template
  60. {
  61. static std::size_t const value[];
  62. #if !defined(SUNPRO_CC)
  63. static std::ptrdiff_t const length;
  64. #else
  65. static std::ptrdiff_t const length
  66. = BOOST_PP_SEQ_SIZE(BOOST_UNORDERED_PRIMES);
  67. #endif
  68. };
  69. template<class T>
  70. std::size_t const prime_list_template<T>::value[] = {
  71. BOOST_PP_SEQ_ENUM(BOOST_UNORDERED_PRIMES)
  72. };
  73. #if !defined(SUNPRO_CC)
  74. template<class T>
  75. std::ptrdiff_t const prime_list_template<T>::length
  76. = BOOST_PP_SEQ_SIZE(BOOST_UNORDERED_PRIMES);
  77. #endif
  78. #undef BOOST_UNORDERED_PRIMES
  79. typedef prime_list_template<std::size_t> prime_list;
  80. // no throw
  81. inline std::size_t next_prime(std::size_t num) {
  82. std::size_t const* const prime_list_begin = prime_list::value;
  83. std::size_t const* const prime_list_end = prime_list_begin +
  84. prime_list::length;
  85. std::size_t const* bound =
  86. std::lower_bound(prime_list_begin, prime_list_end, num);
  87. if(bound == prime_list_end)
  88. bound--;
  89. return *bound;
  90. }
  91. // no throw
  92. inline std::size_t prev_prime(std::size_t num) {
  93. std::size_t const* const prime_list_begin = prime_list::value;
  94. std::size_t const* const prime_list_end = prime_list_begin +
  95. prime_list::length;
  96. std::size_t const* bound =
  97. std::upper_bound(prime_list_begin,prime_list_end, num);
  98. if(bound != prime_list_begin)
  99. bound--;
  100. return *bound;
  101. }
  102. ////////////////////////////////////////////////////////////////////////////
  103. // insert_size/initial_size
  104. #if !defined(BOOST_NO_STD_DISTANCE)
  105. using ::std::distance;
  106. #else
  107. template <class ForwardIterator>
  108. inline std::size_t distance(ForwardIterator i, ForwardIterator j) {
  109. std::size_t x;
  110. std::distance(i, j, x);
  111. return x;
  112. }
  113. #endif
  114. template <class I>
  115. inline typename
  116. boost::unordered::detail::enable_if_forward<I, std::size_t>::type
  117. insert_size(I i, I j)
  118. {
  119. return std::distance(i, j);
  120. }
  121. template <class I>
  122. inline typename
  123. boost::unordered::detail::disable_if_forward<I, std::size_t>::type
  124. insert_size(I, I)
  125. {
  126. return 1;
  127. }
  128. template <class I>
  129. inline std::size_t initial_size(I i, I j,
  130. std::size_t num_buckets =
  131. boost::unordered::detail::default_bucket_count)
  132. {
  133. // TODO: Why +1?
  134. return (std::max)(
  135. boost::unordered::detail::insert_size(i, j) + 1,
  136. num_buckets);
  137. }
  138. ////////////////////////////////////////////////////////////////////////////
  139. // compressed
  140. template <typename T, int Index>
  141. struct compressed_base : private T
  142. {
  143. compressed_base(T const& x) : T(x) {}
  144. compressed_base(T& x, move_tag) : T(boost::move(x)) {}
  145. T& get() { return *this; }
  146. T const& get() const { return *this; }
  147. };
  148. template <typename T, int Index>
  149. struct uncompressed_base
  150. {
  151. uncompressed_base(T const& x) : value_(x) {}
  152. uncompressed_base(T& x, move_tag) : value_(boost::move(x)) {}
  153. T& get() { return value_; }
  154. T const& get() const { return value_; }
  155. private:
  156. T value_;
  157. };
  158. template <typename T, int Index>
  159. struct generate_base
  160. : boost::detail::if_true<
  161. boost::is_empty<T>::value
  162. >:: BOOST_NESTED_TEMPLATE then<
  163. boost::unordered::detail::compressed_base<T, Index>,
  164. boost::unordered::detail::uncompressed_base<T, Index>
  165. >
  166. {};
  167. template <typename T1, typename T2>
  168. struct compressed
  169. : private boost::unordered::detail::generate_base<T1, 1>::type,
  170. private boost::unordered::detail::generate_base<T2, 2>::type
  171. {
  172. typedef typename generate_base<T1, 1>::type base1;
  173. typedef typename generate_base<T2, 2>::type base2;
  174. typedef T1 first_type;
  175. typedef T2 second_type;
  176. first_type& first() {
  177. return static_cast<base1*>(this)->get();
  178. }
  179. first_type const& first() const {
  180. return static_cast<base1 const*>(this)->get();
  181. }
  182. second_type& second() {
  183. return static_cast<base2*>(this)->get();
  184. }
  185. second_type const& second() const {
  186. return static_cast<base2 const*>(this)->get();
  187. }
  188. template <typename First, typename Second>
  189. compressed(First const& x1, Second const& x2)
  190. : base1(x1), base2(x2) {}
  191. compressed(compressed const& x)
  192. : base1(x.first()), base2(x.second()) {}
  193. compressed(compressed& x, move_tag m)
  194. : base1(x.first(), m), base2(x.second(), m) {}
  195. void assign(compressed const& x)
  196. {
  197. first() = x.first();
  198. second() = x.second();
  199. }
  200. void move_assign(compressed& x)
  201. {
  202. first() = boost::move(x.first());
  203. second() = boost::move(x.second());
  204. }
  205. void swap(compressed& x)
  206. {
  207. boost::swap(first(), x.first());
  208. boost::swap(second(), x.second());
  209. }
  210. private:
  211. // Prevent assignment just to make use of assign or
  212. // move_assign explicit.
  213. compressed& operator=(compressed const&);
  214. };
  215. }}}
  216. #endif