use super::*;
impl<E: Environment> BitXor<Boolean<E>> for Boolean<E> {
type Output = Boolean<E>;
fn bitxor(self, other: Boolean<E>) -> Self::Output {
self ^ &other
}
}
impl<E: Environment> BitXor<Boolean<E>> for &Boolean<E> {
type Output = Boolean<E>;
fn bitxor(self, other: Boolean<E>) -> Self::Output {
self ^ &other
}
}
impl<E: Environment> BitXor<&Boolean<E>> for Boolean<E> {
type Output = Boolean<E>;
fn bitxor(self, other: &Boolean<E>) -> Self::Output {
&self ^ other
}
}
impl<E: Environment> BitXor<&Boolean<E>> for &Boolean<E> {
type Output = Boolean<E>;
fn bitxor(self, other: &Boolean<E>) -> Self::Output {
let mut output = self.clone();
output ^= other;
output
}
}
impl<E: Environment> BitXorAssign<Boolean<E>> for Boolean<E> {
fn bitxor_assign(&mut self, other: Boolean<E>) {
*self ^= &other;
}
}
impl<E: Environment> BitXorAssign<&Boolean<E>> for Boolean<E> {
fn bitxor_assign(&mut self, other: &Boolean<E>) {
*self =
if self.is_constant() {
match self.eject_value() {
true => !other.clone(),
false => other.clone(),
}
}
else if other.is_constant() {
match other.eject_value() {
true => !self.clone(),
false => self.clone(),
}
}
else {
let output = Boolean(
E::new_variable(Mode::Private, match self.eject_value() ^ other.eject_value() {
true => E::BaseField::one(),
false => E::BaseField::zero(),
})
.into(),
);
E::enforce(|| ((&self.0 + &self.0), other, (&self.0 + &other.0 - &output.0)));
output
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use snarkvm_circuit_environment::Circuit;
fn check_xor(
name: &str,
expected: bool,
a: Boolean<Circuit>,
b: Boolean<Circuit>,
num_constants: u64,
num_public: u64,
num_private: u64,
num_constraints: u64,
) {
Circuit::scope(name, || {
let candidate = &a ^ &b;
assert_eq!(expected, candidate.eject_value(), "({} != {})", a.eject_value(), b.eject_value());
assert_scope!(num_constants, num_public, num_private, num_constraints);
});
}
#[test]
fn test_constant_xor_constant() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_xor("false != false", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_xor("false != true", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_xor("true != false", expected, a, b, 0, 0, 0, 0);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_xor("true != true", expected, a, b, 0, 0, 0, 0);
}
#[test]
fn test_constant_xor_public() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_xor("false != false", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_xor("false != true", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_xor("true != false", expected, a, b, 0, 0, 0, 0);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_xor("true != true", expected, a, b, 0, 0, 0, 0);
}
#[test]
fn test_constant_xor_private() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_xor("false != false", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_xor("false != true", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_xor("true != false", expected, a, b, 0, 0, 0, 0);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_xor("true != true", expected, a, b, 0, 0, 0, 0);
}
#[test]
fn test_public_xor_constant() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_xor("false != false", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_xor("false != true", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_xor("true != false", expected, a, b, 0, 0, 0, 0);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_xor("true != true", expected, a, b, 0, 0, 0, 0);
}
#[test]
fn test_public_xor_public() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_xor("false != false", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_xor("false != true", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_xor("true != false", expected, a, b, 0, 0, 1, 1);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_xor("true != true", expected, a, b, 0, 0, 1, 1);
}
#[test]
fn test_public_xor_private() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_xor("false != false", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_xor("false != true", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_xor("true != false", expected, a, b, 0, 0, 1, 1);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_xor("true != true", expected, a, b, 0, 0, 1, 1);
}
#[test]
fn test_private_xor_constant() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_xor("false != false", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_xor("false != true", expected, a, b, 0, 0, 0, 0);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_xor("true != false", expected, a, b, 0, 0, 0, 0);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_xor("true != true", expected, a, b, 0, 0, 0, 0);
}
#[test]
fn test_private_xor_public() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_xor("false != false", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_xor("false != true", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_xor("true != false", expected, a, b, 0, 0, 1, 1);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_xor("true != true", expected, a, b, 0, 0, 1, 1);
}
#[test]
fn test_private_xor_private() {
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_xor("false != false", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_xor("false != true", expected, a, b, 0, 0, 1, 1);
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_xor("true != false", expected, a, b, 0, 0, 1, 1);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_xor("true != true", expected, a, b, 0, 0, 1, 1);
}
}