From a2f3495fac396e2f4e7c0d51826c790aa55b541e Mon Sep 17 00:00:00 2001 From: Tom French Date: Tue, 16 Jan 2024 23:30:09 +0000 Subject: [PATCH 1/2] feat: move `Instruction::Constrain` as close to where it can be processed as possible. --- compiler/noirc_evaluator/src/ssa.rs | 3 +- .../src/ssa/opt/bubble_up_constrains.rs | 43 +++++++++++++++++++ compiler/noirc_evaluator/src/ssa/opt/mod.rs | 1 + 3 files changed, 46 insertions(+), 1 deletion(-) create mode 100644 compiler/noirc_evaluator/src/ssa/opt/bubble_up_constrains.rs diff --git a/compiler/noirc_evaluator/src/ssa.rs b/compiler/noirc_evaluator/src/ssa.rs index deffe84baea..f87471a851d 100644 --- a/compiler/noirc_evaluator/src/ssa.rs +++ b/compiler/noirc_evaluator/src/ssa.rs @@ -62,7 +62,8 @@ pub(crate) fn optimize_into_acir( // Run mem2reg once more with the flattened CFG to catch any remaining loads/stores .run_pass(Ssa::mem2reg, "After Mem2Reg:") .run_pass(Ssa::fold_constants, "After Constant Folding:") - .run_pass(Ssa::dead_instruction_elimination, "After Dead Instruction Elimination:"); + .run_pass(Ssa::dead_instruction_elimination, "After Dead Instruction Elimination:") + .run_pass(Ssa::bubble_up_constrains, "After Constraint Bubbling:"); let brillig = ssa_builder.to_brillig(print_brillig_trace); diff --git a/compiler/noirc_evaluator/src/ssa/opt/bubble_up_constrains.rs b/compiler/noirc_evaluator/src/ssa/opt/bubble_up_constrains.rs new file mode 100644 index 00000000000..e0144005a0f --- /dev/null +++ b/compiler/noirc_evaluator/src/ssa/opt/bubble_up_constrains.rs @@ -0,0 +1,43 @@ +use crate::ssa::{ir::instruction::Instruction, ssa_gen::Ssa}; + +impl Ssa { + /// A simple SSA pass to go through each instruction and move every `Instruction::Constrain` to immediately + /// after when all of its inputs are available. + #[tracing::instrument(level = "trace", skip(self))] + pub(crate) fn bubble_up_constrains(mut self) -> Ssa { + for function in self.functions.values_mut() { + for block in function.reachable_blocks() { + let instructions = function.dfg[block].take_instructions(); + let mut filtered_instructions = Vec::with_capacity(instructions.len()); + + let dfg = &function.dfg; + for instruction in instructions { + let (lhs, rhs) = match dfg[instruction] { + Instruction::Constrain(lhs, rhs, ..) => (lhs, rhs), + _ => { + filtered_instructions.push(instruction); + continue; + } + }; + + let index = filtered_instructions + .iter() + .rev() + .position(|instruction_id| { + let results = dfg.instruction_results(*instruction_id).to_vec(); + results.contains(&lhs) || results.contains(&rhs) + }) + // We iterate through the previous instructions in reverse order so the index is from the + // back of the vector. Subtract from vector length to get correct index. + .map(|reversed_index| filtered_instructions.len() - reversed_index) + .unwrap_or(filtered_instructions.len()); + + filtered_instructions.insert(index, instruction); + } + + *function.dfg[block].instructions_mut() = filtered_instructions; + } + } + self + } +} diff --git a/compiler/noirc_evaluator/src/ssa/opt/mod.rs b/compiler/noirc_evaluator/src/ssa/opt/mod.rs index 95784194d28..7f03edceff2 100644 --- a/compiler/noirc_evaluator/src/ssa/opt/mod.rs +++ b/compiler/noirc_evaluator/src/ssa/opt/mod.rs @@ -5,6 +5,7 @@ //! Generally, these passes are also expected to minimize the final amount of instructions. mod array_use; mod assert_constant; +mod bubble_up_constrains; mod constant_folding; mod defunctionalize; mod die; From 4e0041b03ff94470286dc7a010aa64fe877f102d Mon Sep 17 00:00:00 2001 From: Tom French Date: Wed, 17 Jan 2024 18:09:48 +0000 Subject: [PATCH 2/2] chore: fix build --- compiler/noirc_evaluator/src/ssa.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/compiler/noirc_evaluator/src/ssa.rs b/compiler/noirc_evaluator/src/ssa.rs index 52cba2c5e1a..e2da5652faf 100644 --- a/compiler/noirc_evaluator/src/ssa.rs +++ b/compiler/noirc_evaluator/src/ssa.rs @@ -63,7 +63,7 @@ pub(crate) fn optimize_into_acir( .run_pass(Ssa::mem2reg, "After Mem2Reg:") .run_pass(Ssa::fold_constants, "After Constant Folding:") .run_pass(Ssa::dead_instruction_elimination, "After Dead Instruction Elimination:") - .run_pass(Ssa::bubble_up_constrains, "After Constraint Bubbling:"); + .run_pass(Ssa::bubble_up_constrains, "After Constraint Bubbling:") .finish(); let brillig = ssa.to_brillig(print_brillig_trace);