From 808fd963821357bcf9189bf54e2246301d8e1427 Mon Sep 17 00:00:00 2001 From: Kevaundray Wedderburn Date: Thu, 20 Apr 2023 18:47:36 +0100 Subject: [PATCH 1/6] add range optimization --- acvm/src/compiler/optimizers/mod.rs | 2 + acvm/src/compiler/optimizers/range.rs | 19 -- .../compiler/optimizers/redundant_range.rs | 231 ++++++++++++++++++ 3 files changed, 233 insertions(+), 19 deletions(-) delete mode 100644 acvm/src/compiler/optimizers/range.rs create mode 100644 acvm/src/compiler/optimizers/redundant_range.rs diff --git a/acvm/src/compiler/optimizers/mod.rs b/acvm/src/compiler/optimizers/mod.rs index 2b7b95f28..b69364507 100644 --- a/acvm/src/compiler/optimizers/mod.rs +++ b/acvm/src/compiler/optimizers/mod.rs @@ -1,3 +1,5 @@ mod general; +mod redundant_range; pub use general::GeneralOpt as GeneralOptimizer; +pub use redundant_range::RangeOptimizer; diff --git a/acvm/src/compiler/optimizers/range.rs b/acvm/src/compiler/optimizers/range.rs deleted file mode 100644 index 9bfa86d08..000000000 --- a/acvm/src/compiler/optimizers/range.rs +++ /dev/null @@ -1,19 +0,0 @@ -// XXX: We could alleviate a runtime check from noir -// By casting directly -// Example: -// priv z1 = x as u32 -// priv z2 = x as u16 -// -// The IR would see both casts and replace it with -// -// -// priv z1 = x as u16; -// priv z2 = x as u16; -// -// -// Then maybe another optimization could be done so that it transforms into -// -// priv z1 = x as u16 -// priv z2 = z1 -// This is what I would call a general optimization, so it could live inside of the IR module -// A more specific optimization would be to have z2 = z1 not use a gate (copy_from_to), this is more specific to plonk-aztec and would not live in this module diff --git a/acvm/src/compiler/optimizers/redundant_range.rs b/acvm/src/compiler/optimizers/redundant_range.rs new file mode 100644 index 000000000..558522150 --- /dev/null +++ b/acvm/src/compiler/optimizers/redundant_range.rs @@ -0,0 +1,231 @@ +use acir::{ + circuit::{Circuit, Opcode}, + native_types::Witness, + BlackBoxFunc, +}; +use std::collections::{BTreeMap, HashSet}; +/// RangeOptimizer will remove redundant range constraints. +/// +/// Suppose we had the following pseudo-code: +/// +/// let z1 = x as u16; +// let z2 = x as u32; +/// +/// It is clear that if `x` fits inside of a 16-bit integer, +/// it must also fit inside of a 32-bit integer. +/// +/// The generated ACIR may produce two range opcodes however; +/// - One for the 16 bit range constraint of `x` +/// - One for the 32-bit range constraint of `x` +/// +/// This optimization pass will keep the 16-bit range constraint +/// and remove the 32-bit range constraint opcode. +pub struct RangeOptimizer { + /// Maps witnesses to their lowest known bit sizes. + lists: BTreeMap, + circuit: Circuit, +} + +impl RangeOptimizer { + /// Creates a new `RangeOptimizer` by collecting all known range + /// constraints from `Circuit`. + pub fn new(circuit: Circuit) -> Self { + let range_list = Self::collect_ranges(&circuit); + Self { circuit, lists: range_list } + } + + /// Stores the lowest bit range, that a witness + /// has been constrained to be. + /// For example, if we constrain a witness `x` to be + /// both 32 bits and 16 bits. This function will + /// only store the fact that we have constrained it to + /// be 16 bits. + fn collect_ranges(circuit: &Circuit) -> BTreeMap { + let mut witness_to_bit_sizes = BTreeMap::new(); + + for opcode in &circuit.opcodes { + // Extract the witness index and number of bits, + // if it is a range constraint + let (witness, num_bits) = match extract_range_opcode(opcode) { + Some(func_inputs) => func_inputs, + None => continue, + }; + + // Check if the witness has already been recorded and if the witness + // size is more than the current one, we replace it + let should_replace = match witness_to_bit_sizes.get(&witness).cloned() { + Some(old_range_bits) => old_range_bits > num_bits, + None => true, + }; + if should_replace { + witness_to_bit_sizes.insert(witness, num_bits); + } + } + witness_to_bit_sizes + } + + /// Returns a `Circuit` where each Witness is only range constrained + /// once to the lowest number `bit size` possible. + pub fn replace_redundant_ranges(self) -> Circuit { + let mut already_seen_witness = HashSet::new(); + + let mut optimized_opcodes = Vec::with_capacity(self.circuit.opcodes.len()); + + for opcode in self.circuit.opcodes { + let (witness, num_bits) = match extract_range_opcode(&opcode) { + Some(range_opcode) => range_opcode, + None => { + // If its not the range opcode, add it to the opcode + // list and continue; + optimized_opcodes.push(opcode); + continue; + } + }; + + // Check if this is the lowest number of bits + // in the circuit and that we have not added it + // in the list of opcodes yet. + let stored_num_bits = self.lists.get(&witness).expect("Could not find witness. This should never be the case if `collect_ranges` is called"); + let is_lowest_bit_size = num_bits <= *stored_num_bits; + let already_added = already_seen_witness.contains(&witness); + + // If the opcode is associated with the lowest bit size + // and we have not added a duplicate of this opcode yet, + // then we should add retain this range opcode. + if is_lowest_bit_size && !already_added { + already_seen_witness.insert(witness); + optimized_opcodes.push(opcode); + } + } + + Circuit { + current_witness_index: self.circuit.current_witness_index, + opcodes: optimized_opcodes, + public_parameters: self.circuit.public_parameters, + return_values: self.circuit.return_values, + } + } +} + +/// Extract the range opcode from the `Opcode` enum +/// Returns None, if `Opcode` is not the range opcode. +fn extract_range_opcode(opcode: &Opcode) -> Option<(Witness, u32)> { + // Range constraints are blackbox function calls + // so we first extract the function call + let func_call = match opcode { + acir::circuit::Opcode::BlackBoxFuncCall(func_call) => func_call, + _ => return None, + }; + + // Skip if it is not a range constraint + if func_call.name != BlackBoxFunc::RANGE { + return None; + } + + let func_input = + func_call.inputs.first().expect("we expect there to be one input for a range call"); + Some((func_input.witness, func_input.num_bits)) +} + +#[cfg(test)] +mod tests { + use crate::compiler::optimizers::{redundant_range::extract_range_opcode, RangeOptimizer}; + use acir::{ + circuit::{ + opcodes::{BlackBoxFuncCall, FunctionInput}, + Circuit, Opcode, PublicInputs, + }, + native_types::{Expression, Witness}, + BlackBoxFunc, + }; + + fn test_circuit(ranges: Vec<(Witness, u32)>) -> Circuit { + fn test_range_constraint(witness: Witness, num_bits: u32) -> Opcode { + Opcode::BlackBoxFuncCall(BlackBoxFuncCall { + name: BlackBoxFunc::RANGE, + inputs: vec![FunctionInput { witness, num_bits }], + outputs: vec![], + }) + } + + let opcodes: Vec<_> = ranges + .into_iter() + .map(|(witness, num_bits)| test_range_constraint(witness, num_bits)) + .collect(); + + Circuit { + current_witness_index: 1, + opcodes, + public_parameters: PublicInputs::default(), + return_values: PublicInputs::default(), + } + } + + #[test] + fn retain_lowest_range_size() { + // The optimizer should keep the lowest bit size range constraint + let circuit = test_circuit(vec![(Witness(1), 32), (Witness(1), 16)]); + + let optimizer = RangeOptimizer::new(circuit); + + let range_size = *optimizer + .lists + .get(&Witness(1)) + .expect("Witness(1) was inserted, but it is missing from the map"); + assert_eq!( + range_size, 16, + "expected a range size of 16 since that was the lowest bit size provided" + ); + + let optimized_circuit = optimizer.replace_redundant_ranges(); + assert_eq!(optimized_circuit.opcodes.len(), 1); + + let (witness, num_bits) = + extract_range_opcode(&optimized_circuit.opcodes[0]).expect("expected one range opcode"); + + assert_eq!(witness, Witness(1)); + assert_eq!(num_bits, 16); + } + + #[test] + fn remove_duplicates() { + // The optimizer should remove all duplicate range opcodes. + + let circuit = test_circuit(vec![ + (Witness(1), 16), + (Witness(1), 16), + (Witness(2), 23), + (Witness(2), 23), + ]); + + let optimizer = RangeOptimizer::new(circuit); + let optimized_circuit = optimizer.replace_redundant_ranges(); + assert_eq!(optimized_circuit.opcodes.len(), 2); + + let (witness_a, num_bits_a) = + extract_range_opcode(&optimized_circuit.opcodes[0]).expect("expected two range opcode"); + let (witness_b, num_bits_b) = + extract_range_opcode(&optimized_circuit.opcodes[1]).expect("expected two range opcode"); + + assert_eq!(witness_a, Witness(1)); + assert_eq!(witness_b, Witness(2)); + assert_eq!(num_bits_a, 16); + assert_eq!(num_bits_b, 23); + } + + #[test] + fn non_range_opcodes() { + // The optimizer should not remove or change non-range opcodes + // The four Arithmetic opcodes should remain unchanged. + let mut circuit = test_circuit(vec![(Witness(1), 16), (Witness(1), 16)]); + + circuit.opcodes.push(Opcode::Arithmetic(Expression::default())); + circuit.opcodes.push(Opcode::Arithmetic(Expression::default())); + circuit.opcodes.push(Opcode::Arithmetic(Expression::default())); + circuit.opcodes.push(Opcode::Arithmetic(Expression::default())); + + let optimizer = RangeOptimizer::new(circuit); + let optimized_circuit = optimizer.replace_redundant_ranges(); + assert_eq!(optimized_circuit.opcodes.len(), 5) + } +} From 15fe870afac02ba809580274de9260d43f3159bb Mon Sep 17 00:00:00 2001 From: Kevaundray Wedderburn Date: Thu, 20 Apr 2023 19:15:16 +0100 Subject: [PATCH 2/6] add range optimizer to compile method --- acvm/src/compiler.rs | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/acvm/src/compiler.rs b/acvm/src/compiler.rs index 316117169..c62bb7432 100644 --- a/acvm/src/compiler.rs +++ b/acvm/src/compiler.rs @@ -13,6 +13,8 @@ use optimizers::GeneralOptimizer; use thiserror::Error; use transformers::{CSatTransformer, FallbackTransformer, IsOpcodeSupported, R1CSTransformer}; +use self::optimizers::RangeOptimizer; + #[derive(PartialEq, Eq, Debug, Error)] pub enum CompileError { #[error("The blackbox function {0} is not supported by the backend and acvm does not have a fallback implementation")] @@ -43,6 +45,10 @@ pub fn compile( } let acir = Circuit { opcodes, ..acir }; + // Range optimization pass + let range_optimizer = RangeOptimizer::new(acir); + let acir = range_optimizer.replace_redundant_ranges(); + let transformer = match &np_language { crate::Language::R1CS => { let transformer = R1CSTransformer::new(acir); From 40d3f9767a7281c38b88de61e861d46c828594a1 Mon Sep 17 00:00:00 2001 From: kevaundray Date: Thu, 20 Apr 2023 20:22:43 +0100 Subject: [PATCH 3/6] Update acvm/src/compiler/optimizers/redundant_range.rs Co-authored-by: Tom French <15848336+TomAFrench@users.noreply.github.com> --- acvm/src/compiler/optimizers/redundant_range.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/acvm/src/compiler/optimizers/redundant_range.rs b/acvm/src/compiler/optimizers/redundant_range.rs index 558522150..c0c21ca55 100644 --- a/acvm/src/compiler/optimizers/redundant_range.rs +++ b/acvm/src/compiler/optimizers/redundant_range.rs @@ -4,6 +4,7 @@ use acir::{ BlackBoxFunc, }; use std::collections::{BTreeMap, HashSet}; + /// RangeOptimizer will remove redundant range constraints. /// /// Suppose we had the following pseudo-code: From 772f5c30e459f1d6f1e4b9b3d70c63e6d236719e Mon Sep 17 00:00:00 2001 From: kevaundray Date: Thu, 20 Apr 2023 20:23:51 +0100 Subject: [PATCH 4/6] Update acvm/src/compiler/optimizers/redundant_range.rs Co-authored-by: Tom French <15848336+TomAFrench@users.noreply.github.com> --- acvm/src/compiler/optimizers/redundant_range.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/acvm/src/compiler/optimizers/redundant_range.rs b/acvm/src/compiler/optimizers/redundant_range.rs index c0c21ca55..8893e1951 100644 --- a/acvm/src/compiler/optimizers/redundant_range.rs +++ b/acvm/src/compiler/optimizers/redundant_range.rs @@ -54,7 +54,7 @@ impl RangeOptimizer { // Check if the witness has already been recorded and if the witness // size is more than the current one, we replace it - let should_replace = match witness_to_bit_sizes.get(&witness).cloned() { + let should_replace = match witness_to_bit_sizes.get(&witness).copied() { Some(old_range_bits) => old_range_bits > num_bits, None => true, }; From a5892f3de08014af13ebd0d555e7bf2745f16f14 Mon Sep 17 00:00:00 2001 From: kevaundray Date: Thu, 20 Apr 2023 20:25:09 +0100 Subject: [PATCH 5/6] Update acvm/src/compiler/optimizers/redundant_range.rs Co-authored-by: Tom French <15848336+TomAFrench@users.noreply.github.com> --- acvm/src/compiler/optimizers/redundant_range.rs | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/acvm/src/compiler/optimizers/redundant_range.rs b/acvm/src/compiler/optimizers/redundant_range.rs index 8893e1951..15608a6ab 100644 --- a/acvm/src/compiler/optimizers/redundant_range.rs +++ b/acvm/src/compiler/optimizers/redundant_range.rs @@ -82,18 +82,20 @@ impl RangeOptimizer { continue; } }; + // If we've already applied the range constraint for this witness then skip this opcode. + let already_added = already_seen_witness.contains(&witness); + if already_added { + continue; + } - // Check if this is the lowest number of bits - // in the circuit and that we have not added it - // in the list of opcodes yet. + // Check if this is the lowest number of bits in the circuit let stored_num_bits = self.lists.get(&witness).expect("Could not find witness. This should never be the case if `collect_ranges` is called"); let is_lowest_bit_size = num_bits <= *stored_num_bits; - let already_added = already_seen_witness.contains(&witness); // If the opcode is associated with the lowest bit size // and we have not added a duplicate of this opcode yet, // then we should add retain this range opcode. - if is_lowest_bit_size && !already_added { + if is_lowest_bit_size { already_seen_witness.insert(witness); optimized_opcodes.push(opcode); } From a1b06dbfef4751789c7d74067bb0c368968aa895 Mon Sep 17 00:00:00 2001 From: Kevaundray Wedderburn Date: Thu, 20 Apr 2023 20:59:16 +0100 Subject: [PATCH 6/6] make range optimizer crate visible --- acvm/src/compiler/optimizers/mod.rs | 2 +- acvm/src/compiler/optimizers/redundant_range.rs | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/acvm/src/compiler/optimizers/mod.rs b/acvm/src/compiler/optimizers/mod.rs index b69364507..be4dae5cb 100644 --- a/acvm/src/compiler/optimizers/mod.rs +++ b/acvm/src/compiler/optimizers/mod.rs @@ -2,4 +2,4 @@ mod general; mod redundant_range; pub use general::GeneralOpt as GeneralOptimizer; -pub use redundant_range::RangeOptimizer; +pub(crate) use redundant_range::RangeOptimizer; diff --git a/acvm/src/compiler/optimizers/redundant_range.rs b/acvm/src/compiler/optimizers/redundant_range.rs index 15608a6ab..0dd77d96f 100644 --- a/acvm/src/compiler/optimizers/redundant_range.rs +++ b/acvm/src/compiler/optimizers/redundant_range.rs @@ -21,7 +21,7 @@ use std::collections::{BTreeMap, HashSet}; /// /// This optimization pass will keep the 16-bit range constraint /// and remove the 32-bit range constraint opcode. -pub struct RangeOptimizer { +pub(crate) struct RangeOptimizer { /// Maps witnesses to their lowest known bit sizes. lists: BTreeMap, circuit: Circuit, @@ -30,7 +30,7 @@ pub struct RangeOptimizer { impl RangeOptimizer { /// Creates a new `RangeOptimizer` by collecting all known range /// constraints from `Circuit`. - pub fn new(circuit: Circuit) -> Self { + pub(crate) fn new(circuit: Circuit) -> Self { let range_list = Self::collect_ranges(&circuit); Self { circuit, lists: range_list } } @@ -67,7 +67,7 @@ impl RangeOptimizer { /// Returns a `Circuit` where each Witness is only range constrained /// once to the lowest number `bit size` possible. - pub fn replace_redundant_ranges(self) -> Circuit { + pub(crate) fn replace_redundant_ranges(self) -> Circuit { let mut already_seen_witness = HashSet::new(); let mut optimized_opcodes = Vec::with_capacity(self.circuit.opcodes.len()); @@ -132,7 +132,7 @@ fn extract_range_opcode(opcode: &Opcode) -> Option<(Witness, u32)> { #[cfg(test)] mod tests { - use crate::compiler::optimizers::{redundant_range::extract_range_opcode, RangeOptimizer}; + use crate::compiler::optimizers::redundant_range::{extract_range_opcode, RangeOptimizer}; use acir::{ circuit::{ opcodes::{BlackBoxFuncCall, FunctionInput},