Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fibonacci Stark Prover #59

Merged
merged 89 commits into from
Mar 2, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
89 commits
Select commit Hold shift + click to select a range
5b71de3
prover sub crate created
pablodeymo Feb 13, 2023
6c2789e
working on fold function
pablodeymo Feb 13, 2023
49bb783
merge
pablodeymo Feb 13, 2023
9e4a345
test working
pablodeymo Feb 13, 2023
bd2b05f
fold test completed
pablodeymo Feb 13, 2023
855d4b3
next_fri_layer function
pablodeymo Feb 13, 2023
0d022a8
Dependencies removed
pablodeymo Feb 13, 2023
f18b906
using iterator step_by
pablodeymo Feb 14, 2023
97c1179
fmt
pablodeymo Feb 14, 2023
b44022e
reordering fri functions
pablodeymo Feb 14, 2023
35abba4
fri_decommit init
pablodeymo Feb 14, 2023
6b8f62c
evaluate_vec in polynomial and reference in evaluate
pablodeymo Feb 14, 2023
a9bd07d
using evaluate_vec
pablodeymo Feb 14, 2023
cff637a
evaluate_vec changed to evaluate_slice
pablodeymo Feb 14, 2023
5a8e336
evaluate_slice changed
pablodeymo Feb 14, 2023
8036639
fri_commitment
pablodeymo Feb 14, 2023
57d19fc
fri continuation
pablodeymo Feb 14, 2023
a207853
comment moved
pablodeymo Feb 15, 2023
85ecd77
fri_decommit_layers
pablodeymo Feb 15, 2023
cf2a09b
comments added
pablodeymo Feb 16, 2023
688a8c9
polynomial.rs merge confilct
pablodeymo Feb 16, 2023
e151449
adapting to the new code
pablodeymo Feb 16, 2023
f15af6b
conflicts solved
pablodeymo Feb 16, 2023
9068b27
append in transcript
pablodeymo Feb 16, 2023
f667207
Merge branch 'main' into fri-operation
pablodeymo Feb 16, 2023
623e374
insert last_evaluation in transcript
pablodeymo Feb 16, 2023
d69145b
beta from transcript.challenge()
pablodeymo Feb 16, 2023
f6dbc60
test: generating subgroups
pablodeymo Feb 16, 2023
a21bc11
prover sub crate created
pablodeymo Feb 13, 2023
e635f0e
Save work in progress
entropidelic Feb 13, 2023
24ffd2a
Add first iteration of function to get composition polynomials from t…
entropidelic Feb 14, 2023
7772c38
Add test for get_composition_poly
entropidelic Feb 14, 2023
0f72803
Add get_coefficients function
entropidelic Feb 14, 2023
999b209
Tidy up code
entropidelic Feb 15, 2023
1e65005
Add docs
entropidelic Feb 15, 2023
9b3fdfb
Fix tests
entropidelic Feb 15, 2023
5ce7556
Add u128_prime field and make get_composition_poly return a Polynomia…
entropidelic Feb 15, 2023
73299e8
Fixes from rebasing
entropidelic Feb 15, 2023
faf992f
Apply clippy suggestions
entropidelic Feb 15, 2023
81f1eb4
Make functions pub crate
entropidelic Feb 15, 2023
d2aeb77
Tidy up code
entropidelic Feb 15, 2023
ee52e75
Tidy up code
entropidelic Feb 15, 2023
9b9258e
Minor fixes
entropidelic Feb 16, 2023
88ba28e
Use U384 instead of U128
entropidelic Feb 22, 2023
8c45aa3
Tidy up code and remove unnecessary u128 field element module
entropidelic Feb 22, 2023
9cd60e0
Merge branch 'arithmetization' into fri-operation
jrchatruc Feb 22, 2023
f34f61d
generate_vec_roots
pablodeymo Feb 22, 2023
b27b769
generate_vec_roots in lib
pablodeymo Feb 22, 2023
0049484
Return trace polynomial from get_composition_poly
entropidelic Feb 22, 2023
6e1e300
coset_factor
pablodeymo Feb 22, 2023
58657b2
Add coset evaluation and fri commitment steps
jrchatruc Feb 22, 2023
1bba26f
Add result to get_cp_and_tp
entropidelic Feb 22, 2023
e363b02
Change error description and module name
entropidelic Feb 22, 2023
11d1f08
Merge branch 'arithmetization' into fri-operation
jrchatruc Feb 23, 2023
082c553
Add decommitment step
jrchatruc Feb 23, 2023
f343e1a
Start filling the stark proof struct
jrchatruc Feb 23, 2023
290114e
Small comments
jrchatruc Feb 23, 2023
331df41
Add first verifier step
jrchatruc Feb 24, 2023
a387a5e
Switch to hardcoded fibonacci trace
jrchatruc Feb 24, 2023
f340b5a
Start FRI verification step
jrchatruc Feb 27, 2023
5bbfa09
More progress
jrchatruc Feb 27, 2023
86075cd
Merge branch 'main' into fri-operation
jrchatruc Feb 28, 2023
3cb371c
Improve code, change field to 17 for testing purposes
MauroToscano Feb 28, 2023
1aa01c3
Fix FRI operation
MauroToscano Feb 28, 2023
331d683
Go back to fibonacci example with test passing
jrchatruc Feb 28, 2023
ce8b1ed
Refactor functions that use fiat shamir to take in a transcript
jrchatruc Feb 28, 2023
2b2514e
Add TODO
jrchatruc Feb 28, 2023
9aa4e9d
Add comments
MauroToscano Mar 1, 2023
609da6f
Moved field definition to lib, removed duplicated definitions
MauroToscano Mar 1, 2023
72db780
Renamed types
MauroToscano Mar 1, 2023
e09904b
Simplified operations
MauroToscano Mar 1, 2023
25d5b1d
Refactor roots of unity generator
MauroToscano Mar 1, 2023
47230dd
Small refactor
MauroToscano Mar 1, 2023
0437286
Refactor roots of unity generator
MauroToscano Mar 1, 2023
635386f
Update comment
MauroToscano Mar 1, 2023
ee8e868
Extracted FRI
MauroToscano Mar 1, 2023
cd6f341
Refactor verify
MauroToscano Mar 1, 2023
9fd980c
Refactor clippy
MauroToscano Mar 1, 2023
f0d5cd3
Re ordered prover
MauroToscano Mar 1, 2023
e635195
cargo fmt
MauroToscano Mar 1, 2023
a1b4d25
fix roots of unity
MauroToscano Mar 1, 2023
d874f5e
Remove air
MauroToscano Mar 1, 2023
7e26c7e
Prover -> Stark
MauroToscano Mar 1, 2023
d2ad4a4
Move folders
MauroToscano Mar 1, 2023
c589654
Uncomment tests, remove unused code
jrchatruc Mar 1, 2023
22a691c
Fix fri_functions tests
jrchatruc Mar 1, 2023
c4b7328
Remove fri_merkle_tree module, move to mod.rs
jrchatruc Mar 1, 2023
5f4ccc2
Clippy
jrchatruc Mar 1, 2023
9c63aaa
Remove TODOs
jrchatruc Mar 1, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -3,4 +3,5 @@
members = [
"math",
"crypto",
"proving-system/stark",
]
2 changes: 1 addition & 1 deletion crypto/src/fiat_shamir/mod.rs
Original file line number Diff line number Diff line change
@@ -1 +1 @@
mod transcript;
pub mod transcript;
10 changes: 5 additions & 5 deletions crypto/src/fiat_shamir/transcript.rs
Original file line number Diff line number Diff line change
@@ -1,24 +1,24 @@
use sha3::{Digest, Sha3_256};

struct Transcript {
pub struct Transcript {
hasher: Sha3_256,
}

impl Transcript {
#[allow(unused)]
fn new() -> Self {
#[allow(clippy::new_without_default)]
pub fn new() -> Self {
Self {
hasher: Sha3_256::new(),
}
}

#[allow(unused)]
fn append(&mut self, new_data: &[u8]) {
pub fn append(&mut self, new_data: &[u8]) {
self.hasher.update(&mut new_data.to_owned());
}

#[allow(unused)]
fn challenge(&mut self) -> [u8; 32] {
pub fn challenge(&mut self) -> [u8; 32] {
let mut result_hash = [0_u8; 32];
result_hash.copy_from_slice(&self.hasher.finalize_reset());
self.hasher.update(result_hash);
Expand Down
2 changes: 1 addition & 1 deletion crypto/src/merkle_tree/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ use crate::hash::traits::IsCryptoHash;
use self::{merkle::MerkleTree, proof::Proof};

pub mod merkle;
mod proof;
pub mod proof;
mod utils;

pub type U64F = U64PrimeField<0xFFFF_FFFF_0000_0001_u64>;
Expand Down
17 changes: 17 additions & 0 deletions crypto/src/merkle_tree/proof.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,29 @@ use lambdaworks_math::{
traits::ByteConversion,
};

#[derive(Debug, Clone)]
pub struct Proof<F: IsField, H: IsCryptoHash<F>> {
pub value: FieldElement<F>,
pub merkle_path: Vec<(FieldElement<F>, bool)>,
pub hasher: H,
}

impl<F: IsField, H: IsCryptoHash<F>> Proof<F, H> {
pub fn verify(&self, root_hash: FieldElement<F>) -> bool {
let mut hashed_value = self.hasher.hash_one(self.value.clone());

for (sibling_node, is_left) in self.merkle_path.iter().rev() {
if *is_left {
hashed_value = self.hasher.hash_two(hashed_value, sibling_node.clone());
} else {
hashed_value = self.hasher.hash_two(sibling_node.clone(), hashed_value);
}
}

root_hash == hashed_value
}
}

impl<F, H> ByteConversion for Proof<F, H>
where
F: IsField,
Expand Down
5 changes: 5 additions & 0 deletions math/src/field/element.rs
Original file line number Diff line number Diff line change
Expand Up @@ -300,6 +300,11 @@ where
&self.value
}

// Returns the representative of the value stored
pub fn representative(&self) -> F::BaseType {
F::representative(self.value.clone())
}

/// Returns the multiplicative inverse of `self`
pub fn inv(&self) -> Self {
Self {
Expand Down
4 changes: 4 additions & 0 deletions math/src/field/extensions/cubic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -127,6 +127,10 @@ where
fn from_base_type(x: [FieldElement<Q::BaseField>; 3]) -> [FieldElement<Q::BaseField>; 3] {
x
}

fn representative(_x: Self::BaseType) -> Self::BaseType {
todo!()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same for this

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

}
}

#[cfg(test)]
Expand Down
4 changes: 4 additions & 0 deletions math/src/field/extensions/quadratic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,10 @@ where
fn from_base_type(x: [FieldElement<Q::BaseField>; 2]) -> [FieldElement<Q::BaseField>; 2] {
x
}

fn representative(_x: Self::BaseType) -> Self::BaseType {
todo!()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same for this

}
}

#[cfg(test)]
Expand Down
5 changes: 5 additions & 0 deletions math/src/field/fields/u384_prime_field.rs
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,11 @@ where
fn from_base_type(x: Self::BaseType) -> Self::BaseType {
MontgomeryAlgorithms::cios(&x, &C::R2, &C::MODULUS, &C::MU)
}

// TO DO: Add tests for representatives
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's make an issue to track this TODO

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We could label it as "good first issue".

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sounds good, created it here #138

fn representative(x: Self::BaseType) -> Self::BaseType {
MontgomeryAlgorithms::cios(&x, &U384::from_u64(1), &C::MODULUS, &C::MU)
}
}

impl<C> ByteConversion for FieldElement<MontgomeryBackendPrimeField<C>>
Expand Down
4 changes: 4 additions & 0 deletions math/src/field/fields/u64_prime_field.rs
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,10 @@ impl<const MODULUS: u64> IsField for U64PrimeField<MODULUS> {
fn from_base_type(x: u64) -> u64 {
Self::from_u64(x)
}

fn representative(x: u64) -> u64 {
x
}
}

impl<const MODULUS: u64> Copy for U64FieldElement<MODULUS> {}
Expand Down
4 changes: 4 additions & 0 deletions math/src/field/test_fields/u64_test_field.rs
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,10 @@ impl<const MODULUS: u64> IsField for U64TestField<MODULUS> {
fn from_base_type(x: u64) -> u64 {
Self::from_u64(x)
}

fn representative(x: u64) -> u64 {
x
}
}

impl<const MODULUS: u64> IsTwoAdicField for U64TestField<MODULUS> {
Expand Down
3 changes: 3 additions & 0 deletions math/src/field/traits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -93,4 +93,7 @@ pub trait IsField: Debug + Clone {
/// Takes as input an element of BaseType and returns the internal representation
/// of that element in the field.
fn from_base_type(x: Self::BaseType) -> Self::BaseType;

// Returns the representative of the value stored
fn representative(a: Self::BaseType) -> Self::BaseType;
}
53 changes: 52 additions & 1 deletion math/src/polynomial.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use std::ops;
/// as a vector of coefficients `[c_0, c_1, ... , c_n]`
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct Polynomial<FE> {
coefficients: Vec<FE>,
pub coefficients: Vec<FE>,
}

impl<F: IsField> Polynomial<FieldElement<F>> {
Expand Down Expand Up @@ -160,6 +160,32 @@ impl<F: IsField> Polynomial<FieldElement<F>> {
}
}

// TODO: This is not an optimal implementation, it should use FFT to interpolate.
pub fn compose<F>(
poly_1: &Polynomial<FieldElement<F>>,
poly_2: &Polynomial<FieldElement<F>>,
) -> Polynomial<FieldElement<F>>
where
F: IsField,
{
let max_degree: u64 = (poly_1.degree() * poly_2.degree()) as u64;

let mut interpolation_points = vec![];
for i in 0_u64..max_degree + 1 {
interpolation_points.push(FieldElement::<F>::from(i));
}

let values: Vec<_> = interpolation_points
.iter()
.map(|value| {
let intermediate_value = poly_2.evaluate(value);
poly_1.evaluate(&intermediate_value)
})
.collect();

Polynomial::interpolate(interpolation_points.as_slice(), values.as_slice())
}

impl<F: IsField> ops::Add<&Polynomial<FieldElement<F>>> for &Polynomial<FieldElement<F>> {
type Output = Polynomial<FieldElement<F>>;

Expand Down Expand Up @@ -232,6 +258,21 @@ impl<F: IsField> ops::Mul<Polynomial<FieldElement<F>>> for Polynomial<FieldEleme
}
}

impl<F: IsField> ops::Mul<FieldElement<F>> for Polynomial<FieldElement<F>> {
type Output = Polynomial<FieldElement<F>>;

fn mul(self, multiplicand: FieldElement<F>) -> Polynomial<FieldElement<F>> {
let new_coefficients = self
.coefficients
.iter()
.map(|value| value * &multiplicand)
.collect();
Polynomial {
coefficients: new_coefficients,
}
}
}

#[cfg(test)]
mod tests {
use crate::field::fields::u64_prime_field::U64PrimeField;
Expand Down Expand Up @@ -494,4 +535,14 @@ mod tests {
let p = Polynomial::interpolate(&[FE::new(0)], &[FE::new(0)]);
assert_eq!(FE::new(0), p.evaluate(&FE::new(0)));
}

#[test]
fn composition_works() {
let p = Polynomial::new(&[FE::new(0), FE::new(2)]);
let q = Polynomial::new(&[FE::new(0), FE::new(0), FE::new(1)]);
assert_eq!(
compose(&p, &q),
Polynomial::new(&[FE::new(0), FE::new(0), FE::new(2)])
);
}
}
1 change: 1 addition & 0 deletions math/src/unsigned_integer/element.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ use std::fmt::Debug;

pub type U384 = UnsignedInteger<6>;
pub type U256 = UnsignedInteger<4>;
pub type U128 = UnsignedInteger<2>;

/// A big unsigned integer in base 2^{64} represented
/// as fixed-size array `limbs` of `u64` components.
Expand Down
12 changes: 12 additions & 0 deletions proving-system/stark/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
[package]
name = "lambdaworks-stark"
version = "0.1.0"
edition = "2021"

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[dependencies]
rand = "0.8.5"
lambdaworks-math = { path = "../../math" }
lambdaworks-crypto = { path = "../../crypto"}
thiserror = "1.0.38"
10 changes: 10 additions & 0 deletions proving-system/stark/src/fri/fri_commitment.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
pub use super::{FriMerkleTree, Polynomial, F, FE};

pub struct FriCommitment<FE> {
pub poly: Polynomial<FE>,
pub domain: Vec<FE>,
pub evaluation: Vec<FE>,
pub merkle_tree: FriMerkleTree,
}

pub type FriCommitmentVec<FE> = Vec<FriCommitment<FE>>;
94 changes: 94 additions & 0 deletions proving-system/stark/src/fri/fri_decommit.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
use super::FE;
use crate::{fri::fri_commitment::FriCommitmentVec, PrimeField};
pub use lambdaworks_crypto::fiat_shamir::transcript::Transcript;
use lambdaworks_crypto::merkle_tree::DefaultHasher;

use lambdaworks_crypto::merkle_tree::proof::Proof;

#[derive(Debug, Clone)]
pub struct FriDecommitment {
pub layer_merkle_paths: Vec<(
Proof<PrimeField, DefaultHasher>,
Proof<PrimeField, DefaultHasher>,
)>,
pub last_layer_evaluation: FE,
}

// verifier chooses a randomness and get the index where
// they want to evaluate the poly
// TODO: encapsulate the return type of this function in a struct.
// This returns a list of authentication paths for evaluations on points and their symmetric counterparts.
pub fn fri_decommit_layers(
commit: &FriCommitmentVec<FE>,
index_to_verify: usize,
) -> FriDecommitment {
let mut index = index_to_verify;

let mut layer_merkle_paths = vec![];

// with every element of the commit, we look for that one in
// the merkle tree and get the corresponding element
for commit_i in commit {
let length_i = commit_i.domain.len();
index %= length_i;
let evaluation_i = commit_i.evaluation[index].clone();
let auth_path = commit_i.merkle_tree.get_proof(&evaluation_i).unwrap();

// symmetrical element
let index_sym = (index + length_i / 2) % length_i;
let evaluation_i_sym = commit_i.evaluation[index_sym].clone();
let auth_path_sym = commit_i.merkle_tree.get_proof(&evaluation_i_sym).unwrap();

layer_merkle_paths.push((auth_path, auth_path_sym));
}

// send the last element of the polynomial
let last = commit.last().unwrap();
let last_evaluation = last.poly.coefficients[0].clone();

FriDecommitment {
layer_merkle_paths,
last_layer_evaluation: last_evaluation,
}
}

// Integration test:
// * get an arbitrary polynomial
// * have a domain containing roots of the unity (# is power of two)
// p = 65_537
// * apply FRI commitment
// * apply FRI decommitment
// assert:
// * evaluations of the polynomials coincide with calculations from the decommitment
// * show a fail example: with a monomial

#[cfg(test)]
mod tests {
use crate::fri::U64PrimeField;
use lambdaworks_math::field::element::FieldElement;
use std::collections::HashSet;
const PRIME_GENERATOR: (u64, u64) = (0xFFFF_FFFF_0000_0001_u64, 2717_u64);
pub type F = U64PrimeField<{ PRIME_GENERATOR.0 }>;
pub type FeGoldilocks = FieldElement<F>;

#[test]
fn test() {
let subgroup_size = 1024_u64;
let generator_field = FeGoldilocks::new(PRIME_GENERATOR.1);
let exp = (PRIME_GENERATOR.0 - 1) / subgroup_size;
let generator_of_subgroup = generator_field.pow(exp);
let mut numbers = HashSet::new();

let mut i = 0;
for exp in 0..1024_u64 {
i += 1;
let ret = generator_of_subgroup.pow(exp);
numbers.insert(*ret.value());
println!("{ret:?}");
}

let count = numbers.len();
println!("count: {count}");
println!("iter: {i}");
}
}
Loading