1#[allow(unused_imports)]
4use crate::{
5 _core::{fmt, hash, ops},
6 compile, iif, isize_up, paste, usize_up,
7};
8
9#[doc = crate::doc_!(vendor: "quickdiv")]
28#[must_use]
29#[derive(Clone, Copy)]
30pub struct Divisor<T> {
31 inner: DivisorInner<T>,
32}
33
34#[derive(Clone, Copy)]
36enum DivisorInner<T> {
37 Shift(T, u8),
38 MultiplyShift(T, T, u8),
39 MultiplyAddShift(T, T, u8),
40 #[allow(dead_code)]
42 ShiftAndNegate(T, u8),
43 #[allow(dead_code)]
45 MultiplyAddShiftNegate(T, T, u8),
46}
47
48macro_rules! impl_divisor {
50 () => {
51 impl_divisor![signed
52 i8|u8|i16|u16:Y:"_int_i8", i16|u16|i32|u32:Y:"_int_i16", i32|u32|i64|u64:Y:"_int_i32",
53 i64|u64|i128|u128:PW:"_int_i64", i128|u128|i128|u128:N:"_int_i128",
54 isize|usize|isize_up|usize_up:Y:"_int_isize"
55 ];
56 impl_divisor![unsigned
57 u8|u16:Y:"_int_u8", u16|u32:Y:"_int_u16", u32|u64:Y:"_int_u32",
58 u64|u128:PW:"_int_u64", u128|u128:N:"_int_u128", usize|usize_up:Y:"_int_usize"
59 ];
60 };
61
62 (
63 signed $( $t:ty | $un:ty | $up:ty | $unup:ty : $is_up:ident : $cap:literal),+) => {
73 $( impl_divisor![@signed $t|$un|$up|$unup:$is_up:$cap]; )+
74 };
75 (unsigned $( $t:ty | $up:ty : $is_up:ident : $cap:literal),+) => {
76 $( impl_divisor![@unsigned $t|$up:$is_up:$cap]; )+
77 };
78 (
79 @signed $t:ty | $un:ty | $up:ty | $unup:ty : $is_up:ident : $cap:literal) => {
81 #[cfg(feature = $cap )]
82 impl_divisor![@traits $t];
83
84 #[cfg(feature = $cap )]
85 #[doc = crate::doc_availability!(feature = $cap)]
86 impl Divisor<$t> {
88 impl_divisor![@shared $t|$un|$up|$unup:$is_up:$cap]; #[must_use]
92 const fn abs(n: $t) -> $un {
93 iif![n < 0; ((-1i8) as $un).wrapping_mul(n as $un); n as $un]
94 }
95
96 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(-21).unwrap();"]]
105 #[must_use]
107 pub const fn new(d: $t) -> Option<Divisor<$t>> {
108 if d == 0 {
109 Self::cold_0_divisor()
110 } else {
111 let ud = Self::abs(d);
112 let shift = ud.ilog2() as u8;
113 let inner = if ud.is_power_of_two() {
114 iif![d > 0; DivisorInner::Shift(d, shift); DivisorInner::ShiftAndNegate(d, shift)]
115 } else {
116 let (mut magic, rem) = Self::div_rem_wide_by_base(1 << (shift - 1), ud);
117 let e = ud - rem;
118 if e < 1 << shift {
119 DivisorInner::MultiplyShift(d, d.signum() * (magic as $t + 1), shift - 1)
120 } else {
121 magic *= 2;
122 let (doubled_rem, overflowed) = rem.overflowing_mul(2);
123 iif![doubled_rem >= ud || overflowed; magic += 1];
124 magic += 1;
125 if d > 0 {
126 DivisorInner::MultiplyAddShift(d, magic as $t, shift)
127 } else {
128 DivisorInner::MultiplyAddShiftNegate(d, -(magic as $t), shift)
129 }
130 }
131 };
132
133 Some(Self { inner })
134 }
135 }
136
137 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(-15).unwrap();"]]
143 #[must_use]
146 pub const fn get(&self) -> $t {
147 match self.inner {
148 DivisorInner::Shift(d, _) => d,
149 DivisorInner::ShiftAndNegate(d, _) => d,
150 DivisorInner::MultiplyShift(d, _, _) => d,
151 DivisorInner::MultiplyAddShift(d, _, _) => d,
152 DivisorInner::MultiplyAddShiftNegate(d, _, _) => d,
153 }
154 }
155
156 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(-9).unwrap();"]]
164 #[must_use]
167 pub const fn divides(&self, n: $t) -> bool {
168 self.rem_of(n) == 0
169 }
170
171 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(21).unwrap();"]]
177 #[must_use]
181 pub const fn rem_of(&self, n: $t) -> $t {
182 n.wrapping_add((self.get().wrapping_mul(self.div_of(n))).wrapping_mul(-1))
183 }
184
185 #[doc = concat!("`Divisor::<", stringify!($t), ">::new(-1).unwrap().div_of(",
189 stringify!($t) ,"::MIN)`")]
190 #[doc = concat!("`", stringify!($t) ,"::MIN`")]
192 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(13).unwrap();"]]
198 #[must_use]
202 pub const fn div_of(&self, n: $t) -> $t {
203 match self.inner {
204 DivisorInner::Shift(_, shift) => {
205 let mask = (1 as $t << shift).wrapping_sub(1);
206 let b = (n >> (<$t>::BITS - 1)) & mask;
207 n.wrapping_add(b) >> shift
208 },
209 DivisorInner::ShiftAndNegate(_, shift) => {
210 let mask = (1 as $t << shift).wrapping_sub(1);
211 let b = (n >> (<$t>::BITS - 1)) & mask;
212 let t = n.wrapping_add(b) >> shift;
213 t.wrapping_mul(-1)
214 },
215 DivisorInner::MultiplyShift(_, magic, shift) => {
216 let q = Self::mulh(magic, n) >> shift;
217 iif![q < 0; q + 1; q]
218 },
219 DivisorInner::MultiplyAddShift(_, magic, shift) => {
220 let q = Self::mulh(magic, n);
221 let t = q.wrapping_add(n) >> shift;
222 iif![t < 0; t + 1; t]
223 },
224 DivisorInner::MultiplyAddShiftNegate(_, magic, shift) => {
225 let q = Self::mulh(magic, n);
226 let t = q.wrapping_add(n.wrapping_mul(-1)) >> shift;
227 iif![t < 0; t + 1; t]
228 }
229 }
230 }
231 }
232 };
233
234 (@unsigned $t:ty | $up:ty : $is_up:ident : $cap:literal) => {
235 #[cfg(feature = $cap )]
236 impl_divisor![@traits $t];
237
238 #[cfg(feature = $cap )]
239 #[doc = crate::doc_availability!(feature = $cap)]
240 impl Divisor<$t> {
242 impl_divisor![@shared $t|$t|$up|$up:$is_up:$cap]; #[doc = concat!["let _d = Divisor::<", stringify![$t], ">::new(5);"]]
253 #[must_use]
255 pub const fn new(d: $t) -> Option<Divisor<$t>> {
256 if d == 0 {
257 Self::cold_0_divisor()
258 } else {
259 let shift = d.ilog2() as u8;
260 let inner = if d.is_power_of_two() {
261 DivisorInner::Shift(d, shift)
262 } else {
263 let (mut magic, rem) = Self::div_rem_wide_by_base(1 << shift, d);
264 let e = d - rem;
265 if e < 1 << shift {
266 DivisorInner::MultiplyShift(d, magic + 1, shift)
267 } else {
268 magic = magic.wrapping_mul(2);
269 let (doubled_rem, overflowed) = rem.overflowing_mul(2);
270 if doubled_rem >= d || overflowed { magic += 1; }
271 DivisorInner::MultiplyAddShift(d, magic + 1, shift)
272 }
273 };
274 Some(Self { inner })
275 }
276 }
277
278 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(7).unwrap();"]]
284 #[must_use]
287 pub const fn get(&self) -> $t {
288 match self.inner {
289 DivisorInner::Shift(d, _) => d,
290 DivisorInner::MultiplyShift(d, _, _) => d,
291 DivisorInner::MultiplyAddShift(d, _, _) => d,
292 _ => unreachable![],
293 }
294 }
295
296 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(17).unwrap();"]]
304 #[must_use]
307 pub const fn divides(&self, n: $t) -> bool {
308 self.rem_of(n) == 0
309 }
310
311 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(11).unwrap();"]]
317 #[must_use]
321 pub const fn rem_of(&self, n: $t) -> $t {
322 n - self.get() * self.div_of(n)
323 }
324
325 #[doc = concat!["let d = Divisor::<", stringify![$t], ">::new(17).unwrap();"]]
331 #[must_use]
335 pub const fn div_of(&self, n: $t) -> $t {
336 match self.inner {
337 DivisorInner::Shift(_, shift) => n >> shift,
338 DivisorInner::MultiplyShift(_, magic, shift) => Self::mulh(magic, n) >> shift,
339 DivisorInner::MultiplyAddShift(_, magic, shift) => {
340 let q = Self::mulh(magic, n);
341 let t = ((n - q) >> 1) + q;
342 t >> shift
343 },
344 _ => unreachable![], }
346 }
347 }
348 };
349
350 (@shared $t:ty | $un:ty | $up:ty | $unup:ty : $is_up:ident : $cap:literal) => {
351 paste!{
352 pub const fn [<new_ $t>](d: $t) -> Option<Divisor<$t>> { Self::new(d) }
354 }
355
356 #[cold] #[inline(never)]
358 const fn cold_0_divisor() -> Option<Self> { None }
359
360 #[compile(any(same($is_up, Y), all(same($is_up, PW), pointer_width_eq(64))))]
365 const fn mulh(x: $t, y: $t) -> $t {
366 (((x as $up) * (y as $up)) >> <$t>::BITS) as $t
367 }
368 #[compile(any(same($is_up, N), all(same($is_up, PW), not(pointer_width_eq(64)))))]
370 const fn mulh(x: $t, y: $t) -> $t {
371 const HALF_WIDTH_BITS: u32 = <$t>::BITS / 2;
372 const LOWER_HALF_MASK: $t = (1 << HALF_WIDTH_BITS) - 1;
373
374 let x_low = x & LOWER_HALF_MASK;
375 let y_low = y & LOWER_HALF_MASK;
376 let t = x_low.wrapping_mul(y_low);
377 let k = t >> HALF_WIDTH_BITS;
378
379 let x_high = x >> HALF_WIDTH_BITS;
380 let t = x_high.wrapping_mul(y_low) + k;
381 let k = t & LOWER_HALF_MASK;
382 let w1 = t >> HALF_WIDTH_BITS;
383
384 let y_high = y >> HALF_WIDTH_BITS;
385 let t = x_low.wrapping_mul(y_high) + k;
386 let k = t >> HALF_WIDTH_BITS;
387
388 x_high.wrapping_mul(y_high) + w1 + k
389 }
390
391 #[compile(any(same($is_up, Y), all(same($is_up, PW), pointer_width_eq(64))))]
398 const fn div_rem_wide_by_base(top_half: $un, d: $un) -> ($un, $un) {
399 let n = (top_half as $unup) << <$un>::BITS;
400 let quot = (n / (d as $unup)) as $un;
401 let rem = (n % (d as $unup)) as $un;
402 (quot, rem)
403 }
404 #[compile(any(same($is_up, N), all(same($is_up, PW), not(pointer_width_eq(64)))))]
406 const fn div_rem_wide_by_base(top_half: $un, d: $un) -> ($un, $un) {
407 const HALF_WORD_BITS: u32 = <$un>::BITS / 2;
408 const BASE: $un = 1 << HALF_WORD_BITS;
409 let s = d.leading_zeros();
410 let v = d << s;
411 let vn1 = v >> HALF_WORD_BITS;
412 let vn0 = v & (BASE - 1);
413 let un32 = top_half << s;
414 let mut q1 = un32 / vn1;
415 let mut rhat = un32 - q1 * vn1;
416 loop {
417 if q1 >= BASE || q1 * vn0 > (rhat << HALF_WORD_BITS) {
418 q1 -= 1;
419 rhat += vn1;
420 iif![rhat < BASE; continue];
421 }
422 break;
423 }
424 let un21 = (un32 << HALF_WORD_BITS).wrapping_sub(q1.wrapping_mul(v));
425 let mut q0 = un21 / vn1;
426 rhat = un21 - q0 * vn1;
427 loop {
428 if q0 >= BASE || q0 * vn0 > (rhat << HALF_WORD_BITS) {
429 q0 -= 1;
430 rhat += vn1;
431 iif![rhat < BASE; continue];
432 }
433 break;
434 }
435 let r = ((un21 << HALF_WORD_BITS).wrapping_sub(q0.wrapping_mul(v))) >> s;
436 ((q1 << HALF_WORD_BITS) + q0, r)
437 }
438 };
439
440 (@traits $t:ty) => {
441 impl PartialEq for Divisor<$t> {
442 fn eq(&self, other: &Self) -> bool { self.get() == other.get() }
443 }
444 impl Eq for Divisor<$t> {}
445 impl fmt::Debug for Divisor<$t> {
446 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!(f, "{}", self.get()) }
447 }
448 impl fmt::Display for Divisor<$t> {
449 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!(f, "{}", self.get()) }
450 }
451 impl hash::Hash for Divisor<$t> {
452 fn hash<H: hash::Hasher>(&self, state: &mut H) { self.get().hash(state); }
453 }
454 impl ops::Div<Divisor<$t>> for $t {
455 type Output = $t;
456 fn div(self, rhs: Divisor<$t>) -> Self::Output { rhs.div_of(self) }
457 }
458 impl ops::DivAssign<Divisor<$t>> for $t {
459 fn div_assign(&mut self, rhs: Divisor<$t>) { *self = rhs.div_of(*self) }
460 }
461 impl ops::Rem<Divisor<$t>> for $t {
462 type Output = $t;
463 fn rem(self, rhs: Divisor<$t>) -> Self::Output { rhs.rem_of(self) }
464 }
465 impl ops::RemAssign<Divisor<$t>> for $t {
466 fn rem_assign(&mut self, rhs: Divisor<$t>) { *self = rhs.rem_of(*self) }
467 }
468 };
469}
470impl_divisor!();