From b8abc2ee8c41c510afd4d01203080b88249f66a7 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Thu, 26 Sep 2024 15:23:16 +0800 Subject: [PATCH 1/2] chore: Bump crate-ci/typos from 1.22.7 to 1.24.6 (#369) Bumps [crate-ci/typos](https://github.com/crate-ci/typos) from 1.22.7 to 1.24.6. - [Release notes](https://github.com/crate-ci/typos/releases) - [Changelog](https://github.com/crate-ci/typos/blob/master/CHANGELOG.md) - [Commits](https://github.com/crate-ci/typos/compare/v1.22.7...v1.24.6) --- updated-dependencies: - dependency-name: crate-ci/typos dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/ci.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 8d9982636..42ed2c67f 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -34,7 +34,7 @@ jobs: runs-on: ubuntu-latest steps: - uses: actions/checkout@v4 - - uses: crate-ci/typos@v1.22.7 + - uses: crate-ci/typos@v1.24.6 with: config: ./typos.toml isolated: true From 7c368af34ed119680e576829405892d9df0bad37 Mon Sep 17 00:00:00 2001 From: guorong009 Date: Thu, 26 Sep 2024 17:16:38 +0800 Subject: [PATCH 2/2] feat: improve error handling(Error::Synthesis) (#368) * feat: add "WitnessMissing" variant to "AssignmentError" * feat: rename "AssignmentError" as "AssignError" --- halo2_frontend/src/circuit.rs | 18 ++++++++++++---- halo2_frontend/src/dev.rs | 33 +++++++++++++++++++++--------- halo2_frontend/src/plonk/error.rs | 29 +++++++++++++++----------- halo2_frontend/src/plonk/keygen.rs | 27 ++++++++++++++++++++---- 4 files changed, 77 insertions(+), 30 deletions(-) diff --git a/halo2_frontend/src/circuit.rs b/halo2_frontend/src/circuit.rs index 48cbc4c09..cef77072d 100644 --- a/halo2_frontend/src/circuit.rs +++ b/halo2_frontend/src/circuit.rs @@ -1,6 +1,6 @@ //! Traits and structs for implementing circuit components. -use crate::plonk::{self, AssignmentError}; +use crate::plonk::{self, AssignError}; use crate::plonk::{ permutation, sealed::{self, SealedPhase}, @@ -154,7 +154,7 @@ impl<'a, F: Field> Assignment for WitnessCollection<'a, F> { fn query_instance(&self, column: Column, row: usize) -> Result, Error> { if !self.usable_rows.contains(&row) { - return Err(Error::AssignmentError(AssignmentError::QueryInstance { + return Err(Error::AssignError(AssignError::QueryInstance { col: column.into(), row, usable_rows: (0, self.usable_rows.end), @@ -189,7 +189,7 @@ impl<'a, F: Field> Assignment for WitnessCollection<'a, F> { } if !self.usable_rows.contains(&row) { - return Err(Error::AssignmentError(AssignmentError::AssignAdvice { + return Err(Error::AssignError(AssignError::AssignAdvice { desc: desc().into(), col: column.into(), row, @@ -198,11 +198,21 @@ impl<'a, F: Field> Assignment for WitnessCollection<'a, F> { })); } + let value = match to().into_field().assign() { + Ok(v) => v, + Err(_) => { + return Err(Error::AssignError(AssignError::WitnessMissing { + func: "assign_advice".to_string(), + desc: desc().into(), + })) + } + }; + *self .advice .get_mut(column.index()) .and_then(|v| v.get_mut(row)) - .ok_or(Error::BoundsFailure)? = to().into_field().assign()?; + .ok_or(Error::BoundsFailure)? = value; Ok(()) } diff --git a/halo2_frontend/src/dev.rs b/halo2_frontend/src/dev.rs index 98657d291..fa5446ddc 100644 --- a/halo2_frontend/src/dev.rs +++ b/halo2_frontend/src/dev.rs @@ -7,7 +7,7 @@ use std::ops::{Add, Mul, Neg, Range}; use blake2b_simd::blake2b; -use crate::plonk::AssignmentError; +use crate::plonk::AssignError; use crate::{ circuit, plonk::{ @@ -415,7 +415,7 @@ impl Assignment for MockProver { } if !self.usable_rows.contains(&row) { - return Err(Error::AssignmentError(AssignmentError::EnableSelector { + return Err(Error::AssignError(AssignError::EnableSelector { desc: desc().into(), selector: *selector, row, @@ -445,7 +445,7 @@ impl Assignment for MockProver { row: usize, ) -> Result, Error> { if !self.usable_rows.contains(&row) { - return Err(Error::AssignmentError(AssignmentError::QueryInstance { + return Err(Error::AssignError(AssignError::QueryInstance { col: column.into(), row, usable_rows: (self.usable_rows.start, self.usable_rows.end), @@ -476,7 +476,7 @@ impl Assignment for MockProver { { if self.in_phase(FirstPhase) { if !self.usable_rows.contains(&row) { - return Err(Error::AssignmentError(AssignmentError::AssignAdvice { + return Err(Error::AssignError(AssignError::AssignAdvice { desc: desc().into(), col: column.into(), row, @@ -504,11 +504,14 @@ impl Assignment for MockProver { .expect("bounds failure"); *value = CellValue::Assigned(to); } - Err(err) => { + Err(_) => { // Propagate `assign` error if the column is in current phase. let phase = self.cs.advice_column_phase[column.index]; if self.in_phase(phase) { - return Err(err); + return Err(Error::AssignError(AssignError::WitnessMissing { + func: "assign_advice".to_string(), + desc: desc().into(), + })); } } } @@ -534,7 +537,7 @@ impl Assignment for MockProver { } if !self.usable_rows.contains(&row) { - return Err(Error::AssignmentError(AssignmentError::AssignFixed { + return Err(Error::AssignError(AssignError::AssignFixed { desc: desc().into(), col: column.into(), row, @@ -552,11 +555,21 @@ impl Assignment for MockProver { .or_default(); } + let value = match to().into_field().evaluate().assign() { + Ok(v) => CellValue::Assigned(v), + Err(_) => { + return Err(Error::AssignError(AssignError::WitnessMissing { + func: "assign_fixed".to_string(), + desc: desc().into(), + })) + } + }; + *self .fixed .get_mut(column.index()) .and_then(|v| v.get_mut(row)) - .expect("bounds failure") = CellValue::Assigned(to().into_field().evaluate().assign()?); + .expect("bounds failure") = value; Ok(()) } @@ -573,7 +586,7 @@ impl Assignment for MockProver { } if !self.usable_rows.contains(&left_row) || !self.usable_rows.contains(&right_row) { - return Err(Error::AssignmentError(AssignmentError::Copy { + return Err(Error::AssignError(AssignError::Copy { left_col: left_column, left_row, right_col: right_column, @@ -598,7 +611,7 @@ impl Assignment for MockProver { } if !self.usable_rows.contains(&from_row) { - return Err(Error::AssignmentError(AssignmentError::FillFromRow { + return Err(Error::AssignError(AssignError::FillFromRow { col: col.into(), from_row, usable_rows: (self.usable_rows.start, self.usable_rows.end), diff --git a/halo2_frontend/src/plonk/error.rs b/halo2_frontend/src/plonk/error.rs index 89f2e9709..61bee8675 100644 --- a/halo2_frontend/src/plonk/error.rs +++ b/halo2_frontend/src/plonk/error.rs @@ -6,7 +6,7 @@ use halo2_middleware::circuit::Any; /// This is an error that could occur during circuit synthesis. /// -/// **NOTE**: [`AssignmentError`] is introduced to provide more debugging info +/// **NOTE**: [`AssignError`] is introduced to provide more debugging info /// to developers when assigning witnesses to circuit cells. /// Hence, they are used for [`MockProver`] and [`WitnessCollection`]. /// The [`keygen`] process use the [`NotEnoughRowsAvailable`], since it is just enough. @@ -32,8 +32,8 @@ pub enum Error { ColumnNotInPermutation(Column), /// An error relating to a lookup table. TableError(TableError), - /// An error relating to an `Assignment`. - AssignmentError(AssignmentError), + /// An error relating to a circuit assignment. + AssignError(AssignError), /// Generic error not covered by previous cases Other(String), } @@ -65,7 +65,7 @@ impl fmt::Display for Error { "Column {column:?} must be included in the permutation. Help: try applying `meta.enable_equalty` on the column", ), Error::TableError(error) => write!(f, "{error}"), - Error::AssignmentError(error) => write!(f, "{error}"), + Error::AssignError(error) => write!(f, "{error}"), Error::Other(error) => write!(f, "Other: {error}"), } } @@ -112,7 +112,7 @@ impl fmt::Display for TableError { /// This is an error that could occur during `assign_advice`, `assign_fixed`, `copy`, etc. #[derive(Debug)] -pub enum AssignmentError { +pub enum AssignError { AssignAdvice { desc: String, col: Column, @@ -154,12 +154,16 @@ pub enum AssignmentError { usable_rows: (usize, usize), k: u32, }, + WitnessMissing { + func: String, + desc: String, + }, } -impl fmt::Display for AssignmentError { +impl fmt::Display for AssignError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match self { - AssignmentError::AssignAdvice { desc, col, row, usable_rows:(start, end), k } => write!( + AssignError::AssignAdvice { desc, col, row, usable_rows:(start, end), k } => write!( f, "assign_advice `{}` error: column={:?}({}), row={}, usable_rows={}..{}, k={}", desc, @@ -169,7 +173,7 @@ impl fmt::Display for AssignmentError { start, end, k, ), - AssignmentError::AssignFixed {desc, col, row, usable_rows: (start, end), k } => write!( + AssignError::AssignFixed {desc, col, row, usable_rows: (start, end), k } => write!( f, "assign_fixed `{}` error: column={:?}({}), row={}, usable_rows={}..{}, k={}", desc, @@ -179,7 +183,7 @@ impl fmt::Display for AssignmentError { start, end, k, ), - AssignmentError::EnableSelector { desc, selector, row, usable_rows: (start, end), k } => write!( + AssignError::EnableSelector { desc, selector, row, usable_rows: (start, end), k } => write!( f, "enable_selector `{}` error: column=Selector({:?}), row={}, usable_rows={}..{}, k={}", desc, @@ -188,7 +192,7 @@ impl fmt::Display for AssignmentError { start, end, k, ), - AssignmentError::QueryInstance { col, row, usable_rows:(start, end), k } => write!( + AssignError::QueryInstance { col, row, usable_rows:(start, end), k } => write!( f, "query_instance error: column={:?}({}), row={}, usable_rows={}..{}, k={}", col.column_type, @@ -198,7 +202,7 @@ impl fmt::Display for AssignmentError { end, k, ), - AssignmentError::Copy { left_col, left_row, right_col, right_row, usable_rows:(start, end), k } => write!( + AssignError::Copy { left_col, left_row, right_col, right_row, usable_rows:(start, end), k } => write!( f, "copy error: left_column={:?}({}), left_row={}, right_column={:?}({}), right_row={}, usable_rows={}..{}, k={}", left_col.column_type(), @@ -210,7 +214,7 @@ impl fmt::Display for AssignmentError { start, end, k, ), - AssignmentError::FillFromRow { col, from_row, usable_rows:(start, end), k } => write!( + AssignError::FillFromRow { col, from_row, usable_rows:(start, end), k } => write!( f, "fill_from_row error: column={:?}({}), from_row={}, usable_rows={}..{}, k={}", col.column_type(), @@ -219,6 +223,7 @@ impl fmt::Display for AssignmentError { start, end, k, ), + AssignError::WitnessMissing { func, desc } => write!(f, "witness missing/unknown when {} `{}`", func, desc), } } } diff --git a/halo2_frontend/src/plonk/keygen.rs b/halo2_frontend/src/plonk/keygen.rs index 8e9b82903..e7afe3034 100644 --- a/halo2_frontend/src/plonk/keygen.rs +++ b/halo2_frontend/src/plonk/keygen.rs @@ -5,7 +5,8 @@ use halo2_middleware::ff::Field; use crate::circuit::Value; use crate::plonk::{ - permutation, Advice, Assigned, Assignment, Challenge, Column, Error, Fixed, Instance, Selector, + permutation, Advice, AssignError, Assigned, Assignment, Challenge, Column, Error, Fixed, + Instance, Selector, }; /// Assembly to be used in circuit synthesis. @@ -75,7 +76,7 @@ impl Assignment for Assembly { fn assign_fixed( &mut self, - _: A, + desc: A, column: Column, row: usize, to: V, @@ -90,11 +91,21 @@ impl Assignment for Assembly { return Err(Error::not_enough_rows_available(self.k)); } + let value = match to().into_field().assign() { + Ok(v) => v, + Err(_) => { + return Err(Error::AssignError(AssignError::WitnessMissing { + func: "assign_fixed".to_string(), + desc: desc().into(), + })) + } + }; + *self .fixed .get_mut(column.index()) .and_then(|v| v.get_mut(row)) - .ok_or(Error::BoundsFailure)? = to().into_field().assign()?; + .ok_or(Error::BoundsFailure)? = value; Ok(()) } @@ -129,7 +140,15 @@ impl Assignment for Assembly { .get_mut(column.index()) .ok_or(Error::BoundsFailure)?; - let filler = to.assign()?; + let filler = match to.assign() { + Ok(v) => v, + Err(_) => { + return Err(Error::AssignError(AssignError::WitnessMissing { + func: "fill_from_row".to_string(), + desc: "".to_string(), + })) + } + }; for row in self.usable_rows.clone().skip(from_row) { col[row] = filler; }