From d0acd9d21d68094b8440cbf65b4391f3850704a6 Mon Sep 17 00:00:00 2001 From: Zhang Zhuo Date: Fri, 30 Dec 2022 16:37:23 +0800 Subject: [PATCH] opt(assignment): speedup EndBlock padding --- zkevm-circuits/src/evm_circuit.rs | 28 ++- zkevm-circuits/src/evm_circuit/execution.rs | 229 +++++++++++------- zkevm-circuits/src/evm_circuit/util.rs | 42 +++- .../evm_circuit/util/math_gadget/test_util.rs | 3 +- 4 files changed, 209 insertions(+), 93 deletions(-) diff --git a/zkevm-circuits/src/evm_circuit.rs b/zkevm-circuits/src/evm_circuit.rs index 7a2947cf1..ec5f86218 100644 --- a/zkevm-circuits/src/evm_circuit.rs +++ b/zkevm-circuits/src/evm_circuit.rs @@ -483,19 +483,39 @@ pub mod test { mod evm_circuit_stats { use super::test::*; use super::*; - use crate::evm_circuit::step::ExecutionState; + use crate::{evm_circuit::step::ExecutionState, witness::block_convert}; + use bus_mapping::{circuit_input_builder::CircuitsParams, mock::BlockData}; use eth_types::{bytecode, evm_types::OpcodeId, geth_types::GethData}; use halo2_proofs::halo2curves::bn256::Fr; use halo2_proofs::plonk::ConstraintSystem; use mock::test_ctx::{helpers::*, TestContext}; use strum::IntoEnumIterator; - #[test] - pub fn empty_evm_circuit() { + fn get_empty_witness_block() -> Block { let block: GethData = TestContext::<0, 0>::new(None, |_| {}, |_, _| {}, |b, _| b) .unwrap() .into(); - run_test_circuit_geth_data_default::(block).unwrap(); + let mut builder = + BlockData::new_from_geth_data_with_params(block.clone(), CircuitsParams::default()) + .new_circuit_input_builder(); + builder + .handle_block(&block.eth_block, &block.geth_traces) + .unwrap(); + + block_convert(&builder.block, &builder.code_db).unwrap() + } + + #[test] + pub fn empty_evm_circuit_no_padding() { + let block = get_empty_witness_block(); + run_test_circuit(block).unwrap(); + } + + #[test] + pub fn empty_evm_circuit_with_padding() { + let mut block = get_empty_witness_block(); + block.evm_circuit_pad_to = (1 << 18) - 100; + run_test_circuit(block).unwrap(); } /// This function prints to stdout a table with all the implemented states diff --git a/zkevm-circuits/src/evm_circuit/execution.rs b/zkevm-circuits/src/evm_circuit/execution.rs index a9d31a52a..fcd9a2ba1 100644 --- a/zkevm-circuits/src/evm_circuit/execution.rs +++ b/zkevm-circuits/src/evm_circuit/execution.rs @@ -818,6 +818,7 @@ impl ExecutionConfig { layouter.assign_region( || "Execution step", |mut region| { + log::info!("start execution step assignment"); if is_first_time { is_first_time = false; region.assign_advice( @@ -850,46 +851,52 @@ impl ExecutionConfig { .iter() .map(move |step| (tx, &tx.calls[step.call_index], step)) }) + .chain(std::iter::once((&dummy_tx, &last_call, end_block_not_last))) .peekable(); let evm_rows = block.evm_circuit_pad_to; - let exact = evm_rows == 0; - - let mut no_next_step = false; - let mut get_next = |cur_state: ExecutionState, offset: &usize| match steps.next() { - Some((transaction, _, step)) => Ok(Some(( - transaction, - &transaction.calls[step.call_index], - step, - ))), - None => { - if no_next_step { - return Ok(None); - } - - let mut block_step = end_block_not_last; - let cur_state_height = self.get_step_height(cur_state); - if !exact && offset + cur_state_height >= evm_rows { - log::error!( - "evm circuit larger than evm_rows: {} >= {}", - offset + cur_state_height, - evm_rows - ); - return Err(Error::Synthesis); - } - if exact || evm_rows - (offset + cur_state_height) == 1 { - block_step = end_block_last; - no_next_step = true; + let no_padding = evm_rows == 0; + let assign_q_step = + |region: &mut Region<'_, F>, offset, height| -> Result<(), Error> { + for idx in 0..height { + let offset = offset + idx; + self.q_usable.enable(region, offset)?; + region.assign_advice( + || "step selector", + self.q_step, + offset, + || Value::known(if idx == 0 { F::one() } else { F::zero() }), + )?; + let value = if idx == 0 { + F::zero() + } else { + F::from((height - idx) as u64) + }; + region.assign_advice( + || "step height", + self.num_rows_until_next_step, + offset, + || Value::known(value), + )?; + region.assign_advice( + || "step height inv", + self.num_rows_inv, + offset, + || Value::known(value.invert().unwrap_or(F::zero())), + )?; } + Ok(()) + }; - Ok(Some((&dummy_tx, &last_call, block_step))) + // part1: assign real steps + loop { + let (transaction, call, step) = steps.next().expect("should not be empty"); + let next = steps.peek(); + if next.is_none() { + break; } - }; - - let mut next = get_next(ExecutionState::BeginTx, &offset)?; - while let Some((transaction, call, step)) = next { - next = get_next(step.execution_state, &offset)?; let height = self.get_step_height(step.execution_state); + // Assign the step witness if step.execution_state == ExecutionState::EndTx { let mut tx = transaction.clone(); @@ -930,40 +937,19 @@ impl ExecutionConfig { call, step, height, - next, + next.copied(), power_of_randomness, )?; + // q_step logic - for idx in 0..height { - let offset = offset + idx; - self.q_usable.enable(&mut region, offset)?; - region.assign_advice( - || "step selector", - self.q_step, - offset, - || Value::known(if idx == 0 { F::one() } else { F::zero() }), - )?; - let value = if idx == 0 { - F::zero() - } else { - F::from((height - idx) as u64) - }; - region.assign_advice( - || "step height", - self.num_rows_until_next_step, - offset, - || Value::known(value), - )?; - region.assign_advice( - || "step height inv", - self.num_rows_inv, - offset, - || Value::known(value.invert().unwrap_or(F::zero())), - )?; - } + assign_q_step(&mut region, offset, height)?; + offset += height; + } - if !exact && offset > evm_rows { + // part2: assign non-last EndBlock steps when padding needed + if !no_padding { + if offset >= evm_rows { log::error!( "evm circuit offset larger than padding: {} > {}", offset, @@ -971,13 +957,53 @@ impl ExecutionConfig { ); return Err(Error::Synthesis); } + let height = self.get_step_height(ExecutionState::EndBlock); + debug_assert_eq!(height, 1); + let last_row = evm_rows - 1; + log::trace!( + "assign non-last EndBlock in range [{},{})", + offset, + last_row + ); + self.assign_same_exec_step_in_range( + &mut region, + offset, + last_row, + block, + &dummy_tx, + &last_call, + end_block_not_last, + height, + power_of_randomness, + )?; - if next.is_none() { - // Assert that EndBlock height is 1 - debug_assert_eq!(height, 1); + for row_idx in offset..last_row { + assign_q_step(&mut region, row_idx, height)?; } + offset = last_row; } + // part3: assign the last EndBlock at offset `evm_rows - 1` + let height = self.get_step_height(ExecutionState::EndBlock); + debug_assert_eq!(height, 1); + log::trace!("assign last EndBlock at offset {}", offset); + self.assign_exec_step( + &mut region, + offset, + block, + &dummy_tx, + &last_call, + end_block_last, + height, + None, + power_of_randomness, + )?; + assign_q_step(&mut region, offset, height)?; + // enable q_step_last + self.q_step_last.enable(&mut region, offset)?; + offset += height; + + // part4: // These are still referenced (but not used) in next rows region.assign_advice( || "step height", @@ -992,10 +1018,7 @@ impl ExecutionConfig { || Value::known(F::zero()), )?; - const END_BLOCK_HEIGHT: usize = 1; - self.q_step_last - .enable(&mut region, offset - END_BLOCK_HEIGHT)?; - + log::info!("finish execution step assignment"); log::debug!("assign for region done at offset {}", offset); Ok(()) }, @@ -1004,6 +1027,45 @@ impl ExecutionConfig { Ok(()) } + #[allow(clippy::too_many_arguments)] + fn assign_same_exec_step_in_range( + &self, + region: &mut Region<'_, F>, + offset_begin: usize, + offset_end: usize, + block: &Block, + transaction: &Transaction, + call: &Call, + step: &ExecStep, + height: usize, + power_of_randomness: [F; 31], + ) -> Result<(), Error> { + if offset_end <= offset_begin { + return Ok(()); + } + assert_eq!(height, 1); + assert!(step.rw_indices.is_empty()); + assert!(matches!(step.execution_state, ExecutionState::EndBlock)); + + // Disable access to next step deliberately for "repeatable" step + let region = &mut CachedRegion::<'_, '_, F>::new( + region, + power_of_randomness, + self.advices.to_vec(), + 1, + offset_begin, + ); + self.assign_exec_step_int(region, offset_begin, block, transaction, call, step)?; + + region.replicate_assignment_for_range( + || format!("repeat {:?} rows", step.execution_state), + offset_begin + 1, + offset_end, + )?; + + Ok(()) + } + #[allow(clippy::too_many_arguments)] fn assign_exec_step( &self, @@ -1033,9 +1095,8 @@ impl ExecutionConfig { let region = &mut CachedRegion::<'_, '_, F>::new( region, power_of_randomness, - STEP_WIDTH, + self.advices.to_vec(), MAX_STEP_HEIGHT * 3, - self.advices[0].index(), offset, ); @@ -1235,18 +1296,20 @@ impl ExecutionConfig { let assigned_stored_expressions = self.assign_stored_expressions(region, offset, step)?; // enable with `RUST_LOG=debug` - if log::log_enabled!(log::Level::Debug) - && !(step.execution_state == ExecutionState::EndBlock && step.rw_indices.is_empty()) - { - // expensive function call - Self::check_rw_lookup( - &assigned_stored_expressions, - offset, - step, - call, - transaction, - block, - ); + if log::log_enabled!(log::Level::Debug) { + let is_padding_step = matches!(step.execution_state, ExecutionState::EndBlock) + && step.rw_indices.is_empty(); + if !is_padding_step { + // expensive function call + Self::check_rw_lookup( + &assigned_stored_expressions, + offset, + step, + call, + transaction, + block, + ); + } } Ok(()) } diff --git a/zkevm-circuits/src/evm_circuit/util.rs b/zkevm-circuits/src/evm_circuit/util.rs index e204ab6cd..883536ab9 100644 --- a/zkevm-circuits/src/evm_circuit/util.rs +++ b/zkevm-circuits/src/evm_circuit/util.rs @@ -12,6 +12,7 @@ use halo2_proofs::{ plonk::{Advice, Assigned, Column, ConstraintSystem, Error, Expression, VirtualCells}, poly::Rotation, }; +use itertools::Itertools; use std::collections::BTreeMap; pub(crate) mod common_gadget; @@ -74,6 +75,7 @@ impl Expr for &Cell { pub struct CachedRegion<'r, 'b, F: FieldExt> { region: &'r mut Region<'b, F>, advice: Vec>, + advice_columns: Vec>, power_of_randomness: [F; 31], width_start: usize, height_start: usize, @@ -84,20 +86,52 @@ impl<'r, 'b, F: FieldExt> CachedRegion<'r, 'b, F> { pub(crate) fn new( region: &'r mut Region<'b, F>, power_of_randomness: [F; 31], - width: usize, + advice_columns: Vec>, height: usize, - width_start: usize, height_start: usize, ) -> Self { Self { region, - advice: vec![vec![F::zero(); height]; width], + advice: vec![vec![F::zero(); height]; advice_columns.len()], power_of_randomness, - width_start, + width_start: advice_columns[0].index(), height_start, + advice_columns, } } + /// Repeatedly assign the locally cached witnesses. + /// This method can be used as a "quick" path for assignment for repeated + /// padding rows + pub fn replicate_assignment_for_range( + &mut self, + annotation: A, + offset_begin: usize, + offset_end: usize, + ) -> Result<(), Error> + where + A: Fn() -> AR, + AR: Into, + { + for (v, column) in self + .advice + .iter() + .map(|values| values[0]) + .zip_eq(self.advice_columns.iter()) + { + if v.is_zero_vartime() { + continue; + } + let annotation: &String = &annotation().into(); + for offset in offset_begin..offset_end { + self.region + .assign_advice(|| annotation, *column, offset, || Value::known(v))?; + } + } + + Ok(()) + } + /// Assign an advice column value (witness). pub fn assign_advice<'v, V, VR, A, AR>( &'v mut self, diff --git a/zkevm-circuits/src/evm_circuit/util/math_gadget/test_util.rs b/zkevm-circuits/src/evm_circuit/util/math_gadget/test_util.rs index faafb9ca2..6b17eac44 100644 --- a/zkevm-circuits/src/evm_circuit/util/math_gadget/test_util.rs +++ b/zkevm-circuits/src/evm_circuit/util/math_gadget/test_util.rs @@ -162,9 +162,8 @@ impl> Circuit for UnitTestMathGadgetBaseC let cached_region = &mut CachedRegion::<'_, '_, F>::new( &mut region, power_of_randomness.try_into().unwrap(), - STEP_WIDTH, + config.advices.to_vec(), MAX_STEP_HEIGHT * 3, - config.advices[0].index(), // TODO offset, ); config.step.state.execution_state.assign(