lexical_util/
format_flags.rs

1//! Bitmask flags and masks for numeric formats.
2//!
3//! These bitflags and masks comprise a compressed struct as a 128-bit
4//! integer, allowing its use in const generics. This comprises two parts:
5//! flags designating which numerical components are valid in a string,
6//! and masks to designate the control characters.
7//!
8//! The flags are designated in the lower 64 bits that modify
9//! the syntax of strings that are parsed by lexical.
10//!
11//! Bits 8-32 are reserved for float component flags, such
12//! as for example if base prefixes or postfixes are case-sensitive,
13//! if leading zeros in a float are valid, etc.
14//!
15//! Bits 32-64 are reserved for digit separator flags. These
16//! define which locations within a float or integer digit separators
17//! are valid, for example, before any digits in the integer component,
18//! whether consecutive digit separators are allowed, and more.
19//!
20//! ```text
21//! 0   1   2   3   4   5   6   7   8   9   10  11  12  13  14  15  16
22//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
23//! |I/R|F/R|E/R|M/R|+/M|R/M|e/e|+/E|R/E|e/F|S/S|S/C|N/I|N/F|R/e|e/C|
24//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
25//!
26//! 16  17  18  19  20  21  22  23  24  25  26  27  28  29  30  31  32
27//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
28//! |e/P|e/S|                                                       |
29//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
30//!
31//! 32  33  34  35  36  37  38  39  40  41 42  43  44  45  46  47   48
32//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
33//! |I/I|F/I|E/I|I/L|F/L|E/L|I/T|F/T|E/T|I/C|F/C|E/C|S/D|           |
34//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
35//!
36//! 48  49  50  51  52  53  54  55  56  57  58  59  60  62  62  63  64
37//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
38//! |                                                               |
39//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
40//!
41//! Where:
42//!     Non-Digit Separator Flags:
43//!         I/R = Required integer digits.
44//!         F/R = Required fraction digits.
45//!         E/R = Required exponent digits.
46//!         M/R = Required mantissa digits.
47//!         +/M = No mantissa positive sign.
48//!         R/M = Required positive sign.
49//!         e/e = No exponent notation.
50//!         +/E = No exponent positive sign.
51//!         R/E = Required exponent sign.
52//!         e/F = No exponent without fraction.
53//!         S/S = No special (non-finite) values.
54//!         S/C = Case-sensitive special (non-finite) values.
55//!         N/I = No integer leading zeros.
56//!         N/F = No float leading zeros.
57//!         R/e = Required exponent characters.
58//!         e/C = Case-sensitive exponent character.
59//!         e/P = Case-sensitive base prefix.
60//!         e/S = Case-sensitive base suffix.
61//!
62//!     Digit Separator Flags:
63//!         I/I = Integer internal digit separator.
64//!         F/I = Fraction internal digit separator.
65//!         E/I = Exponent internal digit separator.
66//!         I/L = Integer leading digit separator.
67//!         F/L = Fraction leading digit separator.
68//!         E/L = Exponent leading digit separator.
69//!         I/T = Integer trailing digit separator.
70//!         F/T = Fraction trailing digit separator.
71//!         E/T = Exponent trailing digit separator.
72//!         I/C = Integer consecutive digit separator.
73//!         F/C = Fraction consecutive digit separator.
74//!         E/C = Exponent consecutive digit separator.
75//!         S/D = Special (non-finite) digit separator.
76//! ```
77//!
78//! The upper 64-bits are designated for control characters and radixes,
79//! such as the digit separator and base prefix characters, radixes,
80//! and more.
81//!
82//! ```text
83//! 64  65  66  67  68  69  70  71  72  73  74  75  76  77  78  79  80
84//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
85//! |     Digit Separator       |                                   |
86//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
87//!
88//! 80  81  82  83  84  85  86  87  88  89  90  91  92  93  94  95  96
89//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
90//! |                               |        Base Prefix        |   |
91//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
92//!
93//! 96  97  98  99  100 101 102 103 104 105 106 107 108 109 110 111 112
94//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
95//! |        Base Suffix        |   |    Mantissa Radix     |       |
96//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
97//!
98//! 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
99//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
100//! |     Exponent Base     |       |    Exponent Radix     |       |
101//! +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
102//! ```
103//!
104//!
105//! Note:
106//! -----
107//!
108//! In order to limit the format specification and avoid parsing
109//! non-numerical data, all number formats require some significant
110//! digits. Examples of always invalid numbers include:
111//! - ` `
112//! - `.`
113//! - `e`
114//! - `e7`
115//!
116//! Test Cases:
117//! -----------
118//!
119//! The following test-cases are used to define whether a literal or
120//! a string float is valid in a given language, and these tests are
121//! used to denote features in pre-defined formats. Only a few
122//! of these flags may modify the parsing behavior of integers.
123//! Integer parsing is assumed to be derived from float parsing,
124//! so if consecutive digit separators are valid in the integer
125//! component of a float, they are also valid in an integer.
126//!
127//! ```text
128//! 0: '.3'         // Non-required integer.
129//! 1: '3.'         // Non-required fraction.
130//! 2: '3e'         // Non-required exponent.
131//! 3. '+3.0'       // Mantissa positive sign.
132//! 4: '3.0e7'      // Exponent notation.
133//! 5: '3.0e+7'     // Exponent positive sign.
134//! 6. '3e7'        // Exponent notation without fraction.
135//! 7: 'NaN'        // Special (non-finite) values.
136//! 8: 'NAN'        // Case-sensitive special (non-finite) values.
137//! 9: '3_4.01'     // Integer internal digit separator.
138//! A: '3.0_1'      // Fraction internal digit separator.
139//! B: '3.0e7_1'    // Exponent internal digit separator.
140//! C: '_3.01'      // Integer leading digit separator.
141//! D: '3._01'      // Fraction leading digit separator.
142//! E: '3.0e_71'    // Exponent leading digit separator.
143//! F: '3_.01'      // Integer trailing digit separator.
144//! G: '3.01_'      // Fraction trailing digit separator.
145//! H: '3.0e71_'    // Exponent trailing digit separator.
146//! I: '3__4.01'    // Integer consecutive digit separator.
147//! J: '3.0__1'     // Fraction consecutive digit separator.
148//! K: '3.0e7__1'   // Exponent consecutive digit separator.
149//! L: 'In_f'       // Special (non-finite) digit separator.
150//! M: '010'        // No integer leading zeros.
151//! N: '010.0'      // No float leading zeros.
152//! O: '1.0'        // No required exponent notation.
153//! P: '3.0E7'      // Case-insensitive exponent character.
154//! P: '0x3.0'      // Case-insensitive base prefix.
155//! P: '3.0H'       // Case-insensitive base postfix.
156//! ```
157//!
158//! Currently Supported Programming and Data Languages:
159//! ---------------------------------------------------
160//!
161//! 1. `Rust`
162//! 2. `Python`
163//! 3. `C++` (98, 03, 11, 14, 17)
164//! 4. `C` (89, 90, 99, 11, 18)
165//! 5. `Ruby`
166//! 6. `Swift`
167//! 7. `Go`
168//! 8. `Haskell`
169//! 9. `Javascript`
170//! 10. `Perl`
171//! 11. `PHP`
172//! 12. `Java`
173//! 13. `R`
174//! 14. `Kotlin`
175//! 15. `Julia`
176//! 16. `C#` (ISO-1, ISO-2, 3, 4, 5, 6, 7)
177//! 17. `Kawa`
178//! 18. `Gambit-C`
179//! 19. `Guile`
180//! 20. `Clojure`
181//! 21. `Erlang`
182//! 22. `Elm`
183//! 23. `Scala`
184//! 24. `Elixir`
185//! 25. `FORTRAN`
186//! 26. `D`
187//! 27. `Coffeescript`
188//! 28. `Cobol`
189//! 29. `F#`
190//! 30. `Visual Basic`
191//! 31. `OCaml`
192//! 32. `Objective-C`
193//! 33. `ReasonML`
194//! 34. `Octave`
195//! 35. `Matlab`
196//! 36. `Zig`
197//! 37. `SageMath`
198//! 38. `JSON`
199//! 39. `TOML`
200//! 40. `XML`
201//! 41. `SQLite`
202//! 42. `PostgreSQL`
203//! 43. `MySQL`
204//! 44. `MongoDB`
205
206#![cfg_attr(rustfmt, rustfmt::skip)]
207
208use static_assertions::const_assert;
209
210// ASSERTIONS
211// ----------
212
213// Ensure all our bit flags are valid.
214macro_rules! check_subsequent_flags {
215    ($x:ident, $y:ident) => {
216        const_assert!($x << 1 == $y);
217    };
218}
219
220// Ensure all our bit masks don't overlap.
221macro_rules! check_subsequent_masks {
222    ($x:ident, $y:ident) => {
223        const_assert!($x & $y == 0);
224    };
225}
226
227// Check all our masks are in the range `[0, 255]` after shifting.
228macro_rules! check_mask_shifts {
229    ($mask:ident, $shift:ident) => {
230        const_assert!(0 < $mask >> $shift && 255 >= $mask >> $shift);
231    };
232}
233
234// Ensure all our bit masks don't overlap with existing flags.
235macro_rules! check_masks_and_flags {
236    ($x:ident, $y:ident) => {
237        const_assert!($x & $y == 0);
238    };
239}
240
241// NON-DIGIT SEPARATOR FLAGS & MASKS
242// ---------------------------------
243
244/// Digits are required before the decimal point.
245pub const REQUIRED_INTEGER_DIGITS: u128 = 1 << 0;
246
247/// Digits are required after the decimal point.
248/// This check will only occur if the decimal point is present.
249pub const REQUIRED_FRACTION_DIGITS: u128 = 1 << 1;
250
251/// Digits are required after the exponent character.
252/// This check will only occur if the exponent character is present.
253pub const REQUIRED_EXPONENT_DIGITS: u128 = 1 << 2;
254
255/// Mantissa digits are required (either before or after the decimal point).
256pub const REQUIRED_MANTISSA_DIGITS: u128 = 1 << 3;
257
258/// At least 1 digit in the number is required.
259pub const REQUIRED_DIGITS: u128 =
260    REQUIRED_INTEGER_DIGITS |
261    REQUIRED_FRACTION_DIGITS |
262    REQUIRED_EXPONENT_DIGITS |
263    REQUIRED_MANTISSA_DIGITS;
264
265/// Positive sign before the mantissa is not allowed.
266pub const NO_POSITIVE_MANTISSA_SIGN: u128 = 1 << 4;
267
268/// Positive sign before the mantissa is required.
269pub const REQUIRED_MANTISSA_SIGN: u128 = 1 << 5;
270
271/// Exponent notation is not allowed.
272pub const NO_EXPONENT_NOTATION: u128 = 1 << 6;
273
274/// Positive sign before the exponent is not allowed.
275pub const NO_POSITIVE_EXPONENT_SIGN: u128 = 1 << 7;
276
277/// Positive sign before the exponent is required.
278pub const REQUIRED_EXPONENT_SIGN: u128 = 1 << 8;
279
280/// Exponent without a fraction component is not allowed.
281///
282/// This only checks if a decimal point precedes the exponent character.
283/// To require fraction digits or exponent digits with this check,
284/// please use the appropriate flags.
285pub const NO_EXPONENT_WITHOUT_FRACTION: u128 = 1 << 9;
286
287/// Special (non-finite) values are not allowed.
288pub const NO_SPECIAL: u128 = 1 << 10;
289
290/// Special (non-finite) values are case-sensitive.
291pub const CASE_SENSITIVE_SPECIAL: u128 = 1 << 11;
292
293/// Leading zeros before an integer value are not allowed.
294///
295/// If the value is a literal, then this distinction applies
296/// when the value is treated like an integer literal, typically
297/// when there is no decimal point. If the value is parsed,
298/// then this distinction applies when the value as parsed
299/// as an integer.
300///
301/// # Warning
302///
303/// This also does not mean that the value parsed will be correct,
304/// for example, in languages like C, this will not auto-
305/// deduce that the radix is 8 with leading zeros, for an octal
306/// literal.
307pub const NO_INTEGER_LEADING_ZEROS: u128 = 1 << 12;
308
309/// Leading zeros before a float value are not allowed.
310///
311/// If the value is a literal, then this distinction applies
312/// when the value is treated like an integer float, typically
313/// when there is a decimal point. If the value is parsed,
314/// then this distinction applies when the value as parsed
315/// as a float.
316///
317/// # Warning
318///
319/// This also does not mean that the value parsed will be correct,
320/// for example, in languages like C, this will not auto-
321/// deduce that the radix is 8 with leading zeros, for an octal
322/// literal.
323pub const NO_FLOAT_LEADING_ZEROS: u128 = 1 << 13;
324
325/// Exponent notation is required.
326///
327/// Valid floats must contain an exponent notation character, and if
328/// applicable, a sign character and digits afterwards.
329pub const REQUIRED_EXPONENT_NOTATION: u128 = 1 << 14;
330
331/// Exponent characters are case-sensitive.
332pub const CASE_SENSITIVE_EXPONENT: u128 = 1 << 15;
333
334/// Base prefixes are case-sensitive.
335pub const CASE_SENSITIVE_BASE_PREFIX: u128 = 1 << 16;
336
337/// Base suffixes are case-sensitive.
338pub const CASE_SENSITIVE_BASE_SUFFIX: u128 = 1 << 17;
339
340// Non-digit separator flags.
341const_assert!(REQUIRED_INTEGER_DIGITS == 1);
342check_subsequent_flags!(REQUIRED_INTEGER_DIGITS, REQUIRED_FRACTION_DIGITS);
343check_subsequent_flags!(REQUIRED_FRACTION_DIGITS, REQUIRED_EXPONENT_DIGITS);
344check_subsequent_flags!(REQUIRED_EXPONENT_DIGITS, REQUIRED_MANTISSA_DIGITS);
345check_subsequent_flags!(REQUIRED_MANTISSA_DIGITS, NO_POSITIVE_MANTISSA_SIGN);
346check_subsequent_flags!(NO_POSITIVE_MANTISSA_SIGN, REQUIRED_MANTISSA_SIGN);
347check_subsequent_flags!(REQUIRED_MANTISSA_SIGN, NO_EXPONENT_NOTATION);
348check_subsequent_flags!(NO_EXPONENT_NOTATION, NO_POSITIVE_EXPONENT_SIGN);
349check_subsequent_flags!(NO_POSITIVE_EXPONENT_SIGN, REQUIRED_EXPONENT_SIGN);
350check_subsequent_flags!(REQUIRED_EXPONENT_SIGN, NO_EXPONENT_WITHOUT_FRACTION);
351check_subsequent_flags!(NO_EXPONENT_WITHOUT_FRACTION, NO_SPECIAL);
352check_subsequent_flags!(NO_SPECIAL, CASE_SENSITIVE_SPECIAL);
353check_subsequent_flags!(NO_SPECIAL, CASE_SENSITIVE_SPECIAL);
354check_subsequent_flags!(CASE_SENSITIVE_SPECIAL, NO_INTEGER_LEADING_ZEROS);
355check_subsequent_flags!(NO_INTEGER_LEADING_ZEROS, NO_FLOAT_LEADING_ZEROS);
356check_subsequent_flags!(NO_FLOAT_LEADING_ZEROS, REQUIRED_EXPONENT_NOTATION);
357check_subsequent_flags!(REQUIRED_EXPONENT_NOTATION, CASE_SENSITIVE_EXPONENT);
358check_subsequent_flags!(CASE_SENSITIVE_EXPONENT, CASE_SENSITIVE_BASE_PREFIX);
359check_subsequent_flags!(CASE_SENSITIVE_BASE_PREFIX, CASE_SENSITIVE_BASE_SUFFIX);
360
361// DIGIT SEPARATOR FLAGS & MASKS
362// -----------------------------
363
364/// Digit separators are allowed between integer digits.
365pub const INTEGER_INTERNAL_DIGIT_SEPARATOR: u128 = 1 << 32;
366
367/// Digit separators are allowed between fraction digits.
368pub const FRACTION_INTERNAL_DIGIT_SEPARATOR: u128 = 1 << 33;
369
370/// Digit separators are allowed between exponent digits.
371pub const EXPONENT_INTERNAL_DIGIT_SEPARATOR: u128 = 1 << 34;
372
373/// A digit separator is allowed before any integer digits.
374pub const INTEGER_LEADING_DIGIT_SEPARATOR: u128 = 1 << 35;
375
376/// A digit separator is allowed before any fraction digits.
377pub const FRACTION_LEADING_DIGIT_SEPARATOR: u128 = 1 << 36;
378
379/// A digit separator is allowed before any exponent digits.
380pub const EXPONENT_LEADING_DIGIT_SEPARATOR: u128 = 1 << 37;
381
382/// A digit separator is allowed after any integer digits.
383pub const INTEGER_TRAILING_DIGIT_SEPARATOR: u128 = 1 << 38;
384
385/// A digit separator is allowed after any fraction digits.
386pub const FRACTION_TRAILING_DIGIT_SEPARATOR: u128 = 1 << 39;
387
388/// A digit separator is allowed after any exponent digits.
389pub const EXPONENT_TRAILING_DIGIT_SEPARATOR: u128 = 1 << 40;
390
391/// Multiple consecutive integer digit separators are allowed.
392pub const INTEGER_CONSECUTIVE_DIGIT_SEPARATOR: u128 = 1 << 41;
393
394/// Multiple consecutive fraction digit separators are allowed.
395pub const FRACTION_CONSECUTIVE_DIGIT_SEPARATOR: u128 = 1 << 42;
396
397/// Multiple consecutive exponent digit separators are allowed.
398pub const EXPONENT_CONSECUTIVE_DIGIT_SEPARATOR: u128 = 1 << 43;
399
400/// Digit separators are allowed between digits.
401pub const INTERNAL_DIGIT_SEPARATOR: u128 =
402    INTEGER_INTERNAL_DIGIT_SEPARATOR |
403    FRACTION_INTERNAL_DIGIT_SEPARATOR |
404    EXPONENT_INTERNAL_DIGIT_SEPARATOR;
405
406/// A digit separator is allowed before any digits.
407pub const LEADING_DIGIT_SEPARATOR: u128 =
408    INTEGER_LEADING_DIGIT_SEPARATOR |
409    FRACTION_LEADING_DIGIT_SEPARATOR |
410    EXPONENT_LEADING_DIGIT_SEPARATOR;
411
412/// A digit separator is allowed after any digits.
413pub const TRAILING_DIGIT_SEPARATOR: u128 =
414    INTEGER_TRAILING_DIGIT_SEPARATOR |
415    FRACTION_TRAILING_DIGIT_SEPARATOR |
416    EXPONENT_TRAILING_DIGIT_SEPARATOR;
417
418/// Multiple consecutive digit separators are allowed.
419pub const CONSECUTIVE_DIGIT_SEPARATOR: u128 =
420    INTEGER_CONSECUTIVE_DIGIT_SEPARATOR |
421    FRACTION_CONSECUTIVE_DIGIT_SEPARATOR |
422    EXPONENT_CONSECUTIVE_DIGIT_SEPARATOR;
423
424/// Any digit separators are allowed in special (non-finite) values.
425pub const SPECIAL_DIGIT_SEPARATOR: u128 = 1 << 44;
426
427// Digit separator flags.
428const_assert!(INTEGER_INTERNAL_DIGIT_SEPARATOR == 1 << 32);
429check_subsequent_flags!(INTEGER_INTERNAL_DIGIT_SEPARATOR, FRACTION_INTERNAL_DIGIT_SEPARATOR);
430check_subsequent_flags!(FRACTION_INTERNAL_DIGIT_SEPARATOR, EXPONENT_INTERNAL_DIGIT_SEPARATOR);
431check_subsequent_flags!(EXPONENT_INTERNAL_DIGIT_SEPARATOR, INTEGER_LEADING_DIGIT_SEPARATOR);
432check_subsequent_flags!(INTEGER_LEADING_DIGIT_SEPARATOR, FRACTION_LEADING_DIGIT_SEPARATOR);
433check_subsequent_flags!(FRACTION_LEADING_DIGIT_SEPARATOR, EXPONENT_LEADING_DIGIT_SEPARATOR);
434check_subsequent_flags!(EXPONENT_LEADING_DIGIT_SEPARATOR, INTEGER_TRAILING_DIGIT_SEPARATOR);
435check_subsequent_flags!(INTEGER_TRAILING_DIGIT_SEPARATOR, FRACTION_TRAILING_DIGIT_SEPARATOR);
436check_subsequent_flags!(FRACTION_TRAILING_DIGIT_SEPARATOR, EXPONENT_TRAILING_DIGIT_SEPARATOR);
437check_subsequent_flags!(EXPONENT_TRAILING_DIGIT_SEPARATOR, INTEGER_CONSECUTIVE_DIGIT_SEPARATOR);
438check_subsequent_flags!(INTEGER_CONSECUTIVE_DIGIT_SEPARATOR, FRACTION_CONSECUTIVE_DIGIT_SEPARATOR);
439check_subsequent_flags!(FRACTION_CONSECUTIVE_DIGIT_SEPARATOR, EXPONENT_CONSECUTIVE_DIGIT_SEPARATOR);
440check_subsequent_flags!(EXPONENT_CONSECUTIVE_DIGIT_SEPARATOR, SPECIAL_DIGIT_SEPARATOR);
441
442// CONTROL CHARACTER & RADIX MASKS
443// -------------------------------
444
445/// Shift to convert to and from a digit separator as a `u8`.
446pub const DIGIT_SEPARATOR_SHIFT: i32 = 64;
447
448/// Mask to extract the digit separator character.
449pub const DIGIT_SEPARATOR: u128 = 0xFF << DIGIT_SEPARATOR_SHIFT;
450
451/// Shift to convert to and from a base prefix as a `u8`.
452pub const BASE_PREFIX_SHIFT: i32 = 88;
453
454/// Mask to extract the base prefix character.
455pub const BASE_PREFIX: u128 = 0xFF << BASE_PREFIX_SHIFT;
456
457/// Shift to convert to and from a base suffix as a `u8`.
458pub const BASE_SUFFIX_SHIFT: i32 = 96;
459
460/// Mask to extract the base suffix character.
461pub const BASE_SUFFIX: u128 = 0xFF << BASE_SUFFIX_SHIFT;
462
463/// Shift to convert to and from a mantissa radix as a `u32`.
464pub const MANTISSA_RADIX_SHIFT: i32 = 104;
465
466/// Mask to extract the mantissa radix: the radix for the significant digits.
467pub const MANTISSA_RADIX: u128 = 0xFF << MANTISSA_RADIX_SHIFT;
468
469/// Alias for `MANTISSA_RADIX_SHIFT`.
470pub const RADIX_SHIFT: i32 = MANTISSA_RADIX_SHIFT;
471
472/// Alias for `MANTISSA_RADIX`.
473pub const RADIX: u128 = MANTISSA_RADIX;
474
475/// Shift to convert to and from an exponent base as a `u32`.
476pub const EXPONENT_BASE_SHIFT: i32 = 112;
477
478/// Mask to extract the exponent base: the base the exponent is raised to.
479pub const EXPONENT_BASE: u128 = 0xFF << EXPONENT_BASE_SHIFT;
480
481/// Shift to convert to and from an exponent radix as a `u32`.
482pub const EXPONENT_RADIX_SHIFT: i32 = 120;
483
484/// Mask to extract the exponent radix: the radix for the exponent digits.
485pub const EXPONENT_RADIX: u128 = 0xFF << EXPONENT_RADIX_SHIFT;
486
487// Masks do not overlap.
488check_subsequent_masks!(DIGIT_SEPARATOR, BASE_PREFIX);
489check_subsequent_masks!(BASE_PREFIX, BASE_SUFFIX);
490check_subsequent_masks!(BASE_SUFFIX, MANTISSA_RADIX);
491check_subsequent_masks!(MANTISSA_RADIX, EXPONENT_BASE);
492check_subsequent_masks!(EXPONENT_BASE, EXPONENT_RADIX);
493
494// Check all our shifts shift the masks to a single byte.
495check_mask_shifts!(DIGIT_SEPARATOR, DIGIT_SEPARATOR_SHIFT);
496check_mask_shifts!(BASE_PREFIX, BASE_PREFIX_SHIFT);
497check_mask_shifts!(BASE_SUFFIX, BASE_SUFFIX_SHIFT);
498check_mask_shifts!(MANTISSA_RADIX, MANTISSA_RADIX_SHIFT);
499check_mask_shifts!(EXPONENT_BASE, EXPONENT_BASE_SHIFT);
500check_mask_shifts!(EXPONENT_RADIX, EXPONENT_RADIX_SHIFT);
501
502// Check masks don't overlap with neighboring flags.
503check_masks_and_flags!(DIGIT_SEPARATOR, SPECIAL_DIGIT_SEPARATOR);
504
505// HIDDEN MASKS
506// ------------
507
508/// Mask to extract the flag bits.
509#[doc(hidden)]
510pub const FLAG_MASK: u128 =
511    REQUIRED_DIGITS |
512    NO_POSITIVE_MANTISSA_SIGN |
513    REQUIRED_MANTISSA_SIGN |
514    NO_EXPONENT_NOTATION |
515    NO_POSITIVE_EXPONENT_SIGN |
516    REQUIRED_EXPONENT_SIGN |
517    NO_EXPONENT_WITHOUT_FRACTION |
518    NO_SPECIAL |
519    CASE_SENSITIVE_SPECIAL |
520    NO_INTEGER_LEADING_ZEROS |
521    NO_FLOAT_LEADING_ZEROS |
522    REQUIRED_EXPONENT_NOTATION |
523    CASE_SENSITIVE_EXPONENT |
524    CASE_SENSITIVE_BASE_PREFIX |
525    CASE_SENSITIVE_BASE_SUFFIX |
526    INTERNAL_DIGIT_SEPARATOR |
527    LEADING_DIGIT_SEPARATOR |
528    TRAILING_DIGIT_SEPARATOR |
529    CONSECUTIVE_DIGIT_SEPARATOR |
530    SPECIAL_DIGIT_SEPARATOR;
531
532/// Mask to extract the flag bits controlling interface parsing.
533///
534/// This mask controls all the flags handled by the interface,
535/// omitting those that are handled prior. This limits the
536/// number of match paths required to determine the correct
537/// interface.
538#[doc(hidden)]
539pub const INTERFACE_FLAG_MASK: u128 =
540    REQUIRED_DIGITS |
541    NO_EXPONENT_NOTATION |
542    NO_POSITIVE_EXPONENT_SIGN |
543    REQUIRED_EXPONENT_SIGN |
544    NO_EXPONENT_WITHOUT_FRACTION |
545    NO_FLOAT_LEADING_ZEROS |
546    REQUIRED_EXPONENT_NOTATION |
547    INTERNAL_DIGIT_SEPARATOR |
548    LEADING_DIGIT_SEPARATOR |
549    TRAILING_DIGIT_SEPARATOR |
550    CONSECUTIVE_DIGIT_SEPARATOR;
551
552/// Mask to extract digit separator flags.
553#[doc(hidden)]
554pub const DIGIT_SEPARATOR_FLAG_MASK: u128 =
555    INTERNAL_DIGIT_SEPARATOR |
556    LEADING_DIGIT_SEPARATOR |
557    TRAILING_DIGIT_SEPARATOR |
558    CONSECUTIVE_DIGIT_SEPARATOR |
559    SPECIAL_DIGIT_SEPARATOR;
560
561/// Mask to extract exponent flags.
562#[doc(hidden)]
563pub const EXPONENT_FLAG_MASK: u128 =
564    REQUIRED_EXPONENT_DIGITS |
565    NO_EXPONENT_NOTATION |
566    NO_POSITIVE_EXPONENT_SIGN |
567    REQUIRED_EXPONENT_SIGN |
568    NO_EXPONENT_WITHOUT_FRACTION |
569    REQUIRED_EXPONENT_NOTATION |
570    EXPONENT_INTERNAL_DIGIT_SEPARATOR |
571    EXPONENT_LEADING_DIGIT_SEPARATOR |
572    EXPONENT_TRAILING_DIGIT_SEPARATOR |
573    EXPONENT_CONSECUTIVE_DIGIT_SEPARATOR;
574
575/// Mask to extract integer digit separator flags.
576#[doc(hidden)]
577pub const INTEGER_DIGIT_SEPARATOR_FLAG_MASK: u128 =
578    INTEGER_INTERNAL_DIGIT_SEPARATOR |
579    INTEGER_LEADING_DIGIT_SEPARATOR |
580    INTEGER_TRAILING_DIGIT_SEPARATOR |
581    INTEGER_CONSECUTIVE_DIGIT_SEPARATOR;
582
583/// Mask to extract fraction digit separator flags.
584#[doc(hidden)]
585pub const FRACTION_DIGIT_SEPARATOR_FLAG_MASK: u128 =
586    FRACTION_INTERNAL_DIGIT_SEPARATOR |
587    FRACTION_LEADING_DIGIT_SEPARATOR |
588    FRACTION_TRAILING_DIGIT_SEPARATOR |
589    FRACTION_CONSECUTIVE_DIGIT_SEPARATOR;
590
591/// Mask to extract exponent digit separator flags.
592#[doc(hidden)]
593pub const EXPONENT_DIGIT_SEPARATOR_FLAG_MASK: u128 =
594    EXPONENT_INTERNAL_DIGIT_SEPARATOR |
595    EXPONENT_LEADING_DIGIT_SEPARATOR |
596    EXPONENT_TRAILING_DIGIT_SEPARATOR |
597    EXPONENT_CONSECUTIVE_DIGIT_SEPARATOR;
598
599// EXTRACTORS
600// ----------
601
602/// Extract the digit separator from the format packed struct.
603#[inline(always)]
604pub const fn digit_separator(format: u128) -> u8 {
605    ((format & DIGIT_SEPARATOR) >> DIGIT_SEPARATOR_SHIFT) as u8
606}
607
608/// Extract the base prefix character from the format packed struct.
609#[inline(always)]
610pub const fn base_prefix(format: u128) -> u8 {
611    ((format & BASE_PREFIX) >> BASE_PREFIX_SHIFT) as u8
612}
613
614/// Extract the base suffix character from the format packed struct.
615#[inline(always)]
616pub const fn base_suffix(format: u128) -> u8 {
617    ((format & BASE_SUFFIX) >> BASE_SUFFIX_SHIFT) as u8
618}
619
620/// Extract the mantissa radix from the format packed struct.
621#[inline(always)]
622pub const fn mantissa_radix(format: u128) -> u32 {
623    ((format & MANTISSA_RADIX) >> MANTISSA_RADIX_SHIFT) as u32
624}
625
626/// Extract the exponent base from the format packed struct.
627/// If not provided, defaults to `mantissa_radix`.
628#[inline(always)]
629pub const fn exponent_base(format: u128) -> u32 {
630    let radix = ((format & EXPONENT_BASE) >> EXPONENT_BASE_SHIFT) as u32;
631    if radix == 0 {
632        mantissa_radix(format)
633    } else {
634        radix
635    }
636}
637
638/// Extract the exponent radix from the format packed struct.
639/// If not provided, defaults to `mantissa_radix`.
640#[inline(always)]
641pub const fn exponent_radix(format: u128) -> u32 {
642    let radix = ((format & EXPONENT_RADIX) >> EXPONENT_RADIX_SHIFT) as u32;
643    if radix == 0 {
644        mantissa_radix(format)
645    } else {
646        radix
647    }
648}
649
650/// Extract a generic radix from the format and bitflags.
651#[inline(always)]
652pub const fn radix_from_flags(format: u128, mask: u128, shift: i32) -> u32 {
653    let radix = ((format & mask) >> shift) as u32;
654    if radix == 0 {
655        mantissa_radix(format)
656    } else {
657        radix
658    }
659}
660
661// VALIDATORS
662// ----------
663
664// NOTE: All of these are only used when building formats so it doesn't matter if
665// they have performance issues, since these will be built at compile time.
666
667/// Determine if the provided exponent flags are valid.
668#[inline(always)]
669pub const fn is_valid_exponent_flags(format: u128) -> bool {
670    // Both cannot be set.
671    format & NO_EXPONENT_NOTATION == 0 || format & REQUIRED_EXPONENT_NOTATION == 0
672}
673
674/// Determine if an optional control character is valid.
675#[inline(always)]
676const fn is_valid_optional_control_radix(radix: u32, value: u8) -> bool {
677    // Validate the character isn't a digit or sign character, and is valid ASCII.
678    use crate::ascii::is_valid_ascii;
679    use crate::digit::char_is_digit_const;
680    !char_is_digit_const(value, radix) &&
681        value != b'+' &&
682        value != b'-' &&
683        (is_valid_ascii(value) || value == 0)
684}
685
686/// Determine if an optional control character is valid.
687#[inline(always)]
688const fn is_valid_optional_control(format: u128, value: u8) -> bool {
689    // Need to get the larger of the two radix values, since these
690    // will be the characters that define the valid digits.
691    // const fn doesn't support max as of 1.55 nightly.
692    let mradix = mantissa_radix(format);
693    let eradix = exponent_radix(format);
694    let radix = if mradix > eradix {
695        mradix
696    } else {
697        eradix
698    };
699    is_valid_optional_control_radix(radix, value)
700}
701
702/// Determine if an control character is valid.
703#[inline(always)]
704const fn is_valid_control(format: u128, value: u8) -> bool {
705    value != 0 && is_valid_optional_control(format, value)
706}
707
708/// Determine if the digit separator is valid.
709/// Digit separators must not be valid digits or sign characters.
710#[inline(always)]
711pub const fn is_valid_digit_separator(format: u128) -> bool {
712    let value = digit_separator(format);
713    if cfg!(feature = "format") {
714        is_valid_optional_control(format, value)
715    } else {
716        value == 0
717    }
718}
719
720/// Determine if the base prefix character is valid.
721#[inline(always)]
722pub const fn is_valid_base_prefix(format: u128) -> bool {
723    let value = base_prefix(format);
724    if cfg!(feature = "format") {
725        is_valid_optional_control(format, value)
726    } else {
727        value == 0
728    }
729}
730
731/// Determine if the base suffix character is valid.
732#[inline(always)]
733pub const fn is_valid_base_suffix(format: u128) -> bool {
734    let value = base_suffix(format);
735    if cfg!(feature = "format") {
736        is_valid_optional_control(format, value)
737    } else {
738        value == 0
739    }
740}
741
742/// Determine if all of the "punctuation" characters are valid.
743#[inline(always)]
744#[allow(clippy::if_same_then_else)] // reason="all are different logic conditions"
745pub const fn is_valid_punctuation(format: u128) -> bool {
746    // All the checks against optional characters with mandatory are fine:
747    // if they're not 0, then they can't overlap, and mandatory can't be 0.
748    if cfg!(not(feature = "format")) && digit_separator(format) != 0 {
749        // Digit separator set when not allowed.
750        false
751    } else {
752        let separator = digit_separator(format);
753        let prefix = base_prefix(format);
754        let suffix = base_suffix(format);
755        // Check all are optional, or enough are not present.
756        match (separator, prefix, suffix) {
757            (0, 0, 0) => true,
758            (_, 0, 0) => true,
759            (0, _, 0) => true,
760            (0, 0, _) => true,
761            // Can't have more than 1 0, check they're all different.
762            (x, y, z) => x != y && x != z && y != z,
763        }
764    }
765}
766
767/// Determine if all of the "punctuation" characters for the options API are valid.
768#[inline(always)]
769#[allow(clippy::if_same_then_else)] // reason="all are different logic conditions"
770#[allow(clippy::needless_bool)] // reason="not needless depending on the format condition"
771pub const fn is_valid_options_punctuation(format: u128, exponent: u8, decimal_point: u8) -> bool {
772    // All the checks against optional characters with mandatory are fine:
773    // if they're not 0, then they can't overlap, and mandatory can't be 0.
774    if !is_valid_control(format, decimal_point) || !is_valid_control(format, exponent) {
775        // Must be in the valid range.
776        false
777    } else if decimal_point == exponent {
778        // Can't have overlapping characters.
779        false
780    } else if cfg!(feature = "format") && digit_separator(format) == decimal_point {
781        false
782    } else if cfg!(feature = "format") && digit_separator(format) == exponent {
783        false
784    } else if cfg!(feature = "format") && base_prefix(format) == decimal_point {
785        false
786    } else if cfg!(feature = "format") && base_prefix(format) == exponent {
787        false
788    } else if cfg!(feature = "format") && base_suffix(format) == decimal_point {
789        false
790    } else if cfg!(feature = "format") && base_suffix(format) == exponent {
791        false
792    } else {
793        true
794    }
795}
796
797/// Determine if the radix is valid.
798pub const fn is_valid_radix(radix: u32) -> bool {
799    if cfg!(feature = "radix") {
800        radix >= 2 && radix <= 36
801    } else if cfg!(feature = "power-of-two") {
802        matches!(radix, 2 | 4 | 8 | 10 | 16 | 32)
803    } else {
804        radix == 10
805    }
806}