#![allow(missing_docs)] use core::cmp::Ordering;
mod add;
pub mod div;
mod gcd;
mod mul;
mod mul_redc;
mod ops;
mod shift;
pub use self::{
add::{adc_n, sbb_n},
div::div,
gcd::{gcd, gcd_extended, inv_mod, LehmerMatrix},
mul::{add_nx1, addmul, addmul_n, addmul_nx1, mul_nx1, submul_nx1},
mul_redc::{mul_redc, square_redc},
ops::{adc, sbb},
shift::{shift_left_small, shift_right_small},
};
trait DoubleWord<T>: Sized + Copy {
fn join(high: T, low: T) -> Self;
fn add(a: T, b: T) -> Self;
fn mul(a: T, b: T) -> Self;
fn muladd(a: T, b: T, c: T) -> Self;
fn muladd2(a: T, b: T, c: T, d: T) -> Self;
fn high(self) -> T;
fn low(self) -> T;
fn split(self) -> (T, T);
}
impl DoubleWord<u64> for u128 {
#[inline(always)]
fn join(high: u64, low: u64) -> Self {
(Self::from(high) << 64) | Self::from(low)
}
#[inline(always)]
fn add(a: u64, b: u64) -> Self {
Self::from(a) + Self::from(b)
}
#[inline(always)]
fn mul(a: u64, b: u64) -> Self {
Self::from(a) * Self::from(b)
}
#[inline(always)]
fn muladd(a: u64, b: u64, c: u64) -> Self {
Self::from(a) * Self::from(b) + Self::from(c)
}
#[inline(always)]
fn muladd2(a: u64, b: u64, c: u64, d: u64) -> Self {
Self::from(a) * Self::from(b) + Self::from(c) + Self::from(d)
}
#[inline(always)]
fn high(self) -> u64 {
(self >> 64) as u64
}
#[inline(always)]
#[allow(clippy::cast_possible_truncation)]
fn low(self) -> u64 {
self as u64
}
#[inline(always)]
fn split(self) -> (u64, u64) {
(self.low(), self.high())
}
}
#[inline(always)]
#[must_use]
pub fn cmp(left: &[u64], right: &[u64]) -> Ordering {
let l = core::cmp::min(left.len(), right.len());
let lhs = &left[..l];
let rhs = &right[..l];
for i in (0..l).rev() {
match i8::from(lhs[i] > rhs[i]) - i8::from(lhs[i] < rhs[i]) {
-1 => return Ordering::Less,
0 => {}
1 => return Ordering::Greater,
_ => unsafe { core::hint::unreachable_unchecked() },
}
}
left.len().cmp(&right.len())
}
#[inline]
#[must_use]
pub const fn carrying_add(lhs: u64, rhs: u64, carry: bool) -> (u64, bool) {
let (result, carry_1) = lhs.overflowing_add(rhs);
let (result, carry_2) = result.overflowing_add(carry as u64);
(result, carry_1 | carry_2)
}
#[inline]
#[must_use]
pub const fn borrowing_sub(lhs: u64, rhs: u64, borrow: bool) -> (u64, bool) {
let (result, borrow_1) = lhs.overflowing_sub(rhs);
let (result, borrow_2) = result.overflowing_sub(borrow as u64);
(result, borrow_1 | borrow_2)
}