Skip to content

Commit 41e0363

Browse files
committed
Auto merge of rust-lang#104602 - petrochenkov:effvisperf5, r=oli-obk
privacy: Fix more (potential) issues with effective visibilities Continuation of rust-lang#103965. See individual commits for more detailed description of the changes. The shortcuts removed in rust-lang@4eb63f6 and rust-lang@c7c7d16 could actually be correct (or correct after some tweaks), but they used global reasoning like "we can skip this update because if the code compiles then some other update should do the same thing eventually". I have some expertise in this area, but I still have doubt whether such global reasoning was correct or not, especially in presence of all possible exotic cases with imports. After this PR all table changes should be "locally correct" after every update, even if it may be overcautious. If similar optimizations are introduced again they will need detailed comments explaining why it's legal to do what they do and providing proofs. Fixes rust-lang#104249. Fixes rust-lang#104539.
2 parents af63e3b + 47cd844 commit 41e0363

File tree

6 files changed

+178
-101
lines changed

6 files changed

+178
-101
lines changed

compiler/rustc_middle/src/middle/privacy.rs

+57-50
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@
44
use crate::ty::{DefIdTree, TyCtxt, Visibility};
55
use rustc_data_structures::fx::FxHashMap;
66
use rustc_data_structures::stable_hasher::{HashStable, StableHasher};
7-
use rustc_hir::def::DefKind;
87
use rustc_macros::HashStable;
98
use rustc_query_system::ich::StableHashingContext;
109
use rustc_span::def_id::LocalDefId;
@@ -142,13 +141,13 @@ impl EffectiveVisibilities {
142141
pub fn set_public_at_level(
143142
&mut self,
144143
id: LocalDefId,
145-
default_vis: impl FnOnce() -> Visibility,
144+
lazy_private_vis: impl FnOnce() -> Visibility,
146145
level: Level,
147146
) {
148147
let mut effective_vis = self
149148
.effective_vis(id)
150149
.copied()
151-
.unwrap_or_else(|| EffectiveVisibility::from_vis(default_vis()));
150+
.unwrap_or_else(|| EffectiveVisibility::from_vis(lazy_private_vis()));
152151
for l in Level::all_levels() {
153152
if l <= level {
154153
*effective_vis.at_level_mut(l) = Visibility::Public;
@@ -185,7 +184,6 @@ impl EffectiveVisibilities {
185184
);
186185
}
187186
let nominal_vis = tcx.visibility(def_id);
188-
let def_kind = tcx.opt_def_kind(def_id);
189187
// FIXME: `rustc_privacy` is not yet updated for the new logic and can set
190188
// effective visibilities that are larger than the nominal one.
191189
if !nominal_vis.is_at_least(ev.reachable_through_impl_trait, tcx) && early {
@@ -197,15 +195,15 @@ impl EffectiveVisibilities {
197195
nominal_vis
198196
);
199197
}
200-
// Fully private items are never put into the table, this is important for performance.
201-
// FIXME: Fully private `mod` items are currently put into the table.
202-
if ev.reachable_through_impl_trait == private_vis && def_kind != Some(DefKind::Mod) {
203-
span_bug!(span, "fully private item in the table {:?}: {:?}", def_id, ev.direct);
204-
}
205198
}
206199
}
207200
}
208201

202+
pub trait IntoDefIdTree {
203+
type Tree: DefIdTree;
204+
fn tree(self) -> Self::Tree;
205+
}
206+
209207
impl<Id: Eq + Hash> EffectiveVisibilities<Id> {
210208
pub fn iter(&self) -> impl Iterator<Item = (&Id, &EffectiveVisibility)> {
211209
self.map.iter()
@@ -215,56 +213,65 @@ impl<Id: Eq + Hash> EffectiveVisibilities<Id> {
215213
self.map.get(&id)
216214
}
217215

218-
// `parent_id` is not necessarily a parent in source code tree,
219-
// it is the node from which the maximum effective visibility is inherited.
220-
pub fn update(
216+
// FIXME: Share code with `fn update`.
217+
pub fn effective_vis_or_private(
218+
&mut self,
219+
id: Id,
220+
lazy_private_vis: impl FnOnce() -> Visibility,
221+
) -> &EffectiveVisibility {
222+
self.map.entry(id).or_insert_with(|| EffectiveVisibility::from_vis(lazy_private_vis()))
223+
}
224+
225+
pub fn update<T: IntoDefIdTree>(
221226
&mut self,
222227
id: Id,
223228
nominal_vis: Visibility,
224-
default_vis: Visibility,
225-
inherited_eff_vis: Option<EffectiveVisibility>,
229+
lazy_private_vis: impl FnOnce(T) -> (Visibility, T),
230+
inherited_effective_vis: EffectiveVisibility,
226231
level: Level,
227-
tree: impl DefIdTree,
232+
mut into_tree: T,
228233
) -> bool {
229234
let mut changed = false;
230-
let mut current_effective_vis = self
231-
.map
232-
.get(&id)
233-
.copied()
234-
.unwrap_or_else(|| EffectiveVisibility::from_vis(default_vis));
235-
if let Some(inherited_effective_vis) = inherited_eff_vis {
236-
let mut inherited_effective_vis_at_prev_level =
237-
*inherited_effective_vis.at_level(level);
238-
let mut calculated_effective_vis = inherited_effective_vis_at_prev_level;
239-
for l in Level::all_levels() {
240-
if level >= l {
241-
let inherited_effective_vis_at_level = *inherited_effective_vis.at_level(l);
242-
let current_effective_vis_at_level = current_effective_vis.at_level_mut(l);
243-
// effective visibility for id shouldn't be recalculated if
244-
// inherited from parent_id effective visibility isn't changed at next level
245-
if !(inherited_effective_vis_at_prev_level == inherited_effective_vis_at_level
246-
&& level != l)
247-
{
248-
calculated_effective_vis =
249-
if nominal_vis.is_at_least(inherited_effective_vis_at_level, tree) {
250-
inherited_effective_vis_at_level
251-
} else {
252-
nominal_vis
253-
};
254-
}
255-
// effective visibility can't be decreased at next update call for the
256-
// same id
257-
if *current_effective_vis_at_level != calculated_effective_vis
258-
&& calculated_effective_vis
259-
.is_at_least(*current_effective_vis_at_level, tree)
260-
{
261-
changed = true;
262-
*current_effective_vis_at_level = calculated_effective_vis;
263-
}
264-
inherited_effective_vis_at_prev_level = inherited_effective_vis_at_level;
235+
let mut current_effective_vis = match self.map.get(&id).copied() {
236+
Some(eff_vis) => eff_vis,
237+
None => {
238+
let private_vis;
239+
(private_vis, into_tree) = lazy_private_vis(into_tree);
240+
EffectiveVisibility::from_vis(private_vis)
241+
}
242+
};
243+
let tree = into_tree.tree();
244+
245+
let mut inherited_effective_vis_at_prev_level = *inherited_effective_vis.at_level(level);
246+
let mut calculated_effective_vis = inherited_effective_vis_at_prev_level;
247+
for l in Level::all_levels() {
248+
if level >= l {
249+
let inherited_effective_vis_at_level = *inherited_effective_vis.at_level(l);
250+
let current_effective_vis_at_level = current_effective_vis.at_level_mut(l);
251+
// effective visibility for id shouldn't be recalculated if
252+
// inherited from parent_id effective visibility isn't changed at next level
253+
if !(inherited_effective_vis_at_prev_level == inherited_effective_vis_at_level
254+
&& level != l)
255+
{
256+
calculated_effective_vis =
257+
if nominal_vis.is_at_least(inherited_effective_vis_at_level, tree) {
258+
inherited_effective_vis_at_level
259+
} else {
260+
nominal_vis
261+
};
265262
}
263+
// effective visibility can't be decreased at next update call for the
264+
// same id
265+
if *current_effective_vis_at_level != calculated_effective_vis
266+
&& calculated_effective_vis.is_at_least(*current_effective_vis_at_level, tree)
267+
{
268+
changed = true;
269+
*current_effective_vis_at_level = calculated_effective_vis;
270+
}
271+
inherited_effective_vis_at_prev_level = inherited_effective_vis_at_level;
266272
}
267273
}
274+
268275
self.map.insert(id, current_effective_vis);
269276
changed
270277
}

compiler/rustc_resolve/src/effective_visibilities.rs

+71-45
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,10 @@ use rustc_ast::EnumDef;
77
use rustc_data_structures::intern::Interned;
88
use rustc_hir::def_id::LocalDefId;
99
use rustc_hir::def_id::CRATE_DEF_ID;
10-
use rustc_middle::middle::privacy::{EffectiveVisibilities, EffectiveVisibility, Level};
11-
use rustc_middle::ty::Visibility;
10+
use rustc_middle::middle::privacy::{EffectiveVisibilities, EffectiveVisibility};
11+
use rustc_middle::middle::privacy::{IntoDefIdTree, Level};
12+
use rustc_middle::ty::{DefIdTree, Visibility};
13+
use std::mem;
1214

1315
type ImportId<'a> = Interned<'a, NameBinding<'a>>;
1416

@@ -29,31 +31,71 @@ impl ParentId<'_> {
2931

3032
pub struct EffectiveVisibilitiesVisitor<'r, 'a> {
3133
r: &'r mut Resolver<'a>,
34+
def_effective_visibilities: EffectiveVisibilities,
3235
/// While walking import chains we need to track effective visibilities per-binding, and def id
3336
/// keys in `Resolver::effective_visibilities` are not enough for that, because multiple
3437
/// bindings can correspond to a single def id in imports. So we keep a separate table.
3538
import_effective_visibilities: EffectiveVisibilities<ImportId<'a>>,
39+
// It's possible to recalculate this at any point, but it's relatively expensive.
40+
current_private_vis: Visibility,
3641
changed: bool,
3742
}
3843

44+
impl Resolver<'_> {
45+
fn nearest_normal_mod(&mut self, def_id: LocalDefId) -> LocalDefId {
46+
self.get_nearest_non_block_module(def_id.to_def_id()).nearest_parent_mod().expect_local()
47+
}
48+
49+
fn private_vis_import(&mut self, binding: ImportId<'_>) -> Visibility {
50+
let NameBindingKind::Import { import, .. } = binding.kind else { unreachable!() };
51+
Visibility::Restricted(
52+
import
53+
.id()
54+
.map(|id| self.nearest_normal_mod(self.local_def_id(id)))
55+
.unwrap_or(CRATE_DEF_ID),
56+
)
57+
}
58+
59+
fn private_vis_def(&mut self, def_id: LocalDefId) -> Visibility {
60+
// For mod items `nearest_normal_mod` returns its argument, but we actually need its parent.
61+
let normal_mod_id = self.nearest_normal_mod(def_id);
62+
if normal_mod_id == def_id {
63+
self.opt_local_parent(def_id).map_or(Visibility::Public, Visibility::Restricted)
64+
} else {
65+
Visibility::Restricted(normal_mod_id)
66+
}
67+
}
68+
}
69+
70+
impl<'a, 'b> IntoDefIdTree for &'b mut Resolver<'a> {
71+
type Tree = &'b Resolver<'a>;
72+
fn tree(self) -> Self::Tree {
73+
self
74+
}
75+
}
76+
3977
impl<'r, 'a> EffectiveVisibilitiesVisitor<'r, 'a> {
4078
/// Fills the `Resolver::effective_visibilities` table with public & exported items
4179
/// For now, this doesn't resolve macros (FIXME) and cannot resolve Impl, as we
4280
/// need access to a TyCtxt for that.
4381
pub fn compute_effective_visibilities<'c>(r: &'r mut Resolver<'a>, krate: &'c Crate) {
4482
let mut visitor = EffectiveVisibilitiesVisitor {
4583
r,
84+
def_effective_visibilities: Default::default(),
4685
import_effective_visibilities: Default::default(),
86+
current_private_vis: Visibility::Public,
4787
changed: false,
4888
};
4989

5090
visitor.update(CRATE_DEF_ID, CRATE_DEF_ID);
91+
visitor.current_private_vis = Visibility::Restricted(CRATE_DEF_ID);
5192
visitor.set_bindings_effective_visibilities(CRATE_DEF_ID);
5293

5394
while visitor.changed {
5495
visitor.changed = false;
5596
visit::walk_crate(&mut visitor, krate);
5697
}
98+
visitor.r.effective_visibilities = visitor.def_effective_visibilities;
5799

58100
// Update visibilities for import def ids. These are not used during the
59101
// `EffectiveVisibilitiesVisitor` pass, because we have more detailed binding-based
@@ -90,10 +132,6 @@ impl<'r, 'a> EffectiveVisibilitiesVisitor<'r, 'a> {
90132
info!("resolve::effective_visibilities: {:#?}", r.effective_visibilities);
91133
}
92134

93-
fn nearest_normal_mod(&mut self, def_id: LocalDefId) -> LocalDefId {
94-
self.r.get_nearest_non_block_module(def_id.to_def_id()).nearest_parent_mod().expect_local()
95-
}
96-
97135
/// Update effective visibilities of bindings in the given module,
98136
/// including their whole reexport chains.
99137
fn set_bindings_effective_visibilities(&mut self, module_id: LocalDefId) {
@@ -122,62 +160,47 @@ impl<'r, 'a> EffectiveVisibilitiesVisitor<'r, 'a> {
122160
}
123161
}
124162

125-
fn effective_vis(&self, parent_id: ParentId<'a>) -> Option<EffectiveVisibility> {
126-
match parent_id {
127-
ParentId::Def(def_id) => self.r.effective_visibilities.effective_vis(def_id),
128-
ParentId::Import(binding) => self.import_effective_visibilities.effective_vis(binding),
129-
}
130-
.copied()
163+
fn cheap_private_vis(&self, parent_id: ParentId<'_>) -> Option<Visibility> {
164+
matches!(parent_id, ParentId::Def(_)).then_some(self.current_private_vis)
131165
}
132166

133-
/// The update is guaranteed to not change the table and we can skip it.
134-
fn is_noop_update(
135-
&self,
136-
parent_id: ParentId<'a>,
137-
nominal_vis: Visibility,
138-
default_vis: Visibility,
139-
) -> bool {
140-
nominal_vis == default_vis
141-
|| match parent_id {
142-
ParentId::Def(def_id) => self.r.visibilities[&def_id],
143-
ParentId::Import(binding) => binding.vis.expect_local(),
144-
} == default_vis
167+
fn effective_vis_or_private(&mut self, parent_id: ParentId<'a>) -> EffectiveVisibility {
168+
// Private nodes are only added to the table for caching, they could be added or removed at
169+
// any moment without consequences, so we don't set `changed` to true when adding them.
170+
*match parent_id {
171+
ParentId::Def(def_id) => self
172+
.def_effective_visibilities
173+
.effective_vis_or_private(def_id, || self.r.private_vis_def(def_id)),
174+
ParentId::Import(binding) => self
175+
.import_effective_visibilities
176+
.effective_vis_or_private(binding, || self.r.private_vis_import(binding)),
177+
}
145178
}
146179

147180
fn update_import(&mut self, binding: ImportId<'a>, parent_id: ParentId<'a>) {
148-
let NameBindingKind::Import { import, .. } = binding.kind else { unreachable!() };
149181
let nominal_vis = binding.vis.expect_local();
150-
let default_vis = Visibility::Restricted(
151-
import
152-
.id()
153-
.map(|id| self.nearest_normal_mod(self.r.local_def_id(id)))
154-
.unwrap_or(CRATE_DEF_ID),
155-
);
156-
if self.is_noop_update(parent_id, nominal_vis, default_vis) {
157-
return;
158-
}
182+
let private_vis = self.cheap_private_vis(parent_id);
183+
let inherited_eff_vis = self.effective_vis_or_private(parent_id);
159184
self.changed |= self.import_effective_visibilities.update(
160185
binding,
161186
nominal_vis,
162-
default_vis,
163-
self.effective_vis(parent_id),
187+
|r| (private_vis.unwrap_or_else(|| r.private_vis_import(binding)), r),
188+
inherited_eff_vis,
164189
parent_id.level(),
165-
ResolverTree(&self.r.definitions, &self.r.crate_loader),
190+
&mut *self.r,
166191
);
167192
}
168193

169194
fn update_def(&mut self, def_id: LocalDefId, nominal_vis: Visibility, parent_id: ParentId<'a>) {
170-
let default_vis = Visibility::Restricted(self.nearest_normal_mod(def_id));
171-
if self.is_noop_update(parent_id, nominal_vis, default_vis) {
172-
return;
173-
}
174-
self.changed |= self.r.effective_visibilities.update(
195+
let private_vis = self.cheap_private_vis(parent_id);
196+
let inherited_eff_vis = self.effective_vis_or_private(parent_id);
197+
self.changed |= self.def_effective_visibilities.update(
175198
def_id,
176199
nominal_vis,
177-
if def_id == CRATE_DEF_ID { Visibility::Public } else { default_vis },
178-
self.effective_vis(parent_id),
200+
|r| (private_vis.unwrap_or_else(|| r.private_vis_def(def_id)), r),
201+
inherited_eff_vis,
179202
parent_id.level(),
180-
ResolverTree(&self.r.definitions, &self.r.crate_loader),
203+
&mut *self.r,
181204
);
182205
}
183206

@@ -201,8 +224,11 @@ impl<'r, 'ast> Visitor<'ast> for EffectiveVisibilitiesVisitor<'ast, 'r> {
201224
),
202225

203226
ast::ItemKind::Mod(..) => {
227+
let prev_private_vis =
228+
mem::replace(&mut self.current_private_vis, Visibility::Restricted(def_id));
204229
self.set_bindings_effective_visibilities(def_id);
205230
visit::walk_item(self, item);
231+
self.current_private_vis = prev_private_vis;
206232
}
207233

208234
ast::ItemKind::Enum(EnumDef { ref variants }, _) => {

src/test/ui/privacy/effective_visibilities.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,13 @@ mod outer { //~ ERROR Direct: pub(crate), Reexported: pub(crate), Reachable: pub
1717
}
1818

1919
#[rustc_effective_visibility]
20-
struct PrivStruct; //~ ERROR not in the table
21-
//~| ERROR not in the table
20+
struct PrivStruct; //~ ERROR Direct: pub(self), Reexported: pub(self), Reachable: pub(self), ReachableThroughImplTrait: pub(self)
21+
//~| ERROR Direct: pub(self), Reexported: pub(self), Reachable: pub(self), ReachableThroughImplTrait: pub(self)
2222

2323
#[rustc_effective_visibility]
2424
pub union PubUnion { //~ ERROR Direct: pub(crate), Reexported: pub, Reachable: pub, ReachableThroughImplTrait: pub
2525
#[rustc_effective_visibility]
26-
a: u8, //~ ERROR not in the table
26+
a: u8, //~ ERROR Direct: pub(self), Reexported: pub(self), Reachable: pub(self), ReachableThroughImplTrait: pub(self)
2727
#[rustc_effective_visibility]
2828
pub b: u8, //~ ERROR Direct: pub(crate), Reexported: pub, Reachable: pub, ReachableThroughImplTrait: pub
2929
}

src/test/ui/privacy/effective_visibilities.stderr

+3-3
Original file line numberDiff line numberDiff line change
@@ -22,13 +22,13 @@ error: Direct: pub(crate), Reexported: pub, Reachable: pub, ReachableThroughImpl
2222
LL | pub trait PubTrait {
2323
| ^^^^^^^^^^^^^^^^^^
2424

25-
error: not in the table
25+
error: Direct: pub(self), Reexported: pub(self), Reachable: pub(self), ReachableThroughImplTrait: pub(self)
2626
--> $DIR/effective_visibilities.rs:20:9
2727
|
2828
LL | struct PrivStruct;
2929
| ^^^^^^^^^^^^^^^^^
3030

31-
error: not in the table
31+
error: Direct: pub(self), Reexported: pub(self), Reachable: pub(self), ReachableThroughImplTrait: pub(self)
3232
--> $DIR/effective_visibilities.rs:20:9
3333
|
3434
LL | struct PrivStruct;
@@ -40,7 +40,7 @@ error: Direct: pub(crate), Reexported: pub, Reachable: pub, ReachableThroughImpl
4040
LL | pub union PubUnion {
4141
| ^^^^^^^^^^^^^^^^^^
4242

43-
error: not in the table
43+
error: Direct: pub(self), Reexported: pub(self), Reachable: pub(self), ReachableThroughImplTrait: pub(self)
4444
--> $DIR/effective_visibilities.rs:26:13
4545
|
4646
LL | a: u8,

0 commit comments

Comments
 (0)