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 def_item; | ||||
|  | ||||
| pub mod definition; | ||||
|  | ||||
| pub mod module; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user