numeric_traits.hpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. // (C) Copyright David Abrahams 2001, Howard Hinnant 2001.
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See
  4. // accompanying file LICENSE_1_0.txt or copy at
  5. // http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. // Template class numeric_traits<Number> --
  8. //
  9. // Supplies:
  10. //
  11. // typedef difference_type -- a type used to represent the difference
  12. // between any two values of Number.
  13. //
  14. // Support:
  15. // 1. Not all specializations are supplied
  16. //
  17. // 2. Use of specializations that are not supplied will cause a
  18. // compile-time error
  19. //
  20. // 3. Users are free to specialize numeric_traits for any type.
  21. //
  22. // 4. Right now, specializations are only supplied for integer types.
  23. //
  24. // 5. On implementations which do not supply compile-time constants in
  25. // std::numeric_limits<>, only specializations for built-in integer types
  26. // are supplied.
  27. //
  28. // 6. Handling of numbers whose range of representation is at least as
  29. // great as boost::intmax_t can cause some differences to be
  30. // unrepresentable in difference_type:
  31. //
  32. // Number difference_type
  33. // ------ ---------------
  34. // signed Number
  35. // unsigned intmax_t
  36. //
  37. // template <class Number> typename numeric_traits<Number>::difference_type
  38. // numeric_distance(Number x, Number y)
  39. // computes (y - x), attempting to avoid overflows.
  40. //
  41. // See http://www.boost.org for most recent version including documentation.
  42. // Revision History
  43. // 11 Feb 2001 - Use BOOST_STATIC_CONSTANT (David Abrahams)
  44. // 11 Feb 2001 - Rolled back ineffective Borland-specific code
  45. // (David Abrahams)
  46. // 10 Feb 2001 - Rolled in supposed Borland fixes from John Maddock, but
  47. // not seeing any improvement yet (David Abrahams)
  48. // 06 Feb 2001 - Factored if_true out into boost/detail/select_type.hpp
  49. // (David Abrahams)
  50. // 23 Jan 2001 - Fixed logic of difference_type selection, which was
  51. // completely wack. In the process, added digit_traits<>
  52. // to compute the number of digits in intmax_t even when
  53. // not supplied by numeric_limits<>. (David Abrahams)
  54. // 21 Jan 2001 - Created (David Abrahams)
  55. #ifndef BOOST_NUMERIC_TRAITS_HPP_DWA20001901
  56. # define BOOST_NUMERIC_TRAITS_HPP_DWA20001901
  57. # include <boost/config.hpp>
  58. # include <boost/cstdint.hpp>
  59. # include <boost/static_assert.hpp>
  60. # include <boost/type_traits.hpp>
  61. # include <boost/detail/select_type.hpp>
  62. # include <boost/limits.hpp>
  63. namespace boost { namespace detail {
  64. // Template class is_signed -- determine whether a numeric type is signed
  65. // Requires that T is constructable from the literals -1 and 0. Compile-time
  66. // error results if that requirement is not met (and thus signedness is not
  67. // likely to have meaning for that type).
  68. template <class Number>
  69. struct is_signed
  70. {
  71. #if defined(BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS)
  72. BOOST_STATIC_CONSTANT(bool, value = (Number(-1) < Number(0)));
  73. #else
  74. BOOST_STATIC_CONSTANT(bool, value = std::numeric_limits<Number>::is_signed);
  75. #endif
  76. };
  77. # ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
  78. // digit_traits - compute the number of digits in a built-in integer
  79. // type. Needed for implementations on which numeric_limits is not specialized
  80. // for intmax_t (e.g. VC6).
  81. template <bool is_specialized> struct digit_traits_select;
  82. // numeric_limits is specialized; just select that version of digits
  83. template <> struct digit_traits_select<true>
  84. {
  85. template <class T> struct traits
  86. {
  87. BOOST_STATIC_CONSTANT(int, digits = std::numeric_limits<T>::digits);
  88. };
  89. };
  90. // numeric_limits is not specialized; compute digits from sizeof(T)
  91. template <> struct digit_traits_select<false>
  92. {
  93. template <class T> struct traits
  94. {
  95. BOOST_STATIC_CONSTANT(int, digits = (
  96. sizeof(T) * std::numeric_limits<unsigned char>::digits
  97. - (is_signed<T>::value ? 1 : 0))
  98. );
  99. };
  100. };
  101. // here's the "usable" template
  102. template <class T> struct digit_traits
  103. {
  104. typedef digit_traits_select<
  105. ::std::numeric_limits<T>::is_specialized> selector;
  106. typedef typename selector::template traits<T> traits;
  107. BOOST_STATIC_CONSTANT(int, digits = traits::digits);
  108. };
  109. #endif
  110. // Template class integer_traits<Integer> -- traits of various integer types
  111. // This should probably be rolled into boost::integer_traits one day, but I
  112. // need it to work without <limits>
  113. template <class Integer>
  114. struct integer_traits
  115. {
  116. # ifndef BOOST_NO_LIMITS_COMPILE_TIME_CONSTANTS
  117. private:
  118. typedef Integer integer_type;
  119. typedef std::numeric_limits<integer_type> x;
  120. public:
  121. typedef typename
  122. if_true<(int(x::is_signed)
  123. && (!int(x::is_bounded)
  124. // digits is the number of no-sign bits
  125. || (int(x::digits) + 1 >= digit_traits<boost::intmax_t>::digits)))>::template then<
  126. Integer,
  127. typename if_true<(int(x::digits) + 1 < digit_traits<signed int>::digits)>::template then<
  128. signed int,
  129. typename if_true<(int(x::digits) + 1 < digit_traits<signed long>::digits)>::template then<
  130. signed long,
  131. // else
  132. intmax_t
  133. >::type>::type>::type difference_type;
  134. #else
  135. BOOST_STATIC_ASSERT(boost::is_integral<Integer>::value);
  136. typedef typename
  137. if_true<(sizeof(Integer) >= sizeof(intmax_t))>::template then<
  138. typename if_true<(is_signed<Integer>::value)>::template then<
  139. Integer,
  140. intmax_t
  141. >::type,
  142. typename if_true<(sizeof(Integer) < sizeof(std::ptrdiff_t))>::template then<
  143. std::ptrdiff_t,
  144. intmax_t
  145. >::type
  146. >::type difference_type;
  147. # endif
  148. };
  149. // Right now, only supports integers, but should be expanded.
  150. template <class Number>
  151. struct numeric_traits
  152. {
  153. typedef typename integer_traits<Number>::difference_type difference_type;
  154. };
  155. template <class Number>
  156. typename numeric_traits<Number>::difference_type numeric_distance(Number x, Number y)
  157. {
  158. typedef typename numeric_traits<Number>::difference_type difference_type;
  159. return difference_type(y) - difference_type(x);
  160. }
  161. }}
  162. #endif // BOOST_NUMERIC_TRAITS_HPP_DWA20001901