use std::collections::HashMap; use std::rc::Rc; #[derive(PartialEq, Eq, Copy, Clone, Hash, Debug)] pub struct PrimitiveId(usize); #[derive(PartialEq, Eq, Copy, Clone, Hash, Debug)] pub struct ClassId(usize); #[derive(PartialEq, Eq, Copy, Clone, Hash, Debug)] pub struct ParamId(usize); #[derive(PartialEq, Eq, Copy, Clone, Hash, Debug)] pub struct VariableId(usize); #[derive(PartialEq, Eq, Clone, Hash, Debug)] pub enum Type { BotType, SelfType, PrimitiveType(PrimitiveId), ClassType(ClassId), VirtualClassType(ClassId), ParametricType(ParamId, Vec>), TypeVariable(VariableId), } pub struct FnDef { pub args: Rc, pub result: Option>, } pub struct TypeDef<'a> { pub name: &'a str, pub fields: HashMap<&'a str, Type>, pub methods: HashMap<&'a str, FnDef>, } pub struct ClassDef<'a> { pub base: TypeDef<'a>, pub parents: Vec, } pub struct ParametricDef<'a> { pub base: TypeDef<'a>, pub params: Vec, } pub struct VarDef<'a> { pub name: &'a str, pub bound: Vec>, } pub const TUPLE_TYPE: ParamId = ParamId(0); pub const LIST_TYPE: ParamId = ParamId(1); pub const BOOL_TYPE: PrimitiveId = PrimitiveId(0); pub const INT32_TYPE: PrimitiveId = PrimitiveId(1); pub struct GlobalContext<'a> { primitive_defs: Vec>, class_defs: Vec>, parametric_defs: Vec>, var_defs: Vec>, sym_table: HashMap<&'a str, Type>, fn_table: HashMap<&'a str, FnDef>, } impl<'a> GlobalContext<'a> { pub fn new(primitives: Vec>) -> GlobalContext { let mut sym_table = HashMap::new(); for (i, t) in primitives.iter().enumerate() { sym_table.insert(t.name, Type::PrimitiveType(PrimitiveId(i))); } return GlobalContext { primitive_defs: primitives, class_defs: Vec::new(), parametric_defs: Vec::new(), var_defs: Vec::new(), fn_table: HashMap::new(), sym_table, }; } pub fn add_class(&mut self, def: ClassDef<'a>) { self.sym_table.insert( def.base.name, Type::ClassType(ClassId(self.class_defs.len())), ); self.class_defs.push(def); } pub fn add_parametric(&mut self, def: ParametricDef<'a>) { let params = def .params .iter() .map(|&v| Rc::new(Type::TypeVariable(v))) .collect(); self.sym_table.insert( def.base.name, Type::ParametricType(ParamId(self.parametric_defs.len()), params), ); self.parametric_defs.push(def); } pub fn add_variable(&mut self, def: VarDef<'a>) { self.sym_table.insert( def.name, Type::TypeVariable(VariableId(self.var_defs.len())), ); self.var_defs.push(def); } pub fn add_variable_private(&mut self, def: VarDef<'a>) { self.var_defs.push(def); } pub fn add_fn(&'a mut self, name: &'a str, def: FnDef) { self.fn_table.insert(name, def); } pub fn get_fn(&self, name: &str) -> Option<&FnDef> { self.fn_table.get(name) } pub fn get_primitive_mut(&mut self, id: PrimitiveId) -> &mut TypeDef<'a> { self.primitive_defs.get_mut(id.0).unwrap() } pub fn get_primitive(&self, id: PrimitiveId) -> &TypeDef { self.primitive_defs.get(id.0).unwrap() } pub fn get_class_mut(&mut self, id: ClassId) -> &mut ClassDef<'a> { self.class_defs.get_mut(id.0).unwrap() } pub fn get_class(&self, id: ClassId) -> &ClassDef { self.class_defs.get(id.0).unwrap() } pub fn get_parametric_mut(&mut self, id: ParamId) -> &mut ParametricDef<'a> { self.parametric_defs.get_mut(id.0).unwrap() } pub fn get_parametric(&self, id: ParamId) -> &ParametricDef { self.parametric_defs.get(id.0).unwrap() } pub fn get_variable_mut(&mut self, id: VariableId) -> &mut VarDef<'a> { self.var_defs.get_mut(id.0).unwrap() } pub fn get_variable(&self, id: VariableId) -> &VarDef { self.var_defs.get(id.0).unwrap() } pub fn get_type(&self, name: &str) -> Option { // TODO: change this to handle import self.sym_table.get(name).map(|v| v.clone()) } } impl Type { pub fn subst(&self, map: &HashMap>) -> Type { match self { Type::TypeVariable(id) => map.get(id).map(|v| v.as_ref()).unwrap_or(self).clone(), Type::ParametricType(id, params) => Type::ParametricType( *id, params .iter() .map(|v| v.as_ref().subst(map).into()) .collect(), ), _ => self.clone(), } } pub fn inv_subst(&self, map: &[(Rc, Rc)]) -> Rc { for (from, to) in map.iter() { if self == from.as_ref() { return to.clone(); } } match self { Type::ParametricType(id, params) => Type::ParametricType( *id, params .iter() .map(|v| v.as_ref().inv_subst(map).into()) .collect(), ), _ => self.clone(), }.into() } pub fn get_subst(&self, ctx: &GlobalContext) -> HashMap> { match self { Type::ParametricType(id, params) => { let vars = &ctx.get_parametric(*id).params; vars.iter() .zip(params) .map(|(v, p)| (*v, p.as_ref().clone().into())) .collect() } // if this proves to be slow, we can use option type _ => HashMap::new(), } } }