7 #if !defined(CNL_IMPL_PARSE_H)
8 #define CNL_IMPL_PARSE_H
10 #include "charconv/constants.h"
11 #include "charconv/descale.h"
12 #include "cnl_assert.h"
14 #include "narrow_cast.h"
15 #include "num_traits/digits.h"
16 #include "num_traits/max_digits.h"
17 #include "num_traits/set_digits.h"
18 #include "unreachable.h"
32 constexpr
auto strlen(
const char* str)
38 return narrow_cast<int>(it - str);
44 template<
typename Sum>
45 using scale_op_t = Sum (*)(Sum
const&);
47 constexpr
auto make_scale_op(
int base) -> scale_op_t<std::int64_t>
67 return unreachable<scale_op_t<std::int64_t>>(
"unsupported number base");
71 template<
typename Sum>
72 constexpr
auto make_scale_op_chunk(
int base) -> scale_op_t<Sum>
76 return [](Sum
const& sum) {
77 return Sum{(sum << 63)};
80 return [](Sum
const& sum) {
81 return Sum{(sum << 63)};
84 return [](Sum
const& sum) {
85 return Sum{(sum * 1
'000'000
'000'000
'000'000)};
88 return [](Sum
const& sum) {
89 return Sum{(sum << 60)};
92 return unreachable<scale_op_t<Sum>>(
"unsupported number base");
99 using char_to_digit_t = int (*)(char);
101 constexpr
auto make_char_to_digit_negative(
int base) -> char_to_digit_t
106 return (c >=
'0' && c <=
'1') ?
'0' - c : unreachable<int>(
"invalid binary digit");
110 return (c >=
'0' && c <=
'7') ?
'0' - c : unreachable<int>(
"invalid octal digit");
114 return (c >=
'0' && c <=
'9') ?
'0' - c : unreachable<int>(
"invalid decimal digit");
118 return (c >=
'0' && c <=
'9') ?
'0' - c : (c >=
'a' && c <=
'z') ? (
'a' - 10) - c
119 : (c >=
'A' && c <=
'Z') ? (
'A' - 10) - c
120 : unreachable<int>(
"invalid hexadecimal digit");
123 return unreachable<char_to_digit_t>(
"unsupported number base");
127 constexpr
auto make_char_to_digit_positive(
int base) -> char_to_digit_t
132 return (c >=
'0' && c <=
'1') ? c -
'0' : unreachable<int>(
"invalid binary digit");
136 return (c >=
'0' && c <=
'7') ? c -
'0' : unreachable<int>(
"invalid octal digit");
140 return (c >=
'0' && c <=
'9') ? c -
'0' : unreachable<int>(
"invalid decimal digit");
144 return (c >=
'0' && c <=
'9') ? c -
'0' : (c >=
'a' && c <=
'z') ? c - (
'a' - 10)
145 : (c >=
'A' && c <=
'Z') ? c - (
'A' - 10)
146 : unreachable<int>(
"invalid hexadecimal digit");
149 return unreachable<char_to_digit_t>(
"unsupported number base");
153 constexpr
auto make_char_to_digit(
bool is_negative,
int base)
156 ? make_char_to_digit_negative(base)
157 : make_char_to_digit_positive(base);
170 int num_fractional_digits;
173 constexpr
auto separator{
'\''};
175 [[nodiscard]] constexpr
auto scan_msb(
176 char const* str,
bool is_negative,
177 int base,
int stride,
int offset,
178 int max_num_bits,
int num_digits,
int num_fractional_digits)
186 auto const first_digit_char{str[offset + (str[offset] == radix_char)]};
187 auto const first_digit{make_char_to_digit_positive(base)(first_digit_char)};
192 max_num_bits - (first_digit * 2 < base),
194 num_fractional_digits};
197 [[nodiscard]] constexpr
auto scan_base(
char const* str,
bool is_negative,
int offset,
int length)
199 auto const last{str + length};
200 auto const found_radix{
std::find(str, last, radix_char)};
201 auto const has_radix{found_radix != last};
202 auto const post_radix{found_radix + has_radix};
203 auto const pre_radix_separators{narrow_cast<int>(
std::count(str, found_radix, separator))};
204 auto const post_radix_separators{narrow_cast<int>(
std::count(post_radix, last, separator))};
205 auto const num_separators{pre_radix_separators + post_radix_separators};
206 auto const num_non_separators{length - (num_separators + has_radix)};
207 auto const num_fractional_digits{
208 narrow_cast<int>(
std::distance(post_radix, last)) - post_radix_separators};
210 auto const is_decimal{str[offset] !=
'0' || has_radix};
211 if (is_decimal || offset + 1 >= num_non_separators) {
213 auto const num_digits{num_non_separators};
214 return scan_msb(str, is_negative, 10, 18, offset, (num_digits * 3322 + 678) / 1000, num_digits, num_fractional_digits);
216 switch (str[offset + 1]) {
219 auto const num_digits{num_non_separators - 2};
220 return scan_msb(str, is_negative, 2, 63, offset + 2, num_digits, num_digits, num_fractional_digits);
224 auto const num_digits{num_non_separators - 2};
225 return scan_msb(str, is_negative, 16, 15, offset + 2, num_digits * 4, num_digits, num_fractional_digits);
228 auto const num_digits{num_non_separators - 1};
229 return scan_msb(str, is_negative, 8, 21, offset + 1, num_digits * 3, num_digits, num_fractional_digits);
233 [[nodiscard]]
inline constexpr
auto scan_string(
char const* str,
int length)
237 return scan_base(str,
false, 1, length - 1);
239 return scan_base(str,
true, 1, length - 1);
241 return scan_base(str,
false, 0, length);
247 [[nodiscard]]
inline constexpr
auto scan_string(
char const (&str)[Length])
249 return scan_string(str, Length);
255 template<
typename Result>
256 [[nodiscard]] constexpr
auto parse_string(
257 char const* first,
int num_digits,
bool is_negative,
int base,
int stride)
259 auto const parse_int64 = [&num_digits, &first,
260 char_to_digit = make_char_to_digit(is_negative, base),
261 scale_op = make_scale_op(base)](
int n) {
264 CNL_ASSERT(num_digits >= 0);
266 auto const digit{*first++};
268 if (digit != separator && digit != radix_char) {
269 init = scale_op(init) + char_to_digit(digit);
276 auto const chunk_scale_op{make_scale_op_chunk<Result>(base)};
278 Result init(parse_int64((num_digits + stride) % stride));
283 init = chunk_scale_op(
std::move(init)) + parse_int64(stride);
289 template<
typename Result,
int NumChars>
290 [[nodiscard]] constexpr
auto parse_string(
292 char const (&str)[NumChars],
int num_digits,
bool is_negative,
int base,
int stride,
int first_numeral)
294 return parse_string<Result>(str + first_numeral, num_digits, is_negative, base, stride);
297 template<
typename Result,
int NumDigits,
bool IsNegative,
int Base,
int Stride,
int FirstNumeral,
char... Chars>
298 [[nodiscard]] constexpr
auto parse_string()
300 return parse_string<Result,
sizeof...(Chars)>({Chars...}, NumDigits, IsNegative, Base, Stride, FirstNumeral);
306 template<
typename Narrowest>
307 [[nodiscard]] constexpr
auto parse(
char const* str)
309 auto const length{
strlen(str)};
310 auto params{scan_string(str, length)};
311 return parse_string<Narrowest>(
312 str + params.first_numeral,
319 template<
typename Narrowest,
char... Chars>
320 [[nodiscard]] constexpr
auto parse_real()
322 constexpr
auto params{scan_string({Chars...})};
323 constexpr
auto result_digits{
324 std::max(digits_v<Narrowest>,
std::min(params.num_bits, max_digits<Narrowest>))};
325 using result_type = set_digits_t<Narrowest, result_digits>;
327 return descaled<result_type, params.base>{
334 params.first_numeral,
336 -params.num_fractional_digits};
339 template<
typename Narrowest,
char... Chars>
340 [[nodiscard]] constexpr
auto parse()
342 constexpr
auto parsed{parse_real<Narrowest, Chars...>()};
344 parsed.exponent == 0,
345 "non-integer number");
346 return parsed.significand;
351 #endif // CNL_IMPL_PARSE_H