use crate::{CompilerState, RenameTable};
use leo_ast::{Expression, Identifier, Node, Statement};
use leo_span::Symbol;
pub struct SsaFormingVisitor<'a> {
pub state: &'a mut CompilerState,
pub rename_table: RenameTable,
pub program: Symbol,
pub rename_defs: bool,
}
impl SsaFormingVisitor<'_> {
pub(crate) fn push(&mut self) {
let parent_table = core::mem::take(&mut self.rename_table);
self.rename_table = RenameTable::new(Some(Box::from(parent_table)));
}
pub(crate) fn pop(&mut self) -> RenameTable {
let parent = self.rename_table.parent.clone().unwrap_or_default();
core::mem::replace(&mut self.rename_table, *parent)
}
pub(crate) fn rename_identifier(&mut self, mut identifier: Identifier) -> Identifier {
self.rename_table.update(identifier.name, identifier.name, identifier.id);
let new_name = self.state.assigner.unique_symbol(identifier.name, "$#");
self.rename_table.update(identifier.name, new_name, identifier.id);
identifier.name = new_name;
identifier
}
pub(crate) fn simple_definition(&mut self, identifier: Identifier, rhs: Expression) -> Statement {
let type_ = match self.state.type_table.get(&rhs.id()) {
Some(type_) => type_,
None => unreachable!("Type checking guarantees that all expressions have a type."),
};
self.state.type_table.insert(identifier.id(), type_);
self.rename_table.update(identifier.name, identifier.name, identifier.id);
self.state.assigner.simple_definition(identifier, rhs, self.state.node_builder.next_id())
}
pub(crate) fn unique_simple_definition(&mut self, expr: Expression) -> (Identifier, Statement) {
let name = self.state.assigner.unique_symbol("$var", "$");
let place = Identifier { name, span: Default::default(), id: self.state.node_builder.next_id() };
let statement = self.simple_definition(place, expr);
(place, statement)
}
}