2020-08-25 12:57:15 +02:00
|
|
|
//! Attributes & documentation for hir types.
|
2021-08-02 14:33:09 +02:00
|
|
|
|
2023-11-24 16:38:48 +01:00
|
|
|
use base_db::FileId;
|
2020-08-25 12:56:01 +02:00
|
|
|
use hir_def::{
|
2023-09-02 16:26:48 +02:00
|
|
|
attr::AttrsWithOwner,
|
2021-08-02 14:33:09 +02:00
|
|
|
item_scope::ItemInNs,
|
2023-08-07 00:59:35 +09:00
|
|
|
path::{ModPath, Path},
|
2023-08-18 11:46:35 +02:00
|
|
|
per_ns::Namespace,
|
2023-08-07 00:59:35 +09:00
|
|
|
resolver::{HasResolver, Resolver, TypeNs},
|
2023-09-02 17:12:57 +02:00
|
|
|
AssocItemId, AttrDefId, ModuleDefId,
|
2020-08-25 12:56:01 +02:00
|
|
|
};
|
2023-11-24 16:38:48 +01:00
|
|
|
use hir_expand::{
|
|
|
|
name::Name,
|
|
|
|
span::{RealSpanMap, SpanMapRef},
|
|
|
|
};
|
2020-08-25 12:56:01 +02:00
|
|
|
use hir_ty::db::HirDatabase;
|
2021-12-28 15:48:07 +03:00
|
|
|
use syntax::{ast, AstNode};
|
2020-08-25 12:56:01 +02:00
|
|
|
|
|
|
|
use crate::{
|
2023-09-02 17:12:57 +02:00
|
|
|
Adt, AsAssocItem, AssocItem, BuiltinType, Const, ConstParam, DocLinkDef, Enum, ExternCrateDecl,
|
|
|
|
Field, Function, GenericParam, Impl, LifetimeParam, Macro, Module, ModuleDef, Static, Struct,
|
|
|
|
Trait, TraitAlias, TypeAlias, TypeParam, Union, Variant, VariantDef,
|
2020-08-25 12:56:01 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
pub trait HasAttrs {
|
2021-03-19 21:23:57 +01:00
|
|
|
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner;
|
2023-09-02 17:12:57 +02:00
|
|
|
#[doc(hidden)]
|
|
|
|
fn attr_id(self) -> AttrDefId;
|
2023-08-07 00:59:35 +09:00
|
|
|
}
|
|
|
|
|
2020-08-25 14:44:15 +02:00
|
|
|
macro_rules! impl_has_attrs {
|
|
|
|
($(($def:ident, $def_id:ident),)*) => {$(
|
|
|
|
impl HasAttrs for $def {
|
2021-03-19 21:23:57 +01:00
|
|
|
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner {
|
2020-08-25 14:44:15 +02:00
|
|
|
let def = AttrDefId::$def_id(self.into());
|
2023-04-14 12:47:33 +02:00
|
|
|
db.attrs_with_owner(def)
|
2020-08-25 14:44:15 +02:00
|
|
|
}
|
2023-09-02 17:12:57 +02:00
|
|
|
fn attr_id(self) -> AttrDefId {
|
|
|
|
AttrDefId::$def_id(self.into())
|
2020-08-26 18:56:41 +02:00
|
|
|
}
|
2020-08-25 14:44:15 +02:00
|
|
|
}
|
|
|
|
)*};
|
2020-08-25 12:56:01 +02:00
|
|
|
}
|
|
|
|
|
2020-08-25 14:44:15 +02:00
|
|
|
impl_has_attrs![
|
|
|
|
(Field, FieldId),
|
2020-12-20 10:05:24 +03:00
|
|
|
(Variant, EnumVariantId),
|
2020-08-25 14:44:15 +02:00
|
|
|
(Static, StaticId),
|
|
|
|
(Const, ConstId),
|
|
|
|
(Trait, TraitId),
|
2023-03-04 00:24:07 +09:00
|
|
|
(TraitAlias, TraitAliasId),
|
2020-08-25 14:44:15 +02:00
|
|
|
(TypeAlias, TypeAliasId),
|
2022-03-08 23:51:48 +01:00
|
|
|
(Macro, MacroId),
|
2020-08-25 14:44:15 +02:00
|
|
|
(Function, FunctionId),
|
|
|
|
(Adt, AdtId),
|
|
|
|
(Module, ModuleId),
|
2021-01-02 00:42:07 +01:00
|
|
|
(GenericParam, GenericParamId),
|
2021-03-16 18:57:47 +01:00
|
|
|
(Impl, ImplId),
|
2023-09-02 17:12:57 +02:00
|
|
|
(ExternCrateDecl, ExternCrateId),
|
2020-08-25 14:44:15 +02:00
|
|
|
];
|
|
|
|
|
2021-01-02 12:11:46 +01:00
|
|
|
macro_rules! impl_has_attrs_enum {
|
|
|
|
($($variant:ident),* for $enum:ident) => {$(
|
|
|
|
impl HasAttrs for $variant {
|
2021-03-19 21:23:57 +01:00
|
|
|
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner {
|
2021-01-02 12:11:46 +01:00
|
|
|
$enum::$variant(self).attrs(db)
|
2020-08-25 14:44:15 +02:00
|
|
|
}
|
2023-09-02 17:12:57 +02:00
|
|
|
fn attr_id(self) -> AttrDefId {
|
|
|
|
$enum::$variant(self).attr_id()
|
2020-08-26 18:56:41 +02:00
|
|
|
}
|
2020-08-25 14:44:15 +02:00
|
|
|
}
|
|
|
|
)*};
|
|
|
|
}
|
|
|
|
|
2021-01-02 12:11:46 +01:00
|
|
|
impl_has_attrs_enum![Struct, Union, Enum for Adt];
|
|
|
|
impl_has_attrs_enum![TypeParam, ConstParam, LifetimeParam for GenericParam];
|
2020-08-25 14:44:15 +02:00
|
|
|
|
2021-07-23 15:36:43 +02:00
|
|
|
impl HasAttrs for AssocItem {
|
|
|
|
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner {
|
|
|
|
match self {
|
|
|
|
AssocItem::Function(it) => it.attrs(db),
|
|
|
|
AssocItem::Const(it) => it.attrs(db),
|
|
|
|
AssocItem::TypeAlias(it) => it.attrs(db),
|
|
|
|
}
|
|
|
|
}
|
2023-09-02 17:12:57 +02:00
|
|
|
fn attr_id(self) -> AttrDefId {
|
2021-07-23 15:36:43 +02:00
|
|
|
match self {
|
2023-09-02 17:12:57 +02:00
|
|
|
AssocItem::Function(it) => it.attr_id(),
|
|
|
|
AssocItem::Const(it) => it.attr_id(),
|
|
|
|
AssocItem::TypeAlias(it) => it.attr_id(),
|
2021-07-23 15:36:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-02 17:12:57 +02:00
|
|
|
/// Resolves the item `link` points to in the scope of `def`.
|
|
|
|
pub fn resolve_doc_path_on(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: impl HasAttrs,
|
|
|
|
link: &str,
|
|
|
|
ns: Option<Namespace>,
|
|
|
|
) -> Option<DocLinkDef> {
|
|
|
|
// AttrDefId::FieldId(it) => it.parent.resolver(db.upcast()),
|
|
|
|
// AttrDefId::EnumVariantId(it) => it.parent.resolver(db.upcast()),
|
|
|
|
|
|
|
|
resolve_doc_path_on_(db, link, def.attr_id(), ns)
|
2023-08-02 11:52:55 +02:00
|
|
|
}
|
|
|
|
|
2023-09-02 17:12:57 +02:00
|
|
|
fn resolve_doc_path_on_(
|
2020-08-26 18:56:41 +02:00
|
|
|
db: &dyn HirDatabase,
|
|
|
|
link: &str,
|
2023-09-02 17:12:57 +02:00
|
|
|
attr_id: AttrDefId,
|
2020-08-26 18:56:41 +02:00
|
|
|
ns: Option<Namespace>,
|
2023-08-07 00:59:35 +09:00
|
|
|
) -> Option<DocLinkDef> {
|
2023-09-02 17:12:57 +02:00
|
|
|
let resolver = match attr_id {
|
2020-08-26 18:56:41 +02:00
|
|
|
AttrDefId::ModuleId(it) => it.resolver(db.upcast()),
|
|
|
|
AttrDefId::FieldId(it) => it.parent.resolver(db.upcast()),
|
|
|
|
AttrDefId::AdtId(it) => it.resolver(db.upcast()),
|
|
|
|
AttrDefId::FunctionId(it) => it.resolver(db.upcast()),
|
|
|
|
AttrDefId::EnumVariantId(it) => it.parent.resolver(db.upcast()),
|
|
|
|
AttrDefId::StaticId(it) => it.resolver(db.upcast()),
|
|
|
|
AttrDefId::ConstId(it) => it.resolver(db.upcast()),
|
|
|
|
AttrDefId::TraitId(it) => it.resolver(db.upcast()),
|
2023-03-04 00:24:07 +09:00
|
|
|
AttrDefId::TraitAliasId(it) => it.resolver(db.upcast()),
|
2020-08-26 18:56:41 +02:00
|
|
|
AttrDefId::TypeAliasId(it) => it.resolver(db.upcast()),
|
|
|
|
AttrDefId::ImplId(it) => it.resolver(db.upcast()),
|
2021-12-07 17:31:26 +01:00
|
|
|
AttrDefId::ExternBlockId(it) => it.resolver(db.upcast()),
|
2023-08-02 14:53:45 +02:00
|
|
|
AttrDefId::UseId(it) => it.resolver(db.upcast()),
|
2022-03-08 23:51:48 +01:00
|
|
|
AttrDefId::MacroId(it) => it.resolver(db.upcast()),
|
2023-06-15 12:28:40 +02:00
|
|
|
AttrDefId::ExternCrateId(it) => it.resolver(db.upcast()),
|
2023-09-02 17:12:57 +02:00
|
|
|
AttrDefId::GenericParamId(_) => return None,
|
2020-08-26 18:56:41 +02:00
|
|
|
};
|
2021-12-28 15:48:07 +03:00
|
|
|
|
2023-08-07 00:59:35 +09:00
|
|
|
let mut modpath = modpath_from_str(db, link)?;
|
|
|
|
|
|
|
|
let resolved = resolver.resolve_module_path_in_items(db.upcast(), &modpath);
|
|
|
|
if resolved.is_none() {
|
|
|
|
let last_name = modpath.pop_segment()?;
|
|
|
|
resolve_assoc_or_field(db, resolver, modpath, last_name, ns)
|
|
|
|
} else {
|
|
|
|
let def = match ns {
|
|
|
|
Some(Namespace::Types) => resolved.take_types(),
|
|
|
|
Some(Namespace::Values) => resolved.take_values(),
|
|
|
|
Some(Namespace::Macros) => resolved.take_macros().map(ModuleDefId::MacroId),
|
2023-08-17 12:17:41 +02:00
|
|
|
None => resolved.iter_items().next().map(|(it, _)| match it {
|
2023-08-07 00:59:35 +09:00
|
|
|
ItemInNs::Types(it) => it,
|
|
|
|
ItemInNs::Values(it) => it,
|
|
|
|
ItemInNs::Macros(it) => ModuleDefId::MacroId(it),
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
Some(DocLinkDef::ModuleDef(def?.into()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn resolve_assoc_or_field(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
resolver: Resolver,
|
|
|
|
path: ModPath,
|
|
|
|
name: Name,
|
|
|
|
ns: Option<Namespace>,
|
|
|
|
) -> Option<DocLinkDef> {
|
|
|
|
let path = Path::from_known_path_with_no_generic(path);
|
|
|
|
// FIXME: This does not handle `Self` on trait definitions, which we should resolve to the
|
|
|
|
// trait itself.
|
|
|
|
let base_def = resolver.resolve_path_in_type_ns_fully(db.upcast(), &path)?;
|
|
|
|
|
|
|
|
let ty = match base_def {
|
|
|
|
TypeNs::SelfType(id) => Impl::from(id).self_ty(db),
|
|
|
|
TypeNs::GenericParam(_) => {
|
|
|
|
// Even if this generic parameter has some trait bounds, rustdoc doesn't
|
|
|
|
// resolve `name` to trait items.
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
TypeNs::AdtId(id) | TypeNs::AdtSelfType(id) => Adt::from(id).ty(db),
|
|
|
|
TypeNs::EnumVariantId(id) => {
|
|
|
|
// Enum variants don't have path candidates.
|
|
|
|
let variant = Variant::from(id);
|
|
|
|
return resolve_field(db, variant.into(), name, ns);
|
|
|
|
}
|
|
|
|
TypeNs::TypeAliasId(id) => {
|
|
|
|
let alias = TypeAlias::from(id);
|
|
|
|
if alias.as_assoc_item(db).is_some() {
|
|
|
|
// We don't normalize associated type aliases, so we have nothing to
|
|
|
|
// resolve `name` to.
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
alias.ty(db)
|
|
|
|
}
|
|
|
|
TypeNs::BuiltinType(id) => BuiltinType::from(id).ty(db),
|
|
|
|
TypeNs::TraitId(id) => {
|
|
|
|
// Doc paths in this context may only resolve to an item of this trait
|
|
|
|
// (i.e. no items of its supertraits), so we need to handle them here
|
|
|
|
// independently of others.
|
|
|
|
return db.trait_data(id).items.iter().find(|it| it.0 == name).map(|(_, assoc_id)| {
|
|
|
|
let def = match *assoc_id {
|
|
|
|
AssocItemId::FunctionId(it) => ModuleDef::Function(it.into()),
|
|
|
|
AssocItemId::ConstId(it) => ModuleDef::Const(it.into()),
|
|
|
|
AssocItemId::TypeAliasId(it) => ModuleDef::TypeAlias(it.into()),
|
|
|
|
};
|
|
|
|
DocLinkDef::ModuleDef(def)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
TypeNs::TraitAliasId(_) => {
|
|
|
|
// XXX: Do these get resolved?
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// FIXME: Resolve associated items here, e.g. `Option::map`. Note that associated items take
|
|
|
|
// precedence over fields.
|
|
|
|
|
|
|
|
let variant_def = match ty.as_adt()? {
|
|
|
|
Adt::Struct(it) => it.into(),
|
|
|
|
Adt::Union(it) => it.into(),
|
|
|
|
Adt::Enum(_) => return None,
|
|
|
|
};
|
|
|
|
resolve_field(db, variant_def, name, ns)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn resolve_field(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: VariantDef,
|
|
|
|
name: Name,
|
|
|
|
ns: Option<Namespace>,
|
|
|
|
) -> Option<DocLinkDef> {
|
|
|
|
if let Some(Namespace::Types | Namespace::Macros) = ns {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
def.fields(db).into_iter().find(|f| f.name(db) == name).map(DocLinkDef::Field)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn modpath_from_str(db: &dyn HirDatabase, link: &str) -> Option<ModPath> {
|
|
|
|
// FIXME: this is not how we should get a mod path here.
|
|
|
|
let try_get_modpath = |link: &str| {
|
2022-12-23 13:42:58 -05:00
|
|
|
let ast_path = ast::SourceFile::parse(&format!("type T = {link};"))
|
2021-12-28 15:48:07 +03:00
|
|
|
.syntax_node()
|
|
|
|
.descendants()
|
|
|
|
.find_map(ast::Path::cast)?;
|
2023-05-11 15:55:41 +09:00
|
|
|
if ast_path.syntax().text() != link {
|
2021-12-28 15:48:07 +03:00
|
|
|
return None;
|
|
|
|
}
|
2023-11-24 16:38:48 +01:00
|
|
|
ModPath::from_src(
|
|
|
|
db.upcast(),
|
|
|
|
ast_path,
|
2023-12-06 11:53:28 +01:00
|
|
|
SpanMapRef::RealSpanMap(&RealSpanMap::absolute(FileId::BOGUS)),
|
2023-11-24 16:38:48 +01:00
|
|
|
)
|
2021-12-28 15:48:07 +03:00
|
|
|
};
|
|
|
|
|
2023-08-07 00:59:35 +09:00
|
|
|
let full = try_get_modpath(link);
|
|
|
|
if full.is_some() {
|
|
|
|
return full;
|
2021-07-19 20:36:44 +02:00
|
|
|
}
|
2023-08-07 00:59:35 +09:00
|
|
|
|
|
|
|
// Tuple field names cannot be a part of `ModPath` usually, but rustdoc can
|
|
|
|
// resolve doc paths like `TupleStruct::0`.
|
|
|
|
// FIXME: Find a better way to handle these.
|
|
|
|
let (base, maybe_tuple_field) = link.rsplit_once("::")?;
|
|
|
|
let tuple_field = Name::new_tuple_field(maybe_tuple_field.parse().ok()?);
|
|
|
|
let mut modpath = try_get_modpath(base)?;
|
|
|
|
modpath.push_segment(tuple_field);
|
|
|
|
Some(modpath)
|
2020-08-25 12:56:01 +02:00
|
|
|
}
|