CNL  2.0.2 (development)
Compositional Numeric Library
descale.h
1 
2 // Copyright John McFarlane 2021.
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_CHARCONV_DESCALE_H)
8 #define CNL_IMPL_CHARCONV_DESCALE_H
9 
10 #include "../../integer.h"
11 #include "../cstdint/types.h"
12 #include "../numbers/signedness.h"
13 #include "../scaled/declaration.h"
14 #include "../unreachable.h"
15 
16 #include <limits>
17 
19 namespace cnl::_impl {
20  template<integer Rep, int Radix>
21  struct descaled {
22  Rep significand;
23  int exponent;
24  static constexpr int radix = Radix;
25  };
26 
27  template<
28  integer Significand = std::int64_t, int OutRadix = 10,
29  bool Precise = false,
30  int InExponent = 0, int InRadix = 2,
31  integer Rep = int>
32  [[nodiscard]] constexpr auto descale(Rep const& input, power<InExponent, InRadix>)
33  {
34  descaled<Significand, OutRadix> output{static_cast<Significand>(input), 0};
35 
36  if (!input) {
37  return output;
38  }
39 
40  auto const oob{
41  (input < Rep{0})
42  ? []([[maybe_unused]] Significand const& n) -> bool {
43  if constexpr (numbers::signedness_v<Significand>) {
44  return n < -std::numeric_limits<Significand>::max() / OutRadix;
45  } else {
46  return unreachable<bool>("negative unsigned integer");
47  }
48  }
49  : [](Significand const& n) {
50  return n > Significand{std::numeric_limits<Significand>::max() / OutRadix};
51  }};
52 
53  if constexpr (InExponent < 0) {
54  for (int in_exponent = InExponent;
55  in_exponent != 0 || (Precise && !(output.significand % OutRadix));) {
56  if (output.significand % InRadix) {
57  if (oob(output.significand)) {
58  if (Precise) {
59  unreachable<descaled<Significand, OutRadix>>("number cannot be represented in this form");
60  }
61  } else {
62  output.significand *= OutRadix;
63  output.exponent--;
64  continue;
65  }
66  }
67 
68  output.significand /= InRadix;
69  in_exponent++;
70  }
71  } else {
72  for (int in_exponent = InExponent;
73  in_exponent != 0 || !(output.significand % OutRadix);) {
74  if (!(output.significand % OutRadix)) {
75  output.significand /= OutRadix;
76  output.exponent++;
77  continue;
78  }
79 
80  if (!oob(output.significand)) {
81  output.significand *= InRadix;
82  in_exponent--;
83  }
84  }
85  }
86 
87  return output;
88  }
89 }
90 
91 #endif // CNL_IMPL_CHARCONV_DESCALE_H
std::int64_t
std::numeric_limits::max
T max(T... args)