penumbra_sdk_proof_setup/single/
phase2.rsuse ark_ec::Group;
use ark_ff::Zero;
use ark_serialize::{CanonicalDeserialize, CanonicalSerialize, Compress, Validate};
use rand_core::{CryptoRngCore, OsRng};
use crate::single::log::{ContributionHash, Hashable, Phase};
use crate::single::{
dlog,
group::{BatchedPairingChecker11, GroupHasher, F, G1, G2},
};
#[derive(Clone, Debug, CanonicalSerialize, CanonicalDeserialize, PartialEq)]
pub struct RawCRSElements {
pub delta_1: G1,
pub delta_2: G2,
pub inv_delta_p_1: Vec<G1>,
pub inv_delta_t_1: Vec<G1>,
}
impl RawCRSElements {
#[must_use]
pub fn validate<R: CryptoRngCore>(
self,
rng: &mut R,
root: &CRSElements,
) -> Option<CRSElements> {
if self.inv_delta_p_1.len() != root.raw.inv_delta_p_1.len()
|| self.inv_delta_t_1.len() != root.raw.inv_delta_t_1.len()
{
return None;
}
if self.delta_1.is_zero() || self.delta_2.is_zero() {
return None;
}
let mut checker = BatchedPairingChecker11::new(self.delta_2, G2::generator());
checker.add(G1::generator(), self.delta_1);
for (&inv_delta_p_i, &p_i) in self.inv_delta_p_1.iter().zip(root.raw.inv_delta_p_1.iter()) {
checker.add(inv_delta_p_i, p_i);
}
for (&inv_delta_t_i, &t_i) in self.inv_delta_t_1.iter().zip(root.raw.inv_delta_t_1.iter()) {
checker.add(inv_delta_t_i, t_i);
}
if !checker.check(rng) {
return None;
}
Some(CRSElements { raw: self })
}
pub(crate) fn assume_valid(self) -> CRSElements {
CRSElements { raw: self }
}
#[cfg(not(feature = "parallel"))]
pub(crate) fn checked_deserialize_parallel(
compress: Compress,
data: &[u8],
) -> anyhow::Result<Self> {
Ok(Self::deserialize_with_mode(data, compress, Validate::Yes)?)
}
#[cfg(feature = "parallel")]
pub(crate) fn checked_deserialize_parallel(
compress: Compress,
data: &[u8],
) -> anyhow::Result<Self> {
use ark_serialize::Valid;
use rayon::prelude::*;
let out = Self::deserialize_with_mode(data, compress, Validate::No)?;
out.delta_1.check()?;
out.delta_2.check()?;
let mut check_inv_delta_p_1 = Ok(());
let mut check_inv_delta_t_1 = Ok(());
rayon::join(
|| {
check_inv_delta_p_1 = out
.inv_delta_p_1
.par_iter()
.map(|x| x.check())
.collect::<Result<_, _>>();
},
|| {
check_inv_delta_t_1 = out
.inv_delta_t_1
.par_iter()
.map(|x| x.check())
.collect::<Result<_, _>>();
},
);
check_inv_delta_p_1?;
check_inv_delta_t_1?;
Ok(out)
}
}
impl Hashable for RawCRSElements {
fn hash(&self) -> ContributionHash {
let mut hasher = GroupHasher::new(b"PC$:crs_elmnts2");
hasher.eat_g1(&self.delta_1);
hasher.eat_g2(&self.delta_2);
hasher.eat_usize(self.inv_delta_p_1.len());
for v in &self.inv_delta_p_1 {
hasher.eat_g1(v);
}
hasher.eat_usize(self.inv_delta_t_1.len());
for v in &self.inv_delta_t_1 {
hasher.eat_g1(v);
}
ContributionHash(hasher.finalize_bytes())
}
}
#[derive(Clone, Debug, PartialEq)]
pub struct CRSElements {
pub(crate) raw: RawCRSElements,
}
impl Hashable for CRSElements {
fn hash(&self) -> ContributionHash {
self.raw.hash()
}
}
impl CRSElements {
pub(crate) fn dummy_root(degree: usize) -> Self {
Self {
raw: RawCRSElements {
delta_1: G1::generator(),
delta_2: G2::generator(),
inv_delta_p_1: vec![G1::generator(); degree],
inv_delta_t_1: vec![G1::generator(); degree],
},
}
}
}
#[derive(Clone, Debug)]
pub struct RawContribution {
pub parent: ContributionHash,
pub new_elements: RawCRSElements,
pub(crate) linking_proof: dlog::Proof,
}
impl RawContribution {
pub fn validate<R: CryptoRngCore>(
self,
rng: &mut R,
root: &CRSElements,
) -> Option<Contribution> {
self.new_elements
.validate(rng, root)
.map(|new_elements| Contribution {
parent: self.parent,
new_elements,
linking_proof: self.linking_proof,
})
}
pub(crate) fn assume_valid(self) -> Contribution {
Contribution {
parent: self.parent,
new_elements: self.new_elements.assume_valid(),
linking_proof: self.linking_proof,
}
}
}
impl Hashable for RawContribution {
fn hash(&self) -> ContributionHash {
let mut hasher = GroupHasher::new(b"PC$:contrbution2");
hasher.eat_bytes(self.parent.as_ref());
hasher.eat_bytes(self.new_elements.hash().as_ref());
hasher.eat_bytes(self.linking_proof.hash().as_ref());
ContributionHash(hasher.finalize_bytes())
}
}
impl From<Contribution> for RawContribution {
fn from(value: Contribution) -> Self {
Self {
parent: value.parent,
new_elements: value.new_elements.raw,
linking_proof: value.linking_proof,
}
}
}
#[derive(Clone, Debug)]
pub struct Contribution {
pub parent: ContributionHash,
pub new_elements: CRSElements,
pub(crate) linking_proof: dlog::Proof,
}
impl Hashable for Contribution {
fn hash(&self) -> ContributionHash {
RawContribution::from(self.to_owned()).hash()
}
}
impl Contribution {
pub fn make<R: CryptoRngCore>(
rng: &mut R,
parent: ContributionHash,
old: &CRSElements,
) -> Self {
let delta = F::rand(rng);
let delta_inv = delta.inverse().expect("unable to inverse delta");
let mut new = old.clone();
new.raw.delta_1 *= delta;
new.raw.delta_2 *= delta;
for v in &mut new.raw.inv_delta_p_1 {
*v *= delta_inv;
}
for v in &mut new.raw.inv_delta_t_1 {
*v *= delta_inv;
}
let linking_proof = dlog::prove(
rng,
b"phase2 delta proof",
dlog::Statement {
result: new.raw.delta_1,
base: old.raw.delta_1,
},
dlog::Witness { dlog: delta },
);
Contribution {
parent,
new_elements: new,
linking_proof,
}
}
#[must_use]
pub fn is_linked_to(&self, parent: &CRSElements) -> bool {
if self.new_elements.raw.inv_delta_p_1.len() != parent.raw.inv_delta_p_1.len()
|| self.new_elements.raw.inv_delta_t_1.len() != parent.raw.inv_delta_t_1.len()
{
return false;
}
if !dlog::verify(
b"phase2 delta proof",
dlog::Statement {
result: self.new_elements.raw.delta_1,
base: parent.raw.delta_1,
},
&self.linking_proof,
) {
return false;
}
true
}
}
#[derive(Clone, Debug, Default)]
struct Phase2;
impl Phase for Phase2 {
type CRSElements = CRSElements;
type RawContribution = RawContribution;
type Contribution = Contribution;
fn parent_hash(contribution: &Self::RawContribution) -> ContributionHash {
contribution.parent
}
fn elements(contribution: &Self::Contribution) -> &Self::CRSElements {
&contribution.new_elements
}
fn validate(
root: &Self::CRSElements,
contribution: &Self::RawContribution,
) -> Option<Self::Contribution> {
contribution.to_owned().validate(&mut OsRng, root)
}
fn is_linked_to(contribution: &Self::Contribution, elements: &Self::CRSElements) -> bool {
contribution.is_linked_to(elements)
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::single::log::CONTRIBUTION_HASH_SIZE;
use rand_core::OsRng;
fn make_crs(delta: F, delta_inv: F) -> (CRSElements, RawCRSElements) {
let x = F::rand(&mut OsRng);
let root = CRSElements {
raw: RawCRSElements {
delta_1: G1::generator(),
delta_2: G2::generator(),
inv_delta_p_1: vec![G1::generator() * x],
inv_delta_t_1: vec![G1::generator() * (x * x)],
},
};
let new = RawCRSElements {
delta_1: root.raw.delta_1 * delta,
delta_2: root.raw.delta_2 * delta,
inv_delta_p_1: root
.raw
.inv_delta_p_1
.iter()
.map(|&x| x * delta_inv)
.collect(),
inv_delta_t_1: root
.raw
.inv_delta_t_1
.iter()
.map(|&x| x * delta_inv)
.collect(),
};
(root, new)
}
fn non_trivial_crs() -> (CRSElements, RawCRSElements) {
let delta = F::rand(&mut OsRng);
let delta_inv = delta.inverse().expect("unable to inverse delta");
make_crs(delta, delta_inv)
}
#[test]
fn test_nontrivial_crs_is_valid() {
let (root, crs) = non_trivial_crs();
assert!(crs.validate(&mut OsRng, &root).is_some());
}
#[test]
fn test_changing_delta_makes_crs_invalid() {
let (root, mut crs) = non_trivial_crs();
crs.delta_1 = G1::generator();
crs.delta_2 = G2::generator();
assert!(crs.validate(&mut OsRng, &root).is_none());
}
#[test]
fn test_different_deltas_makes_crs_invalid() {
let (root, mut crs) = non_trivial_crs();
crs.delta_1 = G1::generator();
assert!(crs.validate(&mut OsRng, &root).is_none());
}
#[test]
fn test_different_length_from_root_is_invalid_crs() {
let (root, mut crs) = non_trivial_crs();
crs.inv_delta_p_1.clear();
crs.inv_delta_t_1.clear();
assert!(crs.validate(&mut OsRng, &root).is_none());
}
#[test]
fn test_setting_zero_elements_makes_crs_invalid() {
let (root, crs) = make_crs(F::zero(), F::zero());
assert!(crs.validate(&mut OsRng, &root).is_none());
}
#[test]
fn test_not_inverting_delta_makes_crs_invalid() {
let delta = F::rand(&mut OsRng);
let (root, crs) = make_crs(delta, delta);
assert!(crs.validate(&mut OsRng, &root).is_none());
}
#[test]
fn test_contribution_produces_valid_crs() {
let (root, start) = non_trivial_crs();
let start = start
.validate(&mut OsRng, &root)
.expect("unable to validate start");
let contribution = Contribution::make(
&mut OsRng,
ContributionHash([0u8; CONTRIBUTION_HASH_SIZE]),
&start,
);
assert!(contribution
.new_elements
.raw
.validate(&mut OsRng, &root)
.is_some());
}
#[test]
fn test_can_calculate_contribution_hash() {
let (root, start) = non_trivial_crs();
let start = start
.validate(&mut OsRng, &root)
.expect("unable to validate start");
let contribution = Contribution::make(
&mut OsRng,
ContributionHash([0u8; CONTRIBUTION_HASH_SIZE]),
&start,
);
assert_ne!(contribution.hash(), contribution.parent);
}
#[test]
fn test_contribution_is_linked_to_parent() {
let (root, start) = non_trivial_crs();
let start = start
.validate(&mut OsRng, &root)
.expect("unable to validate start");
let contribution = Contribution::make(
&mut OsRng,
ContributionHash([0u8; CONTRIBUTION_HASH_SIZE]),
&start,
);
assert!(contribution.is_linked_to(&start));
}
#[test]
fn test_contribution_is_not_linked_to_itself() {
let (root, start) = non_trivial_crs();
let start = start
.validate(&mut OsRng, &root)
.expect("unable to validate start");
let contribution = Contribution::make(
&mut OsRng,
ContributionHash([0u8; CONTRIBUTION_HASH_SIZE]),
&start,
);
assert!(!contribution.is_linked_to(&contribution.new_elements));
}
}