use crate::{wasm_flat_vector::WasmFlatVector, wasm_vector::WasmVector};
use ark_poly::{univariate::DensePolynomial, DenseUVPolynomial, EvaluationDomain, Evaluations};
use core::ops::Deref;
use paste::paste;
use poly_commitment::{
commitment::b_poly_coefficients, hash_map_cache::HashMapCache, ipa::SRS, SRS as ISRS,
};
use serde::{Deserialize, Serialize};
use std::{
fs::{File, OpenOptions},
io::{BufReader, BufWriter, Seek, SeekFrom::Start},
sync::Arc,
};
use wasm_bindgen::prelude::*;
macro_rules! impl_srs {
($name: ident,
$WasmF: ty,
$WasmG: ty,
$F: ty,
$G: ty,
$WasmPolyComm: ty,
$field_name: ident) => {
paste! {
#[wasm_bindgen]
#[derive(Clone)]
pub struct [<Wasm $field_name:camel Srs>](
#[wasm_bindgen(skip)]
pub Arc<SRS<$G>>);
impl Deref for [<Wasm $field_name:camel Srs>] {
type Target = Arc<SRS<$G>>;
fn deref(&self) -> &Self::Target { &self.0 }
}
impl From<Arc<SRS<$G>>> for [<Wasm $field_name:camel Srs>] {
fn from(x: Arc<SRS<$G>>) -> Self {
[<Wasm $field_name:camel Srs>](x)
}
}
impl From<&Arc<SRS<$G>>> for [<Wasm $field_name:camel Srs>] {
fn from(x: &Arc<SRS<$G>>) -> Self {
[<Wasm $field_name:camel Srs>](x.clone())
}
}
impl From<[<Wasm $field_name:camel Srs>]> for Arc<SRS<$G>> {
fn from(x: [<Wasm $field_name:camel Srs>]) -> Self {
x.0
}
}
impl From<&[<Wasm $field_name:camel Srs>]> for Arc<SRS<$G>> {
fn from(x: &[<Wasm $field_name:camel Srs>]) -> Self {
x.0.clone()
}
}
impl<'a> From<&'a [<Wasm $field_name:camel Srs>]> for &'a Arc<SRS<$G>> {
fn from(x: &'a [<Wasm $field_name:camel Srs>]) -> Self {
&x.0
}
}
#[wasm_bindgen]
pub fn [<$name:snake _create>](depth: i32) -> [<Wasm $field_name:camel Srs>] {
Arc::new(SRS::create(depth as usize)).into()
}
#[wasm_bindgen]
pub fn [<$name:snake _add_lagrange_basis>](
srs: &[<Wasm $field_name:camel Srs>],
log2_size: i32,
) {
crate::rayon::run_in_pool(|| {
let domain = EvaluationDomain::<$F>::new(1 << (log2_size as usize)).expect("invalid domain size");
srs.get_lagrange_basis(domain);
});
}
#[wasm_bindgen]
pub fn [<$name:snake _write>](
append: Option<bool>,
srs: &[<Wasm $field_name:camel Srs>],
path: String,
) -> Result<(), JsValue> {
let file = OpenOptions::new()
.append(append.unwrap_or(true))
.open(path)
.map_err(|err| {
JsValue::from_str(format!("caml_pasta_fp_urs_write: {}", err).as_str())
})?;
let file = BufWriter::new(file);
srs.0.serialize(&mut rmp_serde::Serializer::new(file))
.map_err(|e| JsValue::from_str(format!("caml_pasta_fp_urs_write: {}", e).as_str()))
}
#[wasm_bindgen]
pub fn [<$name:snake _read>](
offset: Option<i32>,
path: String,
) -> Result<Option<[<Wasm $field_name:camel Srs>]>, JsValue> {
let file = File::open(path).map_err(|err| {
JsValue::from_str(format!("caml_pasta_fp_urs_read: {}", err).as_str())
})?;
let mut reader = BufReader::new(file);
if let Some(offset) = offset {
reader.seek(Start(offset as u64)).map_err(|err| {
JsValue::from_str(format!("caml_pasta_fp_urs_read: {}", err).as_str())
})?;
}
let srs = match SRS::<$G>::deserialize(&mut rmp_serde::Deserializer::new(reader)) {
Ok(srs) => srs,
Err(_) => return Ok(None),
};
Ok(Some(Arc::new(srs).into()))
}
#[wasm_bindgen]
pub fn [<$name:snake _lagrange_commitments_whole_domain_ptr>](
srs: &[<Wasm $field_name:camel Srs>],
domain_size: i32,
) -> *mut WasmVector<$WasmPolyComm> {
let comm = srs
.get_lagrange_basis_from_domain_size(domain_size as usize)
.clone()
.into_iter()
.map(|x| x.into())
.collect();
let boxed_comm = Box::<WasmVector<WasmPolyComm>>::new(comm);
Box::into_raw(boxed_comm)
}
#[wasm_bindgen]
pub fn [<$name:snake _lagrange_commitments_whole_domain_read_from_ptr>](
ptr: *mut WasmVector<$WasmPolyComm>,
) -> WasmVector<$WasmPolyComm> {
let b = unsafe { Box::from_raw(ptr) };
b.as_ref().clone()
}
#[wasm_bindgen]
pub fn [<$name:snake _lagrange_commitment>](
srs: &[<Wasm $field_name:camel Srs>],
domain_size: i32,
i: i32,
) -> Result<$WasmPolyComm, JsValue> {
let x_domain = EvaluationDomain::<$F>::new(domain_size as usize).ok_or_else(|| {
JsValue::from_str("caml_pasta_fp_urs_lagrange_commitment")
})?;
let basis =
crate::rayon::run_in_pool(|| {
srs.get_lagrange_basis(x_domain)
});
Ok(basis[i as usize].clone().into())
}
#[wasm_bindgen]
pub fn [<$name:snake _commit_evaluations>](
srs: &[<Wasm $field_name:camel Srs>],
domain_size: i32,
evals: WasmFlatVector<$WasmF>,
) -> Result<$WasmPolyComm, JsValue> {
let x_domain = EvaluationDomain::<$F>::new(domain_size as usize).ok_or_else(|| {
JsValue::from_str("caml_pasta_fp_urs_commit_evaluations")
})?;
let evals = evals.into_iter().map(Into::into).collect();
let p = Evaluations::<$F>::from_vec_and_domain(evals, x_domain).interpolate();
Ok(srs.commit_non_hiding(&p, 1).into())
}
#[wasm_bindgen]
pub fn [<$name:snake _b_poly_commitment>](
srs: &[<Wasm $field_name:camel Srs>],
chals: WasmFlatVector<$WasmF>,
) -> Result<$WasmPolyComm, JsValue> {
let result = crate::rayon::run_in_pool(|| {
let chals: Vec<$F> = chals.into_iter().map(Into::into).collect();
let coeffs = b_poly_coefficients(&chals);
let p = DensePolynomial::<$F>::from_coefficients_vec(coeffs);
srs.commit_non_hiding(&p, 1)
});
Ok(result.into())
}
#[wasm_bindgen]
pub fn [<$name:snake _batch_accumulator_check>](
srs: &[<Wasm $field_name:camel Srs>],
comms: WasmVector<$WasmG>,
chals: WasmFlatVector<$WasmF>,
) -> bool {
crate::rayon::run_in_pool(|| {
let comms: Vec<_> = comms.into_iter().map(Into::into).collect();
let chals: Vec<_> = chals.into_iter().map(Into::into).collect();
crate::urs_utils::batch_dlog_accumulator_check(&srs, &comms, &chals)
})
}
#[wasm_bindgen]
pub fn [<$name:snake _batch_accumulator_generate>](
srs: &[<Wasm $field_name:camel Srs>],
comms: i32,
chals: WasmFlatVector<$WasmF>,
) -> WasmVector<$WasmG> {
crate::urs_utils::batch_dlog_accumulator_generate::<$G>(
&srs,
comms as usize,
&chals.into_iter().map(From::from).collect(),
).into_iter().map(Into::into).collect()
}
#[wasm_bindgen]
pub fn [<$name:snake _h>](srs: &[<Wasm $field_name:camel Srs>]) -> $WasmG {
srs.h.into()
}
}
}
}
pub mod fp {
use super::*;
use crate::{
arkworks::{WasmGVesta as WasmG, WasmPastaFp},
poly_comm::vesta::WasmFpPolyComm as WasmPolyComm,
};
use mina_curves::pasta::{Fp, Vesta as G};
impl_srs!(caml_fp_srs, WasmPastaFp, WasmG, Fp, G, WasmPolyComm, Fp);
#[wasm_bindgen]
pub fn caml_fp_srs_create_parallel(depth: i32) -> WasmFpSrs {
crate::rayon::run_in_pool(|| Arc::new(SRS::<G>::create_parallel(depth as usize)).into())
}
#[wasm_bindgen]
pub fn caml_fp_srs_get(srs: &WasmFpSrs) -> WasmVector<WasmG> {
let mut h_and_gs: Vec<WasmG> = vec![srs.0.h.into()];
h_and_gs.extend(srs.0.g.iter().map(|x: &G| WasmG::from(*x)));
h_and_gs.into()
}
#[wasm_bindgen]
pub fn caml_fp_srs_set(h_and_gs: WasmVector<WasmG>) -> WasmFpSrs {
let mut h_and_gs: Vec<G> = h_and_gs.into_iter().map(|x| x.into()).collect();
let h = h_and_gs.remove(0);
let g = h_and_gs;
let srs = SRS::<G> {
h,
g,
lagrange_bases: HashMapCache::new(),
};
Arc::new(srs).into()
}
#[wasm_bindgen]
pub fn caml_fp_srs_maybe_lagrange_commitment(
srs: &WasmFpSrs,
domain_size: i32,
i: i32,
) -> Option<WasmPolyComm> {
if !(srs.0.lagrange_bases.contains_key(&(domain_size as usize))) {
return None;
}
let basis = srs.get_lagrange_basis_from_domain_size(domain_size as usize);
Some(basis[i as usize].clone().into())
}
#[wasm_bindgen]
pub fn caml_fp_srs_set_lagrange_basis(
srs: &WasmFpSrs,
domain_size: i32,
input_bases: WasmVector<WasmPolyComm>,
) {
srs.lagrange_bases
.get_or_generate(domain_size as usize, || {
input_bases.into_iter().map(Into::into).collect()
});
}
#[wasm_bindgen]
pub fn caml_fp_srs_get_lagrange_basis(
srs: &WasmFpSrs,
domain_size: i32,
) -> WasmVector<WasmPolyComm> {
let basis = crate::rayon::run_in_pool(|| {
let domain =
EvaluationDomain::<Fp>::new(domain_size as usize).expect("invalid domain size");
srs.get_lagrange_basis(domain)
});
basis.iter().map(Into::into).collect()
}
}
pub mod fq {
use super::*;
use crate::{
arkworks::{WasmGPallas as WasmG, WasmPastaFq},
poly_comm::pallas::WasmFqPolyComm as WasmPolyComm,
};
use mina_curves::pasta::{Fq, Pallas as G};
impl_srs!(caml_fq_srs, WasmPastaFq, WasmG, Fq, G, WasmPolyComm, Fq);
#[wasm_bindgen]
pub fn caml_fq_srs_create_parallel(depth: i32) -> WasmFqSrs {
crate::rayon::run_in_pool(|| Arc::new(SRS::<G>::create_parallel(depth as usize)).into())
}
#[wasm_bindgen]
pub fn caml_fq_srs_get(srs: &WasmFqSrs) -> WasmVector<WasmG> {
let mut h_and_gs: Vec<WasmG> = vec![srs.0.h.into()];
h_and_gs.extend(srs.0.g.iter().map(|x: &G| WasmG::from(*x)));
h_and_gs.into()
}
#[wasm_bindgen]
pub fn caml_fq_srs_set(h_and_gs: WasmVector<WasmG>) -> WasmFqSrs {
let mut h_and_gs: Vec<G> = h_and_gs.into_iter().map(|x| x.into()).collect();
let h = h_and_gs.remove(0);
let g = h_and_gs;
let srs = SRS::<G> {
h,
g,
lagrange_bases: HashMapCache::new(),
};
Arc::new(srs).into()
}
#[wasm_bindgen]
pub fn caml_fq_srs_maybe_lagrange_commitment(
srs: &WasmFqSrs,
domain_size: i32,
i: i32,
) -> Option<WasmPolyComm> {
if !(srs.0.lagrange_bases.contains_key(&(domain_size as usize))) {
return None;
}
let basis = srs.get_lagrange_basis_from_domain_size(domain_size as usize);
Some(basis[i as usize].clone().into())
}
#[wasm_bindgen]
pub fn caml_fq_srs_set_lagrange_basis(
srs: &WasmFqSrs,
domain_size: i32,
input_bases: WasmVector<WasmPolyComm>,
) {
srs.lagrange_bases
.get_or_generate(domain_size as usize, || {
input_bases.into_iter().map(Into::into).collect()
});
}
#[wasm_bindgen]
pub fn caml_fq_srs_get_lagrange_basis(
srs: &WasmFqSrs,
domain_size: i32,
) -> WasmVector<WasmPolyComm> {
let basis = crate::rayon::run_in_pool(|| {
let domain =
EvaluationDomain::<Fq>::new(domain_size as usize).expect("invalid domain size");
srs.get_lagrange_basis(domain)
});
basis.iter().map(Into::into).collect()
}
}