use crate::{
decl_engine::*, engine_threading::Engines, error::*, language::ty, namespace::*, type_system::*,
};
use super::TraitMap;
use sway_error::{
error::CompileError,
warning::{CompileWarning, Warning},
};
use sway_types::{span::Span, Spanned};
use std::sync::Arc;
#[derive(Clone, Copy, PartialEq, Debug)]
pub(crate) enum GlobImport {
Yes,
No,
}
pub(super) type SymbolMap = im::OrdMap<Ident, ty::TyDeclaration>;
pub(super) type UseSynonyms = im::HashMap<Ident, (Vec<Ident>, GlobImport)>;
pub(super) type UseAliases = im::HashMap<String, Ident>;
#[derive(Clone, Debug, Default)]
pub struct Items {
pub(crate) symbols: SymbolMap,
pub(crate) implemented_traits: TraitMap,
pub(crate) use_synonyms: UseSynonyms,
pub(crate) use_aliases: UseAliases,
pub(crate) declared_storage: Option<DeclId>,
}
impl Items {
pub fn symbols(&self) -> &SymbolMap {
&self.symbols
}
pub fn apply_storage_load(
&self,
engines: Engines<'_>,
fields: Vec<Ident>,
storage_fields: &[ty::TyStorageField],
access_span: &Span,
) -> CompileResult<(ty::TyStorageAccess, TypeId)> {
let mut warnings = vec![];
let mut errors = vec![];
let type_engine = engines.te();
let decl_engine = engines.de();
match self.declared_storage {
Some(ref decl_id) => {
let storage = check!(
CompileResult::from(decl_engine.get_storage(decl_id.clone(), access_span)),
return err(warnings, errors),
warnings,
errors
);
storage.apply_storage_load(type_engine, fields, storage_fields)
}
None => {
errors.push(CompileError::NoDeclaredStorage {
span: fields[0].span(),
});
err(warnings, errors)
}
}
}
pub fn set_storage_declaration(&mut self, decl_id: DeclId) -> CompileResult<()> {
if self.declared_storage.is_some() {
return err(
vec![],
vec![CompileError::MultipleStorageDeclarations {
span: decl_id.span(),
}],
);
}
self.declared_storage = Some(decl_id);
ok((), vec![], vec![])
}
pub fn get_all_declared_symbols(&self) -> impl Iterator<Item = &Ident> {
self.symbols().keys()
}
pub(crate) fn insert_symbol(
&mut self,
name: Ident,
item: ty::TyDeclaration,
) -> CompileResult<()> {
let mut warnings = vec![];
let mut errors = vec![];
if self.symbols.get(&name).is_some() {
match item {
ty::TyDeclaration::EnumDeclaration { .. }
| ty::TyDeclaration::StructDeclaration { .. }
| ty::TyDeclaration::AbiDeclaration { .. }
| ty::TyDeclaration::TraitDeclaration { .. } => {
errors.push(CompileError::ShadowsOtherSymbol { name: name.clone() });
}
ty::TyDeclaration::GenericTypeForFunctionScope { .. } => {
errors.push(CompileError::GenericShadowsGeneric { name: name.clone() });
}
_ => {
warnings.push(CompileWarning {
span: name.span(),
warning_content: Warning::ShadowsOtherSymbol { name: name.clone() },
});
}
}
}
self.symbols.insert(name, item);
ok((), warnings, errors)
}
pub(crate) fn check_symbol(&self, name: &Ident) -> Result<&ty::TyDeclaration, CompileError> {
self.symbols
.get(name)
.ok_or_else(|| CompileError::SymbolNotFound { name: name.clone() })
}
pub(crate) fn insert_trait_implementation_for_type(
&mut self,
engines: Engines<'_>,
type_id: TypeId,
) {
self.implemented_traits.insert_for_type(engines, type_id);
}
pub(crate) fn get_methods_for_type(
&self,
engines: Engines<'_>,
type_id: TypeId,
) -> Vec<DeclId> {
self.implemented_traits
.get_methods_for_type(engines, type_id)
}
pub(crate) fn has_storage_declared(&self) -> bool {
self.declared_storage.is_some()
}
pub(crate) fn get_storage_field_descriptors(
&self,
decl_engine: &DeclEngine,
access_span: &Span,
) -> CompileResult<Vec<ty::TyStorageField>> {
let mut warnings = vec![];
let mut errors = vec![];
match self.declared_storage {
Some(ref decl_id) => {
let storage = check!(
CompileResult::from(decl_engine.get_storage(decl_id.clone(), access_span)),
return err(warnings, errors),
warnings,
errors
);
ok(storage.fields, warnings, errors)
}
None => {
let msg = "unknown source location";
let span = Span::new(Arc::from(msg), 0, msg.len(), None).unwrap();
errors.push(CompileError::NoDeclaredStorage { span });
err(warnings, errors)
}
}
}
pub(crate) fn find_subfield_type(
&self,
engines: Engines<'_>,
base_name: &Ident,
projections: &[ty::ProjectionKind],
) -> CompileResult<(TypeId, TypeId)> {
let mut warnings = vec![];
let mut errors = vec![];
let type_engine = engines.te();
let symbol = match self.symbols.get(base_name).cloned() {
Some(s) => s,
None => {
errors.push(CompileError::UnknownVariable {
var_name: base_name.clone(),
});
return err(warnings, errors);
}
};
let mut symbol = check!(
symbol.return_type(engines, &base_name.span()),
return err(warnings, errors),
warnings,
errors
);
let mut symbol_span = base_name.span();
let mut parent_rover = symbol;
let mut full_name_for_error = base_name.to_string();
let mut full_span_for_error = base_name.span();
for projection in projections {
let resolved_type = match type_engine.to_typeinfo(symbol, &symbol_span) {
Ok(resolved_type) => resolved_type,
Err(error) => {
errors.push(CompileError::TypeError(error));
return err(warnings, errors);
}
};
match (resolved_type, projection) {
(
TypeInfo::Struct {
name: struct_name,
fields,
..
},
ty::ProjectionKind::StructField { name: field_name },
) => {
let field_type_opt = {
fields.iter().find_map(
|ty::TyStructField {
type_id: r#type,
name,
..
}| {
if name == field_name {
Some(r#type)
} else {
None
}
},
)
};
let field_type = match field_type_opt {
Some(field_type) => field_type,
None => {
let available_fields = fields
.iter()
.map(|field| field.name.as_str())
.collect::<Vec<_>>();
errors.push(CompileError::FieldNotFound {
field_name: field_name.clone(),
struct_name,
available_fields: available_fields.join(", "),
});
return err(warnings, errors);
}
};
parent_rover = symbol;
symbol = *field_type;
symbol_span = field_name.span().clone();
full_name_for_error.push_str(field_name.as_str());
full_span_for_error =
Span::join(full_span_for_error, field_name.span().clone());
}
(TypeInfo::Tuple(fields), ty::ProjectionKind::TupleField { index, index_span }) => {
let field_type_opt = {
fields
.get(*index)
.map(|TypeArgument { type_id, .. }| type_id)
};
let field_type = match field_type_opt {
Some(field_type) => field_type,
None => {
errors.push(CompileError::TupleIndexOutOfBounds {
index: *index,
count: fields.len(),
span: Span::join(full_span_for_error, index_span.clone()),
});
return err(warnings, errors);
}
};
parent_rover = symbol;
symbol = *field_type;
symbol_span = index_span.clone();
full_name_for_error.push_str(&index.to_string());
full_span_for_error = Span::join(full_span_for_error, index_span.clone());
}
(
TypeInfo::Array(elem_ty, _),
ty::ProjectionKind::ArrayIndex { index_span, .. },
) => {
parent_rover = symbol;
symbol = elem_ty.type_id;
symbol_span = index_span.clone();
full_span_for_error = index_span.clone();
}
(actually, ty::ProjectionKind::StructField { .. }) => {
errors.push(CompileError::FieldAccessOnNonStruct {
span: full_span_for_error,
actually: engines.help_out(actually).to_string(),
});
return err(warnings, errors);
}
(actually, ty::ProjectionKind::TupleField { .. }) => {
errors.push(CompileError::NotATuple {
name: full_name_for_error,
span: full_span_for_error,
actually: engines.help_out(actually).to_string(),
});
return err(warnings, errors);
}
(actually, ty::ProjectionKind::ArrayIndex { .. }) => {
errors.push(CompileError::NotIndexable {
name: full_name_for_error,
span: full_span_for_error,
actually: engines.help_out(actually).to_string(),
});
return err(warnings, errors);
}
}
}
ok((symbol, parent_rover), warnings, errors)
}
}