2022-01-06 11:30:16 +00:00
|
|
|
//! Post-nameres attribute resolution.
|
|
|
|
|
2024-01-26 09:43:25 +00:00
|
|
|
use base_db::CrateId;
|
|
|
|
use hir_expand::{
|
|
|
|
attrs::{Attr, AttrId, AttrInput},
|
2024-06-04 10:31:04 +00:00
|
|
|
inert_attr_macro::find_builtin_attr_idx,
|
2024-03-21 16:50:58 +00:00
|
|
|
MacroCallId, MacroCallKind, MacroDefId,
|
2024-01-26 09:43:25 +00:00
|
|
|
};
|
2024-03-15 11:47:05 +00:00
|
|
|
use span::SyntaxContextId;
|
2024-07-16 07:59:39 +00:00
|
|
|
use syntax::ast;
|
2024-01-26 09:43:25 +00:00
|
|
|
use triomphe::Arc;
|
2022-01-06 11:30:16 +00:00
|
|
|
|
|
|
|
use crate::{
|
|
|
|
db::DefDatabase,
|
|
|
|
item_scope::BuiltinShadowMode,
|
|
|
|
nameres::path_resolution::ResolveMode,
|
2024-01-26 09:43:25 +00:00
|
|
|
path::{self, ModPath, PathKind},
|
|
|
|
AstIdWithPath, LocalModuleId, MacroId, UnresolvedMacro,
|
2022-01-06 11:30:16 +00:00
|
|
|
};
|
|
|
|
|
2023-05-11 06:52:13 +00:00
|
|
|
use super::{DefMap, MacroSubNs};
|
2022-01-06 11:30:16 +00:00
|
|
|
|
|
|
|
pub enum ResolvedAttr {
|
|
|
|
/// Attribute resolved to an attribute macro.
|
|
|
|
Macro(MacroCallId),
|
|
|
|
/// Attribute resolved to something else that does not require expansion.
|
|
|
|
Other,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DefMap {
|
|
|
|
pub(crate) fn resolve_attr_macro(
|
|
|
|
&self,
|
|
|
|
db: &dyn DefDatabase,
|
|
|
|
original_module: LocalModuleId,
|
|
|
|
ast_id: AstIdWithPath<ast::Item>,
|
|
|
|
attr: &Attr,
|
|
|
|
) -> Result<ResolvedAttr, UnresolvedMacro> {
|
|
|
|
// NB: does not currently work for derive helpers as they aren't recorded in the `DefMap`
|
|
|
|
|
|
|
|
if self.is_builtin_or_registered_attr(&ast_id.path) {
|
|
|
|
return Ok(ResolvedAttr::Other);
|
|
|
|
}
|
|
|
|
|
|
|
|
let resolved_res = self.resolve_path_fp_with_macro(
|
|
|
|
db,
|
|
|
|
ResolveMode::Other,
|
|
|
|
original_module,
|
|
|
|
&ast_id.path,
|
|
|
|
BuiltinShadowMode::Module,
|
2023-05-11 06:52:13 +00:00
|
|
|
Some(MacroSubNs::Attr),
|
2022-01-06 11:30:16 +00:00
|
|
|
);
|
|
|
|
let def = match resolved_res.resolved_def.take_macros() {
|
|
|
|
Some(def) => {
|
2023-05-11 06:52:13 +00:00
|
|
|
// `MacroSubNs` is just a hint, so the path may still resolve to a custom derive
|
|
|
|
// macro, or even function-like macro when the path is qualified.
|
2022-03-08 20:41:19 +00:00
|
|
|
if def.is_attribute(db) {
|
2022-01-06 11:30:16 +00:00
|
|
|
def
|
|
|
|
} else {
|
|
|
|
return Ok(ResolvedAttr::Other);
|
|
|
|
}
|
|
|
|
}
|
2024-06-12 09:43:36 +00:00
|
|
|
None => return Err(UnresolvedMacro { path: ast_id.path.as_ref().clone() }),
|
2022-01-06 11:30:16 +00:00
|
|
|
};
|
|
|
|
|
2022-03-08 20:41:19 +00:00
|
|
|
Ok(ResolvedAttr::Macro(attr_macro_as_call_id(
|
|
|
|
db,
|
|
|
|
&ast_id,
|
|
|
|
attr,
|
|
|
|
self.krate,
|
2023-12-21 08:18:06 +00:00
|
|
|
db.macro_def(def),
|
2022-03-08 20:41:19 +00:00
|
|
|
)))
|
2022-01-06 11:30:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn is_builtin_or_registered_attr(&self, path: &ModPath) -> bool {
|
|
|
|
if path.kind != PathKind::Plain {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let segments = path.segments();
|
|
|
|
|
|
|
|
if let Some(name) = segments.first() {
|
2024-07-16 07:59:39 +00:00
|
|
|
let name = name.symbol();
|
|
|
|
let pred = |n: &_| *n == *name;
|
2022-01-06 11:30:16 +00:00
|
|
|
|
2024-07-16 07:59:39 +00:00
|
|
|
let is_tool = self.data.registered_tools.iter().any(pred);
|
2022-01-06 11:30:16 +00:00
|
|
|
// FIXME: tool modules can be shadowed by actual modules
|
|
|
|
if is_tool {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if segments.len() == 1 {
|
2024-07-16 07:59:39 +00:00
|
|
|
if find_builtin_attr_idx(name).is_some() {
|
2024-06-04 10:31:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
2024-07-16 07:59:39 +00:00
|
|
|
if self.data.registered_attrs.iter().any(pred) {
|
2024-06-04 10:31:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
2022-01-06 11:30:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
2024-03-21 16:50:58 +00:00
|
|
|
|
2024-01-26 09:43:25 +00:00
|
|
|
pub(super) fn attr_macro_as_call_id(
|
|
|
|
db: &dyn DefDatabase,
|
|
|
|
item_attr: &AstIdWithPath<ast::Item>,
|
|
|
|
macro_attr: &Attr,
|
|
|
|
krate: CrateId,
|
|
|
|
def: MacroDefId,
|
|
|
|
) -> MacroCallId {
|
|
|
|
let arg = match macro_attr.input.as_deref() {
|
|
|
|
Some(AttrInput::TokenTree(tt)) => {
|
|
|
|
let mut tt = tt.as_ref().clone();
|
2024-03-15 11:47:05 +00:00
|
|
|
tt.delimiter.kind = tt::DelimiterKind::Invisible;
|
2024-01-26 09:43:25 +00:00
|
|
|
Some(tt)
|
|
|
|
}
|
|
|
|
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
|
2024-03-13 17:05:27 +00:00
|
|
|
def.make_call(
|
2024-01-26 09:43:25 +00:00
|
|
|
db.upcast(),
|
|
|
|
krate,
|
|
|
|
MacroCallKind::Attr {
|
|
|
|
ast_id: item_attr.ast_id,
|
|
|
|
attr_args: arg.map(Arc::new),
|
|
|
|
invoc_attr_index: macro_attr.id,
|
|
|
|
},
|
2024-03-15 11:47:05 +00:00
|
|
|
macro_attr.ctxt,
|
2024-01-26 09:43:25 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn derive_macro_as_call_id(
|
|
|
|
db: &dyn DefDatabase,
|
|
|
|
item_attr: &AstIdWithPath<ast::Adt>,
|
|
|
|
derive_attr_index: AttrId,
|
|
|
|
derive_pos: u32,
|
2024-03-15 11:47:05 +00:00
|
|
|
call_site: SyntaxContextId,
|
2024-01-26 09:43:25 +00:00
|
|
|
krate: CrateId,
|
2024-06-12 09:25:19 +00:00
|
|
|
resolver: impl Fn(&path::ModPath) -> Option<(MacroId, MacroDefId)>,
|
2024-03-14 12:52:13 +00:00
|
|
|
derive_macro_id: MacroCallId,
|
2024-01-26 09:43:25 +00:00
|
|
|
) -> Result<(MacroId, MacroDefId, MacroCallId), UnresolvedMacro> {
|
2024-06-12 09:25:19 +00:00
|
|
|
let (macro_id, def_id) = resolver(&item_attr.path)
|
2024-01-26 09:43:25 +00:00
|
|
|
.filter(|(_, def_id)| def_id.is_derive())
|
2024-06-12 09:43:36 +00:00
|
|
|
.ok_or_else(|| UnresolvedMacro { path: item_attr.path.as_ref().clone() })?;
|
2024-03-13 17:05:27 +00:00
|
|
|
let call_id = def_id.make_call(
|
2024-01-26 09:43:25 +00:00
|
|
|
db.upcast(),
|
|
|
|
krate,
|
|
|
|
MacroCallKind::Derive {
|
|
|
|
ast_id: item_attr.ast_id,
|
|
|
|
derive_index: derive_pos,
|
|
|
|
derive_attr_index,
|
2024-03-14 12:52:13 +00:00
|
|
|
derive_macro_id,
|
2024-01-26 09:43:25 +00:00
|
|
|
},
|
|
|
|
call_site,
|
|
|
|
);
|
|
|
|
Ok((macro_id, def_id, call_id))
|
|
|
|
}
|