CNL  2.0.2 (development)
Compositional Numeric Library
used_digits.h
1 
2 // Copyright John McFarlane 2018.
3 // Distributed under the Boost Software License, Version 1.0.
4 // (See accompanying file ../LICENSE_1_0.txt or copy at
5 // http://www.boost.org/LICENSE_1_0.txt)
6 
7 #if !defined(CNL_IMPL_USED_DIGITS_H)
8 #define CNL_IMPL_USED_DIGITS_H
9 
10 #include "../integer.h"
11 
12 #include <limits>
13 #include <type_traits>
14 
15 namespace cnl {
16  namespace _impl {
17 
19  // cnl::_impl::used_digits
20 
21  template<bool IsSigned>
22  struct used_digits_signed;
23 
24  template<>
25  struct used_digits_signed<false> {
26  template<integer Integer>
27  [[nodiscard]] constexpr auto operator()(Integer const& value, int radix) const -> int
28  {
29  return (value > 0) ? 1 + used_digits_signed<false>{}(value / radix, radix) : 0;
30  }
31  };
32 
33  template<>
34  struct used_digits_signed<true> {
35  template<integer Integer>
36  [[nodiscard]] constexpr auto operator()(Integer const& value, int radix) const
37  {
38  // Most negative number is not exploited;
39  // thus negating the result or subtracting it from something else
40  // will less likely result in overflow.
41  return (value < 0) ? used_digits_signed<false>{}(Integer(-1) - value, radix)
42  : used_digits_signed<false>{}(value, radix);
43  }
44  };
45 
46  template<integer Integer>
47  [[nodiscard]] constexpr auto used_digits(
48  Integer const& value, int radix = std::numeric_limits<Integer>::radix)
49  {
50  return used_digits_signed<std::is_signed<Integer>::value>{}(value, radix);
51  }
52  }
53 }
54 
55 #endif // CNL_IMPL_USED_DIGITS_H
cnl
compositional numeric library
Definition: abort.h:15
std::numeric_limits