use crate::CompilerState;
use leo_ast::{
ArrayExpression,
Expression,
Identifier,
IntegerType,
Literal,
NodeBuilder,
NodeID,
StructExpression,
StructVariableInitializer,
TupleExpression,
};
use leo_errors::StaticAnalyzerError;
use leo_interpreter::Value;
use leo_span::{Span, Symbol};
pub struct ConstPropagationVisitor<'a> {
pub state: &'a mut CompilerState,
pub program: Symbol,
pub changed: bool,
pub const_not_evaluated: Option<Span>,
pub array_index_not_evaluated: Option<Span>,
}
impl ConstPropagationVisitor<'_> {
pub fn in_scope<T>(&mut self, id: NodeID, func: impl FnOnce(&mut Self) -> T) -> T {
self.state.symbol_table.enter_scope(Some(id));
let result = func(self);
self.state.symbol_table.enter_parent();
result
}
pub fn emit_err(&self, err: StaticAnalyzerError) {
self.state.handler.emit_err(err);
}
}
pub fn value_to_expression(value: &Value, span: Span, node_builder: &NodeBuilder) -> Option<Expression> {
use Value::*;
let id = node_builder.next_id();
let result = match value {
Unit => leo_ast::UnitExpression { span, id }.into(),
Bool(x) => Literal::boolean(*x, span, id).into(),
U8(x) => Literal::integer(IntegerType::U8, format!("{x}"), span, id).into(),
U16(x) => Literal::integer(IntegerType::U16, format!("{x}"), span, id).into(),
U32(x) => Literal::integer(IntegerType::U32, format!("{x}"), span, id).into(),
U64(x) => Literal::integer(IntegerType::U64, format!("{x}"), span, id).into(),
U128(x) => Literal::integer(IntegerType::U128, format!("{x}"), span, id).into(),
I8(x) => Literal::integer(IntegerType::I8, format!("{x}"), span, id).into(),
I16(x) => Literal::integer(IntegerType::I16, format!("{x}"), span, id).into(),
I32(x) => Literal::integer(IntegerType::I32, format!("{x}"), span, id).into(),
I64(x) => Literal::integer(IntegerType::I64, format!("{x}"), span, id).into(),
I128(x) => Literal::integer(IntegerType::I128, format!("{x}"), span, id).into(),
Address(x) => Literal::address(format!("{x}"), span, id).into(),
Group(x) => {
let mut s = format!("{x}");
s.truncate(s.len() - 5);
Literal::group(s, span, id).into()
}
Field(x) => {
let mut s = format!("{x}");
s.truncate(s.len() - 5);
Literal::field(s, span, id).into()
}
Scalar(x) => {
let mut s = format!("{x}");
s.truncate(s.len() - 6);
Literal::scalar(s, span, id).into()
}
Tuple(x) => {
let mut elements = Vec::with_capacity(x.len());
for value in x.iter() {
elements.push(value_to_expression(value, span, node_builder)?);
}
TupleExpression { elements, span, id }.into()
}
Array(x) => {
let mut elements = Vec::with_capacity(x.len());
for value in x.iter() {
elements.push(value_to_expression(value, span, node_builder)?);
}
ArrayExpression { elements, span, id }.into()
}
Struct(x) => StructExpression {
name: Identifier { name: x.name, id: node_builder.next_id(), span },
members: {
let mut members = Vec::with_capacity(x.contents.len());
for (name, val) in x.contents.iter() {
let initializer = StructVariableInitializer {
identifier: Identifier { name: *name, id: node_builder.next_id(), span },
expression: Some(value_to_expression(val, span, node_builder)?),
span,
id: node_builder.next_id(),
};
members.push(initializer)
}
members
},
span,
id,
}
.into(),
Future(..) => return None,
};
Some(result)
}