-
Notifications
You must be signed in to change notification settings - Fork 102
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Update public examples #2572
base: main
Are you sure you want to change the base?
Update public examples #2572
Conversation
Out of curiosity: do I understand correctly that we still have a RHS on public declarations, but it is not enforced in the circuit? Is it used for witgen? Can it be omitted? |
Yes we still have a RHS on public declarations and it will be gradually phased out as I implement each backend using our public references. It's not used for witgen. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚠️ Performance Alert ⚠️
Possible performance regression was detected for benchmark 'Benchmarks'.
Benchmark result of this commit is worse than the previous benchmark result exceeding threshold 1.20
.
Benchmark suite | Current: 9a58980 | Previous: 4dc747a | Ratio |
---|---|---|---|
jit-witgen-benchmark/jit_witgen_benchmark |
30250330310 ns/iter (± 93445799 ) |
10493155444 ns/iter (± 118276385 ) |
2.88 |
This comment was automatically generated by workflow using github-action-benchmark.
fn take_public_values(&mut self) -> BTreeMap<String, T> { | ||
if self.publics.is_empty() { | ||
BTreeMap::new() | ||
} else { | ||
let public_values: Vec<_> = self | ||
.value_polys | ||
.iter() | ||
.enumerate() | ||
.flat_map(|(value_index, poly)| { | ||
self.fixed_data.witness_cols[poly] | ||
.external_values | ||
.cloned() | ||
.map(|mut external_values| { | ||
// External witness values might only be provided partially. | ||
external_values.resize(self.degree as usize, T::zero()); | ||
external_values | ||
}) | ||
.unwrap_or_else(|| { | ||
let mut column = vec![T::zero(); 8]; // 8 public values | ||
for (row, values) in self.data.iter() { | ||
column[*row as usize] = values[value_index].unwrap_or_default(); | ||
} | ||
column | ||
}) | ||
}) | ||
.collect(); | ||
|
||
self.publics | ||
.clone() | ||
.into_iter() | ||
.zip(public_values) | ||
.collect() | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This basically slightly modifies the take_witness_values
code but instead only takes the first 8 values. The more proper way to do this should be to solve the public reference constraints using one of our processors, similar to how it's systematically done in dynamic_machine
and block_machine
. I think this is a good simplification, but a bit hacky.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But why 8? It could be any number?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Because there are 8 publics here: https://github.com/powdr-labs/powdr/blob/main/std/machines/write_once_memory_with_8_publics.asm#L24-L31
let (columns, _publics) = | ||
let (columns, publics) = | ||
MutableState::new(machines.into_iter(), &self.query_callback).run(); | ||
|
||
let publics = extract_publics(&columns, self.analyzed); | ||
let publics = extract_publics(publics, self.analyzed); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The key change is that publics are now assembled from public reference witgen and passed off to the backend.
if !parts.identities.is_empty() { | ||
return None; | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
write_once_memory_with_publics
now contain public reference identities, so this can't be used as a filter.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You could check that all identities reference a public though.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
True :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I looked at the first half, but stopping now as I'm not 100% sure I'm looking at the right diff, maybe this needs a rebase?
I'm also pretty confused at this point, because now we also have have two types of publics and sometimes we just set the scalar publics to an empty map. I guess that's because we don't want to have a massive PR that changes everything at once, but I find it hard to track where we are on the road towards only having scalar publics.
} | ||
|
||
impl<'s, F: FieldElement> SubProver<'s, F> { | ||
pub fn resume(self, response: Vec<(String, Vec<F>)>) -> RunStatus<'s, F> { | ||
self.response_sender.send(response).unwrap(); | ||
self.response_sender | ||
.send((response, BTreeMap::new())) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm, I'm always a bit confused by the sender and receiver here, but why do we need to send a map and then always send an empty map?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This no longer appears after rebase as it's added in a prior PR #2567.
However, to answer this question, this is actually inferred by the compiler:
- The witgen callback function now returns both witness and publics, as needed for stage 1 publics: https://github.com/powdr-labs/powdr/blob/main/backend/src/composite/sub_prover.rs#L32-L42
- The returned value
response_receiver.recv().unwrap()
therefore has to be(witness, publics)
.
- It's also required that the sender and receive have the same underlying type (I think somewhere inside this package): https://github.com/powdr-labs/powdr/blob/main/backend/src/composite/sub_prover.rs#L25
- Therefore, the compiler forces that the
response_sender
always send a(witness, publics)
.
I left this out so that #2567 don't grow too large, but I think one potential effect of this is that composite backends won't work for stage 1 publics, but I was planning to work on other backends in future PRs regardless. (It currently isn't an issue because all backends except Mock still use the witness for exposing publics after this PR).
To make sure this isn't forgotten, however, I'm adding a TODO
comment here.
@@ -375,12 +375,22 @@ mod tests { | |||
pol fixed FIRST = [1] + [0]*; | |||
pol witness x; | |||
pol witness y; | |||
y * y = y; | |||
y = 1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why is this change necessary? It seems like before y
was not uniquely constraint, but why did it work before then?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are some witgen errors after I added in the public reference constraints. Previously, y
is simply set to 0 because it can't be uniquely solved and thus passes the constraint "by chance". I think non unique constraints of the type y * y = y
isn't something we are explicitly testing in this public_values()
test and therefore I just made this unique to avoid the witgen error.
However, happy to look deeper into this or defer to another PR for a fix if we wonder why y
is simply set to 0 before but creates a witgen error now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, I think it's fine, just curious :) Thanks!
if !parts.identities.is_empty() { | ||
return None; | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You could check that all identities reference a public though.
fn take_public_values(&mut self) -> BTreeMap<String, T> { | ||
if self.publics.is_empty() { | ||
BTreeMap::new() | ||
} else { | ||
let public_values: Vec<_> = self | ||
.value_polys | ||
.iter() | ||
.enumerate() | ||
.flat_map(|(value_index, poly)| { | ||
self.fixed_data.witness_cols[poly] | ||
.external_values | ||
.cloned() | ||
.map(|mut external_values| { | ||
// External witness values might only be provided partially. | ||
external_values.resize(self.degree as usize, T::zero()); | ||
external_values | ||
}) | ||
.unwrap_or_else(|| { | ||
let mut column = vec![T::zero(); 8]; // 8 public values | ||
for (row, values) in self.data.iter() { | ||
column[*row as usize] = values[value_index].unwrap_or_default(); | ||
} | ||
column | ||
}) | ||
}) | ||
.collect(); | ||
|
||
self.publics | ||
.clone() | ||
.into_iter() | ||
.zip(public_values) | ||
.collect() | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But why 8? It could be any number?
pub fn extract_publics<T>( | ||
public_references: BTreeMap<String, T>, | ||
pil: &Analyzed<T>, | ||
) -> BTreeMap<String, Option<T>> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What does this do exactly? I think the inputs are publics already, but then this makes sure that there is an entry for each public declaration? What would happen if we removed the function and just return the publics as returned from MutableState::run
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Exactly, it makes sure that we have a value for each public declaration, which might not be the case if we have stage-1 publics during stage-0 witgen, in which case extract_publics
will assign a None
value to the stage-1 public declaration. In the end, it's checked in all backend testing functions that all values in the publics map are Some
, for example: https://github.com/powdr-labs/powdr/blob/main/pipeline/src/test_util.rs#L170-L174
@@ -375,12 +375,22 @@ mod tests { | |||
pol fixed FIRST = [1] + [0]*; | |||
pol witness x; | |||
pol witness y; | |||
y * y = y; | |||
y = 1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, I think it's fine, just curious :) Thanks!
// All identities should have a public reference | ||
if !parts.identities.iter().all(|id| { | ||
(*id).children().any(|c| { | ||
c.all_children() | ||
.any(|c| matches!(c, AlgebraicExpression::PublicReference(_))) | ||
}) | ||
}) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// All identities should have a public reference | |
if !parts.identities.iter().all(|id| { | |
(*id).children().any(|c| { | |
c.all_children() | |
.any(|c| matches!(c, AlgebraicExpression::PublicReference(_))) | |
}) | |
}) { | |
// The only identities we'd expect would be to expose public values. | |
if !parts.identities.iter().all(|id| { | |
id.all_children() | |
.any(|c| matches!(c, AlgebraicExpression::PublicReference(_))) | |
}) { |
@@ -263,7 +277,8 @@ impl<'a, T: FieldElement> Machine<'a, T> for WriteOnceMemory<'a, T> { | |||
&mut self, | |||
_mutable_state: &'b MutableState<'a, T, Q>, | |||
) -> HashMap<String, Vec<T>> { | |||
self.value_polys | |||
let witness = self |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this change is not necessary?
public_names: parts | ||
.fixed_data | ||
.analyzed | ||
.public_declarations_in_source_order() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is fragile, because this would include publics of other machines. I opened #2600 as a suggested fix, it also contains the other suggestions to this file.
// Clone the public part | ||
let public = old_arc.1.clone(); | ||
// Replace with an empty witness and the preserved public | ||
self.artifact.witness_and_publics = Some(Arc::new((Vec::new(), public))); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why do we keep the publics?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this causes a few other changes in the file. In my mind, scalar publics could almost be considered part of the witness (you solve for them, you need them to evaluate constraints, ...), so I don't see why they are treated differently.
// Compute the witness once for all tests that follow. | ||
// we will have to include a backend type here or compute_witness will panic | ||
pipeline.compute_witness().unwrap(); | ||
|
||
test_mock_backend(pipeline.clone()); | ||
|
||
// verify with the mock prover | ||
if T::known_field().unwrap() == KnownField::GoldilocksField { | ||
let pipeline_gl: Pipeline<GoldilocksField> = | ||
unsafe { std::mem::transmute(pipeline.clone()) }; | ||
test_mock_backend(pipeline_gl); | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this change related to publics?
Depends on #2567.
Did a blanket search of public declarations and added public references in all such cases (except one halo2 and one Stwo example, which will require more backend refactoring). This will be required once public reference values instead of witness is solely used for proving.
To make all such updated public examples work, also make the following changes:
set_witness
-->set_witness_and_publics
).