mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-10 04:08:47 +00:00
91 lines
2.7 KiB
Rust
91 lines
2.7 KiB
Rust
|
//! Post-nameres attribute resolution.
|
||
|
|
||
|
use hir_expand::MacroCallId;
|
||
|
use syntax::{ast, SmolStr};
|
||
|
|
||
|
use crate::{
|
||
|
attr::Attr,
|
||
|
attr_macro_as_call_id, builtin_attr,
|
||
|
db::DefDatabase,
|
||
|
item_scope::BuiltinShadowMode,
|
||
|
nameres::path_resolution::ResolveMode,
|
||
|
path::{ModPath, PathKind},
|
||
|
AstIdWithPath, LocalModuleId, UnresolvedMacro,
|
||
|
};
|
||
|
|
||
|
use super::DefMap;
|
||
|
|
||
|
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,
|
||
|
);
|
||
|
let def = match resolved_res.resolved_def.take_macros() {
|
||
|
Some(def) => {
|
||
|
if def.is_attribute() {
|
||
|
def
|
||
|
} else {
|
||
|
return Ok(ResolvedAttr::Other);
|
||
|
}
|
||
|
}
|
||
|
None => return Err(UnresolvedMacro { path: ast_id.path.clone() }),
|
||
|
};
|
||
|
|
||
|
Ok(ResolvedAttr::Macro(attr_macro_as_call_id(&ast_id, attr, db, self.krate, def)))
|
||
|
}
|
||
|
|
||
|
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() {
|
||
|
let name = name.to_smol_str();
|
||
|
let pred = |n: &_| *n == name;
|
||
|
|
||
|
let registered = self.registered_tools.iter().map(SmolStr::as_str);
|
||
|
let is_tool = builtin_attr::TOOL_MODULES.iter().copied().chain(registered).any(pred);
|
||
|
// FIXME: tool modules can be shadowed by actual modules
|
||
|
if is_tool {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if segments.len() == 1 {
|
||
|
let registered = self.registered_attrs.iter().map(SmolStr::as_str);
|
||
|
let is_inert = builtin_attr::INERT_ATTRIBUTES
|
||
|
.iter()
|
||
|
.map(|it| it.name)
|
||
|
.chain(registered)
|
||
|
.any(pred);
|
||
|
return is_inert;
|
||
|
}
|
||
|
}
|
||
|
false
|
||
|
}
|
||
|
}
|