use rustc_hash::FxHashMap;
use slotmap::{DefaultKey, SlotMap};
use sway_features::ExperimentalFeatures;
use sway_types::SourceEngine;
use crate::{
block::BlockContent,
function::FunctionContent,
metadata::Metadatum,
module::{Kind, ModuleContent, ModuleIterator},
value::ValueContent,
variable::LocalVarContent,
Constant, ConstantContent, GlobalVarContent, StorageKeyContent, Type, TypeContent,
};
#[derive(Clone, Copy, Debug, PartialEq, Eq, Default)]
pub enum Backtrace {
All,
#[default]
AllExceptNever,
OnlyAlways,
None,
}
pub struct Context<'eng> {
pub source_engine: &'eng SourceEngine,
pub(crate) modules: SlotMap<DefaultKey, ModuleContent>,
pub(crate) functions: SlotMap<DefaultKey, FunctionContent>,
pub(crate) blocks: SlotMap<DefaultKey, BlockContent>,
pub(crate) values: SlotMap<DefaultKey, ValueContent>,
pub(crate) local_vars: SlotMap<DefaultKey, LocalVarContent>,
pub(crate) global_vars: SlotMap<DefaultKey, GlobalVarContent>,
pub(crate) storage_keys: SlotMap<DefaultKey, StorageKeyContent>,
pub(crate) types: SlotMap<DefaultKey, TypeContent>,
pub(crate) type_map: FxHashMap<TypeContent, Type>,
pub(crate) constants: SlotMap<DefaultKey, ConstantContent>,
pub(crate) constants_map: FxHashMap<u64, Vec<Constant>>,
pub(crate) metadata: SlotMap<DefaultKey, Metadatum>,
pub program_kind: Kind,
pub experimental: ExperimentalFeatures,
pub backtrace: Backtrace,
next_unique_sym_tag: u64,
next_unique_panic_error_code: u64,
next_unique_panicking_call_id: u64,
}
impl<'eng> Context<'eng> {
pub fn new(
source_engine: &'eng SourceEngine,
experimental: ExperimentalFeatures,
backtrace: Backtrace,
) -> Self {
let mut def = Self {
source_engine,
modules: Default::default(),
functions: Default::default(),
blocks: Default::default(),
values: Default::default(),
local_vars: Default::default(),
global_vars: Default::default(),
storage_keys: Default::default(),
types: Default::default(),
type_map: Default::default(),
constants: Default::default(),
constants_map: Default::default(),
metadata: Default::default(),
next_unique_sym_tag: 0,
next_unique_panic_error_code: 0,
next_unique_panicking_call_id: 1,
program_kind: Kind::Contract,
experimental,
backtrace,
};
Type::create_basic_types(&mut def);
def
}
pub fn source_engine(&self) -> &'eng SourceEngine {
self.source_engine
}
pub fn module_iter(&self) -> ModuleIterator {
ModuleIterator::new(self)
}
pub fn get_unique_name(&mut self) -> String {
format!("anon_{}", self.get_unique_symbol_id())
}
pub fn get_unique_symbol_id(&mut self) -> u64 {
let sym = self.next_unique_sym_tag;
self.next_unique_sym_tag += 1;
sym
}
pub fn get_unique_panic_error_code(&mut self) -> u64 {
let code = self.next_unique_panic_error_code;
self.next_unique_panic_error_code += 1;
code
}
pub fn get_unique_panicking_call_id(&mut self) -> u64 {
let id = self.next_unique_panicking_call_id;
self.next_unique_panicking_call_id += 1;
id
}
}
use std::fmt::{Display, Error, Formatter};
impl Display for Context<'_> {
fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error> {
write!(f, "{}", crate::printer::to_string(self))
}
}
impl From<Context<'_>> for String {
fn from(context: Context) -> Self {
crate::printer::to_string(&context)
}
}