//! [`Uint`] bitwise or operations. use super::Uint; use crate::{Limb, Wrapping}; use core::ops::{BitOr, BitOrAssign}; use subtle::{Choice, CtOption}; impl Uint { /// Computes bitwise `a & b`. #[inline(always)] pub const fn bitor(&self, rhs: &Self) -> Self { let mut limbs = [Limb::ZERO; LIMBS]; let mut i = 0; while i < LIMBS { limbs[i] = self.limbs[i].bitor(rhs.limbs[i]); i += 1; } Self { limbs } } /// Perform wrapping bitwise `OR`. /// /// There's no way wrapping could ever happen. /// This function exists so that all operations are accounted for in the wrapping operations pub const fn wrapping_or(&self, rhs: &Self) -> Self { self.bitor(rhs) } /// Perform checked bitwise `OR`, returning a [`CtOption`] which `is_some` always pub fn checked_or(&self, rhs: &Self) -> CtOption { let result = self.bitor(rhs); CtOption::new(result, Choice::from(1)) } } impl BitOr for Uint { type Output = Self; fn bitor(self, rhs: Self) -> Uint { self.bitor(&rhs) } } impl BitOr<&Uint> for Uint { type Output = Uint; #[allow(clippy::needless_borrow)] fn bitor(self, rhs: &Uint) -> Uint { (&self).bitor(rhs) } } impl BitOr> for &Uint { type Output = Uint; fn bitor(self, rhs: Uint) -> Uint { self.bitor(&rhs) } } impl BitOr<&Uint> for &Uint { type Output = Uint; fn bitor(self, rhs: &Uint) -> Uint { self.bitor(rhs) } } impl BitOrAssign for Uint { fn bitor_assign(&mut self, other: Self) { *self = *self | other; } } impl BitOrAssign<&Uint> for Uint { fn bitor_assign(&mut self, other: &Self) { *self = *self | other; } } impl BitOr for Wrapping> { type Output = Self; fn bitor(self, rhs: Self) -> Wrapping> { Wrapping(self.0.bitor(&rhs.0)) } } impl BitOr<&Wrapping>> for Wrapping> { type Output = Wrapping>; fn bitor(self, rhs: &Wrapping>) -> Wrapping> { Wrapping(self.0.bitor(&rhs.0)) } } impl BitOr>> for &Wrapping> { type Output = Wrapping>; fn bitor(self, rhs: Wrapping>) -> Wrapping> { Wrapping(self.0.bitor(&rhs.0)) } } impl BitOr<&Wrapping>> for &Wrapping> { type Output = Wrapping>; fn bitor(self, rhs: &Wrapping>) -> Wrapping> { Wrapping(self.0.bitor(&rhs.0)) } } impl BitOrAssign for Wrapping> { fn bitor_assign(&mut self, other: Self) { *self = *self | other; } } impl BitOrAssign<&Wrapping>> for Wrapping> { fn bitor_assign(&mut self, other: &Self) { *self = *self | other; } } #[cfg(test)] mod tests { use crate::U128; #[test] fn checked_or_ok() { let result = U128::ZERO.checked_or(&U128::ONE); assert_eq!(result.unwrap(), U128::ONE); } #[test] fn overlapping_or_ok() { let result = U128::MAX.wrapping_or(&U128::ONE); assert_eq!(result, U128::MAX); } }