mirror of
https://github.com/rust-lang/rust-analyzer
synced 2024-12-27 05:23:24 +00:00
Derive block attributes from block item tree
This commit is contained in:
parent
05b0612051
commit
e5b23e3bc1
8 changed files with 41 additions and 27 deletions
|
@ -431,12 +431,10 @@ impl AttrsWithOwner {
|
||||||
.item_tree(db)
|
.item_tree(db)
|
||||||
.raw_attrs(AttrOwner::ModItem(definition_tree_id.value.into()))
|
.raw_attrs(AttrOwner::ModItem(definition_tree_id.value.into()))
|
||||||
.clone(),
|
.clone(),
|
||||||
ModuleOrigin::BlockExpr { block } => RawAttrs::from_attrs_owner(
|
ModuleOrigin::BlockExpr { id, block } => {
|
||||||
db.upcast(),
|
let tree = db.block_item_tree_query(id);
|
||||||
InFile::new(block.file_id, block.to_node(db.upcast()))
|
tree.raw_attrs(AttrOwner::TopLevel).clone()
|
||||||
.as_ref()
|
}
|
||||||
.map(|it| it as &dyn ast::HasAttrs),
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
AttrDefId::FieldId(it) => {
|
AttrDefId::FieldId(it) => {
|
||||||
|
|
|
@ -1100,7 +1100,9 @@ impl ExprCollector<'_> {
|
||||||
ast::Stmt::ExprStmt(es) => matches!(es.expr(), Some(ast::Expr::MacroExpr(_))),
|
ast::Stmt::ExprStmt(es) => matches!(es.expr(), Some(ast::Expr::MacroExpr(_))),
|
||||||
_ => false,
|
_ => false,
|
||||||
});
|
});
|
||||||
statement_has_item || matches!(block.tail_expr(), Some(ast::Expr::MacroExpr(_)))
|
statement_has_item
|
||||||
|
|| matches!(block.tail_expr(), Some(ast::Expr::MacroExpr(_)))
|
||||||
|
|| (block.may_carry_attributes() && block.attrs().next().is_some())
|
||||||
};
|
};
|
||||||
|
|
||||||
let block_id = if block_has_items {
|
let block_id = if block_has_items {
|
||||||
|
|
|
@ -82,6 +82,12 @@ pub trait DefDatabase: InternDatabase + ExpandDatabase + Upcast<dyn ExpandDataba
|
||||||
#[salsa::invoke(ItemTree::file_item_tree_query)]
|
#[salsa::invoke(ItemTree::file_item_tree_query)]
|
||||||
fn file_item_tree(&self, file_id: HirFileId) -> Arc<ItemTree>;
|
fn file_item_tree(&self, file_id: HirFileId) -> Arc<ItemTree>;
|
||||||
|
|
||||||
|
#[salsa::invoke(ItemTree::block_item_tree_query)]
|
||||||
|
// FIXME: Investigate memory usage increase if this were not transparent
|
||||||
|
// Also make sure to `shrink_to_fit` if you do
|
||||||
|
#[salsa::transparent]
|
||||||
|
fn block_item_tree_query(&self, block_id: BlockId) -> Arc<ItemTree>;
|
||||||
|
|
||||||
#[salsa::invoke(crate_def_map_wait)]
|
#[salsa::invoke(crate_def_map_wait)]
|
||||||
#[salsa::transparent]
|
#[salsa::transparent]
|
||||||
fn crate_def_map(&self, krate: CrateId) -> Arc<DefMap>;
|
fn crate_def_map(&self, krate: CrateId) -> Arc<DefMap>;
|
||||||
|
|
|
@ -143,6 +143,14 @@ impl ItemTree {
|
||||||
Arc::new(item_tree)
|
Arc::new(item_tree)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn block_item_tree_query(db: &dyn DefDatabase, block: BlockId) -> Arc<ItemTree> {
|
||||||
|
let loc = db.lookup_intern_block(block);
|
||||||
|
let block = loc.ast_id.to_node(db.upcast());
|
||||||
|
|
||||||
|
let ctx = lower::Ctx::new(db, loc.ast_id.file_id);
|
||||||
|
Arc::new(ctx.lower_block(&block))
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns an iterator over all items located at the top level of the `HirFileId` this
|
/// Returns an iterator over all items located at the top level of the `HirFileId` this
|
||||||
/// `ItemTree` was created from.
|
/// `ItemTree` was created from.
|
||||||
pub fn top_level_items(&self) -> &[ModItem] {
|
pub fn top_level_items(&self) -> &[ModItem] {
|
||||||
|
@ -178,13 +186,6 @@ impl ItemTree {
|
||||||
self.data.get_or_insert_with(Box::default)
|
self.data.get_or_insert_with(Box::default)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn block_item_tree(db: &dyn DefDatabase, block: BlockId) -> Arc<ItemTree> {
|
|
||||||
let loc = db.lookup_intern_block(block);
|
|
||||||
let block = loc.ast_id.to_node(db.upcast());
|
|
||||||
let ctx = lower::Ctx::new(db, loc.ast_id.file_id);
|
|
||||||
Arc::new(ctx.lower_block(&block))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn shrink_to_fit(&mut self) {
|
fn shrink_to_fit(&mut self) {
|
||||||
if let Some(data) = &mut self.data {
|
if let Some(data) = &mut self.data {
|
||||||
let ItemTreeData {
|
let ItemTreeData {
|
||||||
|
@ -382,7 +383,7 @@ impl TreeId {
|
||||||
|
|
||||||
pub(crate) fn item_tree(&self, db: &dyn DefDatabase) -> Arc<ItemTree> {
|
pub(crate) fn item_tree(&self, db: &dyn DefDatabase) -> Arc<ItemTree> {
|
||||||
match self.block {
|
match self.block {
|
||||||
Some(block) => ItemTree::block_item_tree(db, block),
|
Some(block) => ItemTree::block_item_tree_query(db, block),
|
||||||
None => db.file_item_tree(self.file),
|
None => db.file_item_tree(self.file),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,6 +77,9 @@ impl<'a> Ctx<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn lower_block(mut self, block: &ast::BlockExpr) -> ItemTree {
|
pub(super) fn lower_block(mut self, block: &ast::BlockExpr) -> ItemTree {
|
||||||
|
self.tree
|
||||||
|
.attrs
|
||||||
|
.insert(AttrOwner::TopLevel, RawAttrs::new(self.db.upcast(), block, self.hygiene()));
|
||||||
self.tree.top_level = block
|
self.tree.top_level = block
|
||||||
.statements()
|
.statements()
|
||||||
.filter_map(|stmt| match stmt {
|
.filter_map(|stmt| match stmt {
|
||||||
|
|
|
@ -227,6 +227,7 @@ pub enum ModuleOrigin {
|
||||||
},
|
},
|
||||||
/// Pseudo-module introduced by a block scope (contains only inner items).
|
/// Pseudo-module introduced by a block scope (contains only inner items).
|
||||||
BlockExpr {
|
BlockExpr {
|
||||||
|
id: BlockId,
|
||||||
block: AstId<ast::BlockExpr>,
|
block: AstId<ast::BlockExpr>,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -269,7 +270,7 @@ impl ModuleOrigin {
|
||||||
definition.file_id,
|
definition.file_id,
|
||||||
ModuleSource::Module(definition.to_node(db.upcast())),
|
ModuleSource::Module(definition.to_node(db.upcast())),
|
||||||
),
|
),
|
||||||
ModuleOrigin::BlockExpr { block } => {
|
ModuleOrigin::BlockExpr { block, .. } => {
|
||||||
InFile::new(block.file_id, ModuleSource::BlockExpr(block.to_node(db.upcast())))
|
InFile::new(block.file_id, ModuleSource::BlockExpr(block.to_node(db.upcast())))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -325,8 +326,10 @@ impl DefMap {
|
||||||
// modules declared by blocks with items. At the moment, we don't use
|
// modules declared by blocks with items. At the moment, we don't use
|
||||||
// this visibility for anything outside IDE, so that's probably OK.
|
// this visibility for anything outside IDE, so that's probably OK.
|
||||||
let visibility = Visibility::Module(ModuleId { krate, local_id, block: None });
|
let visibility = Visibility::Module(ModuleId { krate, local_id, block: None });
|
||||||
let module_data =
|
let module_data = ModuleData::new(
|
||||||
ModuleData::new(ModuleOrigin::BlockExpr { block: block.ast_id }, visibility);
|
ModuleOrigin::BlockExpr { block: block.ast_id, id: block_id },
|
||||||
|
visibility,
|
||||||
|
);
|
||||||
|
|
||||||
let mut def_map = DefMap::empty(krate, parent_map.data.edition, module_data);
|
let mut def_map = DefMap::empty(krate, parent_map.data.edition, module_data);
|
||||||
def_map.data = parent_map.data.clone();
|
def_map.data = parent_map.data.clone();
|
||||||
|
@ -643,7 +646,7 @@ impl ModuleData {
|
||||||
definition.into()
|
definition.into()
|
||||||
}
|
}
|
||||||
ModuleOrigin::Inline { definition, .. } => definition.file_id,
|
ModuleOrigin::Inline { definition, .. } => definition.file_id,
|
||||||
ModuleOrigin::BlockExpr { block } => block.file_id,
|
ModuleOrigin::BlockExpr { block, .. } => block.file_id,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -342,14 +342,7 @@ fn inner_attributes(
|
||||||
ast::Impl(it) => it.assoc_item_list()?.syntax().clone(),
|
ast::Impl(it) => it.assoc_item_list()?.syntax().clone(),
|
||||||
ast::Module(it) => it.item_list()?.syntax().clone(),
|
ast::Module(it) => it.item_list()?.syntax().clone(),
|
||||||
ast::BlockExpr(it) => {
|
ast::BlockExpr(it) => {
|
||||||
use syntax::SyntaxKind::{BLOCK_EXPR , EXPR_STMT};
|
if !it.may_carry_attributes() {
|
||||||
// Block expressions accept outer and inner attributes, but only when they are the outer
|
|
||||||
// expression of an expression statement or the final expression of another block expression.
|
|
||||||
let may_carry_attributes = matches!(
|
|
||||||
it.syntax().parent().map(|it| it.kind()),
|
|
||||||
Some(BLOCK_EXPR | EXPR_STMT)
|
|
||||||
);
|
|
||||||
if !may_carry_attributes {
|
|
||||||
return None
|
return None
|
||||||
}
|
}
|
||||||
syntax.clone()
|
syntax.clone()
|
||||||
|
|
|
@ -61,6 +61,14 @@ impl ast::BlockExpr {
|
||||||
pub fn tail_expr(&self) -> Option<ast::Expr> {
|
pub fn tail_expr(&self) -> Option<ast::Expr> {
|
||||||
self.stmt_list()?.tail_expr()
|
self.stmt_list()?.tail_expr()
|
||||||
}
|
}
|
||||||
|
/// Block expressions accept outer and inner attributes, but only when they are the outer
|
||||||
|
/// expression of an expression statement or the final expression of another block expression.
|
||||||
|
pub fn may_carry_attributes(&self) -> bool {
|
||||||
|
matches!(
|
||||||
|
self.syntax().parent().map(|it| it.kind()),
|
||||||
|
Some(SyntaxKind::BLOCK_EXPR | SyntaxKind::EXPR_STMT)
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq, Clone)]
|
#[derive(Debug, PartialEq, Eq, Clone)]
|
||||||
|
|
Loading…
Reference in a new issue