Skip to content
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

Open
wants to merge 45 commits into
base: main
Choose a base branch
from
Open

Update public examples #2572

wants to merge 45 commits into from

Conversation

qwang98
Copy link
Collaborator

@qwang98 qwang98 commented Mar 21, 2025

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:

  1. Updated mock backend to use public reference values (this is technically the first backend that's edited to use public reference witgen entirely).
  2. Updated pipeline functions for soundness tests (set_witness --> set_witness_and_publics).

@qwang98 qwang98 changed the base branch from main to stage-2-public-reference March 21, 2025 05:09
@Schaeff
Copy link
Collaborator

Schaeff commented Mar 21, 2025

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?

@qwang98
Copy link
Collaborator Author

qwang98 commented Mar 25, 2025

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.

Copy link

@github-actions github-actions bot left a 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.

Comment on lines 296 to 329
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()
}
}
Copy link
Collaborator Author

@qwang98 qwang98 Mar 26, 2025

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.

Copy link
Collaborator

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?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Comment on lines -250 to +254
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);

Copy link
Collaborator Author

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.

Comment on lines 53 to 56
if !parts.identities.is_empty() {
return None;
}

Copy link
Collaborator Author

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.

Copy link
Collaborator

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.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

True :)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

@qwang98 qwang98 requested a review from georgwiese March 26, 2025 08:53
@qwang98 qwang98 marked this pull request as ready for review March 26, 2025 08:56
Base automatically changed from stage-2-public-reference to main March 26, 2025 11:51
Copy link
Collaborator

@georgwiese georgwiese left a 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()))
Copy link
Collaborator

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?

Copy link
Collaborator Author

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:

  1. 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).
  1. 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
  2. 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;
Copy link
Collaborator

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?

Copy link
Collaborator Author

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.

Copy link
Collaborator

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!

Comment on lines 53 to 56
if !parts.identities.is_empty() {
return None;
}

Copy link
Collaborator

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.

Comment on lines 296 to 329
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()
}
}
Copy link
Collaborator

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?

Comment on lines +295 to +298
pub fn extract_publics<T>(
public_references: BTreeMap<String, T>,
pil: &Analyzed<T>,
) -> BTreeMap<String, Option<T>>
Copy link
Collaborator

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?

Copy link
Collaborator Author

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;
Copy link
Collaborator

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!

Comment on lines +55 to +61
// 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(_)))
})
}) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// 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
Copy link
Collaborator

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()
Copy link
Collaborator

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.

Comment on lines +715 to +718
// 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)));
Copy link
Collaborator

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?

Copy link
Collaborator

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.

Comment on lines -45 to -57
// 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);
}

Copy link
Collaborator

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?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants