From 8cc99194bf537be1072713d1d2e02081381026f5 Mon Sep 17 00:00:00 2001 From: Shahar Papini Date: Fri, 22 Mar 2024 12:37:19 +0200 Subject: [PATCH] Use simple merkle tree --- src/commitment_scheme/blake2_merkle.rs | 29 ++++++----- src/commitment_scheme/ops.rs | 4 +- src/commitment_scheme/prover.rs | 1 + src/commitment_scheme/verifier.rs | 10 ++-- src/core/commitment_scheme/prover.rs | 68 +++++--------------------- src/core/commitment_scheme/verifier.rs | 45 ++++++++++------- 6 files changed, 63 insertions(+), 94 deletions(-) diff --git a/src/commitment_scheme/blake2_merkle.rs b/src/commitment_scheme/blake2_merkle.rs index 7e65d556d..f12ba4479 100644 --- a/src/commitment_scheme/blake2_merkle.rs +++ b/src/commitment_scheme/blake2_merkle.rs @@ -1,15 +1,13 @@ use itertools::Itertools; use num_traits::Zero; +use super::blake2_hash::{Blake2sHash, Blake2sHasher}; use super::blake2s_ref::compress; use super::ops::{MerkleHasher, MerkleOps}; use crate::core::backend::CPUBackend; use crate::core::fields::m31::BaseField; -pub struct Blake2Hasher; -impl MerkleHasher for Blake2Hasher { - type Hash = [u32; 8]; - +impl MerkleHasher for Blake2sHasher { fn hash_node( children_hashes: Option<(Self::Hash, Self::Hash)>, column_values: &[BaseField], @@ -33,19 +31,19 @@ impl MerkleHasher for Blake2Hasher { for chunk in padded_values.array_chunks::<16>() { state = compress(state, unsafe { std::mem::transmute(chunk) }, 0, 0, 0, 0); } - state + unsafe { std::mem::transmute(state) } } } -impl MerkleOps for CPUBackend { +impl MerkleOps for CPUBackend { fn commit_on_layer( log_size: u32, - prev_layer: Option<&Vec<[u32; 8]>>, + prev_layer: Option<&Vec>, columns: &[&Vec], - ) -> Vec<[u32; 8]> { + ) -> Vec { (0..(1 << log_size)) .map(|i| { - Blake2Hasher::hash_node( + Blake2sHasher::hash_node( prev_layer.map(|prev_layer| (prev_layer[2 * i], prev_layer[2 * i + 1])), &columns.iter().map(|column| column[i]).collect_vec(), ) @@ -61,7 +59,8 @@ mod tests { use rand::rngs::StdRng; use rand::{Rng, SeedableRng}; - use crate::commitment_scheme::blake2_merkle::Blake2Hasher; + use crate::commitment_scheme::blake2_hash::Blake2sHash; + use crate::commitment_scheme::blake2_merkle::Blake2sHasher; use crate::commitment_scheme::prover::{Decommitment, MerkleProver}; use crate::commitment_scheme::verifier::{MerkleTreeVerifier, MerkleVerificationError}; use crate::core::backend::CPUBackend; @@ -69,9 +68,9 @@ mod tests { type TestData = ( Vec, - Decommitment, + Decommitment, Vec<(u32, Vec)>, - MerkleTreeVerifier, + MerkleTreeVerifier, ); fn prepare_merkle() -> TestData { const N_COLS: usize = 400; @@ -92,7 +91,7 @@ mod tests { .collect_vec() }) .collect_vec(); - let merkle = MerkleProver::::commit(cols.iter().collect_vec()); + let merkle = MerkleProver::::commit(cols.iter().collect_vec()); let queries = (0..N_QUERIES) .map(|_| rng.gen_range(0..(1 << max_log_size))) @@ -128,7 +127,7 @@ mod tests { #[test] fn test_merkle_invalid_witness() { let (queries, mut decommitment, values, verifier) = prepare_merkle(); - decommitment.witness[20] = [0; 8]; + decommitment.witness[20] = Blake2sHash::from(&[0u8; 32][..]); assert_eq!( verifier.verify(queries, values, decommitment).unwrap_err(), @@ -183,7 +182,7 @@ mod tests { #[test] fn test_merkle_witness_too_long() { let (queries, mut decommitment, values, verifier) = prepare_merkle(); - decommitment.witness.push([0; 8]); + decommitment.witness.push(Blake2sHash::from(&[0u8; 32][..])); assert_eq!( verifier.verify(queries, values, decommitment).unwrap_err(), diff --git a/src/commitment_scheme/ops.rs b/src/commitment_scheme/ops.rs index 94e3c74c7..8c1314231 100644 --- a/src/commitment_scheme/ops.rs +++ b/src/commitment_scheme/ops.rs @@ -1,8 +1,8 @@ +use super::hasher::Hasher; use crate::core::backend::{Col, ColumnOps}; use crate::core::fields::m31::BaseField; -pub trait MerkleHasher { - type Hash: Clone + Eq + std::fmt::Debug; +pub trait MerkleHasher: Hasher { /// Hashes a single Merkle node. /// The node may or may not need to hash 2 hashes from the previous layer - depending if it is a /// leaf or not. diff --git a/src/commitment_scheme/prover.rs b/src/commitment_scheme/prover.rs index b1d364130..e9a28fe53 100644 --- a/src/commitment_scheme/prover.rs +++ b/src/commitment_scheme/prover.rs @@ -60,6 +60,7 @@ impl, H: MerkleHasher> MerkleProver { } } +#[derive(Debug)] pub struct Decommitment { pub witness: Vec, } diff --git a/src/commitment_scheme/verifier.rs b/src/commitment_scheme/verifier.rs index b2c890ef0..de2226569 100644 --- a/src/commitment_scheme/verifier.rs +++ b/src/commitment_scheme/verifier.rs @@ -4,10 +4,12 @@ use std::iter::Peekable; use itertools::Itertools; use thiserror::Error; +use super::hasher::Hasher; use super::ops::MerkleHasher; use super::prover::Decommitment; use crate::core::fields::m31::BaseField; +// TODO(spapini): This struct is not necessary. Make it a function on decommitment? pub struct MerkleTreeVerifier { pub root: H::Hash, } @@ -52,7 +54,7 @@ impl MerkleTreeVerifier { } struct MerkleVerifier { - witness: std::vec::IntoIter<::Hash>, + witness: std::vec::IntoIter<::Hash>, column_values: Peekable)>>, layer_column_values: Vec>, } @@ -62,6 +64,7 @@ impl MerkleVerifier { queries: Vec, ) -> Result { let max_log_size = self.column_values.peek().unwrap().0; + assert!(*queries.iter().max().unwrap() < 1 << max_log_size); // A sequence of queries to the current layer. // Each query is a pair of the query index and the known hashes of the children, if any. @@ -144,10 +147,7 @@ impl MerkleVerifier { } } -type ChildrenHashesAtQuery = Option<( - Option<::Hash>, - Option<::Hash>, -)>; +type ChildrenHashesAtQuery = Option<(Option<::Hash>, Option<::Hash>)>; #[derive(Clone, Copy, Debug, Error, PartialEq, Eq)] pub enum MerkleVerificationError { diff --git a/src/core/commitment_scheme/prover.rs b/src/core/commitment_scheme/prover.rs index 837099e35..1bf51c68b 100644 --- a/src/core/commitment_scheme/prover.rs +++ b/src/core/commitment_scheme/prover.rs @@ -1,5 +1,5 @@ +use std::cmp::Reverse; use std::iter::zip; -use std::ops::Deref; use itertools::Itertools; @@ -20,9 +20,7 @@ use super::super::prover::{ use super::super::ColumnVec; use super::utils::TreeVec; use crate::commitment_scheme::blake2_hash::{Blake2sHash, Blake2sHasher}; -use crate::commitment_scheme::merkle_input::{MerkleTreeColumnLayout, MerkleTreeInput}; -use crate::commitment_scheme::mixed_degree_decommitment::MixedDecommitment; -use crate::commitment_scheme::mixed_degree_merkle_tree::MixedDegreeMerkleTree; +use crate::commitment_scheme::prover::{Decommitment, MerkleProver}; use crate::core::channel::Channel; type MerkleHasher = Blake2sHasher; @@ -48,7 +46,7 @@ impl CommitmentSchemeProver { } pub fn roots(&self) -> TreeVec { - self.trees.as_ref().map(|tree| tree.root()) + self.trees.as_ref().map(|tree| tree.commitment.root()) } pub fn polynomials(&self) -> TreeVec> { @@ -132,7 +130,7 @@ impl CommitmentSchemeProver { #[derive(Debug)] pub struct CommitmentSchemeProof { pub proved_values: TreeVec>>, - pub decommitments: TreeVec>, + pub decommitments: TreeVec>, pub queried_values: TreeVec>>, pub proof_of_work: ProofOfWorkProof, pub fri_proof: FriProof, @@ -143,8 +141,7 @@ pub struct CommitmentSchemeProof { pub struct CommitmentTreeProver { pub polynomials: ColumnVec, pub evaluations: ColumnVec>, - pub commitment: MixedDegreeMerkleTree, - column_layout: MerkleTreeColumnLayout, + pub commitment: MerkleProver, } impl CommitmentTreeProver { @@ -155,6 +152,7 @@ impl CommitmentTreeProver { ) -> Self { let evaluations = polynomials .iter() + .sorted_by_key(|eval| Reverse(eval.log_size())) .map(|poly| { poly.evaluate( CanonicCoset::new(poly.log_size() + log_blowup_factor).circle_domain(), @@ -162,32 +160,15 @@ impl CommitmentTreeProver { }) .collect_vec(); - let mut merkle_input = MerkleTreeInput::new(); - const LOG_N_BASEFIELD_ELEMENTS_IN_SACK: u32 = 4; - - // The desired depth for column of log_length n is such that Blake2s hashes are filled(64B). - // Explicitly: There are 2^(d-1) hash 'sacks' at depth d, hence, with elements of 4 bytes, - // 2^(d-1) = 2^n / 16, => d = n-3. - // Assuming rectangle trace, all columns go to the same depth. - // TOOD(AlonH): remove this assumption. - let inject_depth = std::cmp::max::( - evaluations[0].len().ilog2() as i32 - (LOG_N_BASEFIELD_ELEMENTS_IN_SACK as i32 - 1), - 1, + let tree = MerkleProver::::commit( + evaluations.iter().map(|e| &e.values).collect_vec(), ); - for column in evaluations.iter().map(|eval| &eval.values) { - merkle_input.insert_column(inject_depth as usize, column); - } - let (tree, root) = - MixedDegreeMerkleTree::::commit_default(&merkle_input); - channel.mix_digest(root); - - let column_layout = merkle_input.column_layout(); + channel.mix_digest(tree.root()); CommitmentTreeProver { polynomials, evaluations, commitment: tree, - column_layout, } } @@ -196,39 +177,16 @@ impl CommitmentTreeProver { fn decommit( &self, queries: Vec, - ) -> ( - ColumnVec>, - MixedDecommitment, - ) { + ) -> (ColumnVec>, Decommitment) { + // TODO(spapini): Queries should be the queries to the largest layer. + // When we have more than one component, we should extract the values correctly let values = self .evaluations .iter() .map(|c| queries.iter().map(|p| c[*p]).collect()) .collect(); - // Assuming rectangle trace, queries should be similar for all columns. - // TOOD(AlonH): remove this assumption. - let queries = std::iter::repeat(queries.to_vec()) - .take(self.evaluations.len()) - .collect_vec(); - // Rebuild the merkle input for now. - // TODO(Ohad): change after tree refactor. Consider removing the input struct and have the - // decommitment take queries and columns only. - let eval_vec = self - .evaluations - .iter() - .map(|eval| &eval.values[..]) - .collect_vec(); - let input = self.column_layout.build_input(&eval_vec); - let decommitment = self.commitment.decommit(&input, &queries); + let decommitment = self.commitment.decommit(queries); (values, decommitment) } } - -impl Deref for CommitmentTreeProver { - type Target = MixedDegreeMerkleTree; - - fn deref(&self) -> &Self::Target { - &self.commitment - } -} diff --git a/src/core/commitment_scheme/verifier.rs b/src/core/commitment_scheme/verifier.rs index 3c09b88c6..45c8e6d0f 100644 --- a/src/core/commitment_scheme/verifier.rs +++ b/src/core/commitment_scheme/verifier.rs @@ -1,3 +1,4 @@ +use std::cmp::Reverse; use std::iter::zip; use itertools::Itertools; @@ -17,7 +18,8 @@ use super::super::queries::SparseSubCircleDomain; use super::utils::TreeVec; use super::CommitmentSchemeProof; use crate::commitment_scheme::blake2_hash::{Blake2sHash, Blake2sHasher}; -use crate::commitment_scheme::mixed_degree_decommitment::MixedDecommitment; +use crate::commitment_scheme::prover::Decommitment; +use crate::commitment_scheme::verifier::MerkleTreeVerifier; use crate::core::channel::Channel; use crate::core::prover::VerificationError; use crate::core::ColumnVec; @@ -83,20 +85,19 @@ impl CommitmentSchemeVerifier { if !self .trees .as_ref() - .zip(&proof.decommitments) - .map(|(tree, decommitment)| { + .zip(proof.decommitments) + .zip(proof.queried_values.clone()) + .map(|((tree, decommitment), queried_values)| { // TODO(spapini): Also verify proved_values here. // Assuming columns are of equal lengths, replicate queries for all columns. // TOOD(AlonH): remove this assumption. tree.verify( decommitment, - &std::iter::repeat( - fri_query_domains[&(tree.log_sizes[0] + LOG_BLOWUP_FACTOR)] - .flatten() - .clone(), - ) - .take(tree.log_sizes.len()) - .collect_vec(), + queried_values, + // Queries to the largest size. + fri_query_domains[&(tree.log_sizes[0] + LOG_BLOWUP_FACTOR)] + .flatten() + .clone(), ) }) .iter() @@ -186,13 +187,23 @@ impl CommitmentTreeVerifier { pub fn verify( &self, - decommitment: &MixedDecommitment, - queries: &[Vec], + decommitment: Decommitment, + values: Vec>, + queries: Vec, ) -> bool { - decommitment.verify( - self.commitment, - queries, - decommitment.queried_values.iter().copied(), - ) + let values = self + .log_sizes + .iter() + .map(|log_size| *log_size + LOG_BLOWUP_FACTOR) + .zip(values) + .sorted_by_key(|(log_size, _)| Reverse(*log_size)) + .collect_vec(); + + // TODO(spapini): Propagate error. + MerkleTreeVerifier { + root: self.commitment, + } + .verify(queries, values, decommitment) + .is_ok() } }