//! When *constructing* `hir`, we start at some parent syntax node and recursively //! lower the children. //! //! This modules allows one to go in the opposite direction: start with a syntax //! node for a *child*, and get its hir. use either::Either; use hir_expand::HirFileId; use itertools::Itertools; use syntax::ast::HasAttrs; use crate::{ db::DefDatabase, dyn_map::DynMap, item_scope::ItemScope, keys, src::{HasChildSource, HasSource}, AdtId, AssocItemId, DefWithBodyId, EnumId, EnumVariantId, FieldId, ImplId, Lookup, ModuleDefId, ModuleId, TraitId, VariantId, }; pub trait ChildBySource { fn child_by_source(&self, db: &dyn DefDatabase, file_id: HirFileId) -> DynMap { let mut res = DynMap::default(); self.child_by_source_to(db, &mut res, file_id); res } fn child_by_source_to(&self, db: &dyn DefDatabase, map: &mut DynMap, file_id: HirFileId); } impl ChildBySource for TraitId { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) { let data = db.trait_data(*self); for (_name, item) in data.items.iter() { match *item { AssocItemId::FunctionId(func) => { let loc = func.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); res[keys::FUNCTION].insert(src, func) } } AssocItemId::ConstId(konst) => { let loc = konst.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); res[keys::CONST].insert(src, konst) } } AssocItemId::TypeAliasId(ty) => { let loc = ty.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); res[keys::TYPE_ALIAS].insert(src, ty) } } } } } } impl ChildBySource for ImplId { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) { let data = db.impl_data(*self); for &item in data.items.iter() { match item { AssocItemId::FunctionId(func) => { let loc = func.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); res[keys::FUNCTION].insert(src, func) } } AssocItemId::ConstId(konst) => { let loc = konst.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); res[keys::CONST].insert(src, konst) } } AssocItemId::TypeAliasId(ty) => { let loc = ty.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); res[keys::TYPE_ALIAS].insert(src, ty) } } } } } } impl ChildBySource for ModuleId { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) { let def_map = self.def_map(db); let module_data = &def_map[self.local_id]; module_data.scope.child_by_source_to(db, res, file_id); } } impl ChildBySource for ItemScope { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) { self.declarations().for_each(|item| add_module_def(db, file_id, res, item)); self.macros().for_each(|(_, makro)| { let ast_id = makro.ast_id(); if ast_id.either(|it| it.file_id, |it| it.file_id) == file_id { let src = match ast_id { Either::Left(ast_id) => ast_id.with_value(ast_id.to_node(db.upcast())), // FIXME: Do we need to add proc-macros into a PROCMACRO dynmap here? Either::Right(_fn) => return, }; res[keys::MACRO].insert(src, makro); } }); self.unnamed_consts().for_each(|konst| { let src = konst.lookup(db).source(db); res[keys::CONST].insert(src, konst); }); self.impls().for_each(|imp| add_impl(db, file_id, res, imp)); self.attr_macro_invocs().for_each(|(ast_id, call_id)| { let item = ast_id.with_value(ast_id.to_node(db.upcast())); res[keys::ATTR_MACRO].insert(item, call_id); }); self.derive_macro_invocs().for_each(|(ast_id, calls)| { let item = ast_id.to_node(db.upcast()); let grouped = calls.iter().copied().into_group_map(); for (attr_id, calls) in grouped { if let Some(attr) = item.attrs().nth(attr_id.ast_index as usize) { res[keys::DERIVE_MACRO].insert(ast_id.with_value(attr), calls.into()); } } }); fn add_module_def( db: &dyn DefDatabase, file_id: HirFileId, map: &mut DynMap, item: ModuleDefId, ) { match item { ModuleDefId::FunctionId(func) => { let loc = func.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::FUNCTION].insert(src, func) } } ModuleDefId::ConstId(konst) => { let loc = konst.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::CONST].insert(src, konst) } } ModuleDefId::StaticId(statik) => { let loc = statik.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::STATIC].insert(src, statik) } } ModuleDefId::TypeAliasId(ty) => { let loc = ty.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::TYPE_ALIAS].insert(src, ty) } } ModuleDefId::TraitId(trait_) => { let loc = trait_.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::TRAIT].insert(src, trait_) } } ModuleDefId::AdtId(adt) => match adt { AdtId::StructId(strukt) => { let loc = strukt.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::STRUCT].insert(src, strukt) } } AdtId::UnionId(union_) => { let loc = union_.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::UNION].insert(src, union_) } } AdtId::EnumId(enum_) => { let loc = enum_.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::ENUM].insert(src, enum_) } } }, _ => (), } } fn add_impl(db: &dyn DefDatabase, file_id: HirFileId, map: &mut DynMap, imp: ImplId) { let loc = imp.lookup(db); if loc.id.file_id() == file_id { let src = loc.source(db); map[keys::IMPL].insert(src, imp) } } } } impl ChildBySource for VariantId { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, _: HirFileId) { let arena_map = self.child_source(db); let arena_map = arena_map.as_ref(); let parent = *self; for (local_id, source) in arena_map.value.iter() { let id = FieldId { parent, local_id }; match source.clone() { Either::Left(source) => { res[keys::TUPLE_FIELD].insert(arena_map.with_value(source), id) } Either::Right(source) => { res[keys::RECORD_FIELD].insert(arena_map.with_value(source), id) } } } } } impl ChildBySource for EnumId { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, _: HirFileId) { let arena_map = self.child_source(db); let arena_map = arena_map.as_ref(); for (local_id, source) in arena_map.value.iter() { let id = EnumVariantId { parent: *self, local_id }; res[keys::VARIANT].insert(arena_map.with_value(source.clone()), id) } } } impl ChildBySource for DefWithBodyId { fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) { let body = db.body(*self); for (_, def_map) in body.blocks(db) { // All block expressions are merged into the same map, because they logically all add // inner items to the containing `DefWithBodyId`. def_map[def_map.root()].scope.child_by_source_to(db, res, file_id); } } }