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

chore(ssa refactor): Add loop unrolling pass #1364

Merged
merged 33 commits into from
May 24, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
f719307
Start inlining pass
jfecher May 3, 2023
658b764
Get most of pass working
jfecher May 3, 2023
09a50f9
More progress on inlining
jfecher May 3, 2023
a103b1e
Finish function inlining pass
jfecher May 4, 2023
5252d7e
Merge branch 'master' into jf/ssa-inlining
jfecher May 4, 2023
70608fb
Add basic test
jfecher May 4, 2023
4ec74d6
Address PR comments
jfecher May 5, 2023
59d9459
Start block inlining
jfecher May 8, 2023
d1d3c19
Merge branch 'master' into jf/ssa-blocks
jfecher May 9, 2023
4f42673
Add basic instruction simplification
jfecher May 9, 2023
19f8c1b
Cargo fmt
jfecher May 9, 2023
899e4cf
Add comments
jfecher May 9, 2023
1ffdb0f
Add context object
jfecher May 9, 2023
99284d6
Add push_instruction
jfecher May 10, 2023
7991ba9
Merge branch 'jf/ssa-simplify' into jf/ssa-blocks
jfecher May 10, 2023
86ead4d
Fix bug in inlining pass
jfecher May 11, 2023
e4720c5
Reorder loop unrolling pass
jfecher May 11, 2023
57f0f03
Get it working for most loops. Still missing loops with if inside
jfecher May 11, 2023
8b8ee82
Merge branch 'master' into jf/ssa-blocks
jfecher May 12, 2023
b94bfff
Rework entire pass from scratch
jfecher May 16, 2023
685c9d6
Finish loop unrolling
jfecher May 17, 2023
b242fac
Add doc comment
jfecher May 17, 2023
cb516ee
Merge branch 'master' into jf/ssa-blocks
jfecher May 17, 2023
a53c643
Fix bad merge
jfecher May 17, 2023
4e0a163
Add test
jfecher May 17, 2023
5faa69a
Remove outdated parts of PR
jfecher May 17, 2023
c270aae
Correctly handle loops with non-const indices
jfecher May 18, 2023
c2eaee1
Merge branch 'master' into jf/ssa-blocks
jfecher May 18, 2023
1f47a3c
Address PR comments
jfecher May 22, 2023
bf4fa22
Fix inlining bug and add a test for loops which fail to unroll
jfecher May 22, 2023
8871da9
Merge branch 'master' into jf/ssa-blocks
jfecher May 22, 2023
c677282
Update simplify_cfg to use new inline_block method
jfecher May 22, 2023
83c4b5f
Remove now-unneeded test helper function
jfecher May 22, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
25 changes: 20 additions & 5 deletions crates/noirc_evaluator/src/ssa_refactor.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,13 +7,13 @@
//! This module heavily borrows from Cranelift
#![allow(dead_code)]

use crate::errors::RuntimeError;
use crate::errors::{RuntimeError, RuntimeErrorKind};
use acvm::{acir::circuit::Circuit, compiler::transformers::IsOpcodeSupported, Language};
use noirc_abi::Abi;

use noirc_frontend::monomorphization::ast::Program;

use self::acir_gen::Acir;
use self::ssa_gen::Ssa;

mod acir_gen;
mod ir;
Expand All @@ -24,9 +24,15 @@ pub mod ssa_gen;
/// Optimize the given program by converting it into SSA
/// form and performing optimizations there. When finished,
/// convert the final SSA into ACIR and return it.
pub fn optimize_into_acir(program: Program) -> Acir {
ssa_gen::generate_ssa(program).inline_functions().into_acir()
pub fn optimize_into_acir(program: Program) {
ssa_gen::generate_ssa(program)
.print("Initial SSA:")
.inline_functions()
.print("After Inlining:")
.unroll_loops()
.print("After Unrolling:");
kevaundray marked this conversation as resolved.
Show resolved Hide resolved
}

/// Compiles the Program into ACIR and applies optimizations to the arithmetic gates
/// This is analogous to `ssa:create_circuit` and this method is called when one wants
/// to use the new ssa module to process Noir code.
Expand All @@ -37,5 +43,14 @@ pub fn experimental_create_circuit(
_enable_logging: bool,
_show_output: bool,
) -> Result<(Circuit, Abi), RuntimeError> {
todo!("this is a stub function for the new SSA refactor module")
optimize_into_acir(_program);
let error_kind = RuntimeErrorKind::Spanless("Acir-gen is unimplemented".into());
Err(RuntimeError::new(error_kind, None))
}

impl Ssa {
fn print(self, msg: &str) -> Ssa {
println!("{msg}\n{self}");
self
}
}
2 changes: 1 addition & 1 deletion crates/noirc_evaluator/src/ssa_refactor/acir_gen/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use super::ssa_gen::Ssa;
struct Context {}

/// The output of the Acir-gen pass
pub struct Acir {}
pub(crate) struct Acir {}

impl Ssa {
pub(crate) fn into_acir(self) -> Acir {
Expand Down
57 changes: 35 additions & 22 deletions crates/noirc_evaluator/src/ssa_refactor/ir/basic_block.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,10 +29,10 @@ pub(crate) struct BasicBlock {
pub(crate) type BasicBlockId = Id<BasicBlock>;

impl BasicBlock {
/// Create a new BasicBlock with the given instructions.
/// Create a new BasicBlock with the given parameters.
/// Parameters can also be added later via BasicBlock::add_parameter
pub(crate) fn new(instructions: Vec<InstructionId>) -> Self {
Self { parameters: Vec::new(), instructions, terminator: None }
kevaundray marked this conversation as resolved.
Show resolved Hide resolved
pub(crate) fn new() -> Self {
Self { parameters: Vec::new(), instructions: Vec::new(), terminator: None }
}

/// Returns the parameters of this block
Expand All @@ -52,6 +52,12 @@ impl BasicBlock {
self.parameters.push(parameter);
}

/// Replace this block's current parameters with that of the given Vec.
/// This does not perform any checks that any previous parameters were unused.
pub(crate) fn set_parameters(&mut self, parameters: Vec<ValueId>) {
self.parameters = parameters;
}

/// Insert an instruction at the end of this block
pub(crate) fn insert_instruction(&mut self, instruction: InstructionId) {
self.instructions.push(instruction);
Expand Down Expand Up @@ -83,6 +89,32 @@ impl BasicBlock {
self.terminator.as_ref()
}

/// Returns the terminator of this block, panics if there is None.
///
/// Once this block has finished construction, this is expected to always be Some.
pub(crate) fn unwrap_terminator(&self) -> &TerminatorInstruction {
self.terminator().expect("Expected block to have terminator instruction")
}

/// Returns a mutable reference to the terminator of this block.
///
/// Once this block has finished construction, this is expected to always be Some.
pub(crate) fn unwrap_terminator_mut(&mut self) -> &mut TerminatorInstruction {
self.terminator.as_mut().expect("Expected block to have terminator instruction")
}

/// Take ownership of this block's terminator, replacing it with an empty return terminator
/// so that no clone is needed.
///
/// It is expected that this function is used as an optimization on blocks that are no longer
/// reachable or will have their terminator overwritten afterwards. Using this on a reachable
/// block without setting the terminator afterward will result in the empty return terminator
/// being kept, which is likely unwanted.
pub(crate) fn take_terminator(&mut self) -> TerminatorInstruction {
let terminator = self.terminator.as_mut().expect("Expected block to have a terminator");
std::mem::replace(terminator, TerminatorInstruction::Return { return_values: Vec::new() })
}

/// Iterate over all the successors of the currently block, as determined by
/// the blocks jumped to in the terminator instruction. If there is no terminator
/// instruction yet, this will iterate 0 times.
Expand All @@ -107,23 +139,4 @@ impl BasicBlock {
});
self.instructions.remove(index);
}

/// Take ownership of this block's terminator, replacing it with an empty return terminator
/// so that no clone is needed.
///
/// It is expected that this function is used as an optimization on blocks that are no longer
/// reachable or will have their terminator overwritten afterwards. Using this on a reachable
/// block without setting the terminator afterward will result in the empty return terminator
/// being kept, which is likely unwanted.
pub(crate) fn take_terminator(&mut self) -> TerminatorInstruction {
let terminator = self.terminator.as_mut().expect("Expected block to have a terminator");
std::mem::replace(terminator, TerminatorInstruction::Return { return_values: Vec::new() })
}

/// Returns a mutable reference to the terminator of this block.
///
/// Once this block has finished construction, this is expected to always be Some.
pub(crate) fn unwrap_terminator_mut(&mut self) -> &mut TerminatorInstruction {
self.terminator.as_mut().expect("Expected block to have terminator instruction")
}
}
59 changes: 44 additions & 15 deletions crates/noirc_evaluator/src/ssa_refactor/ir/dfg.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
use std::collections::HashMap;

use crate::ssa_refactor::ir::instruction::SimplifyResult;

use super::{
basic_block::{BasicBlock, BasicBlockId},
constant::{NumericConstant, NumericConstantId},
Expand All @@ -13,6 +15,7 @@ use super::{
};

use acvm::FieldElement;
use iter_extended::vecmap;

/// The DataFlowGraph contains most of the actual data in a function including
/// its blocks, instructions, and values. This struct is largely responsible for
Expand Down Expand Up @@ -65,7 +68,27 @@ impl DataFlowGraph {
/// After being created, the block is unreachable in the current function
/// until another block is made to jump to it.
pub(crate) fn make_block(&mut self) -> BasicBlockId {
self.blocks.insert(BasicBlock::new(Vec::new()))
self.blocks.insert(BasicBlock::new())
}

/// Create a new block with the same parameter count and parameter
/// types from the given block.
/// This is a somewhat niche operation used in loop unrolling but is included
/// here as doing it outside the DataFlowGraph would require cloning the parameters.
pub(crate) fn make_block_with_parameters_from_block(
&mut self,
block: BasicBlockId,
) -> BasicBlockId {
let new_block = self.make_block();
let parameters = self.blocks[block].parameters();

let parameters = vecmap(parameters.iter().enumerate(), |(position, param)| {
let typ = self.values[*param].get_type();
self.values.insert(Value::Param { block: new_block, position, typ })
});

self.blocks[new_block].set_parameters(parameters);
new_block
}

/// Get an iterator over references to each basic block within the dfg, paired with the basic
Expand Down Expand Up @@ -101,17 +124,19 @@ impl DataFlowGraph {
}

/// Inserts a new instruction at the end of the given block and returns its results
pub(crate) fn insert_instruction(
pub(crate) fn insert_instruction_and_results(
&mut self,
instruction: Instruction,
block: BasicBlockId,
ctrl_typevars: Option<Vec<Type>>,
) -> InsertInstructionResult {
use InsertInstructionResult::*;
match instruction.simplify(self) {
Some(simplification) => InsertInstructionResult::SimplifiedTo(simplification),
None => {
SimplifyResult::SimplifiedTo(simplification) => SimplifiedTo(simplification),
SimplifyResult::Remove => InstructionRemoved,
SimplifyResult::None => {
let id = self.make_instruction(instruction, ctrl_typevars);
self.insert_instruction_in_block(block, id);
self.blocks[block].insert_instruction(id);
InsertInstructionResult::Results(self.instruction_results(id))
}
}
Expand Down Expand Up @@ -246,16 +271,6 @@ impl DataFlowGraph {
parameter
}

/// Insert an instruction at the end of a given block.
/// If the block already has a terminator, the instruction is inserted before the terminator.
pub(crate) fn insert_instruction_in_block(
&mut self,
block: BasicBlockId,
instruction: InstructionId,
) {
self.blocks[block].insert_instruction(instruction);
}

/// Returns the field element represented by this value if it is a numeric constant.
/// Returns None if the given value is not a numeric constant.
pub(crate) fn get_numeric_constant(&self, value: Id<Value>) -> Option<FieldElement> {
Expand All @@ -282,6 +297,20 @@ impl DataFlowGraph {
) {
self.blocks[block].set_terminator(terminator);
}

/// Moves the entirety of the given block's contents into the destination block.
/// The source block afterward will be left in a valid but emptied state. The
/// destination block will also have its terminator overwritten with that of the
/// source block.
pub(crate) fn inline_block(&mut self, source: BasicBlockId, destination: BasicBlockId) {
let source = &mut self.blocks[source];
let mut instructions = std::mem::take(source.instructions_mut());
let terminator = source.take_terminator();

let destination = &mut self.blocks[destination];
destination.instructions_mut().append(&mut instructions);
destination.set_terminator(terminator);
}
}

impl std::ops::Index<InstructionId> for DataFlowGraph {
Expand Down
Loading