Skip to content

Commit

Permalink
val/ref forwarding for BigInt bitwise operations
Browse files Browse the repository at this point in the history
  • Loading branch information
tspiteri committed Mar 4, 2018
1 parent 008d2f0 commit 57c3a0e
Showing 1 changed file with 66 additions and 0 deletions.
66 changes: 66 additions & 0 deletions src/bigint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -294,15 +294,45 @@ fn bitand_neg_neg(a: &mut Vec<BigDigit>, b: &[BigDigit]) {
}
}

forward_val_val_binop!(impl BitAnd for BigInt, bitand);
forward_ref_val_binop!(impl BitAnd for BigInt, bitand);

// do not use forward_ref_ref_binop_commutative! for bitand so that we can
// clone as needed, avoiding over-allocation
impl<'a, 'b> BitAnd<&'b BigInt> for &'a BigInt {
type Output = BigInt;

#[inline]
fn bitand(self, other: &BigInt) -> BigInt {
match (self.sign, other.sign) {
(NoSign, _) | (_, NoSign) => BigInt::from_slice(NoSign, &[]),
(Plus, Plus) => BigInt::from_biguint(Plus, &self.data & &other.data),
(Plus, Minus) => self.clone() & other,
(Minus, Plus) => other.clone() & self,
(Minus, Minus) => {
// forward to val-ref, choosing the larger to clone
if self.data.len() >= other.data.len() {
self.clone() & other
} else {
other.clone() & self
}
}
}
}
}

impl<'a> BitAnd<&'a BigInt> for BigInt {
type Output = BigInt;

#[inline]
fn bitand(mut self, other: &BigInt) -> BigInt {
self &= other;
self
}
}

forward_val_assign!(impl BitAndAssign for BigInt, bitand_assign);

impl<'a> BitAndAssign<&'a BigInt> for BigInt {
fn bitand_assign(&mut self, other: &BigInt) {
match (self.sign, other.sign) {
Expand Down Expand Up @@ -399,15 +429,46 @@ fn bitor_neg_neg(a: &mut Vec<BigDigit>, b: &[BigDigit]) {
debug_assert!(carry_or == 0);
}

forward_val_val_binop!(impl BitOr for BigInt, bitor);
forward_ref_val_binop!(impl BitOr for BigInt, bitor);

// do not use forward_ref_ref_binop_commutative! for bitor so that we can
// clone as needed, avoiding over-allocation
impl<'a, 'b> BitOr<&'b BigInt> for &'a BigInt {
type Output = BigInt;

#[inline]
fn bitor(self, other: &BigInt) -> BigInt {
match (self.sign, other.sign) {
(NoSign, _) => other.clone(),
(_, NoSign) => self.clone(),
(Plus, Plus) => BigInt::from_biguint(Plus, &self.data | &other.data),
(Plus, Minus) => other.clone() | self,
(Minus, Plus) => self.clone() | other,
(Minus, Minus) => {
// forward to val-ref, choosing the smaller to clone
if self.data.len() <= other.data.len() {
self.clone() | other
} else {
other.clone() | self
}
}
}
}
}

impl<'a> BitOr<&'a BigInt> for BigInt {
type Output = BigInt;

#[inline]
fn bitor(mut self, other: &BigInt) -> BigInt {
self |= other;
self
}
}

forward_val_assign!(impl BitOrAssign for BigInt, bitor_assign);

impl<'a> BitOrAssign<&'a BigInt> for BigInt {
fn bitor_assign(&mut self, other: &BigInt) {
match (self.sign, other.sign) {
Expand Down Expand Up @@ -520,15 +581,20 @@ fn bitxor_neg_neg(a: &mut Vec<BigDigit>, b: &[BigDigit]) {
}
}

forward_all_binop_to_val_ref_commutative!(impl BitXor for BigInt, bitxor);

impl<'a> BitXor<&'a BigInt> for BigInt {
type Output = BigInt;

#[inline]
fn bitxor(mut self, other: &BigInt) -> BigInt {
self ^= other;
self
}
}

forward_val_assign!(impl BitXorAssign for BigInt, bitxor_assign);

impl<'a> BitXorAssign<&'a BigInt> for BigInt {
fn bitxor_assign(&mut self, other: &BigInt) {
match (self.sign, other.sign) {
Expand Down

0 comments on commit 57c3a0e

Please sign in to comment.