Skip to content

Commit 41f0c6d

Browse files
roynalnarutolispc
andauthored
conditional encode (#1394)
* feat: conditional encoding of batch to blob * chore: add worked example * fix: construction of blob field elements * fix: use actual vs cooked len in appropriate assertions * use batch instead of blob (no encoding) and load seqexconf correct * refactor get_blob_data_bytes to avoid inconsistency * get blob from BatchProvingTask | related changes * lower num advices (but only used for tests) * b64 serde * refactor(prover): move params out of prover/verifier * clean * remove redundant function 'degrees' * add sanity check (decoded blob == batch bytes) * test: decode_blob * make some methods public to work with scroll-prover --------- Co-authored-by: Zhang Zhuo <[email protected]>
1 parent 1e16062 commit 41f0c6d

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

41 files changed

+941
-401
lines changed

.gitignore

+1-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,7 @@
44
.vscode
55
.idea
66
*.log
7-
*.json
87
*.sh
98
*.txt
109
*.srs
11-
tmp
10+
tmp

aggregator/data/batch-task.json

+107
Large diffs are not rendered by default.

aggregator/data/worked-example

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
Romeo and Juliet
2+
Excerpt from Act 2, Scene 2
3+
4+
JULIET
5+
O Romeo, Romeo! wherefore art thou Romeo?
6+
Deny thy father and refuse thy name;
7+
Or, if thou wilt not, be but sworn my love,
8+
And I'll no longer be a Capulet.
9+
10+
ROMEO
11+
[Aside] Shall I hear more, or shall I speak at this?
12+
13+
JULIET
14+
'Tis but thy name that is my enemy;
15+
Thou art thyself, though not a Montague.
16+
What's Montague? it is nor hand, nor foot,
17+
Nor arm, nor face, nor any other part
18+
Belonging to a man. O, be some other name!
19+
What's in a name? that which we call a rose
20+
By any other name would smell as sweet;
21+
So Romeo would, were he not Romeo call'd,
22+
Retain that dear perfection which he owes
23+
Without that title. Romeo, doff thy name,
24+
And for that name which is no part of thee
25+
Take all myself.
26+
27+
ROMEO
28+
I take thee at thy word:
29+
Call me but love, and I'll be new baptized;
30+
Henceforth I never will be Romeo.
31+
32+
JULIET
33+
What man art thou that thus bescreen'd in night
34+
So stumblest on my counsel?

aggregator/src/aggregation/barycentric.rs

+6-2
Original file line numberDiff line numberDiff line change
@@ -351,6 +351,7 @@ mod tests {
351351
use super::*;
352352
use crate::{
353353
blob::{BatchData, KZG_TRUSTED_SETUP},
354+
eip4844::{get_blob_bytes, get_coefficients},
354355
MAX_AGG_SNARKS,
355356
};
356357
use c_kzg::{Blob as RethBlob, KzgProof};
@@ -395,12 +396,15 @@ mod tests {
395396
vec![0; 340],
396397
vec![10; 23],
397398
]);
399+
let batch_bytes = batch.get_batch_data_bytes();
400+
let blob_bytes = get_blob_bytes(&batch_bytes);
401+
let coeffs = get_coefficients(&blob_bytes);
398402

399403
for z in 0..10 {
400404
let z = Scalar::from(u64::try_from(13241234 + z).unwrap());
401405
assert_eq!(
402-
reth_point_evaluation(z, &batch.get_coefficients().map(|c| Scalar::from_raw(c.0))),
403-
interpolate(z, &batch.get_coefficients().map(|c| Scalar::from_raw(c.0)))
406+
reth_point_evaluation(z, &coeffs.map(|c| Scalar::from_raw(c.0))),
407+
interpolate(z, &coeffs.map(|c| Scalar::from_raw(c.0)))
404408
);
405409
}
406410
}

aggregator/src/aggregation/batch_data.rs

+6-2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use eth_types::H256;
12
use ethers_core::utils::keccak256;
23
use halo2_ecc::bigint::CRTInteger;
34
use halo2_proofs::{
@@ -372,6 +373,7 @@ impl<const N_SNARKS: usize> BatchDataConfig<N_SNARKS> {
372373
config
373374
}
374375

376+
#[allow(clippy::too_many_arguments)]
375377
pub fn assign(
376378
&self,
377379
layouter: &mut impl Layouter<Fr>,
@@ -381,13 +383,14 @@ impl<const N_SNARKS: usize> BatchDataConfig<N_SNARKS> {
381383
// `core.rs`. Since these are already constrained, we can just use them as is.
382384
chunks_are_padding: &[AssignedCell<Fr, Fr>],
383385
batch_data: &BatchData<N_SNARKS>,
386+
versioned_hash: H256,
384387
barycentric_assignments: &[CRTInteger<Fr>],
385388
) -> Result<AssignedBatchDataExport, Error> {
386389
self.load_range_tables(layouter)?;
387390

388391
let assigned_rows = layouter.assign_region(
389392
|| "BatchData rows",
390-
|mut region| self.assign_rows(&mut region, challenge_value, batch_data),
393+
|mut region| self.assign_rows(&mut region, challenge_value, batch_data, versioned_hash),
391394
)?;
392395

393396
layouter.assign_region(
@@ -415,13 +418,14 @@ impl<const N_SNARKS: usize> BatchDataConfig<N_SNARKS> {
415418
region: &mut Region<Fr>,
416419
challenge_value: Challenges<Value<Fr>>,
417420
batch_data: &BatchData<N_SNARKS>,
421+
versioned_hash: H256,
418422
) -> Result<Vec<AssignedBatchDataConfig>, Error> {
419423
let n_rows_data = BatchData::<N_SNARKS>::n_rows_data();
420424
let n_rows_metadata = BatchData::<N_SNARKS>::n_rows_metadata();
421425
let n_rows_digest_rlc = BatchData::<N_SNARKS>::n_rows_digest_rlc();
422426
let n_rows_total = BatchData::<N_SNARKS>::n_rows();
423427

424-
let rows = batch_data.to_rows(challenge_value);
428+
let rows = batch_data.to_rows(versioned_hash, challenge_value);
425429
assert_eq!(rows.len(), n_rows_total);
426430

427431
// enable data selector

aggregator/src/aggregation/blob_data.rs

+58-37
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,17 @@
1-
use std::io::Write;
2-
31
use gadgets::util::Expr;
42
use halo2_ecc::bigint::CRTInteger;
53
use halo2_proofs::{
64
circuit::{AssignedCell, Layouter, Region, Value},
75
halo2curves::bn256::Fr,
8-
plonk::{Advice, Column, ConstraintSystem, Error, Expression, SecondPhase, Selector},
6+
plonk::{Advice, Column, ConstraintSystem, Error, Expression, Fixed, SecondPhase, Selector},
97
poly::Rotation,
108
};
119
use itertools::Itertools;
1210
use zkevm_circuits::{table::U8Table, util::Challenges};
1311

1412
use crate::{
15-
aggregation::{decoder::witgen::init_zstd_encoder, rlc::POWS_OF_256},
16-
blob::{BatchData, BLOB_WIDTH, N_BLOB_BYTES, N_DATA_BYTES_PER_COEFFICIENT},
13+
aggregation::rlc::POWS_OF_256,
14+
blob::{BLOB_WIDTH, N_BLOB_BYTES, N_DATA_BYTES_PER_COEFFICIENT},
1715
RlcConfig,
1816
};
1917

@@ -30,7 +28,7 @@ use crate::{
3028
#[derive(Clone, Debug)]
3129
pub struct BlobDataConfig<const N_SNARKS: usize> {
3230
/// Selector to mark the first row in the layout, enabled at offset=0.
33-
q_first: Selector,
31+
q_first: Column<Fixed>,
3432
/// Whether the row is enabled or not. We need exactly N_BLOB_BYTES rows, enabled from offset=1
3533
/// to offset=N_BLOB_BYTES.
3634
q_enabled: Selector,
@@ -47,8 +45,11 @@ pub struct BlobDataConfig<const N_SNARKS: usize> {
4745
}
4846

4947
pub struct AssignedBlobDataExport {
48+
pub enable_encoding_bool: bool,
49+
pub enable_encoding: AssignedCell<Fr, Fr>,
5050
pub bytes_rlc: AssignedCell<Fr, Fr>,
5151
pub bytes_len: AssignedCell<Fr, Fr>,
52+
pub cooked_len: AssignedCell<Fr, Fr>,
5253
}
5354

5455
impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
@@ -58,8 +59,8 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
5859
u8_table: U8Table,
5960
) -> Self {
6061
let config = Self {
62+
q_first: meta.fixed_column(),
6163
q_enabled: meta.selector(),
62-
q_first: meta.complex_selector(),
6364
byte: meta.advice_column(),
6465
is_padding: meta.advice_column(),
6566
bytes_rlc: meta.advice_column_in(SecondPhase),
@@ -76,23 +77,30 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
7677
});
7778

7879
meta.create_gate("BlobDataConfig: first row", |meta| {
79-
let is_first = meta.query_selector(config.q_first);
80+
let is_first = meta.query_fixed(config.q_first, Rotation::cur());
8081

8182
let byte = meta.query_advice(config.byte, Rotation::cur());
8283
let bytes_rlc = meta.query_advice(config.bytes_rlc, Rotation::cur());
8384
let bytes_len = meta.query_advice(config.bytes_len, Rotation::cur());
8485
let is_padding_next = meta.query_advice(config.is_padding, Rotation::next());
8586

87+
let bytes_rlc_next = meta.query_advice(config.bytes_rlc, Rotation::next());
88+
let bytes_len_next = meta.query_advice(config.bytes_len, Rotation::next());
89+
8690
vec![
8791
is_first.expr() * byte,
8892
is_first.expr() * bytes_rlc,
93+
is_first.expr() * bytes_rlc_next,
8994
is_first.expr() * bytes_len,
95+
is_first.expr() * bytes_len_next,
9096
is_first.expr() * is_padding_next,
9197
]
9298
});
9399

94100
meta.create_gate("BlobDataConfig: main gate", |meta| {
95101
let is_enabled = meta.query_selector(config.q_enabled);
102+
let is_skip_rlc = meta.query_fixed(config.q_first, Rotation::prev());
103+
let trigger_rlc = 1.expr() - is_skip_rlc;
96104

97105
let is_padding_curr = meta.query_advice(config.is_padding, Rotation::cur());
98106
let is_padding_prev = meta.query_advice(config.is_padding, Rotation::prev());
@@ -116,6 +124,7 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
116124
// bytes_rlc updates in the non-padded territory
117125
is_enabled.expr()
118126
* (1.expr() - is_padding_curr.expr())
127+
* trigger_rlc.expr()
119128
* (bytes_rlc_prev.expr() * challenges.keccak_input() + byte.expr()
120129
- bytes_rlc_curr.expr()),
121130
// bytes_rlc remains unchanged in padded territory
@@ -125,6 +134,7 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
125134
// bytes_len increments in the non-padded territory
126135
is_enabled.expr()
127136
* (1.expr() - is_padding_curr.expr())
137+
* trigger_rlc.expr()
128138
* (bytes_len_prev.expr() + 1.expr() - bytes_len_curr.expr()),
129139
// bytes_len remains unchanged in padded territory
130140
is_enabled.expr()
@@ -143,15 +153,16 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
143153
layouter: &mut impl Layouter<Fr>,
144154
challenge_value: Challenges<Value<Fr>>,
145155
rlc_config: &RlcConfig,
146-
batch_data: &BatchData<N_SNARKS>,
156+
blob_bytes: &[u8],
147157
barycentric_assignments: &[CRTInteger<Fr>],
148158
) -> Result<AssignedBlobDataExport, Error> {
149-
let (assigned_bytes, bytes_rlc, bytes_len) = layouter.assign_region(
159+
let (assigned_bytes, bytes_rlc, bytes_len, enable_encoding_bool) = layouter.assign_region(
150160
|| "BlobData bytes",
151-
|mut region| self.assign_rows(&mut region, batch_data, &challenge_value),
161+
|mut region| self.assign_rows(&mut region, blob_bytes, &challenge_value),
152162
)?;
163+
let enable_encoding = assigned_bytes[0].clone();
153164

154-
let cooked_bytes_len = layouter.assign_region(
165+
let cooked_len = layouter.assign_region(
155166
|| "BlobData internal checks",
156167
|mut region| {
157168
self.assign_internal_checks(
@@ -165,41 +176,38 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
165176
)?;
166177

167178
Ok(AssignedBlobDataExport {
179+
enable_encoding_bool,
180+
enable_encoding,
168181
bytes_rlc,
169-
bytes_len: cooked_bytes_len,
182+
bytes_len,
183+
cooked_len,
170184
})
171185
}
172186

173187
#[allow(clippy::type_complexity)]
174188
pub fn assign_rows(
175189
&self,
176190
region: &mut Region<Fr>,
177-
batch_data: &BatchData<N_SNARKS>,
191+
blob_bytes: &[u8],
178192
challenges: &Challenges<Value<Fr>>,
179193
) -> Result<
180194
(
181195
Vec<AssignedCell<Fr, Fr>>,
182196
AssignedCell<Fr, Fr>,
183197
AssignedCell<Fr, Fr>,
198+
bool,
184199
),
185200
Error,
186201
> {
187-
let batch_bytes = batch_data.get_batch_data_bytes();
188-
let blob_bytes = {
189-
let mut encoder = init_zstd_encoder(None);
190-
encoder
191-
.set_pledged_src_size(Some(batch_bytes.len() as u64))
192-
.map_err(|_| Error::Synthesis)?;
193-
encoder
194-
.write_all(&batch_bytes)
195-
.map_err(|_| Error::Synthesis)?;
196-
encoder.finish().map_err(|_| Error::Synthesis)?
197-
};
202+
let enable_encoding = blob_bytes[0].eq(&1);
203+
198204
assert!(blob_bytes.len() <= N_BLOB_BYTES, "too many blob bytes");
199205

200-
self.q_first.enable(region, 0)?;
206+
// Assign fixed column and selector.
207+
region.assign_fixed(|| "q_first", self.q_first, 0, || Value::known(Fr::one()))?;
201208
for i in 1..=N_BLOB_BYTES {
202209
self.q_enabled.enable(region, i)?;
210+
region.assign_fixed(|| "q_first", self.q_first, i, || Value::known(Fr::zero()))?;
203211
}
204212

205213
for col in [self.byte, self.bytes_rlc, self.bytes_len, self.is_padding] {
@@ -217,60 +225,69 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
217225
let mut last_bytes_len = None;
218226
for (i, &byte) in blob_bytes.iter().enumerate() {
219227
let byte_value = Value::known(Fr::from(byte as u64));
220-
bytes_rlc = bytes_rlc * challenges.keccak_input() + byte_value;
228+
if i > 0 {
229+
bytes_rlc = bytes_rlc * challenges.keccak_input() + byte_value;
230+
}
221231

232+
let offset = i + 1;
222233
assigned_bytes.push(region.assign_advice(
223234
|| "byte",
224235
self.byte,
225-
i + 1,
236+
offset,
226237
|| byte_value,
227238
)?);
228239
region.assign_advice(
229240
|| "is_padding",
230241
self.is_padding,
231-
i + 1,
242+
offset,
232243
|| Value::known(Fr::zero()),
233244
)?;
234245
last_bytes_rlc =
235-
Some(region.assign_advice(|| "bytes_rlc", self.bytes_rlc, i + 1, || bytes_rlc)?);
246+
Some(region.assign_advice(|| "bytes_rlc", self.bytes_rlc, offset, || bytes_rlc)?);
236247
last_bytes_len = Some(region.assign_advice(
237248
|| "bytes_len",
238249
self.bytes_len,
239-
i + 1,
240-
|| Value::known(Fr::from(i as u64 + 1)),
250+
offset,
251+
|| Value::known(Fr::from(i as u64)),
241252
)?);
242253
}
243254

244255
let mut last_bytes_rlc = last_bytes_rlc.expect("at least 1 byte guaranteed");
245256
let mut last_bytes_len = last_bytes_len.expect("at least 1 byte guaranteed");
246257
for i in blob_bytes.len()..N_BLOB_BYTES {
258+
let offset = i + 1;
247259
assigned_bytes.push(region.assign_advice(
248260
|| "byte",
249261
self.byte,
250-
i + 1,
262+
offset,
251263
|| Value::known(Fr::zero()),
252264
)?);
253265
region.assign_advice(
254266
|| "is_padding",
255267
self.is_padding,
256-
i + 1,
268+
offset,
257269
|| Value::known(Fr::one()),
258270
)?;
259271
last_bytes_rlc = region.assign_advice(
260272
|| "bytes_rlc",
261273
self.bytes_rlc,
262-
i + 1,
274+
offset,
263275
|| last_bytes_rlc.value().cloned(),
264276
)?;
265277
last_bytes_len = region.assign_advice(
266278
|| "bytes_len",
267279
self.bytes_len,
268-
i + 1,
280+
offset,
269281
|| last_bytes_len.value().cloned(),
270282
)?;
271283
}
272284

273-
Ok((assigned_bytes, last_bytes_rlc, last_bytes_len))
285+
Ok((
286+
assigned_bytes,
287+
last_bytes_rlc,
288+
last_bytes_len,
289+
enable_encoding,
290+
))
274291
}
275292

276293
pub fn assign_internal_checks(
@@ -307,6 +324,10 @@ impl<const N_SNARKS: usize> BlobDataConfig<N_SNARKS> {
307324
pows_of_256
308325
};
309326

327+
// The first byte in the blob is a boolean indicating whether or not blob is an encoded
328+
// form of the batch.
329+
rlc_config.enforce_binary(region, &assigned_bytes[0], &mut rlc_config_offset)?;
330+
310331
////////////////////////////////////////////////////////////////////////////////
311332
//////////////////////////////////// LINKING ///////////////////////////////////
312333
////////////////////////////////////////////////////////////////////////////////

0 commit comments

Comments
 (0)