leo_passes/static_single_assignment/
visitor.rs1use crate::{CompilerState, RenameTable};
18
19use leo_ast::{Expression, Identifier, Node, Statement};
20use leo_span::Symbol;
21
22pub struct SsaFormingVisitor<'a> {
23 pub state: &'a mut CompilerState,
24 pub rename_table: RenameTable,
26 pub program: Symbol,
28 pub rename_defs: bool,
30}
31
32impl SsaFormingVisitor<'_> {
33 pub(crate) fn push(&mut self) {
35 let parent_table = core::mem::take(&mut self.rename_table);
36 self.rename_table = RenameTable::new(Some(Box::from(parent_table)));
37 }
38
39 pub(crate) fn pop(&mut self) -> RenameTable {
41 let parent = self.rename_table.parent.clone().unwrap_or_default();
42 core::mem::replace(&mut self.rename_table, *parent)
43 }
44
45 pub(crate) fn rename_identifier(&mut self, mut identifier: Identifier) -> Identifier {
46 self.rename_table.update(identifier.name, identifier.name, identifier.id);
48
49 let new_name = self.state.assigner.unique_symbol(identifier.name, "$#");
50 self.rename_table.update(identifier.name, new_name, identifier.id);
51 identifier.name = new_name;
52 identifier
53 }
54
55 pub(crate) fn simple_definition(&mut self, identifier: Identifier, rhs: Expression) -> Statement {
56 let type_ = match self.state.type_table.get(&rhs.id()) {
58 Some(type_) => type_,
59 None => unreachable!("Type checking guarantees that all expressions have a type."),
60 };
61 self.state.type_table.insert(identifier.id(), type_);
62 self.rename_table.update(identifier.name, identifier.name, identifier.id);
64 self.state.assigner.simple_definition(identifier, rhs, self.state.node_builder.next_id())
66 }
67
68 pub(crate) fn unique_simple_definition(&mut self, expr: Expression) -> (Identifier, Statement) {
72 let name = self.state.assigner.unique_symbol("$var", "$");
74
75 let place = Identifier { name, span: Default::default(), id: self.state.node_builder.next_id() };
77
78 let statement = self.simple_definition(place, expr);
80
81 (place, statement)
82 }
83}