From cc0dc8dddaddc3544aab94aa4a0d9b4f37c544d7 Mon Sep 17 00:00:00 2001 From: schvv31n Date: Tue, 30 Jul 2024 12:27:15 +0100 Subject: [PATCH] rustdoc-json: discard non-local inherent impls --- src/librustdoc/formats/cache.rs | 26 +++++++++++-------- src/librustdoc/formats/mod.rs | 8 +++--- src/librustdoc/html/render/print_item.rs | 5 ++-- src/librustdoc/html/render/sidebar.rs | 2 +- .../primitives/no_primitive_impl_pollution.rs | 2 ++ 5 files changed, 25 insertions(+), 18 deletions(-) create mode 100644 tests/rustdoc-json/primitives/no_primitive_impl_pollution.rs diff --git a/src/librustdoc/formats/cache.rs b/src/librustdoc/formats/cache.rs index 9f284486616a0..975c7d30f940c 100644 --- a/src/librustdoc/formats/cache.rs +++ b/src/librustdoc/formats/cache.rs @@ -442,16 +442,16 @@ impl<'a, 'tcx> DocFolder for CacheBuilder<'a, 'tcx> { // `public_items` map, so we can skip inserting into the // paths map if there was already an entry present and we're // not a public item. - if !self.cache.paths.contains_key(&item.item_id.expect_def_id()) + let item_def_id = item.item_id.expect_def_id(); + if !self.cache.paths.contains_key(&item_def_id) || self .cache .effective_visibilities - .is_directly_public(self.tcx, item.item_id.expect_def_id()) + .is_directly_public(self.tcx, item_def_id) { - self.cache.paths.insert( - item.item_id.expect_def_id(), - (self.cache.stack.clone(), item.type_()), - ); + self.cache + .paths + .insert(item_def_id, (self.cache.stack.clone(), item.type_())); } } } @@ -543,9 +543,14 @@ impl<'a, 'tcx> DocFolder for CacheBuilder<'a, 'tcx> { } } let impl_item = Impl { impl_item: item }; - if impl_item.trait_did().map_or(true, |d| self.cache.traits.contains_key(&d)) { + let impl_def_id = impl_item.def_id(); + let impl_trait_id = impl_item.trait_did(); + if match impl_trait_id { + Some(d) => self.cache.traits.contains_key(&d), + None => impl_def_id.is_local() || impl_item.is_on_local_type(self.cache, self.tcx), + } { for did in dids { - if self.impl_ids.entry(did).or_default().insert(impl_item.def_id()) { + if self.impl_ids.entry(did).or_default().insert(impl_def_id) { self.cache .impls .entry(did) @@ -553,9 +558,8 @@ impl<'a, 'tcx> DocFolder for CacheBuilder<'a, 'tcx> { .push(impl_item.clone()); } } - } else { - let trait_did = impl_item.trait_did().expect("no trait did"); - self.cache.orphan_trait_impls.push((trait_did, dids, impl_item)); + } else if let Some(impl_trait_id) = impl_trait_id { + self.cache.orphan_trait_impls.push((impl_trait_id, dids, impl_item)); } None } else { diff --git a/src/librustdoc/formats/mod.rs b/src/librustdoc/formats/mod.rs index 84adca8efa997..39924c494c6e0 100644 --- a/src/librustdoc/formats/mod.rs +++ b/src/librustdoc/formats/mod.rs @@ -2,11 +2,12 @@ pub(crate) mod cache; pub(crate) mod item_type; pub(crate) mod renderer; +use cache::Cache; pub(crate) use renderer::{run_format, FormatRenderer}; use rustc_hir::def_id::DefId; +use rustc_middle::ty::TyCtxt; use crate::clean::{self, ItemId}; -use crate::html::render::Context; /// Metadata about implementations for a type or trait. #[derive(Clone, Debug)] @@ -43,8 +44,7 @@ impl Impl { // Returns true if this is an implementation on a "local" type, meaning: // the type is in the current crate, or the type and the trait are both // re-exported by the current crate. - pub(crate) fn is_on_local_type(&self, cx: &Context<'_>) -> bool { - let cache = cx.cache(); + pub(crate) fn is_on_local_type(&self, cache: &Cache, tcx: TyCtxt<'_>) -> bool { let for_type = &self.inner_impl().for_; if let Some(for_type_did) = for_type.def_id(cache) { // The "for" type is local if it's in the paths for the current crate. @@ -67,7 +67,7 @@ impl Impl { // a foreign type. To make sure that confusion doesn't pass on to // the reader, consider all implementations in std, core, and alloc // to be on local types. - let crate_name = cx.tcx().crate_name(trait_did.krate); + let crate_name = tcx.crate_name(trait_did.krate); if matches!(crate_name.as_str(), "std" | "core" | "alloc") { return true; } diff --git a/src/librustdoc/html/render/print_item.rs b/src/librustdoc/html/render/print_item.rs index 24476e80778e1..7cab894e10437 100644 --- a/src/librustdoc/html/render/print_item.rs +++ b/src/librustdoc/html/render/print_item.rs @@ -980,8 +980,9 @@ fn item_trait(w: &mut Buffer, cx: &mut Context<'_>, it: &clean::Item, t: &clean: } } - let (local, mut foreign) = - implementors.iter().partition::, _>(|i| i.is_on_local_type(cx)); + let (local, mut foreign) = implementors + .iter() + .partition::, _>(|i| i.is_on_local_type(cx.cache(), cx.tcx())); let (mut synthetic, mut concrete): (Vec<&&Impl>, Vec<&&Impl>) = local.iter().partition(|i| i.inner_impl().kind.is_auto()); diff --git a/src/librustdoc/html/render/sidebar.rs b/src/librustdoc/html/render/sidebar.rs index 6e826446c0e0e..c8098f02894db 100644 --- a/src/librustdoc/html/render/sidebar.rs +++ b/src/librustdoc/html/render/sidebar.rs @@ -199,7 +199,7 @@ fn sidebar_trait<'a>( foreign_impls.extend( implementors .iter() - .filter(|i| !i.is_on_local_type(cx)) + .filter(|i| !i.is_on_local_type(cx.cache(), cx.tcx())) .filter_map(|i| super::extract_for_impl_name(&i.impl_item, cx)) .map(|(name, id)| Link::new(id, name)), ); diff --git a/tests/rustdoc-json/primitives/no_primitive_impl_pollution.rs b/tests/rustdoc-json/primitives/no_primitive_impl_pollution.rs new file mode 100644 index 0000000000000..25336f9a3ebf8 --- /dev/null +++ b/tests/rustdoc-json/primitives/no_primitive_impl_pollution.rs @@ -0,0 +1,2 @@ +//@ !has "$.index[*][?(@.inner=='impl')]" +fn main() {}