-
Notifications
You must be signed in to change notification settings - Fork 301
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: ensure that generated ACIR is solvable (noir-lang/noir#6415)
fix: fix Alias and Error kinds (noir-lang/noir#6426) fix: type-check turbofish in trait before function call (noir-lang/noir#6416) chore: create a regression test for #6420 (noir-lang/noir#6421) chore: Release Noir(0.37.0) (noir-lang/noir#6321) chore: update variable names in stdlib tests to be more correct (noir-lang/noir#6419) feat!: remove mimc from stdlib (noir-lang/noir#6402) chore: Replace dead link with updated route in README (noir-lang/noir#6392) feat(ci): Add report of Brillig opcodes executed (noir-lang/noir#6396) feat: improve malformed test attribute error (noir-lang/noir#6414) fix: aliases in path (noir-lang/noir#6399) feat(profiler): Add Brillig procedure info to debug artifact for more informative profiling (noir-lang/noir#6385) chore(lsp): Remove profile code lens (noir-lang/noir#6411) chore(nargo): Remove old profile as part of info cmd (noir-lang/noir#6406) fix: distinguish TypePath with and without turbofish (noir-lang/noir#6404) fix: numeric generic doesn't have a default type (noir-lang/noir#6405) feat: Sync from aztec-packages (noir-lang/noir#6403) chore: add regression tests for #4372 (noir-lang/noir#6401) chore: add regression tests for #6314 (noir-lang/noir#6381)
- Loading branch information
Showing
12 changed files
with
363 additions
and
46 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1 @@ | ||
f8fd813b09ce870364700659e3ea8499ab51105e | ||
b473d99b2b70b595596b8392617256dbaf5d5642 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
305 changes: 305 additions & 0 deletions
305
noir/noir-repo/acvm-repo/acvm/src/compiler/simulator.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,305 @@ | ||
use acir::{ | ||
circuit::{ | ||
brillig::{BrilligInputs, BrilligOutputs}, | ||
directives::Directive, | ||
opcodes::{BlockId, FunctionInput}, | ||
Circuit, Opcode, | ||
}, | ||
native_types::{Expression, Witness}, | ||
AcirField, | ||
}; | ||
use std::collections::{BTreeSet, HashMap, HashSet}; | ||
|
||
#[derive(PartialEq)] | ||
enum BlockStatus { | ||
Initialized, | ||
Used, | ||
} | ||
|
||
/// Simulate a symbolic solve for a circuit | ||
#[derive(Default)] | ||
pub struct CircuitSimulator { | ||
/// Track the witnesses that can be solved | ||
solvable_witness: HashSet<Witness>, | ||
|
||
/// Tells whether a Memory Block is: | ||
/// - Not initialized if not in the map | ||
/// - Initialized if its status is Initialized in the Map | ||
/// - Used, indicating that the block cannot be written anymore. | ||
resolved_blocks: HashMap<BlockId, BlockStatus>, | ||
} | ||
|
||
impl CircuitSimulator { | ||
/// Simulate a symbolic solve for a circuit by keeping track of the witnesses that can be solved. | ||
/// Returns false if the circuit cannot be solved | ||
#[tracing::instrument(level = "trace", skip_all)] | ||
pub fn check_circuit<F: AcirField>(&mut self, circuit: &Circuit<F>) -> bool { | ||
let circuit_inputs = circuit.circuit_arguments(); | ||
self.solvable_witness.extend(circuit_inputs.iter()); | ||
for op in &circuit.opcodes { | ||
if !self.try_solve(op) { | ||
return false; | ||
} | ||
} | ||
true | ||
} | ||
|
||
/// Check if the Opcode can be solved, and if yes, add the solved witness to set of solvable witness | ||
fn try_solve<F: AcirField>(&mut self, opcode: &Opcode<F>) -> bool { | ||
let mut unresolved = HashSet::new(); | ||
match opcode { | ||
Opcode::AssertZero(expr) => { | ||
for (_, w1, w2) in &expr.mul_terms { | ||
if !self.solvable_witness.contains(w1) { | ||
if !self.solvable_witness.contains(w2) { | ||
return false; | ||
} | ||
unresolved.insert(*w1); | ||
} | ||
if !self.solvable_witness.contains(w2) && w1 != w2 { | ||
unresolved.insert(*w2); | ||
} | ||
} | ||
for (_, w) in &expr.linear_combinations { | ||
if !self.solvable_witness.contains(w) { | ||
unresolved.insert(*w); | ||
} | ||
} | ||
if unresolved.len() == 1 { | ||
self.mark_solvable(*unresolved.iter().next().unwrap()); | ||
return true; | ||
} | ||
unresolved.is_empty() | ||
} | ||
Opcode::BlackBoxFuncCall(black_box_func_call) => { | ||
let inputs = black_box_func_call.get_inputs_vec(); | ||
for input in inputs { | ||
if !self.can_solve_function_input(&input) { | ||
return false; | ||
} | ||
} | ||
let outputs = black_box_func_call.get_outputs_vec(); | ||
for output in outputs { | ||
self.mark_solvable(output); | ||
} | ||
true | ||
} | ||
Opcode::Directive(directive) => match directive { | ||
Directive::ToLeRadix { a, b, .. } => { | ||
if !self.can_solve_expression(a) { | ||
return false; | ||
} | ||
for w in b { | ||
self.mark_solvable(*w); | ||
} | ||
true | ||
} | ||
}, | ||
Opcode::MemoryOp { block_id, op, predicate } => { | ||
if !self.can_solve_expression(&op.index) { | ||
return false; | ||
} | ||
if let Some(predicate) = predicate { | ||
if !self.can_solve_expression(predicate) { | ||
return false; | ||
} | ||
} | ||
if op.operation.is_zero() { | ||
let w = op.value.to_witness().unwrap(); | ||
self.mark_solvable(w); | ||
true | ||
} else { | ||
if let Some(BlockStatus::Used) = self.resolved_blocks.get(block_id) { | ||
// Writing after having used the block should not be allowed | ||
return false; | ||
} | ||
self.try_solve(&Opcode::AssertZero(op.value.clone())) | ||
} | ||
} | ||
Opcode::MemoryInit { block_id, init, .. } => { | ||
for w in init { | ||
if !self.solvable_witness.contains(w) { | ||
return false; | ||
} | ||
} | ||
self.resolved_blocks.insert(*block_id, BlockStatus::Initialized); | ||
true | ||
} | ||
Opcode::BrilligCall { id: _, inputs, outputs, predicate } => { | ||
for input in inputs { | ||
if !self.can_solve_brillig_input(input) { | ||
return false; | ||
} | ||
} | ||
if let Some(predicate) = predicate { | ||
if !self.can_solve_expression(predicate) { | ||
return false; | ||
} | ||
} | ||
for output in outputs { | ||
match output { | ||
BrilligOutputs::Simple(w) => self.mark_solvable(*w), | ||
BrilligOutputs::Array(arr) => { | ||
for w in arr { | ||
self.mark_solvable(*w); | ||
} | ||
} | ||
} | ||
} | ||
true | ||
} | ||
Opcode::Call { id: _, inputs, outputs, predicate } => { | ||
for w in inputs { | ||
if !self.solvable_witness.contains(w) { | ||
return false; | ||
} | ||
} | ||
if let Some(predicate) = predicate { | ||
if !self.can_solve_expression(predicate) { | ||
return false; | ||
} | ||
} | ||
for w in outputs { | ||
self.mark_solvable(*w); | ||
} | ||
true | ||
} | ||
} | ||
} | ||
|
||
/// Adds the witness to set of solvable witness | ||
pub(crate) fn mark_solvable(&mut self, witness: Witness) { | ||
self.solvable_witness.insert(witness); | ||
} | ||
|
||
pub fn can_solve_function_input<F: AcirField>(&self, input: &FunctionInput<F>) -> bool { | ||
if !input.is_constant() { | ||
return self.solvable_witness.contains(&input.to_witness()); | ||
} | ||
true | ||
} | ||
fn can_solve_expression<F>(&self, expr: &Expression<F>) -> bool { | ||
for w in Self::expr_wit(expr) { | ||
if !self.solvable_witness.contains(&w) { | ||
return false; | ||
} | ||
} | ||
true | ||
} | ||
fn can_solve_brillig_input<F>(&mut self, input: &BrilligInputs<F>) -> bool { | ||
match input { | ||
BrilligInputs::Single(expr) => self.can_solve_expression(expr), | ||
BrilligInputs::Array(exprs) => { | ||
for expr in exprs { | ||
if !self.can_solve_expression(expr) { | ||
return false; | ||
} | ||
} | ||
true | ||
} | ||
|
||
BrilligInputs::MemoryArray(block_id) => match self.resolved_blocks.entry(*block_id) { | ||
std::collections::hash_map::Entry::Vacant(_) => false, | ||
std::collections::hash_map::Entry::Occupied(entry) | ||
if *entry.get() == BlockStatus::Used => | ||
{ | ||
true | ||
} | ||
std::collections::hash_map::Entry::Occupied(mut entry) => { | ||
entry.insert(BlockStatus::Used); | ||
true | ||
} | ||
}, | ||
} | ||
} | ||
|
||
pub(crate) fn expr_wit<F>(expr: &Expression<F>) -> BTreeSet<Witness> { | ||
let mut result = BTreeSet::new(); | ||
result.extend(expr.mul_terms.iter().flat_map(|i| vec![i.1, i.2])); | ||
result.extend(expr.linear_combinations.iter().map(|i| i.1)); | ||
result | ||
} | ||
} | ||
|
||
#[cfg(test)] | ||
mod tests { | ||
use std::collections::BTreeSet; | ||
|
||
use crate::compiler::CircuitSimulator; | ||
use acir::{ | ||
acir_field::AcirField, | ||
circuit::{Circuit, ExpressionWidth, Opcode, PublicInputs}, | ||
native_types::{Expression, Witness}, | ||
FieldElement, | ||
}; | ||
|
||
fn test_circuit( | ||
opcodes: Vec<Opcode<FieldElement>>, | ||
private_parameters: BTreeSet<Witness>, | ||
public_parameters: PublicInputs, | ||
) -> Circuit<FieldElement> { | ||
Circuit { | ||
current_witness_index: 1, | ||
expression_width: ExpressionWidth::Bounded { width: 4 }, | ||
opcodes, | ||
private_parameters, | ||
public_parameters, | ||
return_values: PublicInputs::default(), | ||
assert_messages: Default::default(), | ||
recursive: false, | ||
} | ||
} | ||
|
||
#[test] | ||
fn reports_true_for_empty_circuit() { | ||
let empty_circuit = test_circuit(vec![], BTreeSet::default(), PublicInputs::default()); | ||
|
||
assert!(CircuitSimulator::default().check_circuit(&empty_circuit)); | ||
} | ||
|
||
#[test] | ||
fn reports_true_for_connected_circuit() { | ||
let connected_circuit = test_circuit( | ||
vec![Opcode::AssertZero(Expression { | ||
mul_terms: Vec::new(), | ||
linear_combinations: vec![ | ||
(FieldElement::one(), Witness(1)), | ||
(-FieldElement::one(), Witness(2)), | ||
], | ||
q_c: FieldElement::zero(), | ||
})], | ||
BTreeSet::from([Witness(1)]), | ||
PublicInputs::default(), | ||
); | ||
|
||
assert!(CircuitSimulator::default().check_circuit(&connected_circuit)); | ||
} | ||
|
||
#[test] | ||
fn reports_false_for_disconnected_circuit() { | ||
let disconnected_circuit = test_circuit( | ||
vec![ | ||
Opcode::AssertZero(Expression { | ||
mul_terms: Vec::new(), | ||
linear_combinations: vec![ | ||
(FieldElement::one(), Witness(1)), | ||
(-FieldElement::one(), Witness(2)), | ||
], | ||
q_c: FieldElement::zero(), | ||
}), | ||
Opcode::AssertZero(Expression { | ||
mul_terms: Vec::new(), | ||
linear_combinations: vec![ | ||
(FieldElement::one(), Witness(3)), | ||
(-FieldElement::one(), Witness(4)), | ||
], | ||
q_c: FieldElement::zero(), | ||
}), | ||
], | ||
BTreeSet::from([Witness(1)]), | ||
PublicInputs::default(), | ||
); | ||
|
||
assert!(!CircuitSimulator::default().check_circuit(&disconnected_circuit)); | ||
} | ||
} |
Oops, something went wrong.