7431: Handle `super` paths inside blocks correctly r=jonas-schievink a=jonas-schievink

We now intern `BlockLoc` and use `BlockId` to refer to block expressions. This is needed to keep `ModuleId` simple, since it would otherwise have to store an arbitrarily long chain of blocks and couldn't be `Copy`.

The `DefMap` hierarchy is now created as the caller descends into an item body. This is necessary to link the correct module as the block's parent, which is important for correct name resolution.

As a result, we can now resolve `super` paths inside block expressions by climbing the `DefMap` chain.

bors r+

Co-authored-by: Jonas Schievink <jonasschievink@gmail.com>
This commit is contained in:
bors[bot] 2021-01-25 18:24:04 +00:00 committed by GitHub
commit a37091d2d0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 159 additions and 75 deletions

View file

@ -2,9 +2,9 @@
use std::sync::Arc; use std::sync::Arc;
use base_db::{salsa, CrateId, SourceDatabase, Upcast}; use base_db::{salsa, CrateId, SourceDatabase, Upcast};
use hir_expand::{db::AstDatabase, AstId, HirFileId}; use hir_expand::{db::AstDatabase, HirFileId};
use la_arena::ArenaMap; use la_arena::ArenaMap;
use syntax::{ast, SmolStr}; use syntax::SmolStr;
use crate::{ use crate::{
adt::{EnumData, StructData}, adt::{EnumData, StructData},
@ -16,9 +16,10 @@ use crate::{
item_tree::ItemTree, item_tree::ItemTree,
lang_item::{LangItemTarget, LangItems}, lang_item::{LangItemTarget, LangItems},
nameres::DefMap, nameres::DefMap,
AttrDefId, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, FunctionId, FunctionLoc, AttrDefId, BlockId, BlockLoc, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, FunctionId,
GenericDefId, ImplId, ImplLoc, LocalEnumVariantId, LocalFieldId, StaticId, StaticLoc, StructId, FunctionLoc, GenericDefId, ImplId, ImplLoc, LocalEnumVariantId, LocalFieldId, StaticId,
StructLoc, TraitId, TraitLoc, TypeAliasId, TypeAliasLoc, UnionId, UnionLoc, VariantId, StaticLoc, StructId, StructLoc, TraitId, TraitLoc, TypeAliasId, TypeAliasLoc, UnionId,
UnionLoc, VariantId,
}; };
#[salsa::query_group(InternDatabaseStorage)] #[salsa::query_group(InternDatabaseStorage)]
@ -41,6 +42,8 @@ pub trait InternDatabase: SourceDatabase {
fn intern_type_alias(&self, loc: TypeAliasLoc) -> TypeAliasId; fn intern_type_alias(&self, loc: TypeAliasLoc) -> TypeAliasId;
#[salsa::interned] #[salsa::interned]
fn intern_impl(&self, loc: ImplLoc) -> ImplId; fn intern_impl(&self, loc: ImplLoc) -> ImplId;
#[salsa::interned]
fn intern_block(&self, loc: BlockLoc) -> BlockId;
} }
#[salsa::query_group(DefDatabaseStorage)] #[salsa::query_group(DefDatabaseStorage)]
@ -56,7 +59,7 @@ pub trait DefDatabase: InternDatabase + AstDatabase + Upcast<dyn AstDatabase> {
fn crate_def_map_query(&self, krate: CrateId) -> Arc<DefMap>; fn crate_def_map_query(&self, krate: CrateId) -> Arc<DefMap>;
#[salsa::invoke(DefMap::block_def_map_query)] #[salsa::invoke(DefMap::block_def_map_query)]
fn block_def_map(&self, krate: CrateId, block: AstId<ast::BlockExpr>) -> Arc<DefMap>; fn block_def_map(&self, block: BlockId) -> Arc<DefMap>;
#[salsa::invoke(StructData::struct_data_query)] #[salsa::invoke(StructData::struct_data_query)]
fn struct_data(&self, id: StructId) -> Arc<StructData>; fn struct_data(&self, id: StructId) -> Arc<StructData>;

View file

@ -74,12 +74,16 @@ use stdx::impl_from;
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct ModuleId { pub struct ModuleId {
krate: CrateId, krate: CrateId,
block: Option<BlockId>,
pub local_id: LocalModuleId, pub local_id: LocalModuleId,
} }
impl ModuleId { impl ModuleId {
pub fn def_map(&self, db: &dyn db::DefDatabase) -> Arc<DefMap> { pub fn def_map(&self, db: &dyn db::DefDatabase) -> Arc<DefMap> {
db.crate_def_map(self.krate) match self.block {
Some(block) => db.block_def_map(block),
None => db.crate_def_map(self.krate),
}
} }
pub fn krate(&self) -> CrateId { pub fn krate(&self) -> CrateId {
@ -230,6 +234,15 @@ pub struct ImplId(salsa::InternId);
type ImplLoc = ItemLoc<Impl>; type ImplLoc = ItemLoc<Impl>;
impl_intern!(ImplId, ImplLoc, intern_impl, lookup_intern_impl); impl_intern!(ImplId, ImplLoc, intern_impl, lookup_intern_impl);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Ord, PartialOrd)]
pub struct BlockId(salsa::InternId);
#[derive(Debug, Hash, PartialEq, Eq, Clone)]
pub struct BlockLoc {
ast_id: AstId<ast::BlockExpr>,
module: ModuleId,
}
impl_intern!(BlockId, BlockLoc, intern_block, lookup_intern_block);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct TypeParamId { pub struct TypeParamId {
pub parent: GenericDefId, pub parent: GenericDefId,

View file

@ -62,7 +62,7 @@ use la_arena::Arena;
use profile::Count; use profile::Count;
use rustc_hash::FxHashMap; use rustc_hash::FxHashMap;
use stdx::format_to; use stdx::format_to;
use syntax::{ast, AstNode}; use syntax::ast;
use crate::{ use crate::{
db::DefDatabase, db::DefDatabase,
@ -70,14 +70,14 @@ use crate::{
nameres::{diagnostics::DefDiagnostic, path_resolution::ResolveMode}, nameres::{diagnostics::DefDiagnostic, path_resolution::ResolveMode},
path::ModPath, path::ModPath,
per_ns::PerNs, per_ns::PerNs,
AstId, LocalModuleId, ModuleDefId, ModuleId, AstId, BlockId, BlockLoc, LocalModuleId, ModuleDefId, ModuleId,
}; };
/// Contains all top-level defs from a macro-expanded crate /// Contains all top-level defs from a macro-expanded crate
#[derive(Debug, PartialEq, Eq)] #[derive(Debug, PartialEq, Eq)]
pub struct DefMap { pub struct DefMap {
_c: Count<Self>, _c: Count<Self>,
parent: Option<Arc<DefMap>>, block: Option<BlockInfo>,
root: LocalModuleId, root: LocalModuleId,
modules: Arena<ModuleData>, modules: Arena<ModuleData>,
krate: CrateId, krate: CrateId,
@ -91,6 +91,13 @@ pub struct DefMap {
diagnostics: Vec<DefDiagnostic>, diagnostics: Vec<DefDiagnostic>,
} }
#[derive(Debug, PartialEq, Eq)]
struct BlockInfo {
block: BlockId,
parent: Arc<DefMap>,
parent_module: LocalModuleId,
}
impl std::ops::Index<LocalModuleId> for DefMap { impl std::ops::Index<LocalModuleId> for DefMap {
type Output = ModuleData; type Output = ModuleData;
fn index(&self, id: LocalModuleId) -> &ModuleData { fn index(&self, id: LocalModuleId) -> &ModuleData {
@ -190,15 +197,12 @@ impl DefMap {
Arc::new(def_map) Arc::new(def_map)
} }
pub(crate) fn block_def_map_query( pub(crate) fn block_def_map_query(db: &dyn DefDatabase, block_id: BlockId) -> Arc<DefMap> {
db: &dyn DefDatabase, let block: BlockLoc = db.lookup_intern_block(block_id);
krate: CrateId, let item_tree = db.item_tree(block.ast_id.file_id);
block: AstId<ast::BlockExpr>, let block_items = item_tree.inner_items_of_block(block.ast_id.value);
) -> Arc<DefMap> {
let item_tree = db.item_tree(block.file_id);
let block_items = item_tree.inner_items_of_block(block.value);
let parent = parent_def_map(db, krate, block); let parent = block.module.def_map(db);
if block_items.is_empty() { if block_items.is_empty() {
// If there are no inner items, nothing new is brought into scope, so we can just return // If there are no inner items, nothing new is brought into scope, so we can just return
@ -206,10 +210,13 @@ impl DefMap {
return parent; return parent;
} }
let mut def_map = DefMap::empty(krate, parent.edition); let block_info =
def_map.parent = Some(parent); BlockInfo { block: block_id, parent, parent_module: block.module.local_id };
let def_map = collector::collect_defs(db, def_map, Some(block.value)); let mut def_map = DefMap::empty(block.module.krate, block_info.parent.edition);
def_map.block = Some(block_info);
let def_map = collector::collect_defs(db, def_map, Some(block.ast_id.value));
Arc::new(def_map) Arc::new(def_map)
} }
@ -218,7 +225,7 @@ impl DefMap {
let root = modules.alloc(ModuleData::default()); let root = modules.alloc(ModuleData::default());
DefMap { DefMap {
_c: Count::new(), _c: Count::new(),
parent: None, block: None,
krate, krate,
edition, edition,
extern_prelude: FxHashMap::default(), extern_prelude: FxHashMap::default(),
@ -266,7 +273,8 @@ impl DefMap {
} }
pub fn module_id(&self, local_id: LocalModuleId) -> ModuleId { pub fn module_id(&self, local_id: LocalModuleId) -> ModuleId {
ModuleId { krate: self.krate, local_id } let block = self.block.as_ref().map(|b| b.block);
ModuleId { krate: self.krate, local_id, block }
} }
pub(crate) fn resolve_path( pub(crate) fn resolve_path(
@ -286,9 +294,9 @@ impl DefMap {
pub fn dump(&self) -> String { pub fn dump(&self) -> String {
let mut buf = String::new(); let mut buf = String::new();
let mut current_map = self; let mut current_map = self;
while let Some(parent) = &current_map.parent { while let Some(block) = &current_map.block {
go(&mut buf, current_map, "block scope", current_map.root); go(&mut buf, current_map, "block scope", current_map.root);
current_map = &**parent; current_map = &*block.parent;
} }
go(&mut buf, current_map, "crate", current_map.root); go(&mut buf, current_map, "crate", current_map.root);
return buf; return buf;
@ -342,35 +350,6 @@ impl ModuleData {
} }
} }
fn parent_def_map(
db: &dyn DefDatabase,
krate: CrateId,
block: AstId<ast::BlockExpr>,
) -> Arc<DefMap> {
// FIXME: store this info in the item tree instead of reparsing here
let ast_id_map = db.ast_id_map(block.file_id);
let block_ptr = ast_id_map.get(block.value);
let root = match db.parse_or_expand(block.file_id) {
Some(it) => it,
None => {
return Arc::new(DefMap::empty(krate, Edition::Edition2018));
}
};
let ast = block_ptr.to_node(&root);
for ancestor in ast.syntax().ancestors().skip(1) {
if let Some(block_expr) = ast::BlockExpr::cast(ancestor) {
let ancestor_id = ast_id_map.ast_id(&block_expr);
let ast_id = InFile::new(block.file_id, ancestor_id);
let parent_map = db.block_def_map(krate, ast_id);
return parent_map;
}
}
// No enclosing block scope, so the parent is the crate-level DefMap.
db.crate_def_map(krate)
}
#[derive(Debug, Clone, PartialEq, Eq)] #[derive(Debug, Clone, PartialEq, Eq)]
pub enum ModuleSource { pub enum ModuleSource {
SourceFile(ast::SourceFile), SourceFile(ast::SourceFile),

View file

@ -10,8 +10,6 @@
//! //!
//! `ReachedFixedPoint` signals about this. //! `ReachedFixedPoint` signals about this.
use std::iter::successors;
use base_db::Edition; use base_db::Edition;
use hir_expand::name; use hir_expand::name;
use hir_expand::name::Name; use hir_expand::name::Name;
@ -131,8 +129,8 @@ impl DefMap {
result.krate = result.krate.or(new.krate); result.krate = result.krate.or(new.krate);
result.segment_index = result.segment_index.min(new.segment_index); result.segment_index = result.segment_index.min(new.segment_index);
match &current_map.parent { match &current_map.block {
Some(map) => current_map = map, Some(block) => current_map = &block.parent,
None => return result, None => return result,
} }
} }
@ -193,14 +191,35 @@ impl DefMap {
self.resolve_name_in_module(db, original_module, &segment, prefer_module) self.resolve_name_in_module(db, original_module, &segment, prefer_module)
} }
PathKind::Super(lvl) => { PathKind::Super(lvl) => {
let m = successors(Some(original_module), |m| self.modules[*m].parent) let mut module = original_module;
.nth(lvl as usize); for i in 0..lvl {
if let Some(local_id) = m { match self.modules[module].parent {
PerNs::types(self.module_id(local_id).into(), Visibility::Public) Some(it) => module = it,
} else { None => match &self.block {
Some(block) => {
// Look up remaining path in parent `DefMap`
let new_path = ModPath {
kind: PathKind::Super(lvl - i),
segments: path.segments.clone(),
};
log::debug!("`super` path: {} -> {} in parent map", path, new_path);
return block.parent.resolve_path_fp_with_macro(
db,
mode,
block.parent_module,
&new_path,
shadow,
);
}
None => {
log::debug!("super path in root module"); log::debug!("super path in root module");
return ResolvePathResult::empty(ReachedFixedPoint::Yes); return ResolvePathResult::empty(ReachedFixedPoint::Yes);
} }
},
}
}
PerNs::types(self.module_id(module).into(), Visibility::Public)
} }
PathKind::Abs => { PathKind::Abs => {
// 2018-style absolute path -- only extern prelude // 2018-style absolute path -- only extern prelude

View file

@ -8,12 +8,12 @@ mod block;
use std::sync::Arc; use std::sync::Arc;
use base_db::{fixture::WithFixture, SourceDatabase}; use base_db::{fixture::WithFixture, FilePosition, SourceDatabase};
use expect_test::{expect, Expect}; use expect_test::{expect, Expect};
use hir_expand::db::AstDatabase; use syntax::AstNode;
use test_utils::mark; use test_utils::mark;
use crate::{db::DefDatabase, nameres::*, test_db::TestDB}; use crate::{db::DefDatabase, nameres::*, test_db::TestDB, Lookup};
fn compute_crate_def_map(ra_fixture: &str) -> Arc<DefMap> { fn compute_crate_def_map(ra_fixture: &str) -> Arc<DefMap> {
let db = TestDB::with_files(ra_fixture); let db = TestDB::with_files(ra_fixture);
@ -23,14 +23,58 @@ fn compute_crate_def_map(ra_fixture: &str) -> Arc<DefMap> {
fn compute_block_def_map(ra_fixture: &str) -> Arc<DefMap> { fn compute_block_def_map(ra_fixture: &str) -> Arc<DefMap> {
let (db, position) = TestDB::with_position(ra_fixture); let (db, position) = TestDB::with_position(ra_fixture);
let module = db.module_for_file(position.file_id);
let ast_map = db.ast_id_map(position.file_id.into());
let ast = db.parse(position.file_id);
let block: ast::BlockExpr =
syntax::algo::find_node_at_offset(&ast.syntax_node(), position.offset).unwrap();
let block_id = ast_map.ast_id(&block);
db.block_def_map(module.krate, InFile::new(position.file_id.into(), block_id)) // FIXME: perhaps we should make this use body lowering tests instead?
let module = db.module_for_file(position.file_id);
let mut def_map = db.crate_def_map(module.krate);
while let Some(new_def_map) = descend_def_map_at_position(&db, position, def_map.clone()) {
def_map = new_def_map;
}
// FIXME: select the right module, not the root
def_map
}
fn descend_def_map_at_position(
db: &dyn DefDatabase,
position: FilePosition,
def_map: Arc<DefMap>,
) -> Option<Arc<DefMap>> {
for (local_id, module_data) in def_map.modules() {
let mod_def = module_data.origin.definition_source(db);
let ast_map = db.ast_id_map(mod_def.file_id);
let item_tree = db.item_tree(mod_def.file_id);
let root = db.parse_or_expand(mod_def.file_id).unwrap();
for item in module_data.scope.declarations() {
match item {
ModuleDefId::FunctionId(it) => {
// Technically blocks can be inside any type (due to arrays and const generics),
// and also in const/static initializers. For tests we only really care about
// functions though.
let ast = ast_map.get(item_tree[it.lookup(db).id.value].ast_id).to_node(&root);
if ast.syntax().text_range().contains(position.offset) {
// Cursor inside function, descend into its body's DefMap.
// Note that we don't handle block *expressions* inside function bodies.
let ast_map = db.ast_id_map(position.file_id.into());
let ast_id = ast_map.ast_id(&ast.body().unwrap());
let block = BlockLoc {
ast_id: InFile::new(position.file_id.into(), ast_id),
module: def_map.module_id(local_id),
};
let block_id = db.intern_block(block);
return Some(db.block_def_map(block_id));
}
}
_ => continue,
}
}
}
None
} }
fn check(ra_fixture: &str, expect: Expect) { fn check(ra_fixture: &str, expect: Expect) {

View file

@ -95,3 +95,29 @@ fn outer() {
"#]], "#]],
); );
} }
#[test]
fn super_imports() {
check_at(
r#"
mod module {
fn f() {
use super::Struct;
$0
}
}
struct Struct {}
"#,
expect![[r#"
block scope
Struct: t
crate
Struct: t
module: t
crate::module
f: v
"#]],
);
}