use super::U512;
impl U512 {
#[inline]
pub const fn overflowing_add(&self, other: &U512) -> (U512, bool) {
let a0 = ((self.0[1] as u128) << 64) | (self.0[0] as u128);
let a1 = ((self.0[3] as u128) << 64) | (self.0[2] as u128);
let a2 = ((self.0[5] as u128) << 64) | (self.0[4] as u128);
let a3 = ((self.0[7] as u128) << 64) | (self.0[6] as u128);
let b0 = ((other.0[1] as u128) << 64) | (other.0[0] as u128);
let b1 = ((other.0[3] as u128) << 64) | (other.0[2] as u128);
let b2 = ((other.0[5] as u128) << 64) | (other.0[4] as u128);
let b3 = ((other.0[7] as u128) << 64) | (other.0[6] as u128);
let (s0, c0) = a0.overflowing_add(b0);
let (s1, c1a) = a1.overflowing_add(b1);
let (s1, c1b) = s1.overflowing_add(c0 as u128);
let (s2, c2a) = a2.overflowing_add(b2);
let (s2, c2b) = s2.overflowing_add((c1a | c1b) as u128);
let (s3, c3a) = a3.overflowing_add(b3);
let (s3, c3b) = s3.overflowing_add((c2a | c2b) as u128);
(
U512([
s0 as u64,
(s0 >> 64) as u64,
s1 as u64,
(s1 >> 64) as u64,
s2 as u64,
(s2 >> 64) as u64,
s3 as u64,
(s3 >> 64) as u64,
]),
c3a | c3b,
)
}
}
#[cfg(test)]
mod ai_tests {
use super::*;
#[test]
fn small_no_overflow() {
let a = U512::from_be_limbs([0, 0, 0, 0, 0, 0, 0, 100]);
let b = U512::from_be_limbs([0, 0, 0, 0, 0, 0, 0, 200]);
let (result, overflow) = a.overflowing_add(&b);
assert_eq!(result, U512::from_be_limbs([0, 0, 0, 0, 0, 0, 0, 300]));
assert!(!overflow);
}
#[test]
fn max_plus_one_overflows() {
let (result, overflow) = U512::MAX.overflowing_add(&U512::ONE);
assert_eq!(result, U512::ZERO);
assert!(overflow);
}
#[test]
fn carry_propagation() {
let a = U512::from_be_limbs([0, 0, 0, 0, 0, 0, 0, u64::MAX]);
let one = U512::ONE;
let (result, overflow) = a.overflowing_add(&one);
assert_eq!(result, U512::from_be_limbs([0, 0, 0, 0, 0, 0, 1, 0]));
assert!(!overflow);
}
#[test]
fn additive_identity() {
let a = U512::from_be_limbs([0x1234, 0x5678, 0x9ABC, 0xDEF0, 1, 2, 3, 4]);
let (result, overflow) = a.overflowing_add(&U512::ZERO);
assert_eq!(result, a);
assert!(!overflow);
}
#[test]
fn full_carry_chain() {
let a = U512::from_be_limbs([0, u64::MAX, u64::MAX, u64::MAX, u64::MAX, u64::MAX, u64::MAX, u64::MAX]);
let (result, overflow) = a.overflowing_add(&U512::ONE);
assert_eq!(result, U512::from_be_limbs([1, 0, 0, 0, 0, 0, 0, 0]));
assert!(!overflow);
}
#[test]
fn commutative() {
let a = U512::from_be_limbs([1, 2, 3, 4, 5, 6, 7, 8]);
let b = U512::from_be_limbs([8, 7, 6, 5, 4, 3, 2, 1]);
assert_eq!(a.overflowing_add(&b), b.overflowing_add(&a));
}
}