cl-typeck: Sketch out a new way to store definition metadata
This commit is contained in:
		
							
								
								
									
										201
									
								
								compiler/cl-typeck/src/def_item.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								compiler/cl-typeck/src/def_item.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,201 @@ | |||||||
|  | //! A [DefItem] contains the [DefSource] and [Item] metadata for any | ||||||
|  | //! [Def](crate::definition::Def), as well as the [Path] of the | ||||||
|  | //! containing [Module]. | ||||||
|  | //! | ||||||
|  | //! [DefItem]s are collected by the [Definition Sorcerer](sorcerer), | ||||||
|  | //! an AST visitor that pairs [DefSource]s with their surrounding | ||||||
|  | //! context ([Path], [Span], [Meta], [Visibility]) | ||||||
|  |  | ||||||
|  | use cl_ast::ast::*; | ||||||
|  | use cl_structures::span::Span; | ||||||
|  | use std::fmt; | ||||||
|  |  | ||||||
|  | #[derive(Clone, Debug, PartialEq, Eq)] | ||||||
|  | pub struct DefItem<'a> { | ||||||
|  |     pub in_path: Path, | ||||||
|  |     pub span: &'a Span, | ||||||
|  |     pub meta: &'a [Meta], | ||||||
|  |     pub vis: Visibility, | ||||||
|  |     pub kind: DefSource<'a>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)] | ||||||
|  | pub enum DefSource<'a> { | ||||||
|  |     Module(&'a Module), | ||||||
|  |     Alias(&'a Alias), | ||||||
|  |     Enum(&'a Enum), | ||||||
|  |     Struct(&'a Struct), | ||||||
|  |     Const(&'a Const), | ||||||
|  |     Static(&'a Static), | ||||||
|  |     Function(&'a Function), | ||||||
|  |     Local(&'a Let), | ||||||
|  |     Impl(&'a Impl), | ||||||
|  |     Use(&'a Use), | ||||||
|  |     Ty(&'a TyKind), | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl<'a> DefSource<'a> { | ||||||
|  |     pub fn name(&self) -> Option<&'a str> { | ||||||
|  |         match self { | ||||||
|  |             DefSource::Module(v) => Some(v.name.0.as_str()), | ||||||
|  |             DefSource::Alias(v) => Some(v.to.0.as_str()), | ||||||
|  |             DefSource::Enum(v) => Some(v.name.0.as_str()), | ||||||
|  |             DefSource::Struct(v) => Some(v.name.0.as_str()), | ||||||
|  |             DefSource::Const(v) => Some(v.name.0.as_str()), | ||||||
|  |             DefSource::Static(v) => Some(v.name.0.as_str()), | ||||||
|  |             DefSource::Function(v) => Some(v.name.0.as_str()), | ||||||
|  |             DefSource::Local(l) => Some(l.name.0.as_str()), | ||||||
|  |             DefSource::Impl(_) | DefSource::Use(_) | DefSource::Ty(_) => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns `true` if this [DefSource] defines a named value | ||||||
|  |     pub fn is_named_value(&self) -> bool { | ||||||
|  |         matches!(self, Self::Const(_) | Self::Static(_) | Self::Function(_)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns `true` if this [DefSource] defines a named type | ||||||
|  |     pub fn is_named_type(&self) -> bool { | ||||||
|  |         matches!( | ||||||
|  |             self, | ||||||
|  |             Self::Module(_) | Self::Alias(_) | Self::Enum(_) | Self::Struct(_) | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns `true` if this [DefSource] refers to a [Ty] with no name | ||||||
|  |     pub fn is_anon_type(&self) -> bool { | ||||||
|  |         matches!(self, Self::Ty(_)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns `true` if this [DefSource] refers to an [Impl] block | ||||||
|  |     pub fn is_impl(&self) -> bool { | ||||||
|  |         matches!(self, Self::Impl(_)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns `true` if this [DefSource] refers to a [Use] import | ||||||
|  |     pub fn is_use_import(&self) -> bool { | ||||||
|  |         matches!(self, Self::Use(_)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl fmt::Display for DefSource<'_> { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||||
|  |         match self { | ||||||
|  |             Self::Module(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Alias(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Enum(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Struct(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Const(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Static(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Function(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Impl(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Use(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Ty(arg0) => arg0.fmt(f), | ||||||
|  |             Self::Local(arg0) => arg0.fmt(f), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub mod sorcerer { | ||||||
|  |     //! An [AST](cl_ast) analysis pass that collects [DefItem] entries. | ||||||
|  |  | ||||||
|  |     use super::{DefItem, DefSource}; | ||||||
|  |     use cl_ast::{ast::*, ast_visitor::visit::*}; | ||||||
|  |     use cl_structures::span::Span; | ||||||
|  |     use std::mem; | ||||||
|  |  | ||||||
|  |     /// An AST analysis pass that collects [DefItem]s | ||||||
|  |     #[derive(Clone, Debug)] | ||||||
|  |     pub struct DefinitionSorcerer<'a> { | ||||||
|  |         path: Path, | ||||||
|  |         parts: Parts<'a>, | ||||||
|  |         defs: Vec<DefItem<'a>>, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     type Parts<'a> = (&'a Span, &'a [Meta], Visibility); | ||||||
|  |  | ||||||
|  |     impl<'a> DefinitionSorcerer<'a> { | ||||||
|  |         pub fn into_defs(self) -> Vec<DefItem<'a>> { | ||||||
|  |             self.defs | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fn with_parts<F>(&mut self, s: &'a Span, a: &'a [Meta], v: Visibility, f: F) | ||||||
|  |         where F: FnOnce(&mut Self) { | ||||||
|  |             let parts = mem::replace(&mut self.parts, (s, a, v)); | ||||||
|  |             f(self); | ||||||
|  |             self.parts = parts; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fn with_only_span<F>(&mut self, span: &'a Span, f: F) | ||||||
|  |         where F: FnOnce(&mut Self) { | ||||||
|  |             self.with_parts(span, &[], Visibility::Public, f) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fn push(&mut self, kind: DefSource<'a>) { | ||||||
|  |             let Self { path, parts, defs } = self; | ||||||
|  |             let (span, meta, vis) = *parts; | ||||||
|  |  | ||||||
|  |             defs.push(DefItem { in_path: path.clone(), span, meta, vis, kind }) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     impl Default for DefinitionSorcerer<'_> { | ||||||
|  |         fn default() -> Self { | ||||||
|  |             const DPARTS: Parts = (&Span::dummy(), &[], Visibility::Private); | ||||||
|  |             Self { | ||||||
|  |                 path: Path { absolute: true, ..Default::default() }, | ||||||
|  |                 parts: DPARTS, | ||||||
|  |                 defs: Default::default(), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     impl<'a> Visit<'a> for DefinitionSorcerer<'a> { | ||||||
|  |         fn visit_module(&mut self, m: &'a Module) { | ||||||
|  |             let Module { name, kind } = m; | ||||||
|  |             self.path.push(PathPart::Ident(name.clone())); | ||||||
|  |             self.visit_module_kind(kind); | ||||||
|  |             self.path.pop(); | ||||||
|  |         } | ||||||
|  |         fn visit_item(&mut self, i: &'a Item) { | ||||||
|  |             let Item { extents, attrs, vis, kind } = i; | ||||||
|  |             self.with_parts(extents, &attrs.meta, *vis, |v| { | ||||||
|  |                 v.visit_item_kind(kind); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |         fn visit_ty(&mut self, t: &'a Ty) { | ||||||
|  |             let Ty { extents, kind } = t; | ||||||
|  |             self.with_only_span(extents, |v| { | ||||||
|  |                 v.push(DefSource::Ty(kind)); | ||||||
|  |                 v.visit_ty_kind(kind); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |         fn visit_stmt(&mut self, s: &'a Stmt) { | ||||||
|  |             let Stmt { extents, kind, semi } = s; | ||||||
|  |             self.with_only_span(extents, |d| { | ||||||
|  |                 d.visit_stmt_kind(kind); | ||||||
|  |                 d.visit_semi(semi); | ||||||
|  |             }) | ||||||
|  |         } | ||||||
|  |         fn visit_item_kind(&mut self, kind: &'a ItemKind) { | ||||||
|  |             match kind { | ||||||
|  |                 ItemKind::Module(i) => self.push(DefSource::Module(i)), | ||||||
|  |                 ItemKind::Alias(i) => self.push(DefSource::Alias(i)), | ||||||
|  |                 ItemKind::Enum(i) => self.push(DefSource::Enum(i)), | ||||||
|  |                 ItemKind::Struct(i) => self.push(DefSource::Struct(i)), | ||||||
|  |                 ItemKind::Const(i) => self.push(DefSource::Const(i)), | ||||||
|  |                 ItemKind::Static(i) => self.push(DefSource::Static(i)), | ||||||
|  |                 ItemKind::Function(i) => self.push(DefSource::Function(i)), | ||||||
|  |                 ItemKind::Impl(i) => self.push(DefSource::Impl(i)), | ||||||
|  |                 ItemKind::Use(i) => self.push(DefSource::Use(i)), | ||||||
|  |             } | ||||||
|  |             or_visit_item_kind(self, kind); | ||||||
|  |         } | ||||||
|  |         fn visit_stmt_kind(&mut self, kind: &'a StmtKind) { | ||||||
|  |             if let StmtKind::Local(l) = kind { | ||||||
|  |                 self.push(DefSource::Local(l)) | ||||||
|  |             } | ||||||
|  |             or_visit_stmt_kind(self, kind); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -52,6 +52,8 @@ | |||||||
|  |  | ||||||
| pub mod key; | pub mod key; | ||||||
|  |  | ||||||
|  | pub mod def_item; | ||||||
|  |  | ||||||
| pub mod definition; | pub mod definition; | ||||||
|  |  | ||||||
| pub mod module; | pub mod module; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user