CNL  2.0.2 (development)
Compositional Numeric Library
to_chars.h
1 
2 // Copyright John McFarlane 2019.
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_TO_CHARS_H)
8 #define CNL_IMPL_TO_CHARS_H
9 
10 #include "../../integer.h"
11 #include "../cnl_assert.h"
12 #include "../narrow_cast.h"
13 #include "../num_traits/digits.h"
14 #include "../num_traits/rounding.h"
15 #include "../num_traits/set_rounding.h"
16 #include "../numbers/signedness.h"
17 #include "constants.h"
18 
19 #include <array>
20 #include <charconv>
21 #include <limits>
22 #include <string_view>
23 #include <system_error>
24 
26 namespace cnl {
27  namespace _impl {
28  // cnl::_impl::max_to_chars_chars
29  template<typename Scalar, int Base = 10>
30  struct max_to_chars_chars {
31  private:
32  static constexpr auto _sign_chars = static_cast<int>(cnl::numbers::signedness_v<Scalar>);
33  static constexpr auto _integer_chars = ((digits_v<Scalar> + 2) / 3);
34 
35  public:
36  static constexpr auto value = _sign_chars + _integer_chars;
37  };
38 
39  // cnl::_impl::itoc
40  template<typename Scalar>
41  [[nodiscard]] constexpr auto itoc(Scalar value)
42  {
43  static_assert(
44  std::is_same<typename rounding<Scalar>::type, native_rounding_tag>::value,
45  "wrong rounding type");
46  auto c = zero_char + static_cast<int>(value);
47  return static_cast<char>(c);
48  }
49 
50  // cnl::_impl::to_chars_natural
51  template<class Integer>
52  [[nodiscard]] constexpr auto to_chars_natural(char* ptr, char* last, Integer const& value) -> char*
53  {
54  auto const quotient = value / 10;
55 
56  auto const next_ptr = quotient ? to_chars_natural(ptr, last, quotient) : ptr;
57 
58  if (next_ptr == last || next_ptr == nullptr) {
59  return nullptr;
60  }
61 
62  // Note: linker may struggle with combination of clang, int128_t and sanitizer.
63  // See clang.cmake
64  auto const remainder = value - (quotient * 10);
65  *next_ptr = itoc(remainder);
66 
67  return next_ptr + 1;
68  }
69 
70  template<integer Integer>
71  [[nodiscard]] constexpr auto
72  to_chars_positive(char* const first, char* const last, Integer const& value)
73  {
74  auto const natural_last = to_chars_natural(first, last, value);
75  return std::to_chars_result{
76  natural_last, natural_last ? std::errc{} : std::errc::value_too_large};
77  }
78 
79  template<typename Number>
80  [[nodiscard]] constexpr auto
81  to_chars_non_zero(char* const first, char* const last, Number const& value)
82  {
83  if constexpr (numbers::signedness_v<Number>) {
84  if (value < Number{}) {
85  auto const destination_length = std::distance(first, last);
86  if (destination_length < 2) {
87  return std::to_chars_result{last, std::errc::value_too_large};
88  }
89 
90  // -ve
91  *first = '-';
92 
93  // implementation does not support the most negative number
94  CNL_ASSERT(-std::numeric_limits<decltype(-value)>::max() <= value);
95 
96  return to_chars_positive(first + 1, last, -value);
97  }
98  }
99 
100  return to_chars_positive(first, last, value);
101  }
102  }
103 
104  // partial implementation of std::to_chars overloaded on cnl::integer
105  template<integer Integer>
106  [[nodiscard]] constexpr auto to_chars(
107  char* const first,
108  char* const last, // NOLINT(readability-non-const-parameter)
109  Integer const& value)
110  {
111  if (!value) {
112  if (first == last) {
113  // buffer too small to contain "0"
114  return std::to_chars_result{last, std::errc::value_too_large};
115  }
116 
117  // zero
118  *first = _impl::zero_char;
119  return std::to_chars_result{first + 1, std::errc{}};
120  }
121 
122  using native_rounding_type = set_rounding_t<decltype(value), native_rounding_tag>;
123  auto const& native_rounding_value = static_cast<native_rounding_type>(value);
124 
125  return _impl::to_chars_non_zero<native_rounding_type>(
126  first, last, native_rounding_value);
127  }
128 
129  template<int NumChars>
130  // NOLINTNEXTLINE(cppcoreguidelines-pro-type-member-init,hicpp-member-init)
131  struct to_chars_static_result {
133  int length;
134  };
135 
136  // variant of cnl::to_chars returning fixed-size array of chars
137  // large enough to store any possible result for given input type
138  template<typename Number>
139  [[nodiscard]] constexpr auto to_chars_static(Number const& value)
140  {
141  constexpr auto max_num_chars = _impl::max_to_chars_chars<Number>::value;
142 
143  // NOLINTNEXTLINE(cppcoreguidelines-pro-type-member-init,hicpp-member-init)
144  to_chars_static_result<max_num_chars> result;
145 
146  auto dynamic_result = to_chars(result.chars.data(), result.chars.data() + max_num_chars, value);
147  CNL_ASSERT(dynamic_result.ptr > result.chars.data());
148  CNL_ASSERT(dynamic_result.ptr <= result.chars.data() + max_num_chars);
149  CNL_ASSERT(dynamic_result.ec == std::errc{});
150 
151  *dynamic_result.ptr = '\0';
152  result.length = _impl::narrow_cast<int>(dynamic_result.ptr - result.chars.data());
153 
154  return result;
155  }
156 }
157 
158 #endif // CNL_IMPL_TO_CHARS_H
std::is_same
std::distance
T distance(T... args)
std::remainder
T remainder(T... args)
std::array< char, NumChars+1 >
std::errc
cnl
compositional numeric library
Definition: abort.h:15
cnl::quotient
constexpr auto quotient(Dividend const &dividend, Divisor const &divisor)
calculates the quotient of two scaled_integer values
Definition: named.h:76
cnl::set_rounding_t
typename set_rounding< Number, RoundingTag >::type set_rounding_t
helper alias of set_rounding
Definition: set_rounding.h:46
std::max
T max(T... args)
std::numeric_limits