use super::{EntryPoint, ExitPoint};
use crate::{
language::{
parsed::TreeType,
ty::{
self, TyConfigurableDecl, TyConstGenericDecl, TyConstantDecl, TyFunctionDecl,
TyFunctionSig,
},
CallPath,
},
type_system::TypeInfo,
Ident,
};
use indexmap::IndexMap;
use petgraph::prelude::NodeIndex;
use std::collections::HashMap;
use sway_types::{IdentUnique, Named};
#[derive(Default, Clone)]
pub(crate) struct FunctionNamespaceEntry {
pub(crate) entry_point: EntryPoint,
pub(crate) exit_point: ExitPoint,
pub(crate) return_type: TypeInfo,
}
#[derive(Default, Clone)]
pub(crate) struct StructNamespaceEntry {
pub(crate) struct_decl_ix: NodeIndex,
pub(crate) fields: IndexMap<String, NodeIndex>,
}
#[derive(Clone, Debug)]
pub(crate) struct TraitNamespaceEntry {
pub(crate) trait_idx: NodeIndex,
pub(crate) module_tree_type: TreeType,
}
#[derive(Default, Clone)]
pub(crate) struct VariableNamespaceEntry {
pub(crate) variable_decl_ix: NodeIndex,
}
#[derive(Default, Clone)]
pub struct ControlFlowNamespace {
pub(crate) function_namespace: IndexMap<(IdentUnique, TyFunctionSig), FunctionNamespaceEntry>,
pub(crate) enum_namespace: HashMap<IdentUnique, (NodeIndex, HashMap<Ident, NodeIndex>)>,
pub(crate) trait_namespace: HashMap<CallPath, TraitNamespaceEntry>,
pub(crate) trait_method_namespace: HashMap<CallPath, HashMap<Ident, NodeIndex>>,
pub(crate) struct_namespace: HashMap<String, StructNamespaceEntry>,
pub(crate) const_namespace: HashMap<Ident, NodeIndex>,
pub(crate) configurable_namespace: HashMap<Ident, NodeIndex>,
pub(crate) const_generic_namespace: HashMap<Ident, NodeIndex>,
pub(crate) storage: HashMap<Ident, NodeIndex>,
pub(crate) code_blocks: Vec<ControlFlowCodeBlock>,
pub(crate) alias: HashMap<IdentUnique, NodeIndex>,
}
#[derive(Default, Clone)]
pub struct ControlFlowCodeBlock {
pub(crate) variables: HashMap<Ident, VariableNamespaceEntry>,
}
impl ControlFlowNamespace {
pub(crate) fn get_function(&self, fn_decl: &TyFunctionDecl) -> Option<&FunctionNamespaceEntry> {
let ident: IdentUnique = fn_decl.name.clone().into();
self.function_namespace
.get(&(ident, TyFunctionSig::from_fn_decl(fn_decl)))
}
pub(crate) fn insert_function(
&mut self,
fn_decl: &ty::TyFunctionDecl,
entry: FunctionNamespaceEntry,
) {
let ident = &fn_decl.name;
let ident: IdentUnique = ident.into();
self.function_namespace
.insert((ident, TyFunctionSig::from_fn_decl(fn_decl)), entry);
}
pub(crate) fn get_constant(&self, const_decl: &TyConstantDecl) -> Option<&NodeIndex> {
self.const_namespace.get(&const_decl.name().clone())
}
pub(crate) fn get_configurable(&self, decl: &TyConfigurableDecl) -> Option<&NodeIndex> {
self.configurable_namespace.get(&decl.name().clone())
}
pub(crate) fn get_const_generic(&self, decl: &TyConstGenericDecl) -> Option<&NodeIndex> {
self.const_generic_namespace.get(&decl.name().clone())
}
#[allow(dead_code)]
pub(crate) fn insert_constant(
&mut self,
const_decl: TyConstantDecl,
declaration_node: NodeIndex,
) {
self.const_namespace
.insert(const_decl.name().clone(), declaration_node);
}
pub(crate) fn get_global_constant(&self, ident: &Ident) -> Option<&NodeIndex> {
self.const_namespace.get(ident)
}
pub(crate) fn insert_global_constant(
&mut self,
const_name: Ident,
declaration_node: NodeIndex,
) {
self.const_namespace.insert(const_name, declaration_node);
}
pub(crate) fn insert_configurable(&mut self, const_name: Ident, declaration_node: NodeIndex) {
self.configurable_namespace
.insert(const_name, declaration_node);
}
pub(crate) fn insert_enum(&mut self, enum_name: Ident, enum_decl_index: NodeIndex) {
self.enum_namespace
.insert(enum_name.into(), (enum_decl_index, HashMap::new()));
}
pub(crate) fn insert_enum_variant(
&mut self,
enum_name: Ident,
enum_decl_index: NodeIndex,
variant_name: Ident,
variant_index: NodeIndex,
) {
match self.enum_namespace.get_mut(&enum_name.clone().into()) {
Some((_ix, variants)) => {
variants.insert(variant_name, variant_index);
}
None => {
let variant_space = {
let mut map = HashMap::new();
map.insert(variant_name, variant_index);
map
};
self.enum_namespace
.insert(enum_name.into(), (enum_decl_index, variant_space));
}
}
}
pub(crate) fn find_enum(&self, enum_name: &Ident) -> Option<&NodeIndex> {
self.enum_namespace.get(&enum_name.into()).map(|f| &f.0)
}
pub(crate) fn find_enum_variant_index(
&self,
enum_name: &Ident,
variant_name: &Ident,
) -> Option<(NodeIndex, NodeIndex)> {
let (enum_ix, enum_decl) = self.enum_namespace.get(&enum_name.into())?;
Some((*enum_ix, *enum_decl.get(variant_name)?))
}
pub(crate) fn add_trait(&mut self, trait_name: CallPath, trait_entry: TraitNamespaceEntry) {
self.trait_namespace.insert(trait_name, trait_entry);
}
pub(crate) fn find_trait(&self, name: &CallPath) -> Option<&TraitNamespaceEntry> {
self.trait_namespace.get(name)
}
pub(crate) fn find_trait_method(
&self,
trait_name: &CallPath,
method_name: &Ident,
) -> Option<&NodeIndex> {
self.trait_method_namespace
.get(trait_name)
.and_then(|methods| methods.get(method_name))
}
pub(crate) fn insert_trait_methods(
&mut self,
trait_name: CallPath,
methods: Vec<(Ident, NodeIndex)>,
) {
match self.trait_method_namespace.get_mut(&trait_name) {
Some(methods_ns) => {
for (name, ix) in methods {
methods_ns.insert(name, ix);
}
}
None => {
let mut ns = HashMap::default();
for (name, ix) in methods {
ns.insert(name, ix);
}
self.trait_method_namespace.insert(trait_name, ns);
}
}
}
pub(crate) fn insert_storage(&mut self, field_nodes: Vec<(ty::TyStorageField, NodeIndex)>) {
for (field, node) in field_nodes {
self.storage.insert(field.name, node);
}
}
pub(crate) fn get_struct(&self, ident: &Ident) -> Option<&StructNamespaceEntry> {
self.struct_namespace.get(ident.as_str())
}
pub(crate) fn insert_struct(
&mut self,
struct_name: String,
declaration_node: NodeIndex,
field_nodes: Vec<(Ident, NodeIndex)>,
) {
let entry = StructNamespaceEntry {
struct_decl_ix: declaration_node,
fields: field_nodes
.into_iter()
.map(|(ident, ix)| (ident.as_str().to_string(), ix))
.collect(),
};
self.struct_namespace.insert(struct_name, entry);
}
pub(crate) fn find_struct_decl(&self, struct_name: &str) -> Option<&NodeIndex> {
self.struct_namespace
.get(struct_name)
.map(|StructNamespaceEntry { struct_decl_ix, .. }| struct_decl_ix)
}
pub(crate) fn find_struct_field_idx(
&self,
struct_name: &str,
field_name: &str,
) -> Option<&NodeIndex> {
self.struct_namespace
.get(struct_name)?
.fields
.get(field_name)
}
pub(crate) fn push_code_block(&mut self) {
self.code_blocks.push(ControlFlowCodeBlock {
variables: HashMap::<Ident, VariableNamespaceEntry>::new(),
});
}
pub(crate) fn pop_code_block(&mut self) {
self.code_blocks.pop();
}
pub(crate) fn insert_variable(&mut self, name: Ident, entry: VariableNamespaceEntry) {
if let Some(code_block) = self.code_blocks.last_mut() {
code_block.variables.insert(name, entry);
}
}
pub(crate) fn get_variable(&mut self, name: &Ident) -> Option<VariableNamespaceEntry> {
for code_block in self.code_blocks.iter().rev() {
if let Some(entry) = code_block.variables.get(name).cloned() {
return Some(entry);
}
}
None
}
pub(crate) fn insert_alias(&mut self, name: Ident, entry: NodeIndex) {
self.alias.insert(name.into(), entry);
}
pub(crate) fn get_alias(&self, name: &Ident) -> Option<&NodeIndex> {
self.alias.get(&name.into())
}
}