From 3f1f6b0f23c0459abdc92bf6c79b8bd34b61ca05 Mon Sep 17 00:00:00 2001 From: Zhang Junyu Date: Mon, 13 Nov 2023 22:59:09 +0800 Subject: [PATCH] chore: add hint for insuffient K --- Cargo.lock | 9 +- Cargo.toml | 1 + crates/zkwasm/Cargo.toml | 1 + crates/zkwasm/src/circuits/rtable.rs | 8 +- .../zkwasm/src/circuits/test_circuit/mod.rs | 10 +- .../zkwasm/src/circuits/utils/table_entry.rs | 107 +++++++++++------- 6 files changed, 85 insertions(+), 51 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 3319b08c8..565a446d5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -641,6 +641,7 @@ dependencies = [ "specs", "strum", "strum_macros", + "thiserror", "wabt", "wasmi", "zkwasm-host-circuits", @@ -2589,18 +2590,18 @@ checksum = "222a222a5bfe1bba4a77b45ec488a741b3cb8872e5e499451fd7d0129c9c7c3d" [[package]] name = "thiserror" -version = "1.0.44" +version = "1.0.50" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "611040a08a0439f8248d1990b111c95baa9c704c805fa1f62104b39655fd7f90" +checksum = "f9a7210f5c9a7156bb50aa36aed4c95afb51df0df00713949448cf9e97d382d2" dependencies = [ "thiserror-impl", ] [[package]] name = "thiserror-impl" -version = "1.0.44" +version = "1.0.50" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "090198534930841fab3a5d1bb637cde49e339654e606195f8d9c76eeb081dc96" +checksum = "266b2e40bc00e5a6c09c3584011e08b06f123c00362c92b975ba9843aaaa14b8" dependencies = [ "proc-macro2", "quote", diff --git a/Cargo.toml b/Cargo.toml index 463a20284..d576aea19 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -7,6 +7,7 @@ anyhow = { version = "1.0.68", features = ["backtrace"] } halo2aggregator-s = { git = "https://github.com/DelphinusLab/halo2aggregator-s.git", branch = "main", features = ["unsafe"] } halo2_proofs = { git = "https://github.com/DelphinusLab/halo2-gpu-specific.git", default-features = true } parity-wasm = { version = "0.42.0", features = ["sign_ext"] } +thiserror = { version = "1.0.50" } wasmi = { path = "third-party/wasmi" } [profile.dev] diff --git a/crates/zkwasm/Cargo.toml b/crates/zkwasm/Cargo.toml index 1393edbe7..cdddbfba2 100644 --- a/crates/zkwasm/Cargo.toml +++ b/crates/zkwasm/Cargo.toml @@ -30,6 +30,7 @@ anyhow.workspace = true halo2aggregator-s.workspace = true halo2_proofs.workspace = true parity-wasm.workspace = true +thiserror.workspace = true wasmi.workspace = true # TODO put the host circuits into features diff --git a/crates/zkwasm/src/circuits/rtable.rs b/crates/zkwasm/src/circuits/rtable.rs index 85cf31574..69742d130 100644 --- a/crates/zkwasm/src/circuits/rtable.rs +++ b/crates/zkwasm/src/circuits/rtable.rs @@ -18,6 +18,10 @@ use strum::IntoEnumIterator; const POW_OP: u64 = 4; +pub fn maximal_common_range() -> u32 { + (1u32 << (zkwasm_k() - 1)) - 1 +} + /* * | Comment | Op | left(u8) | right | result | * | --------- | --- | -------- | --------------------------- | -------- | @@ -163,11 +167,11 @@ impl RangeTableChip { layouter.assign_table( || "common range table", |mut table| { - for i in 0..(1 << (zkwasm_k() - 1)) { + for i in 0..=maximal_common_range() { table.assign_cell( || "range table", self.config.common_range_col, - i, + i as usize, || Ok(F::from(i as u64)), )?; } diff --git a/crates/zkwasm/src/circuits/test_circuit/mod.rs b/crates/zkwasm/src/circuits/test_circuit/mod.rs index 08816e8e5..09a8ea985 100644 --- a/crates/zkwasm/src/circuits/test_circuit/mod.rs +++ b/crates/zkwasm/src/circuits/test_circuit/mod.rs @@ -210,7 +210,15 @@ impl Circuit for TestCircuit { &self.tables.execution_tables.etable, &memory_writing_table, ) - ); + ) + .map_err(|err| match err { + crate::circuits::utils::table_entry::Error::CircuitSizeNotEnough => { + // TODO: maybe put K in self + Error::NotEnoughRowsAvailable { + current_k: zkwasm_k(), + } + } + })?; let etable_permutation_cells = exec_with_profile!( || "Assign etable", diff --git a/crates/zkwasm/src/circuits/utils/table_entry.rs b/crates/zkwasm/src/circuits/utils/table_entry.rs index 02b0f34f8..3955a0902 100644 --- a/crates/zkwasm/src/circuits/utils/table_entry.rs +++ b/crates/zkwasm/src/circuits/utils/table_entry.rs @@ -1,3 +1,4 @@ +use log::error; use serde::Serialize; use specs::etable::EventTable; use specs::etable::EventTableEntry; @@ -10,10 +11,17 @@ use std::collections::BTreeMap; use std::env; use std::io::Write; use std::path::PathBuf; +use thiserror::Error; -use crate::circuits::config::zkwasm_k; +use crate::circuits::rtable::maximal_common_range; use crate::runtime::memory_event_of_step; +#[derive(Error, Debug)] +pub enum Error { + #[error("Failed to lookup memory writing in MemoryWritingTable. It is usually because K is insufficient for the execution, please increase K and try again.")] + CircuitSizeNotEnough, +} + #[derive(Clone, Debug, Serialize)] pub(in crate::circuits) struct MemoryWritingEntry { index: usize, @@ -32,7 +40,7 @@ pub struct MemoryWritingTable(pub(in crate::circuits) Vec); impl From for MemoryWritingTable { fn from(value: MTable) -> Self { - let maximal_eid = (1u32 << (zkwasm_k() - 1)) - 1; + let maximal_eid = maximal_common_range(); let mut index = 0; let mut entries: Vec = value @@ -128,58 +136,69 @@ impl EventTableWithMemoryInfo { pub(in crate::circuits) fn new( event_table: &EventTable, memory_writing_table: &MemoryWritingTable, - ) -> Self { + ) -> Result { let lookup = memory_writing_table.build_lookup_mapping(); - let lookup_mtable_eid = |(eid, ltype, offset, is_writing)| { + let lookup_mtable_eid = |(eid, ltype, offset, is_writing)| -> Result<(u32, u32), Error> { let records = lookup.get(&(ltype, offset)).unwrap(); + // The MemoryWritingTable uses maximal_common_range as maximal end eid, hence + // searching will be failed if eid exceeds maximal_common_range. + if eid > maximal_common_range() { + error!("Failed to lookup memory writing in MemoryWritingTable. It is usually because K is insufficient for the execution, please increase K and try again."); + + return Err(Error::CircuitSizeNotEnough); + } + if is_writing { - let idx = records - .binary_search_by(|(start_eid, _)| start_eid.cmp(eid)) - .unwrap(); - records[idx] + let idx = records.binary_search_by(|(start_eid, _)| start_eid.cmp(&eid)); + + Ok(records[idx.unwrap()]) } else { - let idx = records - .binary_search_by(|(start_eid, end_eid)| { - if eid <= start_eid { - Ordering::Greater - } else if eid > end_eid { - Ordering::Less - } else { - Ordering::Equal - } - }) - .unwrap(); - records[idx] + let idx = records.binary_search_by(|(start_eid, end_eid)| { + if eid <= *start_eid { + Ordering::Greater + } else if eid > *end_eid { + Ordering::Less + } else { + Ordering::Equal + } + }); + + idx.map(|idx| records[idx]) + .map_err(|_| Error::CircuitSizeNotEnough) } }; - EventTableWithMemoryInfo( - event_table - .entries() - .iter() - .map(|eentry| EventTableEntryWithMemoryInfo { - eentry: eentry.clone(), - memory_rw_entires: memory_event_of_step(eentry, &mut 1) - .iter() - .map(|mentry| { - let (start_eid, end_eid) = lookup_mtable_eid(( - &eentry.eid, - mentry.ltype, - mentry.offset, - mentry.atype == AccessType::Write, - )); - - MemoryRWEntry { - entry: mentry.clone(), - start_eid, - end_eid, - } + let entries = event_table + .entries() + .iter() + .map(|eentry| { + let memory_rw_entires = memory_event_of_step(eentry, &mut 1) + .into_iter() + .map(|mentry| { + let (start_eid, end_eid) = lookup_mtable_eid(( + eentry.eid, + mentry.ltype, + mentry.offset, + mentry.atype == AccessType::Write, + ))?; + + Ok(MemoryRWEntry { + entry: mentry.clone(), + start_eid, + end_eid, }) - .collect(), + }) + .collect::>()?; + + Ok(EventTableEntryWithMemoryInfo { + eentry: eentry.clone(), + memory_rw_entires, }) - .collect(), - ) + }) + .collect::>()?; + + Ok(EventTableWithMemoryInfo(entries)) } }