use super::*;
impl<E: Environment> Nor<Self> for Boolean<E> {
type Output = Boolean<E>;
fn nor(&self, other: &Self) -> Self::Output {
if self.is_constant() {
match self.eject_value() {
true => !self.clone(),
false => !other.clone(),
}
}
else if other.is_constant() {
match other.eject_value() {
true => !other.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(|| (E::one() - &self.0, E::one() - &other.0, &output))
.expect("Boolean NOR constraint unsatisfied");
output
}
}
}
impl<E: Environment> Metrics<dyn Nor<Boolean<E>, Output = Boolean<E>>> for Boolean<E> {
type Case = (Mode, Mode);
fn count(case: &Self::Case) -> Count {
match case.0.is_constant() || case.1.is_constant() {
true => Count::is(0, 0, 0, 0),
false => Count::is(0, 0, 1, 1),
}
}
}
impl<E: Environment> OutputMode<dyn Nor<Boolean<E>, Output = Boolean<E>>> for Boolean<E> {
type Case = (CircuitType<Boolean<E>>, CircuitType<Boolean<E>>);
fn output_mode(case: &Self::Case) -> Mode {
match (case.0.mode(), case.1.mode()) {
(Mode::Constant, Mode::Constant) => Mode::Constant,
(_, Mode::Constant) => match &case.1 {
CircuitType::Constant(constant) => match constant.eject_value() {
true => Mode::Constant,
false => Mode::Private,
},
_ => E::halt("The constant is required to determine the output mode of Public NOR Constant"),
},
(Mode::Constant, _) => match &case.0 {
CircuitType::Constant(constant) => match constant.eject_value() {
true => Mode::Constant,
false => Mode::Private,
},
_ => E::halt("The constant is required to determine the output mode of Constant NOR Public"),
},
(_, _) => Mode::Private,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use snarkvm_circuit_environment::Circuit;
fn check_nor(name: &str, expected: bool, a: Boolean<Circuit>, b: Boolean<Circuit>) {
Circuit::scope(name, || {
let candidate = a.nor(&b);
assert_eq!(expected, candidate.eject_value(), "({} NOR {})", a.eject_value(), b.eject_value());
assert_count!(Nor(Boolean, Boolean) => Boolean, &(a.eject_mode(), b.eject_mode()));
assert_output_mode!(Nor(Boolean, Boolean) => Boolean, &(CircuitType::from(&a), CircuitType::from(&b)), candidate);
});
Circuit::reset();
}
#[test]
fn test_constant_nor_constant() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_constant_nor_public() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_constant_nor_private() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, false);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Constant, true);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_public_nor_constant() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_public_nor_public() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_public_nor_private() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, false);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Public, true);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_private_nor_constant() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Constant, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Constant, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_private_nor_public() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Public, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Public, true);
check_nor("true NOR true", expected, a, b);
}
#[test]
fn test_private_nor_private() {
let expected = true;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_nor("false NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, false);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_nor("false NOR true", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Private, false);
check_nor("true NOR false", expected, a, b);
let expected = false;
let a = Boolean::<Circuit>::new(Mode::Private, true);
let b = Boolean::<Circuit>::new(Mode::Private, true);
check_nor("true NOR true", expected, a, b);
}
}