1#[cfg(not(feature = "ferrocene_subset"))]
5use crate::num::NonZero;
6
7#[cfg(not(feature = "ferrocene_subset"))]
9#[inline]
10const fn u8_impl(val: u8) -> u32 {
11 let val = val as u32;
12
13 const C1: u32 = 0b11_00000000 - 10; const C2: u32 = 0b10_00000000 - 100; ((val + C1) & (val + C2)) >> 8
27}
28
29#[cfg(not(feature = "ferrocene_subset"))]
31#[inline]
32const fn less_than_5(val: u32) -> u32 {
33 const C1: u32 = 0b011_00000000000000000 - 10; const C2: u32 = 0b100_00000000000000000 - 100; const C3: u32 = 0b111_00000000000000000 - 1000; const C4: u32 = 0b100_00000000000000000 - 10000; (((val + C1) & (val + C2)) ^ ((val + C3) & (val + C4))) >> 17
49}
50
51#[cfg(not(feature = "ferrocene_subset"))]
53#[inline]
54const fn u16_impl(val: u16) -> u32 {
55 less_than_5(val as u32)
56}
57
58#[cfg(not(feature = "ferrocene_subset"))]
60#[inline]
61const fn u32_impl(mut val: u32) -> u32 {
62 let mut log = 0;
63 if val >= 100_000 {
64 val /= 100_000;
65 log += 5;
66 }
67 log + less_than_5(val)
68}
69
70#[cfg(not(feature = "ferrocene_subset"))]
72#[inline]
73const fn u64_impl(mut val: u64) -> u32 {
74 let mut log = 0;
75 if val >= 10_000_000_000 {
76 val /= 10_000_000_000;
77 log += 10;
78 }
79 if val >= 100_000 {
80 val /= 100_000;
81 log += 5;
82 }
83 log + less_than_5(val as u32)
84}
85
86#[cfg(not(feature = "ferrocene_subset"))]
88#[inline]
89const fn u128_impl(mut val: u128) -> u32 {
90 let mut log = 0;
91 if val >= 100_000_000_000_000_000_000_000_000_000_000 {
92 val /= 100_000_000_000_000_000_000_000_000_000_000;
93 log += 32;
94 return log + u32_impl(val as u32);
95 }
96 if val >= 10_000_000_000_000_000 {
97 val /= 10_000_000_000_000_000;
98 log += 16;
99 }
100 log + u64_impl(val as u64)
101}
102
103#[cfg(not(feature = "ferrocene_subset"))]
104macro_rules! define_unsigned_ilog10 {
105 ($($ty:ident => $impl_fn:ident,)*) => {$(
106 #[inline]
107 pub(super) const fn $ty(val: NonZero<$ty>) -> u32 {
108 let result = $impl_fn(val.get());
109
110 unsafe { crate::hint::assert_unchecked(result <= const { $impl_fn($ty::MAX) }) };
113
114 result
115 }
116 )*};
117}
118
119#[cfg(not(feature = "ferrocene_subset"))]
120define_unsigned_ilog10! {
121 u8 => u8_impl,
122 u16 => u16_impl,
123 u32 => u32_impl,
124 u64 => u64_impl,
125 u128 => u128_impl,
126}
127
128#[cfg(not(feature = "ferrocene_subset"))]
129#[inline]
130pub(super) const fn usize(val: NonZero<usize>) -> u32 {
131 #[cfg(target_pointer_width = "16")]
132 let impl_fn = u16;
133
134 #[cfg(target_pointer_width = "32")]
135 let impl_fn = u32;
136
137 #[cfg(target_pointer_width = "64")]
138 let impl_fn = u64;
139
140 impl_fn(unsafe { NonZero::new_unchecked(val.get() as _) })
143}
144
145#[cfg(not(feature = "ferrocene_subset"))]
146macro_rules! define_signed_ilog10 {
147 ($($ty:ident => $impl_fn:ident,)*) => {$(
148 #[inline]
150 pub(super) const fn $ty(val: $ty) -> Option<u32> {
151 if val > 0 {
152 let result = $impl_fn(val.cast_unsigned());
153
154 unsafe {
157 crate::hint::assert_unchecked(result <= const { $impl_fn($ty::MAX.cast_unsigned()) });
158 }
159
160 Some(result)
161 } else {
162 None
163 }
164 }
165 )*};
166}
167
168#[cfg(not(feature = "ferrocene_subset"))]
169define_signed_ilog10! {
170 i8 => u8_impl,
171 i16 => u16_impl,
172 i32 => u32_impl,
173 i64 => u64_impl,
174 i128 => u128_impl,
175}
176
177#[cold]
180#[track_caller]
181pub(super) const fn panic_for_nonpositive_argument() -> ! {
182 panic!("argument of integer logarithm must be positive")
183}