tstr/to_uint.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114
mod sealed {
#[doc(hidden)]
pub trait Sealed: Sized {}
}
use sealed::Sealed;
/// Converts a [`TStr`] to unsigned integers.
///
/// # Example
///
/// ```rust
/// use tstr::{ToUint, TS, ts};
///
/// type Zero = TS!(0);
/// type N8 = TS!(8);
/// type N13 = TS!(13);
/// type N15 = TS!(0xF);
/// type N16 = TS!(0b10000);
///
/// assert_eq!(Zero::USIZE, 0);
/// assert_eq!(N8::USIZE, 8);
/// assert_eq!(N13::USIZE, 13);
/// assert_eq!(N15::USIZE, 15);
/// assert_eq!(N16::USIZE, 16);
///
/// assert_eq!(ts!(0).to_u128(), 0);
/// assert_eq!(ts!(8).to_u128(), 8);
/// assert_eq!(ts!(13).to_u128(), 13);
/// assert_eq!(ts!(0xF).to_u128(), 15);
/// assert_eq!(ts!(0b10000).to_u128(), 16);
///
/// ```
///
/// [`TStr`]: ./struct.TStr.html
pub trait ToUint: Sized {
/// The `usize` value of the type.
///
/// By default this value is a saturated cast from `Self::U128`.
const USIZE: usize = u128_as_usize(Self::U128);
/// The `u128` value of the type.
const U128: u128;
#[doc(hidden)]
const DIGITS: u32;
/// Gets the usize value of this type
///
/// By default this value is a saturated cast from `Self::U128`.
fn to_usize(&self) -> usize {
Self::USIZE
}
/// Gets the u128 value of this type
fn to_u128(&self) -> u128 {
Self::U128
}
}
#[cfg(feature = "const_generics")]
macro_rules! impl_for_const {
() => {
const fn str_to_u128(s: &str) -> u128 {
let s = s.as_bytes();
let mut out = 0u128;
let mut index = 0usize;
while index < s.len() {
let digit = s[index];
// This has the effect of panicking on non to '0' to '9' characters.
#[allow(clippy::no_effect)]
["Expected all characters to be digits"]
[!(b'0' <= digit && digit <= b'9') as usize];
let digit = (digit - b'0') as u128;
out = out * 10 + digit;
index += 1;
}
out
}
impl<const N: &'static str> Sealed for crate::___<N> {}
impl<const N: &'static str> ToUint for crate::___<N> {
const U128: u128 = str_to_u128(N);
const DIGITS: u32 = N.len() as u32;
}
};
}
impl<T> Sealed for crate::TStr<T> where T: Sealed {}
impl<T> ToUint for crate::TStr<T>
where
T: ToUint,
{
// Intentionally not setting this.
// const USIZE: usize = T::USIZE;
const U128: u128 = T::U128;
const DIGITS: u32 = T::DIGITS;
}
#[cfg(feature = "const_generics")]
impl_for_const! {}
#[cfg(not(feature = "const_generics"))]
mod impl_no_const_generics;
const fn u128_as_usize(n: u128) -> usize {
const MAXU: u128 = usize::max_value() as u128;
[n, MAXU][(n > MAXU) as usize] as usize
}