//! `SourceBinder` should be the main entry point for getting info about source code. //! It's main task is to map source syntax trees to hir-level IDs. //! //! It is intended to subsume `FromSource` and `SourceAnalyzer`. use hir_def::{ child_by_source::ChildBySource, dyn_map::DynMap, keys::{self, Key}, resolver::{HasResolver, Resolver}, ConstId, DefWithBodyId, EnumId, EnumVariantId, FunctionId, ImplId, ModuleId, StaticId, StructFieldId, StructId, TraitId, TypeAliasId, UnionId, VariantId, }; use hir_expand::{AstId, InFile, MacroDefId, MacroDefKind}; use ra_prof::profile; use ra_syntax::{ast, match_ast, AstNode, SyntaxNode, TextUnit}; use rustc_hash::FxHashMap; use crate::{db::HirDatabase, ModuleSource, SourceAnalyzer}; pub struct SourceBinder<'a, DB> { pub db: &'a DB, child_by_source_cache: FxHashMap, } impl SourceBinder<'_, DB> { pub fn new(db: &DB) -> SourceBinder { SourceBinder { db, child_by_source_cache: FxHashMap::default() } } pub fn analyze( &mut self, src: InFile<&SyntaxNode>, offset: Option, ) -> SourceAnalyzer { let _p = profile("SourceBinder::analyzer"); let container = match self.find_container(src) { Some(it) => it, None => return SourceAnalyzer::new_for_resolver(Resolver::default(), src), }; let resolver = match container { ChildContainer::DefWithBodyId(def) => { return SourceAnalyzer::new_for_body(self.db, def, src, offset) } ChildContainer::TraitId(it) => it.resolver(self.db), ChildContainer::ImplId(it) => it.resolver(self.db), ChildContainer::ModuleId(it) => it.resolver(self.db), ChildContainer::EnumId(it) => it.resolver(self.db), ChildContainer::VariantId(it) => it.resolver(self.db), }; SourceAnalyzer::new_for_resolver(resolver, src) } pub fn to_def(&mut self, src: InFile) -> Option where D: From, T: ToId, { let id: T::ID = self.to_id(src)?; Some(id.into()) } fn to_id(&mut self, src: InFile) -> Option { T::to_id(self, src) } fn find_container(&mut self, src: InFile<&SyntaxNode>) -> Option { for container in src.cloned().ancestors_with_macros(self.db).skip(1) { let res: ChildContainer = match_ast! { match (container.value) { ast::TraitDef(it) => { let def: TraitId = self.to_id(container.with_value(it))?; def.into() }, ast::ImplBlock(it) => { let def: ImplId = self.to_id(container.with_value(it))?; def.into() }, ast::FnDef(it) => { let def: FunctionId = self.to_id(container.with_value(it))?; DefWithBodyId::from(def).into() }, ast::StaticDef(it) => { let def: StaticId = self.to_id(container.with_value(it))?; DefWithBodyId::from(def).into() }, ast::ConstDef(it) => { let def: ConstId = self.to_id(container.with_value(it))?; DefWithBodyId::from(def).into() }, ast::EnumDef(it) => { let def: EnumId = self.to_id(container.with_value(it))?; def.into() }, ast::StructDef(it) => { let def: StructId = self.to_id(container.with_value(it))?; VariantId::from(def).into() }, ast::UnionDef(it) => { let def: UnionId = self.to_id(container.with_value(it))?; VariantId::from(def).into() }, // FIXME: handle out-of-line modules here _ => { continue }, } }; return Some(res); } let module_source = ModuleSource::from_child_node(self.db, src); let c = crate::Module::from_definition(self.db, src.with_value(module_source))?; Some(c.id.into()) } } #[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)] enum ChildContainer { DefWithBodyId(DefWithBodyId), ModuleId(ModuleId), TraitId(TraitId), ImplId(ImplId), EnumId(EnumId), VariantId(VariantId), } impl_froms! { ChildContainer: DefWithBodyId, ModuleId, TraitId, ImplId, EnumId, VariantId, } pub trait ToId: Sized + AstNode + 'static { type ID: Sized + Copy + 'static; fn to_id(sb: &mut SourceBinder<'_, DB>, src: InFile) -> Option; } pub trait ToIdByKey: Sized + AstNode + 'static { type ID: Sized + Copy + 'static; const KEY: Key; } impl ToId for T { type ID = ::ID; fn to_id( sb: &mut SourceBinder<'_, DB>, src: InFile, ) -> Option { let container = sb.find_container(src.as_ref().map(|it| it.syntax()))?; let db = sb.db; let dyn_map = &*sb.child_by_source_cache.entry(container).or_insert_with(|| match container { ChildContainer::DefWithBodyId(it) => it.child_by_source(db), ChildContainer::ModuleId(it) => it.child_by_source(db), ChildContainer::TraitId(it) => it.child_by_source(db), ChildContainer::ImplId(it) => it.child_by_source(db), ChildContainer::EnumId(it) => it.child_by_source(db), ChildContainer::VariantId(it) => it.child_by_source(db), }); dyn_map[T::KEY].get(&src).copied() } } macro_rules! to_id_key_impls { ($(($id:ident, $ast:path, $key:path)),* ,) => {$( impl ToIdByKey for $ast { type ID = $id; const KEY: Key = $key; } )*} } to_id_key_impls![ (StructId, ast::StructDef, keys::STRUCT), (UnionId, ast::UnionDef, keys::UNION), (EnumId, ast::EnumDef, keys::ENUM), (TraitId, ast::TraitDef, keys::TRAIT), (FunctionId, ast::FnDef, keys::FUNCTION), (StaticId, ast::StaticDef, keys::STATIC), (ConstId, ast::ConstDef, keys::CONST), (TypeAliasId, ast::TypeAliasDef, keys::TYPE_ALIAS), (ImplId, ast::ImplBlock, keys::IMPL), (StructFieldId, ast::RecordFieldDef, keys::RECORD_FIELD), (EnumVariantId, ast::EnumVariant, keys::ENUM_VARIANT), ]; // FIXME: use DynMap as well? impl ToId for ast::MacroCall { type ID = MacroDefId; fn to_id( sb: &mut SourceBinder<'_, DB>, src: InFile, ) -> Option { let kind = MacroDefKind::Declarative; let module_src = ModuleSource::from_child_node(sb.db, src.as_ref().map(|it| it.syntax())); let module = crate::Module::from_definition(sb.db, InFile::new(src.file_id, module_src))?; let krate = Some(module.krate().id); let ast_id = Some(AstId::new(src.file_id, sb.db.ast_id_map(src.file_id).ast_id(&src.value))); Some(MacroDefId { krate, ast_id, kind }) } }