Merge #7431
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:
commit
a37091d2d0
@ -2,9 +2,9 @@
|
||||
use std::sync::Arc;
|
||||
|
||||
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 syntax::{ast, SmolStr};
|
||||
use syntax::SmolStr;
|
||||
|
||||
use crate::{
|
||||
adt::{EnumData, StructData},
|
||||
@ -16,9 +16,10 @@
|
||||
item_tree::ItemTree,
|
||||
lang_item::{LangItemTarget, LangItems},
|
||||
nameres::DefMap,
|
||||
AttrDefId, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, FunctionId, FunctionLoc,
|
||||
GenericDefId, ImplId, ImplLoc, LocalEnumVariantId, LocalFieldId, StaticId, StaticLoc, StructId,
|
||||
StructLoc, TraitId, TraitLoc, TypeAliasId, TypeAliasLoc, UnionId, UnionLoc, VariantId,
|
||||
AttrDefId, BlockId, BlockLoc, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, FunctionId,
|
||||
FunctionLoc, GenericDefId, ImplId, ImplLoc, LocalEnumVariantId, LocalFieldId, StaticId,
|
||||
StaticLoc, StructId, StructLoc, TraitId, TraitLoc, TypeAliasId, TypeAliasLoc, UnionId,
|
||||
UnionLoc, VariantId,
|
||||
};
|
||||
|
||||
#[salsa::query_group(InternDatabaseStorage)]
|
||||
@ -41,6 +42,8 @@ pub trait InternDatabase: SourceDatabase {
|
||||
fn intern_type_alias(&self, loc: TypeAliasLoc) -> TypeAliasId;
|
||||
#[salsa::interned]
|
||||
fn intern_impl(&self, loc: ImplLoc) -> ImplId;
|
||||
#[salsa::interned]
|
||||
fn intern_block(&self, loc: BlockLoc) -> BlockId;
|
||||
}
|
||||
|
||||
#[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>;
|
||||
|
||||
#[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)]
|
||||
fn struct_data(&self, id: StructId) -> Arc<StructData>;
|
||||
|
@ -74,12 +74,16 @@ macro_rules! eprintln {
|
||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
||||
pub struct ModuleId {
|
||||
krate: CrateId,
|
||||
block: Option<BlockId>,
|
||||
pub local_id: LocalModuleId,
|
||||
}
|
||||
|
||||
impl ModuleId {
|
||||
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 {
|
||||
@ -230,6 +234,15 @@ pub struct FieldId {
|
||||
type ImplLoc = ItemLoc<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)]
|
||||
pub struct TypeParamId {
|
||||
pub parent: GenericDefId,
|
||||
|
@ -62,7 +62,7 @@
|
||||
use profile::Count;
|
||||
use rustc_hash::FxHashMap;
|
||||
use stdx::format_to;
|
||||
use syntax::{ast, AstNode};
|
||||
use syntax::ast;
|
||||
|
||||
use crate::{
|
||||
db::DefDatabase,
|
||||
@ -70,14 +70,14 @@
|
||||
nameres::{diagnostics::DefDiagnostic, path_resolution::ResolveMode},
|
||||
path::ModPath,
|
||||
per_ns::PerNs,
|
||||
AstId, LocalModuleId, ModuleDefId, ModuleId,
|
||||
AstId, BlockId, BlockLoc, LocalModuleId, ModuleDefId, ModuleId,
|
||||
};
|
||||
|
||||
/// Contains all top-level defs from a macro-expanded crate
|
||||
#[derive(Debug, PartialEq, Eq)]
|
||||
pub struct DefMap {
|
||||
_c: Count<Self>,
|
||||
parent: Option<Arc<DefMap>>,
|
||||
block: Option<BlockInfo>,
|
||||
root: LocalModuleId,
|
||||
modules: Arena<ModuleData>,
|
||||
krate: CrateId,
|
||||
@ -91,6 +91,13 @@ pub struct DefMap {
|
||||
diagnostics: Vec<DefDiagnostic>,
|
||||
}
|
||||
|
||||
#[derive(Debug, PartialEq, Eq)]
|
||||
struct BlockInfo {
|
||||
block: BlockId,
|
||||
parent: Arc<DefMap>,
|
||||
parent_module: LocalModuleId,
|
||||
}
|
||||
|
||||
impl std::ops::Index<LocalModuleId> for DefMap {
|
||||
type Output = ModuleData;
|
||||
fn index(&self, id: LocalModuleId) -> &ModuleData {
|
||||
@ -190,15 +197,12 @@ pub(crate) fn crate_def_map_query(db: &dyn DefDatabase, krate: CrateId) -> Arc<D
|
||||
Arc::new(def_map)
|
||||
}
|
||||
|
||||
pub(crate) fn block_def_map_query(
|
||||
db: &dyn DefDatabase,
|
||||
krate: CrateId,
|
||||
block: AstId<ast::BlockExpr>,
|
||||
) -> Arc<DefMap> {
|
||||
let item_tree = db.item_tree(block.file_id);
|
||||
let block_items = item_tree.inner_items_of_block(block.value);
|
||||
pub(crate) fn block_def_map_query(db: &dyn DefDatabase, block_id: BlockId) -> Arc<DefMap> {
|
||||
let block: BlockLoc = db.lookup_intern_block(block_id);
|
||||
let item_tree = db.item_tree(block.ast_id.file_id);
|
||||
let block_items = item_tree.inner_items_of_block(block.ast_id.value);
|
||||
|
||||
let parent = parent_def_map(db, krate, block);
|
||||
let parent = block.module.def_map(db);
|
||||
|
||||
if block_items.is_empty() {
|
||||
// If there are no inner items, nothing new is brought into scope, so we can just return
|
||||
@ -206,10 +210,13 @@ pub(crate) fn block_def_map_query(
|
||||
return parent;
|
||||
}
|
||||
|
||||
let mut def_map = DefMap::empty(krate, parent.edition);
|
||||
def_map.parent = Some(parent);
|
||||
let block_info =
|
||||
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)
|
||||
}
|
||||
|
||||
@ -218,7 +225,7 @@ fn empty(krate: CrateId, edition: Edition) -> DefMap {
|
||||
let root = modules.alloc(ModuleData::default());
|
||||
DefMap {
|
||||
_c: Count::new(),
|
||||
parent: None,
|
||||
block: None,
|
||||
krate,
|
||||
edition,
|
||||
extern_prelude: FxHashMap::default(),
|
||||
@ -266,7 +273,8 @@ pub(crate) fn extern_prelude(&self) -> impl Iterator<Item = (&Name, &ModuleDefId
|
||||
}
|
||||
|
||||
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(
|
||||
@ -286,9 +294,9 @@ pub(crate) fn resolve_path(
|
||||
pub fn dump(&self) -> String {
|
||||
let mut buf = String::new();
|
||||
let mut current_map = self;
|
||||
while let Some(parent) = ¤t_map.parent {
|
||||
while let Some(block) = ¤t_map.block {
|
||||
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);
|
||||
return buf;
|
||||
@ -342,35 +350,6 @@ pub fn declaration_source(&self, db: &dyn DefDatabase) -> Option<InFile<ast::Mod
|
||||
}
|
||||
}
|
||||
|
||||
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)]
|
||||
pub enum ModuleSource {
|
||||
SourceFile(ast::SourceFile),
|
||||
|
@ -10,8 +10,6 @@
|
||||
//!
|
||||
//! `ReachedFixedPoint` signals about this.
|
||||
|
||||
use std::iter::successors;
|
||||
|
||||
use base_db::Edition;
|
||||
use hir_expand::name;
|
||||
use hir_expand::name::Name;
|
||||
@ -131,8 +129,8 @@ pub(super) fn resolve_path_fp_with_macro(
|
||||
result.krate = result.krate.or(new.krate);
|
||||
result.segment_index = result.segment_index.min(new.segment_index);
|
||||
|
||||
match ¤t_map.parent {
|
||||
Some(map) => current_map = map,
|
||||
match ¤t_map.block {
|
||||
Some(block) => current_map = &block.parent,
|
||||
None => return result,
|
||||
}
|
||||
}
|
||||
@ -193,14 +191,35 @@ pub(super) fn resolve_path_fp_with_macro_single(
|
||||
self.resolve_name_in_module(db, original_module, &segment, prefer_module)
|
||||
}
|
||||
PathKind::Super(lvl) => {
|
||||
let m = successors(Some(original_module), |m| self.modules[*m].parent)
|
||||
.nth(lvl as usize);
|
||||
if let Some(local_id) = m {
|
||||
PerNs::types(self.module_id(local_id).into(), Visibility::Public)
|
||||
} else {
|
||||
log::debug!("super path in root module");
|
||||
return ResolvePathResult::empty(ReachedFixedPoint::Yes);
|
||||
let mut module = original_module;
|
||||
for i in 0..lvl {
|
||||
match self.modules[module].parent {
|
||||
Some(it) => module = it,
|
||||
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");
|
||||
return ResolvePathResult::empty(ReachedFixedPoint::Yes);
|
||||
}
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
PerNs::types(self.module_id(module).into(), Visibility::Public)
|
||||
}
|
||||
PathKind::Abs => {
|
||||
// 2018-style absolute path -- only extern prelude
|
||||
|
@ -8,12 +8,12 @@
|
||||
|
||||
use std::sync::Arc;
|
||||
|
||||
use base_db::{fixture::WithFixture, SourceDatabase};
|
||||
use base_db::{fixture::WithFixture, FilePosition, SourceDatabase};
|
||||
use expect_test::{expect, Expect};
|
||||
use hir_expand::db::AstDatabase;
|
||||
use syntax::AstNode;
|
||||
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> {
|
||||
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> {
|
||||
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) {
|
||||
|
@ -95,3 +95,29 @@ fn inner2() {}
|
||||
"#]],
|
||||
);
|
||||
}
|
||||
|
||||
#[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
|
||||
"#]],
|
||||
);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user