dynamic_bitset.hpp 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. // -----------------------------------------------------------
  2. //
  3. // Copyright (c) 2001-2002 Chuck Allison and Jeremy Siek
  4. // Copyright (c) 2003-2006, 2008 Gennaro Prota
  5. //
  6. // Copyright (c) 2014 Glen Joseph Fernandes
  7. // glenfe at live dot com
  8. //
  9. // Distributed under the Boost Software License, Version 1.0.
  10. // (See accompanying file LICENSE_1_0.txt or copy at
  11. // http://www.boost.org/LICENSE_1_0.txt)
  12. //
  13. // -----------------------------------------------------------
  14. #ifndef BOOST_DETAIL_DYNAMIC_BITSET_HPP
  15. #define BOOST_DETAIL_DYNAMIC_BITSET_HPP
  16. #include <memory>
  17. #include <cstddef>
  18. #include "boost/config.hpp"
  19. #include "boost/detail/workaround.hpp"
  20. namespace boost {
  21. namespace detail {
  22. namespace dynamic_bitset_impl {
  23. // Gives (read-)access to the object representation
  24. // of an object of type T (3.9p4). CANNOT be used
  25. // on a base sub-object
  26. //
  27. template <typename T>
  28. inline const unsigned char * object_representation (T* p)
  29. {
  30. return static_cast<const unsigned char *>(static_cast<const void *>(p));
  31. }
  32. template<typename T, int amount, int width /* = default */>
  33. struct shifter
  34. {
  35. static void left_shift(T & v) {
  36. amount >= width ? (v = 0)
  37. : (v >>= BOOST_DYNAMIC_BITSET_WRAP_CONSTANT(amount));
  38. }
  39. };
  40. // ------- count function implementation --------------
  41. typedef unsigned char byte_type;
  42. // These two entities
  43. //
  44. // enum mode { access_by_bytes, access_by_blocks };
  45. // template <mode> struct mode_to_type {};
  46. //
  47. // were removed, since the regression logs (as of 24 Aug 2008)
  48. // showed that several compilers had troubles with recognizing
  49. //
  50. // const mode m = access_by_bytes
  51. //
  52. // as a constant expression
  53. //
  54. // * So, we'll use bool, instead of enum *.
  55. //
  56. template <bool value>
  57. struct value_to_type
  58. {
  59. value_to_type() {}
  60. };
  61. const bool access_by_bytes = true;
  62. const bool access_by_blocks = false;
  63. // the table: wrapped in a class template, so
  64. // that it is only instantiated if/when needed
  65. //
  66. template <bool dummy_name = true>
  67. struct count_table { static const byte_type table[]; };
  68. template <>
  69. struct count_table<false> { /* no table */ };
  70. const unsigned int table_width = 8;
  71. template <bool b>
  72. const byte_type count_table<b>::table[] =
  73. {
  74. // Automatically generated by GPTableGen.exe v.1.0
  75. //
  76. 0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
  77. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  78. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  79. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  80. 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
  81. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  82. 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
  83. 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8
  84. };
  85. // overload for access by bytes
  86. //
  87. template <typename Iterator>
  88. inline std::size_t do_count(Iterator first, std::size_t length,
  89. int /*dummy param*/,
  90. value_to_type<access_by_bytes>* )
  91. {
  92. std::size_t num = 0;
  93. if (length)
  94. {
  95. const byte_type * p = object_representation(&*first);
  96. length *= sizeof(*first);
  97. do {
  98. num += count_table<>::table[*p];
  99. ++p;
  100. --length;
  101. } while (length);
  102. }
  103. return num;
  104. }
  105. // overload for access by blocks
  106. //
  107. template <typename Iterator, typename ValueType>
  108. inline std::size_t do_count(Iterator first, std::size_t length, ValueType,
  109. value_to_type<access_by_blocks>*)
  110. {
  111. std::size_t num = 0;
  112. while (length){
  113. ValueType value = *first;
  114. while (value) {
  115. num += count_table<>::table[value & ((1u<<table_width) - 1)];
  116. value >>= table_width;
  117. }
  118. ++first;
  119. --length;
  120. }
  121. return num;
  122. }
  123. // -------------------------------------------------------
  124. // Some library implementations simply return a dummy
  125. // value such as
  126. //
  127. // size_type(-1) / sizeof(T)
  128. //
  129. // from vector<>::max_size. This tries to get more
  130. // meaningful info.
  131. //
  132. template <typename T>
  133. inline typename T::size_type vector_max_size_workaround(const T & v)
  134. BOOST_NOEXCEPT
  135. {
  136. typedef typename T::allocator_type allocator_type;
  137. const allocator_type& alloc = v.get_allocator();
  138. #if !defined(BOOST_NO_CXX11_ALLOCATOR)
  139. typedef std::allocator_traits<allocator_type> allocator_traits;
  140. const typename allocator_traits::size_type alloc_max =
  141. allocator_traits::max_size(alloc);
  142. #else
  143. const typename allocator_type::size_type alloc_max = alloc.max_size();
  144. #endif
  145. const typename T::size_type container_max = v.max_size();
  146. return alloc_max < container_max ? alloc_max : container_max;
  147. }
  148. // for static_asserts
  149. template <typename T>
  150. struct allowed_block_type {
  151. enum { value = T(-1) > 0 }; // ensure T has no sign
  152. };
  153. template <>
  154. struct allowed_block_type<bool> {
  155. enum { value = false };
  156. };
  157. template <typename T>
  158. struct is_numeric {
  159. enum { value = false };
  160. };
  161. # define BOOST_dynamic_bitset_is_numeric(x) \
  162. template<> \
  163. struct is_numeric< x > { \
  164. enum { value = true }; \
  165. } /**/
  166. BOOST_dynamic_bitset_is_numeric(bool);
  167. BOOST_dynamic_bitset_is_numeric(char);
  168. #if !defined(BOOST_NO_INTRINSIC_WCHAR_T)
  169. BOOST_dynamic_bitset_is_numeric(wchar_t);
  170. #endif
  171. BOOST_dynamic_bitset_is_numeric(signed char);
  172. BOOST_dynamic_bitset_is_numeric(short int);
  173. BOOST_dynamic_bitset_is_numeric(int);
  174. BOOST_dynamic_bitset_is_numeric(long int);
  175. BOOST_dynamic_bitset_is_numeric(unsigned char);
  176. BOOST_dynamic_bitset_is_numeric(unsigned short);
  177. BOOST_dynamic_bitset_is_numeric(unsigned int);
  178. BOOST_dynamic_bitset_is_numeric(unsigned long);
  179. #if defined(BOOST_HAS_LONG_LONG)
  180. BOOST_dynamic_bitset_is_numeric(::boost::long_long_type);
  181. BOOST_dynamic_bitset_is_numeric(::boost::ulong_long_type);
  182. #endif
  183. // intentionally omitted
  184. //BOOST_dynamic_bitset_is_numeric(float);
  185. //BOOST_dynamic_bitset_is_numeric(double);
  186. //BOOST_dynamic_bitset_is_numeric(long double);
  187. #undef BOOST_dynamic_bitset_is_numeric
  188. } // dynamic_bitset_impl
  189. } // namespace detail
  190. } // namespace boost
  191. #endif // include guard