Auto merge of #111371 - compiler-errors:revert-110907, r=petrochenkov
Revert "Populate effective visibilities in `rustc_privacy`"
This reverts commit cff85f22f5
, cc #110907. It needs to be fixed, but there are too many issues being reported that I wanted to put up a revert until a proper fix can be committed.
Fixes a ton of issues where private but still reachable impls were missing during codegen:
Fixes #111320
Fixes #111321
Fixes #111334
Fixes #111357
Fixes #111368
Fixes #111373
Fixes #111377
Fixes #111386
Fixes #111387
`@bors` p=1
r? `@petrochenkov`
This commit is contained in:
commit
3a37c2f052
6 changed files with 176 additions and 179 deletions
|
@ -64,7 +64,7 @@ impl EffectiveVisibility {
|
||||||
self.at_level(level).is_public()
|
self.at_level(level).is_public()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const fn from_vis(vis: Visibility) -> EffectiveVisibility {
|
pub fn from_vis(vis: Visibility) -> EffectiveVisibility {
|
||||||
EffectiveVisibility {
|
EffectiveVisibility {
|
||||||
direct: vis,
|
direct: vis,
|
||||||
reexported: vis,
|
reexported: vis,
|
||||||
|
@ -72,18 +72,6 @@ impl EffectiveVisibility {
|
||||||
reachable_through_impl_trait: vis,
|
reachable_through_impl_trait: vis,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
|
||||||
pub fn min(mut self, lhs: EffectiveVisibility, tcx: TyCtxt<'_>) -> Self {
|
|
||||||
for l in Level::all_levels() {
|
|
||||||
let rhs_vis = self.at_level_mut(l);
|
|
||||||
let lhs_vis = *lhs.at_level(l);
|
|
||||||
if rhs_vis.is_at_least(lhs_vis, tcx) {
|
|
||||||
*rhs_vis = lhs_vis;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
self
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Holds a map of effective visibilities for reachable HIR nodes.
|
/// Holds a map of effective visibilities for reachable HIR nodes.
|
||||||
|
@ -149,6 +137,24 @@ impl EffectiveVisibilities {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn set_public_at_level(
|
||||||
|
&mut self,
|
||||||
|
id: LocalDefId,
|
||||||
|
lazy_private_vis: impl FnOnce() -> Visibility,
|
||||||
|
level: Level,
|
||||||
|
) {
|
||||||
|
let mut effective_vis = self
|
||||||
|
.effective_vis(id)
|
||||||
|
.copied()
|
||||||
|
.unwrap_or_else(|| EffectiveVisibility::from_vis(lazy_private_vis()));
|
||||||
|
for l in Level::all_levels() {
|
||||||
|
if l <= level {
|
||||||
|
*effective_vis.at_level_mut(l) = Visibility::Public;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.map.insert(id, effective_vis);
|
||||||
|
}
|
||||||
|
|
||||||
pub fn check_invariants(&self, tcx: TyCtxt<'_>, early: bool) {
|
pub fn check_invariants(&self, tcx: TyCtxt<'_>, early: bool) {
|
||||||
if !cfg!(debug_assertions) {
|
if !cfg!(debug_assertions) {
|
||||||
return;
|
return;
|
||||||
|
@ -213,7 +219,7 @@ impl<Id: Eq + Hash> EffectiveVisibilities<Id> {
|
||||||
pub fn update(
|
pub fn update(
|
||||||
&mut self,
|
&mut self,
|
||||||
id: Id,
|
id: Id,
|
||||||
nominal_vis: Option<Visibility>,
|
nominal_vis: Visibility,
|
||||||
lazy_private_vis: impl FnOnce() -> Visibility,
|
lazy_private_vis: impl FnOnce() -> Visibility,
|
||||||
inherited_effective_vis: EffectiveVisibility,
|
inherited_effective_vis: EffectiveVisibility,
|
||||||
level: Level,
|
level: Level,
|
||||||
|
@ -237,11 +243,12 @@ impl<Id: Eq + Hash> EffectiveVisibilities<Id> {
|
||||||
if !(inherited_effective_vis_at_prev_level == inherited_effective_vis_at_level
|
if !(inherited_effective_vis_at_prev_level == inherited_effective_vis_at_level
|
||||||
&& level != l)
|
&& level != l)
|
||||||
{
|
{
|
||||||
calculated_effective_vis = if let Some(nominal_vis) = nominal_vis && !nominal_vis.is_at_least(inherited_effective_vis_at_level, tcx) {
|
calculated_effective_vis =
|
||||||
nominal_vis
|
if nominal_vis.is_at_least(inherited_effective_vis_at_level, tcx) {
|
||||||
} else {
|
|
||||||
inherited_effective_vis_at_level
|
inherited_effective_vis_at_level
|
||||||
}
|
} else {
|
||||||
|
nominal_vis
|
||||||
|
};
|
||||||
}
|
}
|
||||||
// effective visibility can't be decreased at next update call for the
|
// effective visibility can't be decreased at next update call for the
|
||||||
// same id
|
// same id
|
||||||
|
|
|
@ -25,7 +25,7 @@ use rustc_hir::intravisit::{self, Visitor};
|
||||||
use rustc_hir::{AssocItemKind, HirIdSet, ItemId, Node, PatKind};
|
use rustc_hir::{AssocItemKind, HirIdSet, ItemId, Node, PatKind};
|
||||||
use rustc_middle::bug;
|
use rustc_middle::bug;
|
||||||
use rustc_middle::hir::nested_filter;
|
use rustc_middle::hir::nested_filter;
|
||||||
use rustc_middle::middle::privacy::{EffectiveVisibilities, EffectiveVisibility, Level};
|
use rustc_middle::middle::privacy::{EffectiveVisibilities, Level};
|
||||||
use rustc_middle::span_bug;
|
use rustc_middle::span_bug;
|
||||||
use rustc_middle::ty::query::Providers;
|
use rustc_middle::ty::query::Providers;
|
||||||
use rustc_middle::ty::subst::InternalSubsts;
|
use rustc_middle::ty::subst::InternalSubsts;
|
||||||
|
@ -38,7 +38,7 @@ use rustc_span::Span;
|
||||||
|
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
use std::ops::ControlFlow;
|
use std::ops::ControlFlow;
|
||||||
use std::{fmt, mem};
|
use std::{cmp, fmt, mem};
|
||||||
|
|
||||||
use errors::{
|
use errors::{
|
||||||
FieldIsPrivate, FieldIsPrivateLabel, FromPrivateDependencyInPublicInterface, InPublicInterface,
|
FieldIsPrivate, FieldIsPrivateLabel, FromPrivateDependencyInPublicInterface, InPublicInterface,
|
||||||
|
@ -408,9 +408,8 @@ impl VisibilityLike for ty::Visibility {
|
||||||
min(find.tcx.local_visibility(def_id), find.min, find.tcx)
|
min(find.tcx.local_visibility(def_id), find.min, find.tcx)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
impl VisibilityLike for Option<Level> {
|
||||||
impl VisibilityLike for Option<EffectiveVisibility> {
|
const MAX: Self = Some(Level::Direct);
|
||||||
const MAX: Self = Some(EffectiveVisibility::from_vis(ty::Visibility::Public));
|
|
||||||
// Type inference is very smart sometimes.
|
// Type inference is very smart sometimes.
|
||||||
// It can make an impl reachable even some components of its type or trait are unreachable.
|
// It can make an impl reachable even some components of its type or trait are unreachable.
|
||||||
// E.g. methods of `impl ReachableTrait<UnreachableTy> for ReachableTy<UnreachableTy> { ... }`
|
// E.g. methods of `impl ReachableTrait<UnreachableTy> for ReachableTy<UnreachableTy> { ... }`
|
||||||
|
@ -422,13 +421,7 @@ impl VisibilityLike for Option<EffectiveVisibility> {
|
||||||
// (which require reaching the `DefId`s in them).
|
// (which require reaching the `DefId`s in them).
|
||||||
const SHALLOW: bool = true;
|
const SHALLOW: bool = true;
|
||||||
fn new_min(find: &FindMin<'_, '_, Self>, def_id: LocalDefId) -> Self {
|
fn new_min(find: &FindMin<'_, '_, Self>, def_id: LocalDefId) -> Self {
|
||||||
if let Some(min) = find.min {
|
cmp::min(find.effective_visibilities.public_at_level(def_id), find.min)
|
||||||
return find
|
|
||||||
.effective_visibilities
|
|
||||||
.effective_vis(def_id)
|
|
||||||
.map(|eff_vis| min.min(*eff_vis, find.tcx));
|
|
||||||
}
|
|
||||||
None
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -454,79 +447,49 @@ struct EmbargoVisitor<'tcx> {
|
||||||
/// n::p::f()
|
/// n::p::f()
|
||||||
/// }
|
/// }
|
||||||
macro_reachable: FxHashSet<(LocalDefId, LocalDefId)>,
|
macro_reachable: FxHashSet<(LocalDefId, LocalDefId)>,
|
||||||
|
/// Previous visibility level; `None` means unreachable.
|
||||||
|
prev_level: Option<Level>,
|
||||||
/// Has something changed in the level map?
|
/// Has something changed in the level map?
|
||||||
changed: bool,
|
changed: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ReachEverythingInTheInterfaceVisitor<'a, 'tcx> {
|
struct ReachEverythingInTheInterfaceVisitor<'a, 'tcx> {
|
||||||
effective_vis: Option<EffectiveVisibility>,
|
level: Option<Level>,
|
||||||
item_def_id: LocalDefId,
|
item_def_id: LocalDefId,
|
||||||
ev: &'a mut EmbargoVisitor<'tcx>,
|
ev: &'a mut EmbargoVisitor<'tcx>,
|
||||||
level: Level,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'tcx> EmbargoVisitor<'tcx> {
|
impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
fn get(&self, def_id: LocalDefId) -> Option<EffectiveVisibility> {
|
fn get(&self, def_id: LocalDefId) -> Option<Level> {
|
||||||
self.effective_visibilities.effective_vis(def_id).copied()
|
self.effective_visibilities.public_at_level(def_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Updates node effective visibility.
|
/// Updates node level and returns the updated level.
|
||||||
fn update(
|
fn update(&mut self, def_id: LocalDefId, level: Option<Level>) -> Option<Level> {
|
||||||
&mut self,
|
let old_level = self.get(def_id);
|
||||||
def_id: LocalDefId,
|
// Visibility levels can only grow.
|
||||||
inherited_effective_vis: Option<EffectiveVisibility>,
|
if level > old_level {
|
||||||
level: Level,
|
self.effective_visibilities.set_public_at_level(
|
||||||
) {
|
|
||||||
let nominal_vis = self.tcx.local_visibility(def_id);
|
|
||||||
self.update_eff_vis(def_id, inherited_effective_vis, Some(nominal_vis), level);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update_eff_vis(
|
|
||||||
&mut self,
|
|
||||||
def_id: LocalDefId,
|
|
||||||
inherited_effective_vis: Option<EffectiveVisibility>,
|
|
||||||
nominal_vis: Option<ty::Visibility>,
|
|
||||||
level: Level,
|
|
||||||
) {
|
|
||||||
if let Some(inherited_effective_vis) = inherited_effective_vis {
|
|
||||||
let private_vis =
|
|
||||||
ty::Visibility::Restricted(self.tcx.parent_module_from_def_id(def_id));
|
|
||||||
if Some(private_vis) != nominal_vis {
|
|
||||||
self.changed |= self.effective_visibilities.update(
|
|
||||||
def_id,
|
def_id,
|
||||||
nominal_vis,
|
|| ty::Visibility::Restricted(self.tcx.parent_module_from_def_id(def_id)),
|
||||||
|| private_vis,
|
level.unwrap(),
|
||||||
inherited_effective_vis,
|
|
||||||
level,
|
|
||||||
self.tcx,
|
|
||||||
);
|
);
|
||||||
}
|
self.changed = true;
|
||||||
|
level
|
||||||
|
} else {
|
||||||
|
old_level
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn reach(
|
fn reach(
|
||||||
&mut self,
|
&mut self,
|
||||||
def_id: LocalDefId,
|
def_id: LocalDefId,
|
||||||
effective_vis: Option<EffectiveVisibility>,
|
level: Option<Level>,
|
||||||
) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
|
) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
|
||||||
ReachEverythingInTheInterfaceVisitor {
|
ReachEverythingInTheInterfaceVisitor {
|
||||||
effective_vis,
|
level: cmp::min(level, Some(Level::Reachable)),
|
||||||
item_def_id: def_id,
|
item_def_id: def_id,
|
||||||
ev: self,
|
ev: self,
|
||||||
level: Level::Reachable,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn reach_through_impl_trait(
|
|
||||||
&mut self,
|
|
||||||
def_id: LocalDefId,
|
|
||||||
effective_vis: Option<EffectiveVisibility>,
|
|
||||||
) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
|
|
||||||
ReachEverythingInTheInterfaceVisitor {
|
|
||||||
effective_vis,
|
|
||||||
item_def_id: def_id,
|
|
||||||
ev: self,
|
|
||||||
level: Level::ReachableThroughImplTrait,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -547,18 +510,16 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.effective_visibilities.public_at_level(local_def_id).is_none() {
|
if self.get(local_def_id).is_none() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Since we are starting from an externally visible module,
|
// Since we are starting from an externally visible module,
|
||||||
// all the parents in the loop below are also guaranteed to be modules.
|
// all the parents in the loop below are also guaranteed to be modules.
|
||||||
let mut module_def_id = macro_module_def_id;
|
let mut module_def_id = macro_module_def_id;
|
||||||
let macro_ev = self.get(local_def_id);
|
|
||||||
assert!(macro_ev.is_some());
|
|
||||||
loop {
|
loop {
|
||||||
let changed_reachability =
|
let changed_reachability =
|
||||||
self.update_macro_reachable(module_def_id, macro_module_def_id, macro_ev);
|
self.update_macro_reachable(module_def_id, macro_module_def_id);
|
||||||
if changed_reachability || module_def_id == CRATE_DEF_ID {
|
if changed_reachability || module_def_id == CRATE_DEF_ID {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -572,33 +533,21 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
&mut self,
|
&mut self,
|
||||||
module_def_id: LocalDefId,
|
module_def_id: LocalDefId,
|
||||||
defining_mod: LocalDefId,
|
defining_mod: LocalDefId,
|
||||||
macro_ev: Option<EffectiveVisibility>,
|
|
||||||
) -> bool {
|
) -> bool {
|
||||||
if self.macro_reachable.insert((module_def_id, defining_mod)) {
|
if self.macro_reachable.insert((module_def_id, defining_mod)) {
|
||||||
self.update_macro_reachable_mod(module_def_id, defining_mod, macro_ev);
|
self.update_macro_reachable_mod(module_def_id, defining_mod);
|
||||||
true
|
true
|
||||||
} else {
|
} else {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update_macro_reachable_mod(
|
fn update_macro_reachable_mod(&mut self, module_def_id: LocalDefId, defining_mod: LocalDefId) {
|
||||||
&mut self,
|
|
||||||
module_def_id: LocalDefId,
|
|
||||||
defining_mod: LocalDefId,
|
|
||||||
macro_ev: Option<EffectiveVisibility>,
|
|
||||||
) {
|
|
||||||
let module = self.tcx.hir().get_module(module_def_id).0;
|
let module = self.tcx.hir().get_module(module_def_id).0;
|
||||||
for item_id in module.item_ids {
|
for item_id in module.item_ids {
|
||||||
let def_kind = self.tcx.def_kind(item_id.owner_id);
|
let def_kind = self.tcx.def_kind(item_id.owner_id);
|
||||||
let vis = self.tcx.local_visibility(item_id.owner_id.def_id);
|
let vis = self.tcx.local_visibility(item_id.owner_id.def_id);
|
||||||
self.update_macro_reachable_def(
|
self.update_macro_reachable_def(item_id.owner_id.def_id, def_kind, vis, defining_mod);
|
||||||
item_id.owner_id.def_id,
|
|
||||||
def_kind,
|
|
||||||
vis,
|
|
||||||
defining_mod,
|
|
||||||
macro_ev,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
for child in self.tcx.module_children_local(module_def_id) {
|
for child in self.tcx.module_children_local(module_def_id) {
|
||||||
// FIXME: Use module children for the logic above too.
|
// FIXME: Use module children for the logic above too.
|
||||||
|
@ -607,7 +556,7 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
&& let Res::Def(def_kind, def_id) = child.res
|
&& let Res::Def(def_kind, def_id) = child.res
|
||||||
&& let Some(def_id) = def_id.as_local() {
|
&& let Some(def_id) = def_id.as_local() {
|
||||||
let vis = self.tcx.local_visibility(def_id);
|
let vis = self.tcx.local_visibility(def_id);
|
||||||
self.update_macro_reachable_def(def_id, def_kind, vis, defining_mod, macro_ev);
|
self.update_macro_reachable_def(def_id, def_kind, vis, defining_mod);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -618,14 +567,16 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
def_kind: DefKind,
|
def_kind: DefKind,
|
||||||
vis: ty::Visibility,
|
vis: ty::Visibility,
|
||||||
module: LocalDefId,
|
module: LocalDefId,
|
||||||
macro_ev: Option<EffectiveVisibility>,
|
|
||||||
) {
|
) {
|
||||||
self.update(def_id, macro_ev, Level::Reachable);
|
let level = Some(Level::Reachable);
|
||||||
|
if vis.is_public() {
|
||||||
|
self.update(def_id, level);
|
||||||
|
}
|
||||||
match def_kind {
|
match def_kind {
|
||||||
// No type privacy, so can be directly marked as reachable.
|
// No type privacy, so can be directly marked as reachable.
|
||||||
DefKind::Const | DefKind::Static(_) | DefKind::TraitAlias | DefKind::TyAlias => {
|
DefKind::Const | DefKind::Static(_) | DefKind::TraitAlias | DefKind::TyAlias => {
|
||||||
if vis.is_accessible_from(module, self.tcx) {
|
if vis.is_accessible_from(module, self.tcx) {
|
||||||
self.update(def_id, macro_ev, Level::Reachable);
|
self.update(def_id, level);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -637,7 +588,7 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
let item = self.tcx.hir().expect_item(def_id);
|
let item = self.tcx.hir().expect_item(def_id);
|
||||||
if let hir::ItemKind::Macro(MacroDef { macro_rules: false, .. }, _) = item.kind {
|
if let hir::ItemKind::Macro(MacroDef { macro_rules: false, .. }, _) = item.kind {
|
||||||
if vis.is_accessible_from(module, self.tcx) {
|
if vis.is_accessible_from(module, self.tcx) {
|
||||||
self.update(def_id, macro_ev, Level::Reachable);
|
self.update(def_id, level);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -648,12 +599,13 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
// the module, however may be reachable.
|
// the module, however may be reachable.
|
||||||
DefKind::Mod => {
|
DefKind::Mod => {
|
||||||
if vis.is_accessible_from(module, self.tcx) {
|
if vis.is_accessible_from(module, self.tcx) {
|
||||||
self.update_macro_reachable(def_id, module, macro_ev);
|
self.update_macro_reachable(def_id, module);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DefKind::Struct | DefKind::Union => {
|
DefKind::Struct | DefKind::Union => {
|
||||||
// While structs and unions have type privacy, their fields do not.
|
// While structs and unions have type privacy, their fields do not.
|
||||||
|
if vis.is_public() {
|
||||||
let item = self.tcx.hir().expect_item(def_id);
|
let item = self.tcx.hir().expect_item(def_id);
|
||||||
if let hir::ItemKind::Struct(ref struct_def, _)
|
if let hir::ItemKind::Struct(ref struct_def, _)
|
||||||
| hir::ItemKind::Union(ref struct_def, _) = item.kind
|
| hir::ItemKind::Union(ref struct_def, _) = item.kind
|
||||||
|
@ -661,13 +613,14 @@ impl<'tcx> EmbargoVisitor<'tcx> {
|
||||||
for field in struct_def.fields() {
|
for field in struct_def.fields() {
|
||||||
let field_vis = self.tcx.local_visibility(field.def_id);
|
let field_vis = self.tcx.local_visibility(field.def_id);
|
||||||
if field_vis.is_accessible_from(module, self.tcx) {
|
if field_vis.is_accessible_from(module, self.tcx) {
|
||||||
self.reach(field.def_id, macro_ev).ty();
|
self.reach(field.def_id, level).ty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
bug!("item {:?} with DefKind {:?}", item, def_kind);
|
bug!("item {:?} with DefKind {:?}", item, def_kind);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// These have type privacy, so are not reachable unless they're
|
// These have type privacy, so are not reachable unless they're
|
||||||
// public, or are not namespaced at all.
|
// public, or are not namespaced at all.
|
||||||
|
@ -709,16 +662,14 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_item(&mut self, item: &'tcx hir::Item<'tcx>) {
|
fn visit_item(&mut self, item: &'tcx hir::Item<'tcx>) {
|
||||||
let item_ev = match item.kind {
|
let item_level = match item.kind {
|
||||||
hir::ItemKind::Impl { .. } => {
|
hir::ItemKind::Impl { .. } => {
|
||||||
let impl_ev = Option::<EffectiveVisibility>::of_impl(
|
let impl_level = Option::<Level>::of_impl(
|
||||||
item.owner_id.def_id,
|
item.owner_id.def_id,
|
||||||
self.tcx,
|
self.tcx,
|
||||||
&self.effective_visibilities,
|
&self.effective_visibilities,
|
||||||
);
|
);
|
||||||
|
self.update(item.owner_id.def_id, impl_level)
|
||||||
self.update_eff_vis(item.owner_id.def_id, impl_ev, None, Level::Direct);
|
|
||||||
impl_ev
|
|
||||||
}
|
}
|
||||||
_ => self.get(item.owner_id.def_id),
|
_ => self.get(item.owner_id.def_id),
|
||||||
};
|
};
|
||||||
|
@ -727,32 +678,38 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
match item.kind {
|
match item.kind {
|
||||||
hir::ItemKind::Enum(ref def, _) => {
|
hir::ItemKind::Enum(ref def, _) => {
|
||||||
for variant in def.variants {
|
for variant in def.variants {
|
||||||
self.update(variant.def_id, item_ev, Level::Reachable);
|
let variant_level = self.update(variant.def_id, item_level);
|
||||||
let variant_ev = self.get(variant.def_id);
|
|
||||||
if let Some(ctor_def_id) = variant.data.ctor_def_id() {
|
if let Some(ctor_def_id) = variant.data.ctor_def_id() {
|
||||||
self.update(ctor_def_id, variant_ev, Level::Reachable);
|
self.update(ctor_def_id, item_level);
|
||||||
}
|
}
|
||||||
for field in variant.data.fields() {
|
for field in variant.data.fields() {
|
||||||
self.update(field.def_id, variant_ev, Level::Reachable);
|
self.update(field.def_id, variant_level);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir::ItemKind::Impl(ref impl_) => {
|
hir::ItemKind::Impl(ref impl_) => {
|
||||||
for impl_item_ref in impl_.items {
|
for impl_item_ref in impl_.items {
|
||||||
self.update(impl_item_ref.id.owner_id.def_id, item_ev, Level::Direct);
|
if impl_.of_trait.is_some()
|
||||||
|
|| self.tcx.visibility(impl_item_ref.id.owner_id).is_public()
|
||||||
|
{
|
||||||
|
self.update(impl_item_ref.id.owner_id.def_id, item_level);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir::ItemKind::Trait(.., trait_item_refs) => {
|
hir::ItemKind::Trait(.., trait_item_refs) => {
|
||||||
for trait_item_ref in trait_item_refs {
|
for trait_item_ref in trait_item_refs {
|
||||||
self.update(trait_item_ref.id.owner_id.def_id, item_ev, Level::Reachable);
|
self.update(trait_item_ref.id.owner_id.def_id, item_level);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir::ItemKind::Struct(ref def, _) | hir::ItemKind::Union(ref def, _) => {
|
hir::ItemKind::Struct(ref def, _) | hir::ItemKind::Union(ref def, _) => {
|
||||||
if let Some(ctor_def_id) = def.ctor_def_id() {
|
if let Some(ctor_def_id) = def.ctor_def_id() {
|
||||||
self.update(ctor_def_id, item_ev, Level::Reachable);
|
self.update(ctor_def_id, item_level);
|
||||||
}
|
}
|
||||||
for field in def.fields() {
|
for field in def.fields() {
|
||||||
self.update(field.def_id, item_ev, Level::Reachable);
|
let vis = self.tcx.visibility(field.def_id);
|
||||||
|
if vis.is_public() {
|
||||||
|
self.update(field.def_id, item_level);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir::ItemKind::Macro(ref macro_def, _) => {
|
hir::ItemKind::Macro(ref macro_def, _) => {
|
||||||
|
@ -760,7 +717,9 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
}
|
}
|
||||||
hir::ItemKind::ForeignMod { items, .. } => {
|
hir::ItemKind::ForeignMod { items, .. } => {
|
||||||
for foreign_item in items {
|
for foreign_item in items {
|
||||||
self.update(foreign_item.id.owner_id.def_id, item_ev, Level::Reachable);
|
if self.tcx.visibility(foreign_item.id.owner_id).is_public() {
|
||||||
|
self.update(foreign_item.id.owner_id.def_id, item_level);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -795,11 +754,8 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
// FIXME: This is some serious pessimization intended to workaround deficiencies
|
// FIXME: This is some serious pessimization intended to workaround deficiencies
|
||||||
// in the reachability pass (`middle/reachable.rs`). Types are marked as link-time
|
// in the reachability pass (`middle/reachable.rs`). Types are marked as link-time
|
||||||
// reachable if they are returned via `impl Trait`, even from private functions.
|
// reachable if they are returned via `impl Trait`, even from private functions.
|
||||||
let exist_ev = Some(EffectiveVisibility::from_vis(ty::Visibility::Public));
|
let exist_level = cmp::max(item_level, Some(Level::ReachableThroughImplTrait));
|
||||||
self.reach_through_impl_trait(item.owner_id.def_id, exist_ev)
|
self.reach(item.owner_id.def_id, exist_level).generics().predicates().ty();
|
||||||
.generics()
|
|
||||||
.predicates()
|
|
||||||
.ty();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Visit everything.
|
// Visit everything.
|
||||||
|
@ -807,18 +763,17 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
| hir::ItemKind::Static(..)
|
| hir::ItemKind::Static(..)
|
||||||
| hir::ItemKind::Fn(..)
|
| hir::ItemKind::Fn(..)
|
||||||
| hir::ItemKind::TyAlias(..) => {
|
| hir::ItemKind::TyAlias(..) => {
|
||||||
if item_ev.is_some() {
|
if item_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, item_ev).generics().predicates().ty();
|
self.reach(item.owner_id.def_id, item_level).generics().predicates().ty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir::ItemKind::Trait(.., trait_item_refs) => {
|
hir::ItemKind::Trait(.., trait_item_refs) => {
|
||||||
if item_ev.is_some() {
|
if item_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, item_ev).generics().predicates();
|
self.reach(item.owner_id.def_id, item_level).generics().predicates();
|
||||||
|
|
||||||
for trait_item_ref in trait_item_refs {
|
for trait_item_ref in trait_item_refs {
|
||||||
let tcx = self.tcx;
|
let tcx = self.tcx;
|
||||||
let mut reach = self.reach(trait_item_ref.id.owner_id.def_id, item_ev);
|
let mut reach = self.reach(trait_item_ref.id.owner_id.def_id, item_level);
|
||||||
|
|
||||||
reach.generics().predicates();
|
reach.generics().predicates();
|
||||||
|
|
||||||
if trait_item_ref.kind == AssocItemKind::Type
|
if trait_item_ref.kind == AssocItemKind::Type
|
||||||
|
@ -832,24 +787,23 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir::ItemKind::TraitAlias(..) => {
|
hir::ItemKind::TraitAlias(..) => {
|
||||||
if item_ev.is_some() {
|
if item_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, item_ev).generics().predicates();
|
self.reach(item.owner_id.def_id, item_level).generics().predicates();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Visit everything except for private impl items.
|
// Visit everything except for private impl items.
|
||||||
hir::ItemKind::Impl(ref impl_) => {
|
hir::ItemKind::Impl(ref impl_) => {
|
||||||
if item_ev.is_some() {
|
if item_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, item_ev)
|
self.reach(item.owner_id.def_id, item_level)
|
||||||
.generics()
|
.generics()
|
||||||
.predicates()
|
.predicates()
|
||||||
.ty()
|
.ty()
|
||||||
.trait_ref();
|
.trait_ref();
|
||||||
|
|
||||||
for impl_item_ref in impl_.items {
|
for impl_item_ref in impl_.items {
|
||||||
let impl_item_ev = self.get(impl_item_ref.id.owner_id.def_id);
|
let impl_item_level = self.get(impl_item_ref.id.owner_id.def_id);
|
||||||
|
if impl_item_level.is_some() {
|
||||||
if impl_item_ev.is_some() {
|
self.reach(impl_item_ref.id.owner_id.def_id, impl_item_level)
|
||||||
self.reach(impl_item_ref.id.owner_id.def_id, impl_item_ev)
|
|
||||||
.generics()
|
.generics()
|
||||||
.predicates()
|
.predicates()
|
||||||
.ty();
|
.ty();
|
||||||
|
@ -860,23 +814,23 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
|
|
||||||
// Visit everything, but enum variants have their own levels.
|
// Visit everything, but enum variants have their own levels.
|
||||||
hir::ItemKind::Enum(ref def, _) => {
|
hir::ItemKind::Enum(ref def, _) => {
|
||||||
if item_ev.is_some() {
|
if item_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, item_ev).generics().predicates();
|
self.reach(item.owner_id.def_id, item_level).generics().predicates();
|
||||||
}
|
}
|
||||||
for variant in def.variants {
|
for variant in def.variants {
|
||||||
let variant_ev = self.get(variant.def_id);
|
let variant_level = self.get(variant.def_id);
|
||||||
if variant_ev.is_some() {
|
if variant_level.is_some() {
|
||||||
for field in variant.data.fields() {
|
for field in variant.data.fields() {
|
||||||
self.reach(field.def_id, variant_ev).ty();
|
self.reach(field.def_id, variant_level).ty();
|
||||||
}
|
}
|
||||||
// Corner case: if the variant is reachable, but its
|
// Corner case: if the variant is reachable, but its
|
||||||
// enum is not, make the enum reachable as well.
|
// enum is not, make the enum reachable as well.
|
||||||
self.reach(item.owner_id.def_id, variant_ev).ty();
|
self.reach(item.owner_id.def_id, variant_level).ty();
|
||||||
}
|
}
|
||||||
if let Some(ctor_def_id) = variant.data.ctor_def_id() {
|
if let Some(ctor_def_id) = variant.data.ctor_def_id() {
|
||||||
let ctor_ev = self.get(ctor_def_id);
|
let ctor_level = self.get(ctor_def_id);
|
||||||
if ctor_ev.is_some() {
|
if ctor_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, ctor_ev).ty();
|
self.reach(item.owner_id.def_id, ctor_level).ty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -884,9 +838,9 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
// Visit everything, but foreign items have their own levels.
|
// Visit everything, but foreign items have their own levels.
|
||||||
hir::ItemKind::ForeignMod { items, .. } => {
|
hir::ItemKind::ForeignMod { items, .. } => {
|
||||||
for foreign_item in items {
|
for foreign_item in items {
|
||||||
let foreign_item_ev = self.get(foreign_item.id.owner_id.def_id);
|
let foreign_item_level = self.get(foreign_item.id.owner_id.def_id);
|
||||||
if foreign_item_ev.is_some() {
|
if foreign_item_level.is_some() {
|
||||||
self.reach(foreign_item.id.owner_id.def_id, foreign_item_ev)
|
self.reach(foreign_item.id.owner_id.def_id, foreign_item_level)
|
||||||
.generics()
|
.generics()
|
||||||
.predicates()
|
.predicates()
|
||||||
.ty();
|
.ty();
|
||||||
|
@ -895,32 +849,36 @@ impl<'tcx> Visitor<'tcx> for EmbargoVisitor<'tcx> {
|
||||||
}
|
}
|
||||||
// Visit everything except for private fields.
|
// Visit everything except for private fields.
|
||||||
hir::ItemKind::Struct(ref struct_def, _) | hir::ItemKind::Union(ref struct_def, _) => {
|
hir::ItemKind::Struct(ref struct_def, _) | hir::ItemKind::Union(ref struct_def, _) => {
|
||||||
if item_ev.is_some() {
|
if item_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, item_ev).generics().predicates();
|
self.reach(item.owner_id.def_id, item_level).generics().predicates();
|
||||||
for field in struct_def.fields() {
|
for field in struct_def.fields() {
|
||||||
let field_ev = self.get(field.def_id);
|
let field_level = self.get(field.def_id);
|
||||||
if field_ev.is_some() {
|
if field_level.is_some() {
|
||||||
self.reach(field.def_id, field_ev).ty();
|
self.reach(field.def_id, field_level).ty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let Some(ctor_def_id) = struct_def.ctor_def_id() {
|
if let Some(ctor_def_id) = struct_def.ctor_def_id() {
|
||||||
let ctor_ev = self.get(ctor_def_id);
|
let ctor_level = self.get(ctor_def_id);
|
||||||
if ctor_ev.is_some() {
|
if ctor_level.is_some() {
|
||||||
self.reach(item.owner_id.def_id, ctor_ev).ty();
|
self.reach(item.owner_id.def_id, ctor_level).ty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let orig_level = mem::replace(&mut self.prev_level, item_level);
|
||||||
intravisit::walk_item(self, item);
|
intravisit::walk_item(self, item);
|
||||||
|
self.prev_level = orig_level;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_block(&mut self, b: &'tcx hir::Block<'tcx>) {
|
fn visit_block(&mut self, b: &'tcx hir::Block<'tcx>) {
|
||||||
// Blocks can have public items, for example impls, but they always
|
// Blocks can have public items, for example impls, but they always
|
||||||
// start as completely private regardless of publicity of a function,
|
// start as completely private regardless of publicity of a function,
|
||||||
// constant, type, field, etc., in which this block resides.
|
// constant, type, field, etc., in which this block resides.
|
||||||
|
let orig_level = mem::replace(&mut self.prev_level, None);
|
||||||
intravisit::walk_block(self, b);
|
intravisit::walk_block(self, b);
|
||||||
|
self.prev_level = orig_level;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -974,7 +932,11 @@ impl<'tcx> DefIdVisitor<'tcx> for ReachEverythingInTheInterfaceVisitor<'_, 'tcx>
|
||||||
_descr: &dyn fmt::Display,
|
_descr: &dyn fmt::Display,
|
||||||
) -> ControlFlow<Self::BreakTy> {
|
) -> ControlFlow<Self::BreakTy> {
|
||||||
if let Some(def_id) = def_id.as_local() {
|
if let Some(def_id) = def_id.as_local() {
|
||||||
self.ev.update_eff_vis(def_id, self.effective_vis, None, self.level);
|
if let (ty::Visibility::Public, _) | (_, Some(Level::ReachableThroughImplTrait)) =
|
||||||
|
(self.tcx().visibility(def_id.to_def_id()), self.level)
|
||||||
|
{
|
||||||
|
self.ev.update(def_id, self.level);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ControlFlow::Continue(())
|
ControlFlow::Continue(())
|
||||||
}
|
}
|
||||||
|
@ -2202,6 +2164,7 @@ fn effective_visibilities(tcx: TyCtxt<'_>, (): ()) -> &EffectiveVisibilities {
|
||||||
tcx,
|
tcx,
|
||||||
effective_visibilities: tcx.resolutions(()).effective_visibilities.clone(),
|
effective_visibilities: tcx.resolutions(()).effective_visibilities.clone(),
|
||||||
macro_reachable: Default::default(),
|
macro_reachable: Default::default(),
|
||||||
|
prev_level: Some(Level::Direct),
|
||||||
changed: false,
|
changed: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -199,7 +199,7 @@ impl<'r, 'a, 'tcx> EffectiveVisibilitiesVisitor<'r, 'a, 'tcx> {
|
||||||
let tcx = self.r.tcx;
|
let tcx = self.r.tcx;
|
||||||
self.changed |= self.import_effective_visibilities.update(
|
self.changed |= self.import_effective_visibilities.update(
|
||||||
binding,
|
binding,
|
||||||
Some(nominal_vis),
|
nominal_vis,
|
||||||
|| cheap_private_vis.unwrap_or_else(|| self.r.private_vis_import(binding)),
|
|| cheap_private_vis.unwrap_or_else(|| self.r.private_vis_import(binding)),
|
||||||
inherited_eff_vis,
|
inherited_eff_vis,
|
||||||
parent_id.level(),
|
parent_id.level(),
|
||||||
|
@ -213,7 +213,7 @@ impl<'r, 'a, 'tcx> EffectiveVisibilitiesVisitor<'r, 'a, 'tcx> {
|
||||||
let tcx = self.r.tcx;
|
let tcx = self.r.tcx;
|
||||||
self.changed |= self.def_effective_visibilities.update(
|
self.changed |= self.def_effective_visibilities.update(
|
||||||
def_id,
|
def_id,
|
||||||
Some(nominal_vis),
|
nominal_vis,
|
||||||
|| cheap_private_vis.unwrap_or_else(|| self.r.private_vis_def(def_id)),
|
|| cheap_private_vis.unwrap_or_else(|| self.r.private_vis_def(def_id)),
|
||||||
inherited_eff_vis,
|
inherited_eff_vis,
|
||||||
parent_id.level(),
|
parent_id.level(),
|
||||||
|
|
|
@ -310,7 +310,6 @@ where
|
||||||
/// Real logic of both `Flatten` and `FlatMap` which simply delegate to
|
/// Real logic of both `Flatten` and `FlatMap` which simply delegate to
|
||||||
/// this type.
|
/// this type.
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
#[unstable(feature = "trusted_len", issue = "37572")]
|
|
||||||
struct FlattenCompat<I, U> {
|
struct FlattenCompat<I, U> {
|
||||||
iter: Fuse<I>,
|
iter: Fuse<I>,
|
||||||
frontiter: Option<U>,
|
frontiter: Option<U>,
|
||||||
|
@ -464,7 +463,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[unstable(feature = "trusted_len", issue = "37572")]
|
|
||||||
impl<I, U> Iterator for FlattenCompat<I, U>
|
impl<I, U> Iterator for FlattenCompat<I, U>
|
||||||
where
|
where
|
||||||
I: Iterator<Item: IntoIterator<IntoIter = U, Item = U::Item>>,
|
I: Iterator<Item: IntoIterator<IntoIter = U, Item = U::Item>>,
|
||||||
|
@ -579,7 +577,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[unstable(feature = "trusted_len", issue = "37572")]
|
|
||||||
impl<I, U> DoubleEndedIterator for FlattenCompat<I, U>
|
impl<I, U> DoubleEndedIterator for FlattenCompat<I, U>
|
||||||
where
|
where
|
||||||
I: DoubleEndedIterator<Item: IntoIterator<IntoIter = U, Item = U::Item>>,
|
I: DoubleEndedIterator<Item: IntoIterator<IntoIter = U, Item = U::Item>>,
|
||||||
|
@ -649,7 +646,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[unstable(feature = "trusted_len", issue = "37572")]
|
|
||||||
unsafe impl<const N: usize, I, T> TrustedLen
|
unsafe impl<const N: usize, I, T> TrustedLen
|
||||||
for FlattenCompat<I, <[T; N] as IntoIterator>::IntoIter>
|
for FlattenCompat<I, <[T; N] as IntoIterator>::IntoIter>
|
||||||
where
|
where
|
||||||
|
@ -657,7 +653,6 @@ where
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
#[unstable(feature = "trusted_len", issue = "37572")]
|
|
||||||
unsafe impl<'a, const N: usize, I, T> TrustedLen
|
unsafe impl<'a, const N: usize, I, T> TrustedLen
|
||||||
for FlattenCompat<I, <&'a [T; N] as IntoIterator>::IntoIter>
|
for FlattenCompat<I, <&'a [T; N] as IntoIterator>::IntoIter>
|
||||||
where
|
where
|
||||||
|
@ -665,7 +660,6 @@ where
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
#[unstable(feature = "trusted_len", issue = "37572")]
|
|
||||||
unsafe impl<'a, const N: usize, I, T> TrustedLen
|
unsafe impl<'a, const N: usize, I, T> TrustedLen
|
||||||
for FlattenCompat<I, <&'a mut [T; N] as IntoIterator>::IntoIter>
|
for FlattenCompat<I, <&'a mut [T; N] as IntoIterator>::IntoIter>
|
||||||
where
|
where
|
||||||
|
|
21
tests/ui/reachable/auxiliary/foreign-priv-aux.rs
Normal file
21
tests/ui/reachable/auxiliary/foreign-priv-aux.rs
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
trait PrivTrait {
|
||||||
|
fn priv_fn(&self);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ImplPrivTrait;
|
||||||
|
|
||||||
|
impl PrivTrait for ImplPrivTrait {
|
||||||
|
fn priv_fn(&self) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Wrapper<T>(T);
|
||||||
|
|
||||||
|
pub trait PubTrait {
|
||||||
|
fn pub_fn(&self);
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: PrivTrait> PubTrait for Wrapper<T> {
|
||||||
|
fn pub_fn(&self) {
|
||||||
|
self.0.priv_fn()
|
||||||
|
}
|
||||||
|
}
|
12
tests/ui/reachable/foreign-priv.rs
Normal file
12
tests/ui/reachable/foreign-priv.rs
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
// aux-build:foreign-priv-aux.rs
|
||||||
|
// build-pass
|
||||||
|
|
||||||
|
#![crate_type = "lib"]
|
||||||
|
|
||||||
|
extern crate foreign_priv_aux;
|
||||||
|
|
||||||
|
use foreign_priv_aux::{ImplPrivTrait, PubTrait, Wrapper};
|
||||||
|
|
||||||
|
pub fn foo(x: Wrapper<ImplPrivTrait>) {
|
||||||
|
x.pub_fn();
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue