Skip to content

Commit 47e4cb7

Browse files
fix: propagate chain_id over app SNARKs (batch circuit instances) (#1382)
* propagate chain_id over recursion circuit instances * generic recursion for "propagated" app PI * mark chain id for "propagated" * fix typo * update doc * fix(doc): check CI run --------- Co-authored-by: Rohit Narurkar <[email protected]> Co-authored-by: Rohit Narurkar <[email protected]>
1 parent 54125eb commit 47e4cb7

File tree

4 files changed

+47
-8
lines changed

4 files changed

+47
-8
lines changed

aggregator/README.md

+8-3
Original file line numberDiff line numberDiff line change
@@ -223,27 +223,32 @@ pub trait StateTransition: Sized {
223223
fn state_prev_indices() -> Vec<usize>
224224
fn state_indices() -> Vec<usize>
225225
fn additional_indices() -> Vec<usize>
226+
fn propagate_indices() -> Vec<usize>
226227
}
227228
```
228229

229230
### Public Inputs
230231
All parts of $PI$ in `AppCircuit` is also put into the $PI$ of recursion circuit, the recursion circuit has a single column of $PI$ with following layout:
231232

232233
```markdown
233-
`accumulator` | `preprocessed_digest` | `init_states` | `final_states` | `round`
234+
`accumulator` | `preprocessed_digest` | `init_states` | `final_states` | `propagated_additional_states` | `additional_states` | `round`
234235
```
235236

236237
- `accumulator` accumulates all of the accumulators from the $N$ $snark_{app}$, all the accumulators exported from the $PI$ of these snarks (if there is any), and accumulators generated by the $N$ steps verification of snarks from recursion circuit.
237238
- `preprocessed_digest` represents the Recursion Circuit itself. There would be an unique value for every recursion circuit which can bundle (any number of) snarks from specified `AppCircuit`
238239
- `init_states` represent the initial state $S_0$.
239240
- `final_states` represent the final state, along with the exported $PI$ from $S_N$.
241+
- `propagated_additional_states` represent PIs in app states which do not involved in state transition, however, the PIs in recursion circuit must be "propagated" into the corresponding PI in every app circuit being verified recursively. For example, the PI of `chainID` in each batch circuit must be same when they are verified together in the recursion circuit
242+
- `additional_states` represent the PIs in app state which do not involved in state transition, and only the PIs in the last app circuit for this part are "export" transparently.
240243
- `round` represents the number of batches being bundled recursively, i.e. $N$.
241244

242245
### Statements
243246
To verify the $k_{th}$ snark, we have 3 PIs from the current circuit, the snark of $k_{th}$ `AppCircuit` , and the snark of $(k-1)_{th}$ recursion circuit respectively. We named it $PI$, $PI_{app}$ and $PI_{prev}$. We have following equality constraints for them:
244247

245248
- if $N > 0$, $PI(preprocessed\_digest) = PI_{prev}(preprocessed\_digest)$: ensure the snark forprevious recursion circuitis the same circuit of current one
246249
- if $N > 0$, $PI(round) = PI_{prev}(round) + 1$: ensure the round number is increment so the first snark from app circuit has round = 0
247-
- $PI_{app}(final\_states) = PI(final\_states)$: transparent pass the PI to app circuit
248-
- if $N > 0$, $PI(init\_states) = PI_{prev}(init\_states)$, else $PI(init\_states) = PI_{app}(init\_states)$c: propagate the init state, and for first recursion, the init state part of PI is passed to app circuit
250+
- if $N > 0$, $PI(init\_states) = PI_{prev}(init\_states)$, else $PI(init\_states) = PI_{app}(init\_states)$: propagate the init state, and for first recursion, the init state part of PI is passed to app circuit
251+
- $PI_{app}(final\_states) = PI(final\_states)$: transparent pass the PIs in `final_states` to app circuit
252+
- $PI_{app}(all\_additional\_states) = PI(all\_additional\_states)$: ensure the PIs in `propagated_additional_states` and `additional_states` would be passed to app circuit's PI fields, which are being marked by `additional_indices`
253+
- if $N > 0$, $PI_{app}(propagated\_additional\_states) = PI_{prev}(propagated\_additional\_states)$: propagate the additional state being marked by `propagate_indices`
249254
- $PI_{app}(init\_states) = PI_{prev}(final\_states)$: the init state part of PI for app circuit must bechainedwith previous recursion round

aggregator/src/recursion.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,7 @@ pub trait StateTransition: Sized {
8282
/// Following is the indices of the layout of instance
8383
/// for StateTransition circuit, the default suppose
8484
/// single col of instance, and the layout is:
85-
/// accumulator | prev_state | state | additional
85+
/// accumulator | prev_state | state | additional (propagated) | additional (free)
8686
///
8787
/// Notice we do not verify the layout of accumulator
8888
/// simply suppose they are put in the beginning
@@ -113,4 +113,10 @@ pub trait StateTransition: Sized {
113113
let end = Self::num_instance();
114114
(start..end).collect()
115115
}
116+
117+
/// The indices of any "other instances" which should be propagated, i.e. must remain
118+
/// unchanged in PI of each app circuit.
119+
fn propagate_indices() -> Vec<usize> {
120+
Vec::new()
121+
}
116122
}

aggregator/src/recursion/circuit.rs

+27-4
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,7 @@ pub struct RecursionCircuit<ST> {
8282
default_accumulator: KzgAccumulator<G1Affine, NativeLoader>,
8383
/// The SNARK witness from the k-th BatchCircuit.
8484
app: SnarkWitness,
85-
/// The SNARK witness from the (k-1)-th BatchCircuit.
85+
/// The SNARK witness from the previous RecursionCircuit, i.e. RecursionCircuit up to the (k-1)-th BatchCircuit.
8686
previous: SnarkWitness,
8787
/// The recursion round, starting at round=0 and incrementing at every subsequent recursion.
8888
round: usize,
@@ -302,7 +302,7 @@ impl<ST: StateTransition> Circuit<Fr> for RecursionCircuit<ST> {
302302
// The index of the "state", i.e. the state achieved post the current batch.
303303
let index_state = index_init_state + ST::num_transition_instance();
304304
// The index where the "additional" fields required to define the state are
305-
// present.
305+
// present. The first field in the "additional" fields is the chain ID.
306306
let index_additional_state = index_state + ST::num_transition_instance();
307307
// The index to find the "round" of recursion in the current instance of the
308308
// Recursion Circuit.
@@ -441,6 +441,28 @@ impl<ST: StateTransition> Circuit<Fr> for RecursionCircuit<ST> {
441441
.map(|(&st, &app_inst)| ("passing cur state to app", st, app_inst))
442442
.collect::<Vec<_>>();
443443

444+
// Pick additional inst part in "previous state", verify the items at the front
445+
// is currently propagated to the app inst which is marked as "propagated"
446+
let propagate_app_states = previous_instances[index_additional_state..index_round]
447+
.iter()
448+
.zip(
449+
ST::propagate_indices()
450+
.into_iter()
451+
.map(|i| &app_instances[i]),
452+
)
453+
.map(|(&st, &app_propagated_inst)| {
454+
(
455+
"propagate additional states in app (not first round)",
456+
main_gate.mul(
457+
&mut ctx,
458+
Existing(app_propagated_inst),
459+
Existing(not_first_round),
460+
),
461+
st,
462+
)
463+
})
464+
.collect::<Vec<_>>();
465+
444466
// Verify that the "previous state" (additional state not included) is the same
445467
// as the previous state defined in the current application SNARK. This check is
446468
// meaningful only in subsequent recursion rounds after the first round.
@@ -465,7 +487,7 @@ impl<ST: StateTransition> Circuit<Fr> for RecursionCircuit<ST> {
465487
for (comment, lhs, rhs) in [
466488
// Propagate the preprocessed digest.
467489
(
468-
"chain preprocessed digest",
490+
"propagate preprocessed digest",
469491
main_gate.mul(
470492
&mut ctx,
471493
Existing(preprocessed_digest),
@@ -475,7 +497,7 @@ impl<ST: StateTransition> Circuit<Fr> for RecursionCircuit<ST> {
475497
),
476498
// Verify that "round" increments by 1 when not the first round of recursion.
477499
(
478-
"round increment",
500+
"increment recursion round",
479501
round,
480502
main_gate.add(
481503
&mut ctx,
@@ -488,6 +510,7 @@ impl<ST: StateTransition> Circuit<Fr> for RecursionCircuit<ST> {
488510
.chain(initial_state_propagate)
489511
.chain(verify_app_state)
490512
.chain(verify_app_init_state)
513+
.chain(propagate_app_states)
491514
{
492515
use halo2_proofs::dev::unwrap_value;
493516
debug_assert_eq!(

prover/src/recursion.rs

+5
Original file line numberDiff line numberDiff line change
@@ -67,4 +67,9 @@ impl<'a, const N_SNARK: usize> StateTransition for RecursionTask<'a, N_SNARK> {
6767
fn num_additional_instance() -> usize {
6868
ADD_INSTANCE
6969
}
70+
71+
fn propagate_indices() -> Vec<usize> {
72+
// the first index of additional indices
73+
vec![Self::additional_indices()[0]]
74+
}
7075
}

0 commit comments

Comments
 (0)