mirror of
https://github.com/rust-lang/rust-analyzer.git
synced 2025-09-29 05:15:04 +00:00
Record attribute calls on assoc items in TraitData and ImplData
This commit is contained in:
parent
6cf0cadfaa
commit
6746ba5839
5 changed files with 79 additions and 27 deletions
|
@ -316,8 +316,7 @@ impl SourceToDefCtx<'_, '_> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn macro_to_def(&mut self, src: InFile<ast::Macro>) -> Option<MacroDefId> {
|
pub(super) fn macro_to_def(&mut self, src: InFile<ast::Macro>) -> Option<MacroDefId> {
|
||||||
let makro =
|
let makro = self.dyn_map(src.as_ref()).and_then(|it| it[keys::MACRO].get(&src).copied());
|
||||||
self.dyn_map(src.as_ref()).and_then(|it| it[keys::MACRO_CALL].get(&src).copied());
|
|
||||||
if let res @ Some(_) = makro {
|
if let res @ Some(_) = makro {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,20 +30,31 @@ pub trait ChildBySource {
|
||||||
impl ChildBySource for TraitId {
|
impl ChildBySource for TraitId {
|
||||||
fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) {
|
fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) {
|
||||||
let data = db.trait_data(*self);
|
let data = db.trait_data(*self);
|
||||||
// FIXME attribute macros
|
|
||||||
for &(_, item) in data.items.iter() {
|
data.attribute_calls().filter(|(ast_id, _)| ast_id.file_id == file_id).for_each(
|
||||||
|
|(ast_id, call_id)| {
|
||||||
|
let item = ast_id.with_value(ast_id.to_node(db.upcast()));
|
||||||
|
res[keys::ATTR_MACRO_CALL].insert(item, call_id);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
data.items.iter().for_each(|&(_, item)| {
|
||||||
child_by_source_assoc_items(db, res, file_id, item);
|
child_by_source_assoc_items(db, res, file_id, item);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ChildBySource for ImplId {
|
impl ChildBySource for ImplId {
|
||||||
fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) {
|
fn child_by_source_to(&self, db: &dyn DefDatabase, res: &mut DynMap, file_id: HirFileId) {
|
||||||
let data = db.impl_data(*self);
|
let data = db.impl_data(*self);
|
||||||
// FIXME attribute macros
|
data.attribute_calls().filter(|(ast_id, _)| ast_id.file_id == file_id).for_each(
|
||||||
for &item in data.items.iter() {
|
|(ast_id, call_id)| {
|
||||||
|
let item = ast_id.with_value(ast_id.to_node(db.upcast()));
|
||||||
|
res[keys::ATTR_MACRO_CALL].insert(item, call_id);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
data.items.iter().for_each(|&item| {
|
||||||
child_by_source_assoc_items(db, res, file_id, item);
|
child_by_source_assoc_items(db, res, file_id, item);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,7 +108,7 @@ impl ChildBySource for ItemScope {
|
||||||
// FIXME: Do we need to add proc-macros into a PROCMACRO dynmap here?
|
// FIXME: Do we need to add proc-macros into a PROCMACRO dynmap here?
|
||||||
Either::Right(_fn) => return,
|
Either::Right(_fn) => return,
|
||||||
};
|
};
|
||||||
res[keys::MACRO_CALL].insert(src, makro);
|
res[keys::MACRO].insert(src, makro);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
self.unnamed_consts().for_each(|konst| {
|
self.unnamed_consts().for_each(|konst| {
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
use hir_expand::{name::Name, AstId, ExpandResult, InFile};
|
use hir_expand::{name::Name, AstId, ExpandResult, InFile, MacroCallId};
|
||||||
use syntax::ast;
|
use syntax::ast;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
@ -184,6 +184,7 @@ pub struct TraitData {
|
||||||
/// method calls to this trait's methods when the receiver is an array and the crate edition is
|
/// method calls to this trait's methods when the receiver is an array and the crate edition is
|
||||||
/// 2015 or 2018.
|
/// 2015 or 2018.
|
||||||
pub skip_array_during_method_dispatch: bool,
|
pub skip_array_during_method_dispatch: bool,
|
||||||
|
pub attribute_calls: Option<Box<Vec<(AstId<ast::Item>, MacroCallId)>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TraitData {
|
impl TraitData {
|
||||||
|
@ -207,7 +208,7 @@ impl TraitData {
|
||||||
.by_key("rustc_skip_array_during_method_dispatch")
|
.by_key("rustc_skip_array_during_method_dispatch")
|
||||||
.exists();
|
.exists();
|
||||||
|
|
||||||
let items = collect_items(
|
let (attribute_calls, items) = collect_items(
|
||||||
db,
|
db,
|
||||||
module_id,
|
module_id,
|
||||||
&mut expander,
|
&mut expander,
|
||||||
|
@ -216,6 +217,8 @@ impl TraitData {
|
||||||
container,
|
container,
|
||||||
100,
|
100,
|
||||||
);
|
);
|
||||||
|
let attribute_calls =
|
||||||
|
if attribute_calls.is_empty() { None } else { Some(Box::new(attribute_calls)) };
|
||||||
|
|
||||||
Arc::new(TraitData {
|
Arc::new(TraitData {
|
||||||
name,
|
name,
|
||||||
|
@ -224,6 +227,7 @@ impl TraitData {
|
||||||
is_unsafe,
|
is_unsafe,
|
||||||
visibility,
|
visibility,
|
||||||
skip_array_during_method_dispatch,
|
skip_array_during_method_dispatch,
|
||||||
|
attribute_calls,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -247,6 +251,10 @@ impl TraitData {
|
||||||
_ => None,
|
_ => None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn attribute_calls(&self) -> impl Iterator<Item = (AstId<ast::Item>, MacroCallId)> + '_ {
|
||||||
|
self.attribute_calls.iter().flat_map(|it| it.iter()).copied()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
|
@ -255,6 +263,7 @@ pub struct ImplData {
|
||||||
pub self_ty: Interned<TypeRef>,
|
pub self_ty: Interned<TypeRef>,
|
||||||
pub items: Vec<AssocItemId>,
|
pub items: Vec<AssocItemId>,
|
||||||
pub is_negative: bool,
|
pub is_negative: bool,
|
||||||
|
pub attribute_calls: Option<Box<Vec<(AstId<ast::Item>, MacroCallId)>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ImplData {
|
impl ImplData {
|
||||||
|
@ -271,7 +280,7 @@ impl ImplData {
|
||||||
let container = ItemContainerId::ImplId(id);
|
let container = ItemContainerId::ImplId(id);
|
||||||
let mut expander = Expander::new(db, impl_loc.id.file_id(), module_id);
|
let mut expander = Expander::new(db, impl_loc.id.file_id(), module_id);
|
||||||
|
|
||||||
let items = collect_items(
|
let (attribute_calls, items) = collect_items(
|
||||||
db,
|
db,
|
||||||
module_id,
|
module_id,
|
||||||
&mut expander,
|
&mut expander,
|
||||||
|
@ -281,8 +290,14 @@ impl ImplData {
|
||||||
100,
|
100,
|
||||||
);
|
);
|
||||||
let items = items.into_iter().map(|(_, item)| item).collect();
|
let items = items.into_iter().map(|(_, item)| item).collect();
|
||||||
|
let attribute_calls =
|
||||||
|
if attribute_calls.is_empty() { None } else { Some(Box::new(attribute_calls)) };
|
||||||
|
|
||||||
Arc::new(ImplData { target_trait, self_ty, items, is_negative })
|
Arc::new(ImplData { target_trait, self_ty, items, is_negative, attribute_calls })
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn attribute_calls(&self) -> impl Iterator<Item = (AstId<ast::Item>, MacroCallId)> + '_ {
|
||||||
|
self.attribute_calls.iter().flat_map(|it| it.iter()).copied()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -341,9 +356,9 @@ fn collect_items(
|
||||||
tree_id: item_tree::TreeId,
|
tree_id: item_tree::TreeId,
|
||||||
container: ItemContainerId,
|
container: ItemContainerId,
|
||||||
limit: usize,
|
limit: usize,
|
||||||
) -> Vec<(Name, AssocItemId)> {
|
) -> (Vec<(AstId<ast::Item>, MacroCallId)>, Vec<(Name, AssocItemId)>) {
|
||||||
if limit == 0 {
|
if limit == 0 {
|
||||||
return Vec::new();
|
return Default::default();
|
||||||
}
|
}
|
||||||
|
|
||||||
let item_tree = tree_id.item_tree(db);
|
let item_tree = tree_id.item_tree(db);
|
||||||
|
@ -352,22 +367,25 @@ fn collect_items(
|
||||||
let def_map = module.def_map(db);
|
let def_map = module.def_map(db);
|
||||||
|
|
||||||
let mut items = Vec::new();
|
let mut items = Vec::new();
|
||||||
|
let mut attribute_calls = Vec::new();
|
||||||
|
|
||||||
'items: for item in assoc_items {
|
'items: for item in assoc_items {
|
||||||
let attrs = item_tree.attrs(db, module.krate, ModItem::from(item).into());
|
let attrs = item_tree.attrs(db, module.krate, ModItem::from(item).into());
|
||||||
if !attrs.is_cfg_enabled(cfg_options) {
|
if !attrs.is_cfg_enabled(cfg_options) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
for attr in &*attrs {
|
for attr in &*attrs {
|
||||||
let ast_id = AstIdWithPath {
|
let ast_id = AstId::new(expander.current_file_id(), item.ast_id(&item_tree).upcast());
|
||||||
path: (*attr.path).clone(),
|
let ast_id_with_path = AstIdWithPath { path: (*attr.path).clone(), ast_id };
|
||||||
ast_id: AstId::new(expander.current_file_id(), item.ast_id(&item_tree).upcast()),
|
|
||||||
};
|
|
||||||
if let Ok(ResolvedAttr::Macro(call_id)) =
|
if let Ok(ResolvedAttr::Macro(call_id)) =
|
||||||
def_map.resolve_attr_macro(db, module.local_id, ast_id, attr)
|
def_map.resolve_attr_macro(db, module.local_id, ast_id_with_path, attr)
|
||||||
{
|
{
|
||||||
|
attribute_calls.push((ast_id, call_id));
|
||||||
let res = expander.enter_expand_id(db, call_id);
|
let res = expander.enter_expand_id(db, call_id);
|
||||||
items.extend(collect_macro_items(db, module, expander, container, limit, res));
|
let (mac_attrs, mac_items) =
|
||||||
|
collect_macro_items(db, module, expander, container, limit, res);
|
||||||
|
items.extend(mac_items);
|
||||||
|
attribute_calls.extend(mac_attrs);
|
||||||
continue 'items;
|
continue 'items;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -401,13 +419,16 @@ fn collect_items(
|
||||||
let res = expander.enter_expand(db, call);
|
let res = expander.enter_expand(db, call);
|
||||||
|
|
||||||
if let Ok(res) = res {
|
if let Ok(res) = res {
|
||||||
items.extend(collect_macro_items(db, module, expander, container, limit, res));
|
let (mac_attrs, mac_items) =
|
||||||
|
collect_macro_items(db, module, expander, container, limit, res);
|
||||||
|
items.extend(mac_items);
|
||||||
|
attribute_calls.extend(mac_attrs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
items
|
(attribute_calls, items)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn collect_macro_items(
|
fn collect_macro_items(
|
||||||
|
@ -417,7 +438,7 @@ fn collect_macro_items(
|
||||||
container: ItemContainerId,
|
container: ItemContainerId,
|
||||||
limit: usize,
|
limit: usize,
|
||||||
res: ExpandResult<Option<(Mark, ast::MacroItems)>>,
|
res: ExpandResult<Option<(Mark, ast::MacroItems)>>,
|
||||||
) -> Vec<(Name, AssocItemId)> {
|
) -> (Vec<(AstId<ast::Item>, MacroCallId)>, Vec<(Name, AssocItemId)>) {
|
||||||
if let Some((mark, mac)) = res.value {
|
if let Some((mark, mac)) = res.value {
|
||||||
let src: InFile<ast::MacroItems> = expander.to_source(mac);
|
let src: InFile<ast::MacroItems> = expander.to_source(mac);
|
||||||
let tree_id = item_tree::TreeId::new(src.file_id, None);
|
let tree_id = item_tree::TreeId::new(src.file_id, None);
|
||||||
|
@ -430,5 +451,5 @@ fn collect_macro_items(
|
||||||
return items;
|
return items;
|
||||||
}
|
}
|
||||||
|
|
||||||
Vec::new()
|
Default::default()
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,7 +32,7 @@ pub const TYPE_PARAM: Key<ast::TypeParam, TypeParamId> = Key::new();
|
||||||
pub const LIFETIME_PARAM: Key<ast::LifetimeParam, LifetimeParamId> = Key::new();
|
pub const LIFETIME_PARAM: Key<ast::LifetimeParam, LifetimeParamId> = Key::new();
|
||||||
pub const CONST_PARAM: Key<ast::ConstParam, ConstParamId> = Key::new();
|
pub const CONST_PARAM: Key<ast::ConstParam, ConstParamId> = Key::new();
|
||||||
|
|
||||||
pub const MACRO_CALL: Key<ast::Macro, MacroDefId> = Key::new();
|
pub const MACRO: Key<ast::Macro, MacroDefId> = Key::new();
|
||||||
pub const ATTR_MACRO_CALL: Key<ast::Item, MacroCallId> = Key::new();
|
pub const ATTR_MACRO_CALL: Key<ast::Item, MacroCallId> = Key::new();
|
||||||
pub const DERIVE_MACRO_CALL: Key<ast::Attr, (AttrId, Box<[Option<MacroCallId>]>)> = Key::new();
|
pub const DERIVE_MACRO_CALL: Key<ast::Attr, (AttrId, Box<[Option<MacroCallId>]>)> = Key::new();
|
||||||
|
|
||||||
|
|
|
@ -1514,4 +1514,25 @@ fn func$0() {
|
||||||
"#]],
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn attr_assoc_item() {
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
//- proc_macros: identity
|
||||||
|
|
||||||
|
trait Trait {
|
||||||
|
#[proc_macros::identity]
|
||||||
|
fn func() {
|
||||||
|
Self::func$0();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
func Function FileId(0) 48..87 51..55
|
||||||
|
|
||||||
|
FileId(0) 74..78
|
||||||
|
"#]],
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue