use ark_poly::EvaluationDomain;
use kimchi::circuits::lookup::runtime_tables::RuntimeTableCfg;
use crate::{
arkworks::WasmPastaFq,
gate_vector::fq::WasmGateVector,
srs::fq::WasmFqSrs as WasmSrs,
wasm_flat_vector::WasmFlatVector,
wasm_vector::{fq::*, WasmVector},
};
use kimchi::{
circuits::{constraints::ConstraintSystem, gate::CircuitGate, lookup::tables::LookupTable},
linearization::expr_linearization,
poly_commitment::{ipa::OpeningProof, SRS as _},
prover_index::ProverIndex,
};
use mina_curves::pasta::{Fq, Pallas as GAffine, PallasParameters, Vesta as GAffineOther};
use mina_poseidon::{constants::PlonkSpongeConstantsKimchi, sponge::DefaultFqSponge};
use serde::{Deserialize, Serialize};
use std::{
fs::{File, OpenOptions},
io::{BufReader, BufWriter, Seek, SeekFrom::Start},
};
use wasm_bindgen::prelude::*;
#[wasm_bindgen]
pub struct WasmPastaFqPlonkIndex(
#[wasm_bindgen(skip)] pub Box<ProverIndex<GAffine, OpeningProof<GAffine>>>,
);
#[wasm_bindgen]
pub struct WasmPastaFqLookupTable {
#[wasm_bindgen(skip)]
pub id: i32,
#[wasm_bindgen(skip)]
pub data: WasmVecVecFq,
}
impl From<WasmPastaFqLookupTable> for LookupTable<Fq> {
fn from(wasm_lt: WasmPastaFqLookupTable) -> LookupTable<Fq> {
LookupTable {
id: wasm_lt.id,
data: wasm_lt.data.0,
}
}
}
#[wasm_bindgen]
impl WasmPastaFqLookupTable {
#[wasm_bindgen(constructor)]
pub fn new(id: i32, data: WasmVecVecFq) -> WasmPastaFqLookupTable {
WasmPastaFqLookupTable { id, data }
}
}
#[wasm_bindgen]
pub struct WasmPastaFqRuntimeTableCfg {
#[wasm_bindgen(skip)]
pub id: i32,
#[wasm_bindgen(skip)]
pub first_column: WasmFlatVector<WasmPastaFq>,
}
impl From<WasmPastaFqRuntimeTableCfg> for RuntimeTableCfg<Fq> {
fn from(wasm_rt_cfg: WasmPastaFqRuntimeTableCfg) -> Self {
Self {
id: wasm_rt_cfg.id,
first_column: wasm_rt_cfg
.first_column
.into_iter()
.map(Into::into)
.collect(),
}
}
}
#[wasm_bindgen]
impl WasmPastaFqRuntimeTableCfg {
#[wasm_bindgen(constructor)]
pub fn new(id: i32, first_column: WasmFlatVector<WasmPastaFq>) -> Self {
Self { id, first_column }
}
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_create(
gates: &WasmGateVector,
public_: i32,
lookup_tables: WasmVector<WasmPastaFqLookupTable>,
runtime_table_cfgs: WasmVector<WasmPastaFqRuntimeTableCfg>,
prev_challenges: i32,
srs: &WasmSrs,
) -> Result<WasmPastaFqPlonkIndex, JsError> {
console_error_panic_hook::set_once();
let index = crate::rayon::run_in_pool(|| {
let gates: Vec<_> = gates
.0
.iter()
.map(|gate| CircuitGate::<Fq> {
typ: gate.typ,
wires: gate.wires,
coeffs: gate.coeffs.clone(),
})
.collect();
let rust_runtime_table_cfgs: Vec<RuntimeTableCfg<Fq>> =
runtime_table_cfgs.into_iter().map(Into::into).collect();
let rust_lookup_tables: Vec<LookupTable<Fq>> =
lookup_tables.into_iter().map(Into::into).collect();
let cs = match ConstraintSystem::<Fq>::create(gates)
.public(public_ as usize)
.prev_challenges(prev_challenges as usize)
.lookup(rust_lookup_tables)
.runtime(if rust_runtime_table_cfgs.is_empty() {
None
} else {
Some(rust_runtime_table_cfgs)
})
.build()
{
Err(_) => {
return Err("caml_pasta_fq_plonk_index_create: could not create constraint system");
}
Ok(cs) => cs,
};
let (endo_q, _endo_r) = poly_commitment::ipa::endos::<GAffineOther>();
srs.0.get_lagrange_basis(cs.domain.d1);
let mut index =
ProverIndex::<GAffine, OpeningProof<GAffine>>::create(cs, endo_q, srs.0.clone());
index.compute_verifier_index_digest::<DefaultFqSponge<PallasParameters, PlonkSpongeConstantsKimchi>>();
Ok(index)
});
match index {
Ok(index) => Ok(WasmPastaFqPlonkIndex(Box::new(index))),
Err(str) => Err(JsError::new(str)),
}
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_max_degree(index: &WasmPastaFqPlonkIndex) -> i32 {
index.0.srs.max_poly_size() as i32
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_public_inputs(index: &WasmPastaFqPlonkIndex) -> i32 {
index.0.cs.public as i32
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_domain_d1_size(index: &WasmPastaFqPlonkIndex) -> i32 {
index.0.cs.domain.d1.size() as i32
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_domain_d4_size(index: &WasmPastaFqPlonkIndex) -> i32 {
index.0.cs.domain.d4.size() as i32
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_domain_d8_size(index: &WasmPastaFqPlonkIndex) -> i32 {
index.0.cs.domain.d8.size() as i32
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_decode(
bytes: &[u8],
srs: &WasmSrs,
) -> Result<WasmPastaFqPlonkIndex, JsError> {
let mut deserializer = rmp_serde::Deserializer::new(bytes);
let mut index =
ProverIndex::<GAffine, OpeningProof<GAffine>>::deserialize(&mut deserializer)
.map_err(|e| JsError::new(&format!("caml_pasta_fq_plonk_index_decode: {}", e)))?;
index.srs = srs.0.clone();
let (linearization, powers_of_alpha) = expr_linearization(Some(&index.cs.feature_flags), true);
index.linearization = linearization;
index.powers_of_alpha = powers_of_alpha;
Ok(WasmPastaFqPlonkIndex(Box::new(index)))
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_encode(index: &WasmPastaFqPlonkIndex) -> Result<Vec<u8>, JsError> {
let mut buffer = Vec::new();
let mut serializer = rmp_serde::Serializer::new(&mut buffer);
index
.0
.serialize(&mut serializer)
.map_err(|e| JsError::new(&format!("caml_pasta_fq_plonk_index_encode: {}", e)))?;
Ok(buffer)
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_read(
offset: Option<i32>,
srs: &WasmSrs,
path: String,
) -> Result<WasmPastaFqPlonkIndex, JsValue> {
let file = match File::open(path) {
Err(_) => return Err(JsValue::from_str("caml_pasta_fq_plonk_index_read")),
Ok(file) => file,
};
let mut r = BufReader::new(file);
if let Some(offset) = offset {
r.seek(Start(offset as u64))
.map_err(|err| JsValue::from_str(&format!("caml_pasta_fq_plonk_index_read: {err}")))?;
}
let mut t = ProverIndex::<GAffine, OpeningProof<GAffine>>::deserialize(
&mut rmp_serde::Deserializer::new(r),
)
.map_err(|err| JsValue::from_str(&format!("caml_pasta_fq_plonk_index_read: {err}")))?;
t.srs = srs.0.clone();
let (linearization, powers_of_alpha) = expr_linearization(Some(&t.cs.feature_flags), true);
t.linearization = linearization;
t.powers_of_alpha = powers_of_alpha;
Ok(WasmPastaFqPlonkIndex(Box::new(t)))
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_write(
append: Option<bool>,
index: &WasmPastaFqPlonkIndex,
path: String,
) -> Result<(), JsValue> {
let file = OpenOptions::new()
.append(append.unwrap_or(true))
.open(path)
.map_err(|_| JsValue::from_str("caml_pasta_fq_plonk_index_write"))?;
let w = BufWriter::new(file);
index
.0
.serialize(&mut rmp_serde::Serializer::new(w))
.map_err(|e| JsValue::from_str(&format!("caml_pasta_fq_plonk_index_read: {e}")))
}
#[wasm_bindgen]
pub fn caml_pasta_fq_plonk_index_serialize(index: &WasmPastaFqPlonkIndex) -> String {
let serialized = rmp_serde::to_vec(&index.0).unwrap();
base64::encode(serialized)
}