Skip to content

Commit 7b7b6b5

Browse files
committed
Test and implement reachability for trait objects and generic parameters of functions
1 parent b92af94 commit 7b7b6b5

File tree

6 files changed

+60
-6
lines changed

6 files changed

+60
-6
lines changed

Cargo.lock

+1
Original file line numberDiff line numberDiff line change
@@ -4397,6 +4397,7 @@ dependencies = [
43974397
"rustc_lexer",
43984398
"rustc_macros",
43994399
"rustc_middle",
4400+
"rustc_privacy",
44004401
"rustc_session",
44014402
"rustc_span",
44024403
"rustc_target",

compiler/rustc_passes/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ rustc_index = { path = "../rustc_index" }
1818
rustc_lexer = { path = "../rustc_lexer" }
1919
rustc_macros = { path = "../rustc_macros" }
2020
rustc_middle = { path = "../rustc_middle" }
21+
rustc_privacy = { path = "../rustc_privacy" }
2122
rustc_session = { path = "../rustc_session" }
2223
rustc_span = { path = "../rustc_span" }
2324
rustc_target = { path = "../rustc_target" }

compiler/rustc_passes/src/reachable.rs

+31-4
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,8 @@ use rustc_middle::middle::codegen_fn_attrs::{CodegenFnAttrFlags, CodegenFnAttrs}
1515
use rustc_middle::middle::privacy::{self, Level};
1616
use rustc_middle::mir::interpret::{ConstAllocation, GlobalAlloc};
1717
use rustc_middle::query::Providers;
18-
use rustc_middle::ty::{self, TyCtxt};
18+
use rustc_middle::ty::{self, ExistentialTraitRef, TyCtxt};
19+
use rustc_privacy::DefIdVisitor;
1920
use rustc_session::config::CrateType;
2021
use rustc_target::spec::abi::Abi;
2122

@@ -271,13 +272,22 @@ impl<'tcx> ReachableContext<'tcx> {
271272
self.propagate_item(Res::Def(self.tcx.def_kind(def_id), def_id))
272273
}
273274
GlobalAlloc::Function(instance) => {
275+
// Manually visit to actually see the instance's `DefId`. Type visitors won't see it
274276
self.propagate_item(Res::Def(
275277
self.tcx.def_kind(instance.def_id()),
276278
instance.def_id(),
277-
))
278-
// TODO: walk generic args
279+
));
280+
self.visit(instance.args);
281+
}
282+
GlobalAlloc::VTable(ty, trait_ref) => {
283+
self.visit(ty);
284+
// Manually visit to actually see the trait's `DefId`. Type visitors won't see it
285+
if let Some(trait_ref) = trait_ref {
286+
let ExistentialTraitRef { def_id, args } = trait_ref.skip_binder();
287+
self.visit_def_id(def_id, "", &"");
288+
self.visit(args);
289+
}
279290
}
280-
GlobalAlloc::VTable(ty, trait_ref) => todo!("{ty:?}, {trait_ref:?}"),
281291
GlobalAlloc::Memory(alloc) => self.propagate_from_alloc(root, alloc),
282292
}
283293
}
@@ -303,6 +313,23 @@ impl<'tcx> ReachableContext<'tcx> {
303313
}
304314
}
305315

316+
impl<'tcx> DefIdVisitor<'tcx> for ReachableContext<'tcx> {
317+
type Result = ();
318+
319+
fn tcx(&self) -> TyCtxt<'tcx> {
320+
self.tcx
321+
}
322+
323+
fn visit_def_id(
324+
&mut self,
325+
def_id: DefId,
326+
_kind: &str,
327+
_descr: &dyn std::fmt::Display,
328+
) -> Self::Result {
329+
self.propagate_item(Res::Def(self.tcx.def_kind(def_id), def_id))
330+
}
331+
}
332+
306333
fn check_item<'tcx>(
307334
tcx: TyCtxt<'tcx>,
308335
id: hir::ItemId,

compiler/rustc_privacy/src/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -67,7 +67,7 @@ impl<'tcx> fmt::Display for LazyDefPathStr<'tcx> {
6767
/// First, it doesn't have overridable `fn visit_trait_ref`, so we have to catch trait `DefId`s
6868
/// manually. Second, it doesn't visit some type components like signatures of fn types, or traits
6969
/// in `impl Trait`, see individual comments in `DefIdVisitorSkeleton::visit_ty`.
70-
trait DefIdVisitor<'tcx> {
70+
pub trait DefIdVisitor<'tcx> {
7171
type Result: VisitorResult = ();
7272
const SHALLOW: bool = false;
7373
const SKIP_ASSOC_TYS: bool = false;
@@ -98,7 +98,7 @@ trait DefIdVisitor<'tcx> {
9898
}
9999
}
100100

101-
struct DefIdVisitorSkeleton<'v, 'tcx, V: ?Sized> {
101+
pub struct DefIdVisitorSkeleton<'v, 'tcx, V: ?Sized> {
102102
def_id_visitor: &'v mut V,
103103
visited_opaque_tys: FxHashSet<DefId>,
104104
dummy: PhantomData<TyCtxt<'tcx>>,

tests/ui/cross-crate/auxiliary/static_init_aux.rs

+21
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
pub static V: &u32 = &X;
22
pub static F: fn() = f;
33
pub static G: fn() = G0;
4+
pub static H: &(dyn Fn() + Sync) = &h;
5+
pub static I: fn() = Helper(j).mk();
46

57
static X: u32 = 42;
68
static G0: fn() = g;
@@ -12,3 +14,22 @@ pub fn v() -> *const u32 {
1214
fn f() {}
1315

1416
fn g() {}
17+
18+
fn h() {}
19+
20+
#[derive(Copy, Clone)]
21+
struct Helper<T: Copy>(T);
22+
23+
impl<T: Copy + FnOnce()> Helper<T> {
24+
const fn mk(self) -> fn() {
25+
i::<T>
26+
}
27+
}
28+
29+
fn i<T: FnOnce()>() {
30+
assert_eq!(std::mem::size_of::<T>(), 0);
31+
// unsafe to work around the lack of a `Default` impl for function items
32+
unsafe { (std::mem::transmute_copy::<(), T>(&()))() }
33+
}
34+
35+
fn j() {}

tests/ui/cross-crate/static-init.rs

+4
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,8 @@ extern crate static_init_aux as aux;
66
static V: &u32 = aux::V;
77
static F: fn() = aux::F;
88
static G: fn() = aux::G;
9+
static H: &(dyn Fn() + Sync) = aux::H;
10+
static I: fn() = aux::I;
911

1012
fn v() -> *const u32 {
1113
V
@@ -15,4 +17,6 @@ fn main() {
1517
assert_eq!(aux::v(), crate::v());
1618
F();
1719
G();
20+
H();
21+
I();
1822
}

0 commit comments

Comments
 (0)