From 47266bacf1585ca290cd3baca214497adc200803 Mon Sep 17 00:00:00 2001 From: Noah Lev Date: Thu, 11 Nov 2021 15:05:25 -0800 Subject: [PATCH 1/4] Return the actual `DefId` for assoc. items in `register_res` Before, if `register_res` were called on an associated item or enum variant, it would return the parent's `DefId`. Now, it returns the actual `DefId`. This change is a step toward removing `Type::ResolvedPath.did` and potentially removing `kind_side_channel` in rustdoc. It also just simplifies rustdoc's behavior. --- src/librustdoc/clean/utils.rs | 12 ++---------- src/librustdoc/html/format.rs | 13 ++++++++++++- 2 files changed, 14 insertions(+), 11 deletions(-) diff --git a/src/librustdoc/clean/utils.rs b/src/librustdoc/clean/utils.rs index 11032211236..78643841a0b 100644 --- a/src/librustdoc/clean/utils.rs +++ b/src/librustdoc/clean/utils.rs @@ -393,20 +393,12 @@ crate fn register_res(cx: &mut DocContext<'_>, res: Res) -> DefId { debug!("register_res({:?})", res); let (did, kind) = match res { - Res::Def(DefKind::AssocTy | DefKind::AssocFn | DefKind::AssocConst, i) => { - // associated items are documented, but on the page of their parent - (cx.tcx.parent(i).unwrap(), ItemType::Trait) - } - Res::Def(DefKind::Variant, i) => { - // variant items are documented, but on the page of their parent - (cx.tcx.parent(i).expect("cannot get parent def id"), ItemType::Enum) - } // Each of these have their own page. Res::Def( kind @ - (Fn | TyAlias | Enum | Trait | Struct | Union | Mod | ForeignTy | Const | Static - | Macro(..) | TraitAlias), + (AssocTy | AssocFn | AssocConst | Variant | Fn | TyAlias | Enum | Trait | Struct + | Union | Mod | ForeignTy | Const | Static | Macro(..) | TraitAlias), i, ) => (i, kind.into()), // This is part of a trait definition; document the trait. diff --git a/src/librustdoc/html/format.rs b/src/librustdoc/html/format.rs index 4f204913204..d4c5dda19f9 100644 --- a/src/librustdoc/html/format.rs +++ b/src/librustdoc/html/format.rs @@ -13,8 +13,10 @@ use rustc_attr::{ConstStability, StabilityLevel}; use rustc_data_structures::captures::Captures; use rustc_data_structures::fx::FxHashSet; use rustc_hir as hir; +use rustc_hir::def::DefKind; use rustc_hir::def_id::DefId; use rustc_middle::ty; +use rustc_middle::ty::DefIdTree; use rustc_middle::ty::TyCtxt; use rustc_span::def_id::CRATE_DEF_INDEX; use rustc_target::spec::abi::Abi; @@ -502,7 +504,16 @@ crate fn href_with_root_path( cx: &Context<'_>, root_path: Option<&str>, ) -> Result<(String, ItemType, Vec), HrefError> { - let cache = &cx.cache(); + let tcx = cx.tcx(); + let def_kind = tcx.def_kind(did); + let did = match def_kind { + DefKind::AssocTy | DefKind::AssocFn | DefKind::AssocConst | DefKind::Variant => { + // documented on their parent's page + tcx.parent(did).unwrap() + } + _ => did, + }; + let cache = cx.cache(); let relative_to = &cx.current; fn to_module_fqp(shortty: ItemType, fqp: &[String]) -> &[String] { if shortty == ItemType::Module { fqp } else { &fqp[..fqp.len() - 1] } From db4a06e9bdbfc808e60b06b357c35481a91fc8e6 Mon Sep 17 00:00:00 2001 From: Noah Lev Date: Thu, 11 Nov 2021 15:35:10 -0800 Subject: [PATCH 2/4] Use `path.def_id()` in `Type::inner_def_id()` --- src/librustdoc/clean/types.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/librustdoc/clean/types.rs b/src/librustdoc/clean/types.rs index d28f3ce8778..ed773cf4ad2 100644 --- a/src/librustdoc/clean/types.rs +++ b/src/librustdoc/clean/types.rs @@ -1522,7 +1522,7 @@ impl Type { fn inner_def_id(&self, cache: Option<&Cache>) -> Option { let t: PrimitiveType = match *self { - ResolvedPath { did, .. } => return Some(did), + ResolvedPath { ref path, did: _ } => return Some(path.def_id()), DynTrait(ref bounds, _) => return Some(bounds[0].trait_.def_id()), Primitive(p) => return cache.and_then(|c| c.primitive_locations.get(&p).cloned()), BorrowedRef { type_: box Generic(..), .. } => PrimitiveType::Reference, From 43651125eea6284554e101aaf77af6e296f0b666 Mon Sep 17 00:00:00 2001 From: Noah Lev Date: Thu, 11 Nov 2021 15:45:45 -0800 Subject: [PATCH 3/4] Remove `ResolvedPath.did` --- src/librustdoc/clean/mod.rs | 4 ++-- src/librustdoc/clean/types.rs | 4 ++-- src/librustdoc/clean/utils.rs | 7 ++++--- src/librustdoc/formats/cache.rs | 10 +++++----- src/librustdoc/html/format.rs | 3 ++- src/librustdoc/html/render/cache.rs | 2 +- src/librustdoc/html/render/mod.rs | 6 +++--- src/librustdoc/html/render/print_item.rs | 9 +++++---- src/librustdoc/json/conversions.rs | 14 +++++--------- 9 files changed, 29 insertions(+), 30 deletions(-) diff --git a/src/librustdoc/clean/mod.rs b/src/librustdoc/clean/mod.rs index de1293cd96d..5c893e44a9f 100644 --- a/src/librustdoc/clean/mod.rs +++ b/src/librustdoc/clean/mod.rs @@ -1411,12 +1411,12 @@ impl<'tcx> Clean for Ty<'tcx> { }; inline::record_extern_fqn(cx, did, kind); let path = external_path(cx, did, false, vec![], substs); - ResolvedPath { path, did } + ResolvedPath { path } } ty::Foreign(did) => { inline::record_extern_fqn(cx, did, ItemType::ForeignType); let path = external_path(cx, did, false, vec![], InternalSubsts::empty()); - ResolvedPath { path, did } + ResolvedPath { path } } ty::Dynamic(obj, ref reg) => { // HACK: pick the first `did` as the `did` of the trait object. Someone diff --git a/src/librustdoc/clean/types.rs b/src/librustdoc/clean/types.rs index ed773cf4ad2..9e088ac72ad 100644 --- a/src/librustdoc/clean/types.rs +++ b/src/librustdoc/clean/types.rs @@ -1419,7 +1419,7 @@ crate enum Type { /// A named type, which could be a trait. /// /// This is mostly Rustdoc's version of [`hir::Path`]. It has to be different because Rustdoc's [`PathSegment`] can contain cleaned generics. - ResolvedPath { path: Path, did: DefId }, + ResolvedPath { path: Path }, /// A `dyn Trait` object: `dyn for<'a> Trait<'a> + Send + 'static` DynTrait(Vec, Option), /// A type parameter. @@ -1522,7 +1522,7 @@ impl Type { fn inner_def_id(&self, cache: Option<&Cache>) -> Option { let t: PrimitiveType = match *self { - ResolvedPath { ref path, did: _ } => return Some(path.def_id()), + ResolvedPath { ref path } => return Some(path.def_id()), DynTrait(ref bounds, _) => return Some(bounds[0].trait_.def_id()), Primitive(p) => return cache.and_then(|c| c.primitive_locations.get(&p).cloned()), BorrowedRef { type_: box Generic(..), .. } => PrimitiveType::Reference, diff --git a/src/librustdoc/clean/utils.rs b/src/librustdoc/clean/utils.rs index 78643841a0b..2976dd8f65a 100644 --- a/src/librustdoc/clean/utils.rs +++ b/src/librustdoc/clean/utils.rs @@ -187,7 +187,8 @@ crate fn build_deref_target_impls(cx: &mut DocContext<'_>, items: &[Item], ret: for &did in prim.impls(tcx).iter().filter(|did| !did.is_local()) { inline::build_impl(cx, None, did, None, ret); } - } else if let ResolvedPath { did, .. } = *target { + } else if let ResolvedPath { path } = target { + let did = path.def_id(); if !did.is_local() { inline::build_impls(cx, None, did, None, ret); } @@ -360,8 +361,8 @@ crate fn resolve_type(cx: &mut DocContext<'_>, path: Path) -> Type { Res::SelfTy(..) if path.segments.len() == 1 => Generic(kw::SelfUpper), Res::Def(DefKind::TyParam, _) if path.segments.len() == 1 => Generic(path.segments[0].name), _ => { - let did = register_res(cx, path.res); - ResolvedPath { path, did } + let _ = register_res(cx, path.res); + ResolvedPath { path } } } } diff --git a/src/librustdoc/formats/cache.rs b/src/librustdoc/formats/cache.rs index db2b836de86..dd13c4809bb 100644 --- a/src/librustdoc/formats/cache.rs +++ b/src/librustdoc/formats/cache.rs @@ -401,8 +401,8 @@ impl<'a, 'tcx> DocFolder for CacheBuilder<'a, 'tcx> { clean::ImplItem(ref i) => { self.cache.parent_is_trait_impl = i.trait_.is_some(); match i.for_ { - clean::ResolvedPath { did, .. } => { - self.cache.parent_stack.push(did); + clean::ResolvedPath { ref path } => { + self.cache.parent_stack.push(path.def_id()); true } clean::DynTrait(ref bounds, _) @@ -436,9 +436,9 @@ impl<'a, 'tcx> DocFolder for CacheBuilder<'a, 'tcx> { // Note: matching twice to restrict the lifetime of the `i` borrow. let mut dids = FxHashSet::default(); match i.for_ { - clean::ResolvedPath { did, .. } - | clean::BorrowedRef { type_: box clean::ResolvedPath { did, .. }, .. } => { - dids.insert(did); + clean::ResolvedPath { ref path } + | clean::BorrowedRef { type_: box clean::ResolvedPath { ref path }, .. } => { + dids.insert(path.def_id()); } clean::DynTrait(ref bounds, _) | clean::BorrowedRef { type_: box clean::DynTrait(ref bounds, _), .. } => { diff --git a/src/librustdoc/html/format.rs b/src/librustdoc/html/format.rs index d4c5dda19f9..1ed3ba7ece0 100644 --- a/src/librustdoc/html/format.rs +++ b/src/librustdoc/html/format.rs @@ -762,8 +762,9 @@ fn fmt_type<'cx>( match *t { clean::Generic(name) => write!(f, "{}", name), - clean::ResolvedPath { did, ref path } => { + clean::ResolvedPath { ref path } => { // Paths like `T::Output` and `Self::Output` should be rendered with all segments. + let did = path.def_id(); resolved_path(f, did, path, path.is_assoc_ty(), use_absolute, cx) } clean::DynTrait(ref bounds, ref lt) => { diff --git a/src/librustdoc/html/render/cache.rs b/src/librustdoc/html/render/cache.rs index 9aa69d94215..c114edf1e70 100644 --- a/src/librustdoc/html/render/cache.rs +++ b/src/librustdoc/html/render/cache.rs @@ -371,7 +371,7 @@ crate fn get_real_types<'tcx>( let mut ty_generics = Vec::new(); for bound in bound.get_bounds().unwrap_or(&[]) { if let Some(path) = bound.get_trait_path() { - let ty = Type::ResolvedPath { did: path.def_id(), path }; + let ty = Type::ResolvedPath { path }; get_real_types(generics, &ty, tcx, recurse + 1, &mut ty_generics, cache); } } diff --git a/src/librustdoc/html/render/mod.rs b/src/librustdoc/html/render/mod.rs index ffd09663f82..690c8f59fce 100644 --- a/src/librustdoc/html/render/mod.rs +++ b/src/librustdoc/html/render/mod.rs @@ -1243,8 +1243,8 @@ fn should_render_item(item: &clean::Item, deref_mut_: bool, tcx: TyCtxt<'_>) -> | SelfTy::SelfExplicit(clean::BorrowedRef { mutability, .. }) => { (mutability == Mutability::Mut, false, false) } - SelfTy::SelfExplicit(clean::ResolvedPath { did, .. }) => { - (false, Some(did) == tcx.lang_items().owned_box(), false) + SelfTy::SelfExplicit(clean::ResolvedPath { path }) => { + (false, Some(path.def_id()) == tcx.lang_items().owned_box(), false) } SelfTy::SelfValue => (false, false, true), _ => (false, false, false), @@ -2536,7 +2536,7 @@ fn collect_paths_for_type(first_ty: clean::Type, cache: &Cache) -> Vec { } match ty { - clean::Type::ResolvedPath { did, .. } => process_path(did), + clean::Type::ResolvedPath { path } => process_path(path.def_id()), clean::Type::Tuple(tys) => { work.extend(tys.into_iter()); } diff --git a/src/librustdoc/html/render/print_item.rs b/src/librustdoc/html/render/print_item.rs index 049d17a4b47..e59b94f6b7d 100644 --- a/src/librustdoc/html/render/print_item.rs +++ b/src/librustdoc/html/render/print_item.rs @@ -727,10 +727,11 @@ fn item_trait(w: &mut Buffer, cx: &Context<'_>, it: &clean::Item, t: &clean::Tra let mut implementor_dups: FxHashMap = FxHashMap::default(); for implementor in implementors { match implementor.inner_impl().for_ { - clean::ResolvedPath { ref path, did, .. } - | clean::BorrowedRef { - type_: box clean::ResolvedPath { ref path, did, .. }, .. - } if !path.is_assoc_ty() => { + clean::ResolvedPath { ref path } + | clean::BorrowedRef { type_: box clean::ResolvedPath { ref path }, .. } + if !path.is_assoc_ty() => + { + let did = path.def_id(); let &mut (prev_did, ref mut has_duplicates) = implementor_dups.entry(path.last()).or_insert((did, false)); if prev_did != did { diff --git a/src/librustdoc/json/conversions.rs b/src/librustdoc/json/conversions.rs index a46518ef489..a5c1eb12410 100644 --- a/src/librustdoc/json/conversions.rs +++ b/src/librustdoc/json/conversions.rs @@ -365,8 +365,7 @@ impl FromWithTcx for GenericBound { match bound { TraitBound(clean::PolyTrait { trait_, generic_params }, modifier) => { // FIXME: should `trait_` be a clean::Path equivalent in JSON? - let trait_ = - clean::ResolvedPath { did: trait_.def_id(), path: trait_ }.into_tcx(tcx); + let trait_ = clean::ResolvedPath { path: trait_ }.into_tcx(tcx); GenericBound::TraitBound { trait_, generic_params: generic_params.into_iter().map(|x| x.into_tcx(tcx)).collect(), @@ -391,9 +390,9 @@ impl FromWithTcx for Type { fn from_tcx(ty: clean::Type, tcx: TyCtxt<'_>) -> Self { use clean::Type::*; match ty { - ResolvedPath { path, did } => Type::ResolvedPath { + ResolvedPath { path } => Type::ResolvedPath { name: path.whole_name(), - id: from_item_id(did.into()), + id: from_item_id(path.def_id().into()), args: path.segments.last().map(|args| Box::new(args.clone().args.into_tcx(tcx))), param_names: Vec::new(), }, @@ -436,7 +435,7 @@ impl FromWithTcx for Type { }, QPath { name, self_type, trait_, .. } => { // FIXME: should `trait_` be a clean::Path equivalent in JSON? - let trait_ = ResolvedPath { did: trait_.def_id(), path: trait_ }.into_tcx(tcx); + let trait_ = ResolvedPath { path: trait_ }.into_tcx(tcx); Type::QualifiedPath { name: name.to_string(), self_type: Box::new((*self_type).into_tcx(tcx)), @@ -502,10 +501,7 @@ impl FromWithTcx for Impl { let provided_trait_methods = impl_.provided_trait_methods(tcx); let clean::Impl { unsafety, generics, trait_, for_, items, polarity, kind } = impl_; // FIXME: should `trait_` be a clean::Path equivalent in JSON? - let trait_ = trait_.map(|path| { - let did = path.def_id(); - clean::ResolvedPath { path, did }.into_tcx(tcx) - }); + let trait_ = trait_.map(|path| clean::ResolvedPath { path }.into_tcx(tcx)); // FIXME: use something like ImplKind in JSON? let (synthetic, blanket_impl) = match kind { clean::ImplKind::Normal => (false, None), From 617bbb21f8a4f92a234b1e36d276f1a13b3411ce Mon Sep 17 00:00:00 2001 From: Noah Lev Date: Wed, 24 Nov 2021 18:32:03 -0500 Subject: [PATCH 4/4] Update comment Co-authored-by: Joshua Nelson --- src/librustdoc/clean/utils.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/librustdoc/clean/utils.rs b/src/librustdoc/clean/utils.rs index 2976dd8f65a..1141aff41f2 100644 --- a/src/librustdoc/clean/utils.rs +++ b/src/librustdoc/clean/utils.rs @@ -394,7 +394,7 @@ crate fn register_res(cx: &mut DocContext<'_>, res: Res) -> DefId { debug!("register_res({:?})", res); let (did, kind) = match res { - // Each of these have their own page. + // These should be added to the cache using `record_extern_fqn`. Res::Def( kind @