pub struct MockProver<F: Field> { /* private fields */ }
Expand description
A test prover for debugging circuits.
The normal proving process, when applied to a buggy circuit implementation, might
return proofs that do not validate when they should, but it can’t indicate anything
other than “something is invalid”. MockProver
can be used to figure out why these
are invalid: it stores all the private inputs along with the circuit internals, and
then checks every constraint manually.
§Examples
use halo2_axiom::{
circuit::{Layouter, SimpleFloorPlanner, Value},
dev::{FailureLocation, MockProver, VerifyFailure},
plonk::{Advice, Any, Circuit, Column, ConstraintSystem, Error, Selector},
poly::Rotation,
};
use ff::PrimeField;
use halo2curves::pasta::Fp;
const K: u32 = 5;
#[derive(Copy, Clone)]
struct MyConfig {
a: Column<Advice>,
b: Column<Advice>,
c: Column<Advice>,
s: Selector,
}
#[derive(Clone, Default)]
struct MyCircuit {
a: Value<u64>,
b: Value<u64>,
}
impl<F: PrimeField> Circuit<F> for MyCircuit {
type Config = MyConfig;
type FloorPlanner = SimpleFloorPlanner;
#[cfg(feature = "circuit-params")]
type Params = ();
fn without_witnesses(&self) -> Self {
Self::default()
}
fn configure(meta: &mut ConstraintSystem<F>) -> MyConfig {
let a = meta.advice_column();
let b = meta.advice_column();
let c = meta.advice_column();
let s = meta.selector();
meta.create_gate("R1CS constraint", |meta| {
let a = meta.query_advice(a, Rotation::cur());
let b = meta.query_advice(b, Rotation::cur());
let c = meta.query_advice(c, Rotation::cur());
let s = meta.query_selector(s);
// BUG: Should be a * b - c
Some(("buggy R1CS", s * (a * b + c)))
});
MyConfig { a, b, c, s }
}
fn synthesize(&self, config: MyConfig, mut layouter: impl Layouter<F>) -> Result<(), Error> {
layouter.assign_region(|| "Example region", |mut region| {
config.s.enable(&mut region, 0)?;
region.assign_advice(config.a, 0, self.a.map(F::from));
region.assign_advice(config.b, 0, self.b.map(F::from));
region.assign_advice(config.c, 0, (self.a * self.b).map(F::from));
Ok(())
})
}
}
// Assemble the private inputs to the circuit.
let circuit = MyCircuit {
a: Value::known(2),
b: Value::known(4),
};
// This circuit has no public inputs.
let instance = vec![];
let prover = MockProver::<Fp>::run(K, &circuit, instance).unwrap();
assert_eq!(
prover.verify(),
Err(vec![VerifyFailure::ConstraintNotSatisfied {
constraint: ((0, "R1CS constraint").into(), 0, "buggy R1CS").into(),
location: FailureLocation::OutsideRegion { row: 0 },
cell_values: vec![
(((Any::advice(), 0).into(), 0).into(), "0x2".to_string()),
(((Any::advice(), 1).into(), 0).into(), "0x4".to_string()),
(((Any::advice(), 2).into(), 0).into(), "0x8".to_string()),
],
}])
);
// If we provide a too-small K, we get a panic.
use std::panic;
let result = panic::catch_unwind(|| {
MockProver::<Fp>::run(2, &circuit, vec![]).unwrap_err()
});
assert_eq!(
result.unwrap_err().downcast_ref::<String>().unwrap(),
"n=4, minimum_rows=8, k=2"
);
Implementations§
Source§impl<F: FromUniformBytes<64> + Ord> MockProver<F>
impl<F: FromUniformBytes<64> + Ord> MockProver<F>
Sourcepub fn run<ConcreteCircuit: Circuit<F>>(
k: u32,
circuit: &ConcreteCircuit,
instance: Vec<Vec<F>>,
) -> Result<Self, Error>
pub fn run<ConcreteCircuit: Circuit<F>>( k: u32, circuit: &ConcreteCircuit, instance: Vec<Vec<F>>, ) -> Result<Self, Error>
Runs a synthetic keygen-and-prove operation on the given circuit, collecting data about the constraints and their assignments.
Sourcepub fn advice_values(&self, column: Column<Advice>) -> &[AdviceCellValue<F>]
pub fn advice_values(&self, column: Column<Advice>) -> &[AdviceCellValue<F>]
Return the content of an advice column as assigned by the circuit.
Sourcepub fn fixed_values(&self, column: Column<Fixed>) -> &[CellValue<F>]
pub fn fixed_values(&self, column: Column<Fixed>) -> &[CellValue<F>]
Return the content of a fixed column as assigned by the circuit.
Sourcepub fn verify(&self) -> Result<(), Vec<VerifyFailure>>
pub fn verify(&self) -> Result<(), Vec<VerifyFailure>>
Returns Ok(())
if this MockProver
is satisfied, or a list of errors indicating
the reasons that the circuit is not satisfied.
Sourcepub fn verify_at_rows<I: Clone + Iterator<Item = usize>>(
&self,
gate_row_ids: I,
lookup_input_row_ids: I,
) -> Result<(), Vec<VerifyFailure>>
pub fn verify_at_rows<I: Clone + Iterator<Item = usize>>( &self, gate_row_ids: I, lookup_input_row_ids: I, ) -> Result<(), Vec<VerifyFailure>>
Returns Ok(())
if this MockProver
is satisfied, or a list of errors indicating
the reasons that the circuit is not satisfied.
Constraints are only checked at gate_row_ids
,
and lookup inputs are only checked at lookup_input_row_ids
Sourcepub fn verify_par(&self) -> Result<(), Vec<VerifyFailure>>
pub fn verify_par(&self) -> Result<(), Vec<VerifyFailure>>
Returns Ok(())
if this MockProver
is satisfied, or a list of errors indicating
the reasons that the circuit is not satisfied.
Constraints and lookup are checked at usable_rows
, parallelly.
Sourcepub fn verify_at_rows_par<I: Clone + Iterator<Item = usize>>(
&self,
gate_row_ids: I,
lookup_input_row_ids: I,
) -> Result<(), Vec<VerifyFailure>>
pub fn verify_at_rows_par<I: Clone + Iterator<Item = usize>>( &self, gate_row_ids: I, lookup_input_row_ids: I, ) -> Result<(), Vec<VerifyFailure>>
Returns Ok(())
if this MockProver
is satisfied, or a list of errors indicating
the reasons that the circuit is not satisfied.
Constraints are only checked at gate_row_ids
, and lookup inputs are only checked at lookup_input_row_ids
, parallelly.
Sourcepub fn assert_satisfied(&self)
pub fn assert_satisfied(&self)
Panics if the circuit being checked by this MockProver
is not satisfied.
Any verification failures will be pretty-printed to stderr before the function panics.
Apart from the stderr output, this method is equivalent to:
assert_eq!(prover.verify(), Ok(()));
Sourcepub fn assert_satisfied_par(&self)
pub fn assert_satisfied_par(&self)
Panics if the circuit being checked by this MockProver
is not satisfied.
Any verification failures will be pretty-printed to stderr before the function panics.
Internally, this function uses a parallel aproach in order to verify the MockProver
contents.
Apart from the stderr output, this method is equivalent to:
assert_eq!(prover.verify_par(), Ok(()));
Sourcepub fn assert_satisfied_at_rows_par<I: Clone + Iterator<Item = usize>>(
&self,
gate_row_ids: I,
lookup_input_row_ids: I,
)
pub fn assert_satisfied_at_rows_par<I: Clone + Iterator<Item = usize>>( &self, gate_row_ids: I, lookup_input_row_ids: I, )
Panics if the circuit being checked by this MockProver
is not satisfied.
Any verification failures will be pretty-printed to stderr before the function panics.
Constraints are only checked at gate_row_ids
, and lookup inputs are only checked at lookup_input_row_ids
, parallelly.
Apart from the stderr output, this method is equivalent to:
assert_eq!(prover.verify_at_rows_par(), Ok(()));
Sourcepub fn fixed(&self) -> &Vec<Vec<CellValue<F>>>
pub fn fixed(&self) -> &Vec<Vec<CellValue<F>>>
Returns the list of Fixed Columns used within a MockProver instance and the associated values contained on each Cell.
Sourcepub fn permutation(&self) -> &Assembly
pub fn permutation(&self) -> &Assembly
Returns the permutation argument (Assembly
) used within a MockProver instance.
Trait Implementations§
Source§impl<F: Field> Assignment<F> for MockProver<F>
impl<F: Field> Assignment<F> for MockProver<F>
Source§fn enter_region<NR, N>(&mut self, name: N)
fn enter_region<NR, N>(&mut self, name: N)
Source§fn exit_region(&mut self)
fn exit_region(&mut self)
Source§fn annotate_column<A, AR>(&mut self, annotation: A, column: Column<Any>)
fn annotate_column<A, AR>(&mut self, annotation: A, column: Column<Any>)
Region
. Read moreSource§fn enable_selector<A, AR>(
&mut self,
_: A,
selector: &Selector,
row: usize,
) -> Result<(), Error>
fn enable_selector<A, AR>( &mut self, _: A, selector: &Selector, row: usize, ) -> Result<(), Error>
Source§fn query_instance(
&self,
column: Column<Instance>,
row: usize,
) -> Result<Value<F>, Error>
fn query_instance( &self, column: Column<Instance>, row: usize, ) -> Result<Value<F>, Error>
Source§fn assign_advice<'v>(
&mut self,
column: Column<Advice>,
row: usize,
to: Value<Assigned<F>>,
) -> Value<&'v Assigned<F>>
fn assign_advice<'v>( &mut self, column: Column<Advice>, row: usize, to: Value<Assigned<F>>, ) -> Value<&'v Assigned<F>>
Source§fn assign_fixed(&mut self, column: Column<Fixed>, row: usize, to: Assigned<F>)
fn assign_fixed(&mut self, column: Column<Fixed>, row: usize, to: Assigned<F>)
Source§fn copy(
&mut self,
left_column: Column<Any>,
left_row: usize,
right_column: Column<Any>,
right_row: usize,
)
fn copy( &mut self, left_column: Column<Any>, left_row: usize, right_column: Column<Any>, right_row: usize, )
Source§fn fill_from_row(
&mut self,
col: Column<Fixed>,
from_row: usize,
to: Value<Assigned<F>>,
) -> Result<(), Error>
fn fill_from_row( &mut self, col: Column<Fixed>, from_row: usize, to: Value<Assigned<F>>, ) -> Result<(), Error>
column
starting from the given row
with value to
.Source§fn get_challenge(&self, challenge: Challenge) -> Value<F>
fn get_challenge(&self, challenge: Challenge) -> Value<F>
Source§fn push_namespace<NR, N>(&mut self, _: N)
fn push_namespace<NR, N>(&mut self, _: N)
Source§fn next_phase(&mut self)
fn next_phase(&mut self)
Auto Trait Implementations§
impl<F> Freeze for MockProver<F>
impl<F> RefUnwindSafe for MockProver<F>where
F: RefUnwindSafe,
impl<F> Send for MockProver<F>
impl<F> Sync for MockProver<F>
impl<F> Unpin for MockProver<F>where
F: Unpin,
impl<F> UnwindSafe for MockProver<F>where
F: UnwindSafe + RefUnwindSafe,
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> FmtForward for T
impl<T> FmtForward for T
Source§fn fmt_binary(self) -> FmtBinary<Self>where
Self: Binary,
fn fmt_binary(self) -> FmtBinary<Self>where
Self: Binary,
self
to use its Binary
implementation when Debug
-formatted.Source§fn fmt_display(self) -> FmtDisplay<Self>where
Self: Display,
fn fmt_display(self) -> FmtDisplay<Self>where
Self: Display,
self
to use its Display
implementation when
Debug
-formatted.Source§fn fmt_lower_exp(self) -> FmtLowerExp<Self>where
Self: LowerExp,
fn fmt_lower_exp(self) -> FmtLowerExp<Self>where
Self: LowerExp,
self
to use its LowerExp
implementation when
Debug
-formatted.Source§fn fmt_lower_hex(self) -> FmtLowerHex<Self>where
Self: LowerHex,
fn fmt_lower_hex(self) -> FmtLowerHex<Self>where
Self: LowerHex,
self
to use its LowerHex
implementation when
Debug
-formatted.Source§fn fmt_octal(self) -> FmtOctal<Self>where
Self: Octal,
fn fmt_octal(self) -> FmtOctal<Self>where
Self: Octal,
self
to use its Octal
implementation when Debug
-formatted.Source§fn fmt_pointer(self) -> FmtPointer<Self>where
Self: Pointer,
fn fmt_pointer(self) -> FmtPointer<Self>where
Self: Pointer,
self
to use its Pointer
implementation when
Debug
-formatted.Source§fn fmt_upper_exp(self) -> FmtUpperExp<Self>where
Self: UpperExp,
fn fmt_upper_exp(self) -> FmtUpperExp<Self>where
Self: UpperExp,
self
to use its UpperExp
implementation when
Debug
-formatted.Source§fn fmt_upper_hex(self) -> FmtUpperHex<Self>where
Self: UpperHex,
fn fmt_upper_hex(self) -> FmtUpperHex<Self>where
Self: UpperHex,
self
to use its UpperHex
implementation when
Debug
-formatted.Source§impl<T> Instrument for T
impl<T> Instrument for T
Source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
Source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
Source§impl<T> IntoEither for T
impl<T> IntoEither for T
Source§fn into_either(self, into_left: bool) -> Either<Self, Self>
fn into_either(self, into_left: bool) -> Either<Self, Self>
self
into a Left
variant of Either<Self, Self>
if into_left
is true
.
Converts self
into a Right
variant of Either<Self, Self>
otherwise. Read moreSource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
self
into a Left
variant of Either<Self, Self>
if into_left(&self)
returns true
.
Converts self
into a Right
variant of Either<Self, Self>
otherwise. Read moreSource§impl<T> Pipe for Twhere
T: ?Sized,
impl<T> Pipe for Twhere
T: ?Sized,
Source§fn pipe<R>(self, func: impl FnOnce(Self) -> R) -> Rwhere
Self: Sized,
fn pipe<R>(self, func: impl FnOnce(Self) -> R) -> Rwhere
Self: Sized,
Source§fn pipe_ref<'a, R>(&'a self, func: impl FnOnce(&'a Self) -> R) -> Rwhere
R: 'a,
fn pipe_ref<'a, R>(&'a self, func: impl FnOnce(&'a Self) -> R) -> Rwhere
R: 'a,
self
and passes that borrow into the pipe function. Read moreSource§fn pipe_ref_mut<'a, R>(&'a mut self, func: impl FnOnce(&'a mut Self) -> R) -> Rwhere
R: 'a,
fn pipe_ref_mut<'a, R>(&'a mut self, func: impl FnOnce(&'a mut Self) -> R) -> Rwhere
R: 'a,
self
and passes that borrow into the pipe function. Read moreSource§fn pipe_borrow<'a, B, R>(&'a self, func: impl FnOnce(&'a B) -> R) -> R
fn pipe_borrow<'a, B, R>(&'a self, func: impl FnOnce(&'a B) -> R) -> R
Source§fn pipe_borrow_mut<'a, B, R>(
&'a mut self,
func: impl FnOnce(&'a mut B) -> R,
) -> R
fn pipe_borrow_mut<'a, B, R>( &'a mut self, func: impl FnOnce(&'a mut B) -> R, ) -> R
Source§fn pipe_as_ref<'a, U, R>(&'a self, func: impl FnOnce(&'a U) -> R) -> R
fn pipe_as_ref<'a, U, R>(&'a self, func: impl FnOnce(&'a U) -> R) -> R
self
, then passes self.as_ref()
into the pipe function.Source§fn pipe_as_mut<'a, U, R>(&'a mut self, func: impl FnOnce(&'a mut U) -> R) -> R
fn pipe_as_mut<'a, U, R>(&'a mut self, func: impl FnOnce(&'a mut U) -> R) -> R
self
, then passes self.as_mut()
into the pipe
function.Source§fn pipe_deref<'a, T, R>(&'a self, func: impl FnOnce(&'a T) -> R) -> R
fn pipe_deref<'a, T, R>(&'a self, func: impl FnOnce(&'a T) -> R) -> R
self
, then passes self.deref()
into the pipe function.Source§impl<T> Pointable for T
impl<T> Pointable for T
Source§impl<T> Tap for T
impl<T> Tap for T
Source§fn tap_borrow<B>(self, func: impl FnOnce(&B)) -> Self
fn tap_borrow<B>(self, func: impl FnOnce(&B)) -> Self
Borrow<B>
of a value. Read moreSource§fn tap_borrow_mut<B>(self, func: impl FnOnce(&mut B)) -> Self
fn tap_borrow_mut<B>(self, func: impl FnOnce(&mut B)) -> Self
BorrowMut<B>
of a value. Read moreSource§fn tap_ref<R>(self, func: impl FnOnce(&R)) -> Self
fn tap_ref<R>(self, func: impl FnOnce(&R)) -> Self
AsRef<R>
view of a value. Read moreSource§fn tap_ref_mut<R>(self, func: impl FnOnce(&mut R)) -> Self
fn tap_ref_mut<R>(self, func: impl FnOnce(&mut R)) -> Self
AsMut<R>
view of a value. Read moreSource§fn tap_deref<T>(self, func: impl FnOnce(&T)) -> Self
fn tap_deref<T>(self, func: impl FnOnce(&T)) -> Self
Deref::Target
of a value. Read moreSource§fn tap_deref_mut<T>(self, func: impl FnOnce(&mut T)) -> Self
fn tap_deref_mut<T>(self, func: impl FnOnce(&mut T)) -> Self
Deref::Target
of a value. Read moreSource§fn tap_dbg(self, func: impl FnOnce(&Self)) -> Self
fn tap_dbg(self, func: impl FnOnce(&Self)) -> Self
.tap()
only in debug builds, and is erased in release builds.Source§fn tap_mut_dbg(self, func: impl FnOnce(&mut Self)) -> Self
fn tap_mut_dbg(self, func: impl FnOnce(&mut Self)) -> Self
.tap_mut()
only in debug builds, and is erased in release
builds.Source§fn tap_borrow_dbg<B>(self, func: impl FnOnce(&B)) -> Self
fn tap_borrow_dbg<B>(self, func: impl FnOnce(&B)) -> Self
.tap_borrow()
only in debug builds, and is erased in release
builds.Source§fn tap_borrow_mut_dbg<B>(self, func: impl FnOnce(&mut B)) -> Self
fn tap_borrow_mut_dbg<B>(self, func: impl FnOnce(&mut B)) -> Self
.tap_borrow_mut()
only in debug builds, and is erased in release
builds.Source§fn tap_ref_dbg<R>(self, func: impl FnOnce(&R)) -> Self
fn tap_ref_dbg<R>(self, func: impl FnOnce(&R)) -> Self
.tap_ref()
only in debug builds, and is erased in release
builds.Source§fn tap_ref_mut_dbg<R>(self, func: impl FnOnce(&mut R)) -> Self
fn tap_ref_mut_dbg<R>(self, func: impl FnOnce(&mut R)) -> Self
.tap_ref_mut()
only in debug builds, and is erased in release
builds.Source§fn tap_deref_dbg<T>(self, func: impl FnOnce(&T)) -> Self
fn tap_deref_dbg<T>(self, func: impl FnOnce(&T)) -> Self
.tap_deref()
only in debug builds, and is erased in release
builds.