1#![feature(associated_type_defaults)]
3#![feature(default_field_values)]
4#![feature(try_blocks)]
5mod errors;
8
9use std::marker::PhantomData;
10use std::ops::ControlFlow;
11use std::{debug_assert_matches, fmt};
12
13use errors::{
14 FieldIsPrivate, FieldIsPrivateLabel, FromPrivateDependencyInPublicInterface, InPublicInterface,
15 ItemIsPrivate, PrivateInterfacesOrBoundsLint, ReportEffectiveVisibility, UnnameableTypesLint,
16 UnnamedItemIsPrivate,
17};
18use rustc_ast::visit::{VisitorResult, try_visit};
19use rustc_data_structures::fx::{FxHashMap, FxHashSet};
20use rustc_data_structures::indexmap::IndexSet;
21use rustc_data_structures::intern::Interned;
22use rustc_errors::{MultiSpan, listify};
23use rustc_hir::def::{CtorOf, DefKind, Res};
24use rustc_hir::def_id::{DefId, LocalDefId, LocalModDefId};
25use rustc_hir::intravisit::{self, InferKind, Visitor};
26use rustc_hir::{self as hir, AmbigArg, ForeignItemId, ItemId, OwnerId, PatKind, find_attr};
27use rustc_middle::middle::privacy::{EffectiveVisibilities, EffectiveVisibility, Level};
28use rustc_middle::query::Providers;
29use rustc_middle::ty::print::PrintTraitRefExt as _;
30use rustc_middle::ty::{
31 self, AssocContainer, Const, GenericParamDefKind, TraitRef, Ty, TyCtxt, TypeSuperVisitable,
32 TypeVisitable, TypeVisitor,
33};
34use rustc_middle::{bug, span_bug};
35use rustc_session::lint;
36use rustc_span::{Ident, Span, Symbol, sym};
37use tracing::debug;
38
39struct LazyDefPathStr<'tcx> {
44 def_id: DefId,
45 tcx: TyCtxt<'tcx>,
46}
47
48impl<'tcx> fmt::Display for LazyDefPathStr<'tcx> {
49 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
50 f.write_fmt(format_args!("{0}", self.tcx.def_path_str(self.def_id)))write!(f, "{}", self.tcx.def_path_str(self.def_id))
51 }
52}
53
54pub trait DefIdVisitor<'tcx> {
63 type Result: VisitorResult = ();
64 const SHALLOW: bool = false;
65 fn skip_assoc_tys(&self) -> bool {
66 false
67 }
68
69 fn tcx(&self) -> TyCtxt<'tcx>;
70 fn visit_def_id(&mut self, def_id: DefId, kind: &str, descr: &dyn fmt::Display)
74 -> Self::Result;
75
76 fn skeleton(&mut self) -> DefIdVisitorSkeleton<'_, 'tcx, Self> {
78 DefIdVisitorSkeleton {
79 def_id_visitor: self,
80 visited_tys: Default::default(),
81 dummy: Default::default(),
82 }
83 }
84 fn visit(&mut self, ty_fragment: impl TypeVisitable<TyCtxt<'tcx>>) -> Self::Result {
85 ty_fragment.visit_with(&mut self.skeleton())
86 }
87 fn visit_trait(&mut self, trait_ref: TraitRef<'tcx>) -> Self::Result {
88 self.skeleton().visit_trait(trait_ref)
89 }
90 fn visit_predicates(&mut self, predicates: ty::GenericPredicates<'tcx>) -> Self::Result {
91 self.skeleton().visit_clauses(predicates.predicates)
92 }
93 fn visit_clauses(&mut self, clauses: &[(ty::Clause<'tcx>, Span)]) -> Self::Result {
94 self.skeleton().visit_clauses(clauses)
95 }
96}
97
98pub struct DefIdVisitorSkeleton<'v, 'tcx, V: ?Sized> {
99 def_id_visitor: &'v mut V,
100 visited_tys: FxHashSet<Ty<'tcx>>,
101 dummy: PhantomData<TyCtxt<'tcx>>,
102}
103
104impl<'tcx, V> DefIdVisitorSkeleton<'_, 'tcx, V>
105where
106 V: DefIdVisitor<'tcx> + ?Sized,
107{
108 fn visit_trait(&mut self, trait_ref: TraitRef<'tcx>) -> V::Result {
109 let TraitRef { def_id, args, .. } = trait_ref;
110 match ::rustc_ast_ir::visit::VisitorResult::branch(self.def_id_visitor.visit_def_id(def_id,
"trait", &trait_ref.print_only_trait_path())) {
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.def_id_visitor.visit_def_id(
111 def_id,
112 "trait",
113 &trait_ref.print_only_trait_path()
114 ));
115 if V::SHALLOW { V::Result::output() } else { args.visit_with(self) }
116 }
117
118 fn visit_projection_term(&mut self, projection: ty::AliasTerm<'tcx>) -> V::Result {
119 let tcx = self.def_id_visitor.tcx();
120 let (trait_ref, assoc_args) = projection.trait_ref_and_own_args(tcx);
121 match ::rustc_ast_ir::visit::VisitorResult::branch(self.visit_trait(trait_ref))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.visit_trait(trait_ref));
122 if V::SHALLOW {
123 V::Result::output()
124 } else {
125 V::Result::from_branch(
126 assoc_args.iter().try_for_each(|arg| arg.visit_with(self).branch()),
127 )
128 }
129 }
130
131 fn visit_clause(&mut self, clause: ty::Clause<'tcx>) -> V::Result {
132 match clause.kind().skip_binder() {
133 ty::ClauseKind::Trait(ty::TraitPredicate { trait_ref, polarity: _ }) => {
134 self.visit_trait(trait_ref)
135 }
136 ty::ClauseKind::HostEffect(pred) => {
137 match ::rustc_ast_ir::visit::VisitorResult::branch(self.visit_trait(pred.trait_ref))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.visit_trait(pred.trait_ref));
138 pred.constness.visit_with(self)
139 }
140 ty::ClauseKind::Projection(ty::ProjectionPredicate {
141 projection_term: projection_ty,
142 term,
143 }) => {
144 match ::rustc_ast_ir::visit::VisitorResult::branch(term.visit_with(self)) {
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(term.visit_with(self));
145 self.visit_projection_term(projection_ty)
146 }
147 ty::ClauseKind::TypeOutlives(ty::OutlivesPredicate(ty, _region)) => ty.visit_with(self),
148 ty::ClauseKind::RegionOutlives(..) => V::Result::output(),
149 ty::ClauseKind::ConstArgHasType(ct, ty) => {
150 match ::rustc_ast_ir::visit::VisitorResult::branch(ct.visit_with(self)) {
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(ct.visit_with(self));
151 ty.visit_with(self)
152 }
153 ty::ClauseKind::ConstEvaluatable(ct) => ct.visit_with(self),
154 ty::ClauseKind::WellFormed(term) => term.visit_with(self),
155 ty::ClauseKind::UnstableFeature(_) => V::Result::output(),
156 }
157 }
158
159 fn visit_clauses(&mut self, clauses: &[(ty::Clause<'tcx>, Span)]) -> V::Result {
160 for &(clause, _) in clauses {
161 match ::rustc_ast_ir::visit::VisitorResult::branch(self.visit_clause(clause))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.visit_clause(clause));
162 }
163 V::Result::output()
164 }
165}
166
167impl<'tcx, V> TypeVisitor<TyCtxt<'tcx>> for DefIdVisitorSkeleton<'_, 'tcx, V>
168where
169 V: DefIdVisitor<'tcx> + ?Sized,
170{
171 type Result = V::Result;
172
173 fn visit_predicate(&mut self, p: ty::Predicate<'tcx>) -> Self::Result {
174 self.visit_clause(p.as_clause().unwrap())
175 }
176
177 fn visit_ty(&mut self, ty: Ty<'tcx>) -> Self::Result {
178 let tcx = self.def_id_visitor.tcx();
179 let ty_kind = *ty.kind();
182 match ty_kind {
183 ty::Adt(ty::AdtDef(Interned(&ty::AdtDefData { did: def_id, .. }, _)), ..)
184 | ty::Foreign(def_id)
185 | ty::FnDef(def_id, ..)
186 | ty::Closure(def_id, ..)
187 | ty::CoroutineClosure(def_id, ..)
188 | ty::Coroutine(def_id, ..) => {
189 match ::rustc_ast_ir::visit::VisitorResult::branch(self.def_id_visitor.visit_def_id(def_id,
"type", &ty)) {
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.def_id_visitor.visit_def_id(def_id, "type", &ty));
190 if V::SHALLOW {
191 return V::Result::output();
192 }
193 if let ty::FnDef(..) = ty_kind {
197 match ::rustc_ast_ir::visit::VisitorResult::branch(tcx.fn_sig(def_id).instantiate_identity().skip_norm_wip().visit_with(self))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(
199 tcx.fn_sig(def_id).instantiate_identity().skip_norm_wip().visit_with(self)
200 );
201 }
202 if let Some(assoc_item) = tcx.opt_associated_item(def_id)
207 && let Some(impl_def_id) = assoc_item.impl_container(tcx)
208 {
209 match ::rustc_ast_ir::visit::VisitorResult::branch(tcx.type_of(impl_def_id).instantiate_identity().skip_norm_wip().visit_with(self))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(
210 tcx.type_of(impl_def_id)
211 .instantiate_identity()
212 .skip_norm_wip()
213 .visit_with(self)
214 );
215 }
216 }
217 ty::Alias(
218 data @ ty::AliasTy {
219 kind:
220 kind @ (ty::Inherent { def_id }
221 | ty::Free { def_id }
222 | ty::Projection { def_id }),
223 ..
224 },
225 ) => {
226 if self.def_id_visitor.skip_assoc_tys() {
227 return V::Result::output();
233 }
234 if !self.visited_tys.insert(ty) {
235 return V::Result::output();
239 }
240
241 match ::rustc_ast_ir::visit::VisitorResult::branch(self.def_id_visitor.visit_def_id(def_id,
match kind {
ty::Inherent { .. } | ty::Projection { .. } =>
"associated type",
ty::Free { .. } => "type alias",
ty::Opaque { .. } =>
::core::panicking::panic("internal error: entered unreachable code"),
}, &LazyDefPathStr { def_id, tcx })) {
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.def_id_visitor.visit_def_id(
242 def_id,
243 match kind {
244 ty::Inherent { .. } | ty::Projection { .. } => "associated type",
245 ty::Free { .. } => "type alias",
246 ty::Opaque { .. } => unreachable!(),
247 },
248 &LazyDefPathStr { def_id, tcx },
249 ));
250
251 return if V::SHALLOW {
253 V::Result::output()
254 } else if #[allow(non_exhaustive_omitted_patterns)] match kind {
ty::Projection { .. } => true,
_ => false,
}matches!(kind, ty::Projection { .. }) {
255 self.visit_projection_term(data.into())
256 } else {
257 V::Result::from_branch(
258 data.args.iter().try_for_each(|arg| arg.visit_with(self).branch()),
259 )
260 };
261 }
262 ty::Dynamic(predicates, ..) => {
263 for predicate in predicates {
266 let trait_ref = match predicate.skip_binder() {
267 ty::ExistentialPredicate::Trait(trait_ref) => trait_ref,
268 ty::ExistentialPredicate::Projection(proj) => proj.trait_ref(tcx),
269 ty::ExistentialPredicate::AutoTrait(def_id) => {
270 ty::ExistentialTraitRef::new(tcx, def_id, ty::GenericArgs::empty())
271 }
272 };
273 let ty::ExistentialTraitRef { def_id, .. } = trait_ref;
274 match ::rustc_ast_ir::visit::VisitorResult::branch(self.def_id_visitor.visit_def_id(def_id,
"trait", &trait_ref)) {
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.def_id_visitor.visit_def_id(def_id, "trait", &trait_ref));
275 }
276 }
277 ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, .. }) => {
278 if self.visited_tys.insert(ty) {
280 match ::rustc_ast_ir::visit::VisitorResult::branch(self.visit_clauses(tcx.explicit_item_bounds(def_id).skip_binder()))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};try_visit!(self.visit_clauses(tcx.explicit_item_bounds(def_id).skip_binder()));
288 }
289 }
290 ty::Bool
293 | ty::Char
294 | ty::Int(..)
295 | ty::Uint(..)
296 | ty::Float(..)
297 | ty::Str
298 | ty::Never
299 | ty::Array(..)
300 | ty::Slice(..)
301 | ty::Tuple(..)
302 | ty::RawPtr(..)
303 | ty::Ref(..)
304 | ty::Pat(..)
305 | ty::FnPtr(..)
306 | ty::UnsafeBinder(_)
307 | ty::Param(..)
308 | ty::Bound(..)
309 | ty::Error(_)
310 | ty::CoroutineWitness(..) => {}
311 ty::Placeholder(..) | ty::Infer(..) => {
312 ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected type: {0:?}", ty))bug!("unexpected type: {:?}", ty)
313 }
314 }
315
316 if V::SHALLOW { V::Result::output() } else { ty.super_visit_with(self) }
317 }
318
319 fn visit_const(&mut self, c: Const<'tcx>) -> Self::Result {
320 let tcx = self.def_id_visitor.tcx();
321 tcx.expand_abstract_consts(c).super_visit_with(self)
322 }
323}
324
325fn assoc_has_type_of(tcx: TyCtxt<'_>, item: &ty::AssocItem) -> bool {
326 if let ty::AssocKind::Type { data: ty::AssocTypeData::Normal(..) } = item.kind
327 && let hir::Node::TraitItem(item) =
328 tcx.hir_node(tcx.local_def_id_to_hir_id(item.def_id.expect_local()))
329 && let hir::TraitItemKind::Type(_, None) = item.kind
330 {
331 false
332 } else {
333 true
334 }
335}
336
337fn min(vis1: ty::Visibility, vis2: ty::Visibility, tcx: TyCtxt<'_>) -> ty::Visibility {
338 if vis1.greater_than(vis2, tcx) { vis2 } else { vis1 }
339}
340
341struct FindMin<'a, 'tcx, VL: VisibilityLike, const SHALLOW: bool> {
343 tcx: TyCtxt<'tcx>,
344 effective_visibilities: &'a EffectiveVisibilities,
345 min: VL,
346}
347
348impl<'a, 'tcx, VL: VisibilityLike, const SHALLOW: bool> DefIdVisitor<'tcx>
349 for FindMin<'a, 'tcx, VL, SHALLOW>
350{
351 const SHALLOW: bool = SHALLOW;
352 fn skip_assoc_tys(&self) -> bool {
353 true
354 }
355 fn tcx(&self) -> TyCtxt<'tcx> {
356 self.tcx
357 }
358 fn visit_def_id(&mut self, def_id: DefId, _kind: &str, _descr: &dyn fmt::Display) {
359 if let Some(def_id) = def_id.as_local() {
360 self.min = VL::new_min(self, def_id);
361 }
362 }
363}
364
365trait VisibilityLike: Sized {
366 const MAX: Self;
367 fn new_min<const SHALLOW: bool>(
368 find: &FindMin<'_, '_, Self, SHALLOW>,
369 def_id: LocalDefId,
370 ) -> Self;
371
372 fn of_impl<const SHALLOW: bool>(
375 def_id: LocalDefId,
376 of_trait: bool,
377 tcx: TyCtxt<'_>,
378 effective_visibilities: &EffectiveVisibilities,
379 ) -> Self {
380 let mut find = FindMin::<_, SHALLOW> { tcx, effective_visibilities, min: Self::MAX };
381 find.visit(tcx.type_of(def_id).instantiate_identity().skip_norm_wip());
382 if of_trait {
383 find.visit_trait(tcx.impl_trait_ref(def_id).instantiate_identity().skip_norm_wip());
384 }
385 find.min
386 }
387}
388
389impl VisibilityLike for ty::Visibility {
390 const MAX: Self = ty::Visibility::Public;
391 fn new_min<const SHALLOW: bool>(
392 find: &FindMin<'_, '_, Self, SHALLOW>,
393 def_id: LocalDefId,
394 ) -> Self {
395 min(find.tcx.local_visibility(def_id), find.min, find.tcx)
396 }
397}
398
399impl VisibilityLike for EffectiveVisibility {
400 const MAX: Self = EffectiveVisibility::from_vis(ty::Visibility::Public);
401 fn new_min<const SHALLOW: bool>(
402 find: &FindMin<'_, '_, Self, SHALLOW>,
403 def_id: LocalDefId,
404 ) -> Self {
405 let effective_vis =
406 find.effective_visibilities.effective_vis(def_id).copied().unwrap_or_else(|| {
407 let private_vis = ty::Visibility::Restricted(
408 find.tcx.parent_module_from_def_id(def_id).to_local_def_id(),
409 );
410 EffectiveVisibility::from_vis(private_vis)
411 });
412
413 effective_vis.min(find.min, find.tcx)
414 }
415}
416
417type DefIdsToImpls = FxHashMap<LocalDefId, FxHashSet<LocalDefId>>;
418
419struct DefIdsToImplsCollector<'tcx, 'a> {
422 tcx: TyCtxt<'tcx>,
423 def_ids_to_impls: &'a mut DefIdsToImpls,
424 impl_def_id: LocalDefId,
425}
426
427impl<'tcx, 'a> DefIdsToImplsCollector<'tcx, 'a> {
428 fn collect(tcx: TyCtxt<'tcx>) -> DefIdsToImpls {
429 let mut def_ids_to_impls = Default::default();
430 for item in tcx.hir_free_items() {
431 let impl_def_id = item.owner_id.def_id;
432 let DefKind::Impl { of_trait } = tcx.def_kind(impl_def_id) else {
433 continue;
434 };
435
436 let mut visitor = DefIdsToImplsCollector {
438 tcx,
439 impl_def_id,
440 def_ids_to_impls: &mut def_ids_to_impls,
441 };
442
443 visitor.visit(tcx.type_of(impl_def_id).instantiate_identity().skip_norm_wip());
444 if of_trait {
445 visitor.visit_trait(
446 tcx.impl_trait_ref(impl_def_id).instantiate_identity().skip_norm_wip(),
447 );
448 }
449 }
450
451 def_ids_to_impls
452 }
453}
454
455impl<'tcx, 'a> DefIdVisitor<'tcx> for DefIdsToImplsCollector<'tcx, 'a> {
456 const SHALLOW: bool = true;
457 fn skip_assoc_tys(&self) -> bool {
458 true
459 }
460 fn tcx(&self) -> TyCtxt<'tcx> {
461 self.tcx
462 }
463 fn visit_def_id(&mut self, def_id: DefId, _kind: &str, _descr: &dyn fmt::Display) {
464 if let Some(def_id) = def_id.as_local() {
465 if true {
{
match self.tcx.def_kind(def_id) {
DefKind::Enum | DefKind::Union | DefKind::Struct |
DefKind::ForeignTy | DefKind::Trait => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"DefKind::Enum | DefKind::Union | DefKind::Struct | DefKind::ForeignTy |\nDefKind::Trait",
::core::option::Option::None);
}
}
};
};debug_assert_matches!(
466 self.tcx.def_kind(def_id),
467 DefKind::Enum
468 | DefKind::Union
469 | DefKind::Struct
470 | DefKind::ForeignTy
471 | DefKind::Trait
472 );
473 self.def_ids_to_impls.entry(def_id).or_default().insert(self.impl_def_id);
474 }
475 }
476}
477
478struct EmbargoVisitor<'tcx> {
480 tcx: TyCtxt<'tcx>,
481 effective_visibilities: EffectiveVisibilities,
483 queue: IndexSet<LocalDefId>,
485 def_ids_to_impls: DefIdsToImpls,
487}
488
489struct ReachEverythingInTheInterfaceVisitor<'a, 'tcx> {
490 effective_vis: EffectiveVisibility,
491 item_def_id: LocalDefId,
492 ev: &'a mut EmbargoVisitor<'tcx>,
493 level: Level,
494}
495
496impl<'tcx> EmbargoVisitor<'tcx> {
497 fn get(&self, def_id: LocalDefId) -> Option<EffectiveVisibility> {
498 self.effective_visibilities.effective_vis(def_id).copied()
499 }
500
501 fn update(
503 &mut self,
504 def_id: LocalDefId,
505 inherited_effective_vis: EffectiveVisibility,
506 level: Level,
507 ) {
508 let nominal_vis = self.tcx.local_visibility(def_id);
509 self.update_eff_vis(def_id, inherited_effective_vis, Some(nominal_vis), level);
510 }
511
512 fn update_eff_vis(
513 &mut self,
514 def_id: LocalDefId,
515 inherited_effective_vis: EffectiveVisibility,
516 max_vis: Option<ty::Visibility>,
517 level: Level,
518 ) -> bool {
519 let private_vis =
521 ty::Visibility::Restricted(self.tcx.parent_module_from_def_id(def_id).into());
522 if max_vis != Some(private_vis) {
523 return self.effective_visibilities.update(
524 def_id,
525 max_vis,
526 private_vis,
527 inherited_effective_vis,
528 level,
529 self.tcx,
530 );
531 }
532 false
533 }
534
535 fn reach(
536 &mut self,
537 def_id: LocalDefId,
538 effective_vis: EffectiveVisibility,
539 ) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
540 ReachEverythingInTheInterfaceVisitor {
541 effective_vis,
542 item_def_id: def_id,
543 ev: self,
544 level: Level::Reachable,
545 }
546 }
547
548 fn reach_through_impl_trait(
549 &mut self,
550 def_id: LocalDefId,
551 effective_vis: EffectiveVisibility,
552 ) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
553 ReachEverythingInTheInterfaceVisitor {
554 effective_vis,
555 item_def_id: def_id,
556 ev: self,
557 level: Level::ReachableThroughImplTrait,
558 }
559 }
560}
561
562impl<'tcx> EmbargoVisitor<'tcx> {
563 fn check_assoc_item(&mut self, item: &ty::AssocItem, item_ev: EffectiveVisibility) {
564 let def_id = item.def_id.expect_local();
565 let tcx = self.tcx;
566 let mut reach = self.reach(def_id, item_ev);
567 reach.generics().predicates();
568 if assoc_has_type_of(tcx, item) {
569 reach.ty();
570 }
571 if item.is_type() && item.container == AssocContainer::Trait {
572 reach.bounds();
573 }
574 }
575
576 fn check_def_id(&mut self, def_id: LocalDefId) {
577 let item_ev = self.get(def_id);
580 let def_kind = self.tcx.def_kind(def_id);
581 match def_kind {
582 DefKind::Use | DefKind::ExternCrate | DefKind::GlobalAsm => {}
584 DefKind::Mod => {}
586 DefKind::Macro { .. } => {}
588 DefKind::ForeignTy
589 | DefKind::Const { .. }
590 | DefKind::Static { .. }
591 | DefKind::Fn
592 | DefKind::TyAlias => {
593 if let Some(item_ev) = item_ev {
594 self.reach(def_id, item_ev).generics().predicates().ty();
595 }
596 }
597 DefKind::Trait => {
598 if let Some(item_ev) = item_ev {
599 self.reach(def_id, item_ev).generics().predicates();
600
601 for assoc_item in self.tcx.associated_items(def_id).in_definition_order() {
602 let def_id = assoc_item.def_id.expect_local();
603 self.update(def_id, item_ev, Level::Reachable);
604
605 self.check_assoc_item(assoc_item, item_ev);
606 }
607 }
608 }
609 DefKind::TraitAlias => {
610 if let Some(item_ev) = item_ev {
611 self.reach(def_id, item_ev).generics().predicates();
612 }
613 }
614 DefKind::Impl { of_trait } => {
615 let item_ev = EffectiveVisibility::of_impl::<true>(
626 def_id,
627 of_trait,
628 self.tcx,
629 &self.effective_visibilities,
630 );
631
632 self.update_eff_vis(def_id, item_ev, None, Level::Direct);
633
634 {
635 let mut reach = self.reach(def_id, item_ev);
636 reach.generics().predicates().ty();
637 if of_trait {
638 reach.trait_ref();
639 }
640 }
641
642 for assoc_item in self.tcx.associated_items(def_id).in_definition_order() {
643 let def_id = assoc_item.def_id.expect_local();
644 let max_vis =
645 if of_trait { None } else { Some(self.tcx.local_visibility(def_id)) };
646 self.update_eff_vis(def_id, item_ev, max_vis, Level::Direct);
647
648 if let Some(impl_item_ev) = self.get(def_id) {
649 self.check_assoc_item(assoc_item, impl_item_ev);
650 }
651 }
652 }
653 DefKind::Enum => {
654 if let Some(item_ev) = item_ev {
655 self.reach(def_id, item_ev).generics().predicates();
656 }
657 let def = self.tcx.adt_def(def_id);
658 for variant in def.variants() {
659 if let Some(item_ev) = item_ev {
660 self.update(variant.def_id.expect_local(), item_ev, Level::Reachable);
661 }
662
663 if let Some(variant_ev) = self.get(variant.def_id.expect_local()) {
664 if let Some(ctor_def_id) = variant.ctor_def_id() {
665 self.update(ctor_def_id.expect_local(), variant_ev, Level::Reachable);
666 }
667
668 for field in &variant.fields {
669 let field = field.did.expect_local();
670 self.update(field, variant_ev, Level::Reachable);
671 self.reach(field, variant_ev).ty();
672 }
673 self.reach(def_id, variant_ev).ty();
676 }
677 if let Some(ctor_def_id) = variant.ctor_def_id() {
678 if let Some(ctor_ev) = self.get(ctor_def_id.expect_local()) {
679 self.reach(def_id, ctor_ev).ty();
680 }
681 }
682 }
683 }
684 DefKind::Struct | DefKind::Union => {
685 let def = self.tcx.adt_def(def_id).non_enum_variant();
686 if let Some(item_ev) = item_ev {
687 self.reach(def_id, item_ev).generics().predicates();
688 for field in &def.fields {
689 let field = field.did.expect_local();
690 self.update(field, item_ev, Level::Reachable);
691 if let Some(field_ev) = self.get(field) {
692 self.reach(field, field_ev).ty();
693 }
694 }
695 }
696 if let Some(ctor_def_id) = def.ctor_def_id() {
697 if let Some(item_ev) = item_ev {
698 self.update(ctor_def_id.expect_local(), item_ev, Level::Reachable);
699 }
700 if let Some(ctor_ev) = self.get(ctor_def_id.expect_local()) {
701 self.reach(def_id, ctor_ev).ty();
702 }
703 }
704 }
705 DefKind::ForeignMod => {}
707 DefKind::Field
708 | DefKind::Variant
709 | DefKind::AssocFn
710 | DefKind::AssocTy
711 | DefKind::AssocConst { .. }
712 | DefKind::TyParam
713 | DefKind::AnonConst
714 | DefKind::InlineConst
715 | DefKind::OpaqueTy
716 | DefKind::Closure
717 | DefKind::SyntheticCoroutineBody
718 | DefKind::ConstParam
719 | DefKind::LifetimeParam
720 | DefKind::Ctor(..) => {
721 ::rustc_middle::util::bug::span_bug_fmt(self.tcx.def_span(def_id),
format_args!("{0:?} should be checked while checking parent", def_kind))span_bug!(
722 self.tcx.def_span(def_id),
723 "{def_kind:?} should be checked while checking parent"
724 )
725 }
726 }
727 }
728}
729
730impl ReachEverythingInTheInterfaceVisitor<'_, '_> {
731 fn generics(&mut self) -> &mut Self {
732 for param in &self.ev.tcx.generics_of(self.item_def_id).own_params {
733 if let GenericParamDefKind::Const { .. } = param.kind {
734 self.visit(
735 self.ev.tcx.type_of(param.def_id).instantiate_identity().skip_norm_wip(),
736 );
737 }
738 if let Some(default) = param.default_value(self.ev.tcx) {
739 self.visit(default.instantiate_identity().skip_norm_wip());
740 }
741 }
742 self
743 }
744
745 fn predicates(&mut self) -> &mut Self {
746 self.visit_predicates(self.ev.tcx.explicit_predicates_of(self.item_def_id));
747 self
748 }
749
750 fn bounds(&mut self) -> &mut Self {
751 self.visit_clauses(self.ev.tcx.explicit_item_bounds(self.item_def_id).skip_binder());
752 self
753 }
754
755 fn ty(&mut self) -> &mut Self {
756 self.visit(self.ev.tcx.type_of(self.item_def_id).instantiate_identity().skip_norm_wip());
757 self
758 }
759
760 fn trait_ref(&mut self) -> &mut Self {
761 self.visit_trait(
762 self.ev.tcx.impl_trait_ref(self.item_def_id).instantiate_identity().skip_norm_wip(),
763 );
764 self
765 }
766
767 fn enqueue_def_id(&mut self, def_id: LocalDefId) {
770 let def_kind = self.ev.tcx.def_kind(def_id);
771 match def_kind {
772 DefKind::Enum
773 | DefKind::Union
774 | DefKind::Struct
775 | DefKind::ForeignTy
776 | DefKind::Trait => {
777 self.ev.queue.insert(def_id);
778 if let Some(impls) = self.ev.def_ids_to_impls.get(&def_id) {
780 #[allow(rustc::potential_query_instability)]
782 self.ev.queue.extend(impls);
783 }
784 }
785
786 DefKind::TraitAlias | DefKind::Fn => {
787 self.ev.queue.insert(def_id);
788 }
789
790 DefKind::AssocConst { .. } | DefKind::AssocFn | DefKind::AssocTy => {
791 self.ev.queue.insert(self.ev.tcx.local_parent(def_id));
794 }
795
796 DefKind::Ctor(ctor_of, _) => {
797 let update_id = match ctor_of {
798 CtorOf::Struct => self.ev.tcx.local_parent(def_id),
799 CtorOf::Variant => self.ev.tcx.local_parent(self.ev.tcx.local_parent(def_id)),
800 };
801 self.ev.queue.insert(update_id);
803 }
804
805 DefKind::Closure => {}
808
809 DefKind::Impl { .. }
811 | DefKind::TyAlias
812 | DefKind::Field
813 | DefKind::Variant
814 | DefKind::Static { .. }
815 | DefKind::Macro(_)
816 | DefKind::TyParam
817 | DefKind::AnonConst
818 | DefKind::InlineConst
819 | DefKind::OpaqueTy
820 | DefKind::SyntheticCoroutineBody
821 | DefKind::ConstParam
822 | DefKind::LifetimeParam
823 | DefKind::Mod
824 | DefKind::Use
825 | DefKind::ExternCrate
826 | DefKind::GlobalAsm
827 | DefKind::ForeignMod
828 | DefKind::Const { .. } => {
829 ::rustc_middle::util::bug::span_bug_fmt(self.tcx().def_span(def_id),
format_args!("{0:?} unexpectedly reached by `ReachEverythingInTheInterfaceVisitor`",
def_kind))span_bug!(
830 self.tcx().def_span(def_id),
831 "{def_kind:?} unexpectedly reached by `ReachEverythingInTheInterfaceVisitor`"
832 )
833 }
834 }
835 }
836}
837
838impl<'tcx> DefIdVisitor<'tcx> for ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
839 fn tcx(&self) -> TyCtxt<'tcx> {
840 self.ev.tcx
841 }
842 fn visit_def_id(&mut self, def_id: DefId, _kind: &str, _descr: &dyn fmt::Display) {
843 if let Some(def_id) = def_id.as_local() {
844 let max_vis = (self.level != Level::ReachableThroughImplTrait)
848 .then(|| self.ev.tcx.local_visibility(def_id));
849 if self.ev.update_eff_vis(def_id, self.effective_vis, max_vis, self.level) {
850 self.enqueue_def_id(def_id);
851 }
852 }
853 }
854}
855
856pub struct TestReachabilityVisitor<'a, 'tcx> {
858 tcx: TyCtxt<'tcx>,
859 effective_visibilities: &'a EffectiveVisibilities,
860}
861
862impl<'a, 'tcx> TestReachabilityVisitor<'a, 'tcx> {
863 fn effective_visibility_diagnostic(&self, def_id: LocalDefId) {
864 if {
{
'done:
{
for i in
::rustc_hir::attrs::HasAttrs::get_attrs(def_id, &self.tcx) {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcEffectiveVisibility) => {
break 'done Some(());
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}
}.is_some()find_attr!(self.tcx, def_id, RustcEffectiveVisibility) {
865 let mut error_msg = String::new();
866 let span = self.tcx.def_span(def_id.to_def_id());
867 if let Some(effective_vis) = self.effective_visibilities.effective_vis(def_id) {
868 for level in Level::all_levels() {
869 let vis_str = effective_vis.at_level(level).to_string(def_id, self.tcx);
870 if level != Level::Direct {
871 error_msg.push_str(", ");
872 }
873 error_msg.push_str(&::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0:?}: {1}", level, vis_str))
})format!("{level:?}: {vis_str}"));
874 }
875 } else {
876 error_msg.push_str("not in the table");
877 }
878 self.tcx.dcx().emit_err(ReportEffectiveVisibility { span, descr: error_msg });
879 }
880 }
881}
882
883impl<'a, 'tcx> TestReachabilityVisitor<'a, 'tcx> {
884 fn check_def_id(&self, owner_id: OwnerId) {
885 self.effective_visibility_diagnostic(owner_id.def_id);
886
887 match self.tcx.def_kind(owner_id) {
888 DefKind::Enum => {
889 let def = self.tcx.adt_def(owner_id.def_id);
890 for variant in def.variants() {
891 self.effective_visibility_diagnostic(variant.def_id.expect_local());
892 if let Some(ctor_def_id) = variant.ctor_def_id() {
893 self.effective_visibility_diagnostic(ctor_def_id.expect_local());
894 }
895 for field in &variant.fields {
896 self.effective_visibility_diagnostic(field.did.expect_local());
897 }
898 }
899 }
900 DefKind::Struct | DefKind::Union => {
901 let def = self.tcx.adt_def(owner_id.def_id).non_enum_variant();
902 if let Some(ctor_def_id) = def.ctor_def_id() {
903 self.effective_visibility_diagnostic(ctor_def_id.expect_local());
904 }
905 for field in &def.fields {
906 self.effective_visibility_diagnostic(field.did.expect_local());
907 }
908 }
909 _ => {}
910 }
911 }
912}
913
914struct NamePrivacyVisitor<'tcx> {
920 tcx: TyCtxt<'tcx>,
921 maybe_typeck_results: Option<&'tcx ty::TypeckResults<'tcx>>,
922}
923
924impl<'tcx> NamePrivacyVisitor<'tcx> {
925 #[track_caller]
929 fn typeck_results(&self) -> &'tcx ty::TypeckResults<'tcx> {
930 self.maybe_typeck_results
931 .expect("`NamePrivacyVisitor::typeck_results` called outside of body")
932 }
933
934 fn check_field(
936 &self,
937 hir_id: hir::HirId, use_ctxt: Span, def: ty::AdtDef<'tcx>, field: &'tcx ty::FieldDef,
941 ) -> bool {
942 if def.is_enum() {
943 return true;
944 }
945
946 let ident = Ident::new(sym::dummy, use_ctxt);
948 let (_, def_id) = self.tcx.adjust_ident_and_get_scope(ident, def.did(), hir_id);
949 !field.vis.is_accessible_from(def_id, self.tcx)
950 }
951
952 fn emit_unreachable_field_error(
954 &self,
955 fields: Vec<(Symbol, Span, bool )>,
956 def: ty::AdtDef<'tcx>, update_syntax: Option<Span>,
958 struct_span: Span,
959 ) {
960 if def.is_enum() || fields.is_empty() {
961 return;
962 }
963
964 let Some(field_names) = listify(&fields[..], |(n, _, _)| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", n))
})format!("`{n}`")) else { return };
976 let span: MultiSpan = fields.iter().map(|(_, span, _)| *span).collect::<Vec<Span>>().into();
977
978 let rest_field_names: Vec<_> =
980 fields.iter().filter(|(_, _, is_present)| !is_present).map(|(n, _, _)| n).collect();
981 let rest_len = rest_field_names.len();
982 let rest_field_names =
983 listify(&rest_field_names[..], |n| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", n))
})format!("`{n}`")).unwrap_or_default();
984 let labels = fields
986 .iter()
987 .filter(|(_, _, is_present)| *is_present)
988 .map(|(_, span, _)| FieldIsPrivateLabel::Other { span: *span })
989 .chain(update_syntax.iter().map(|span| FieldIsPrivateLabel::IsUpdateSyntax {
990 span: *span,
991 rest_field_names: rest_field_names.clone(),
992 rest_len,
993 }))
994 .collect();
995
996 self.tcx.dcx().emit_err(FieldIsPrivate {
997 span,
998 struct_span: if self
999 .tcx
1000 .sess
1001 .source_map()
1002 .is_multiline(fields[0].1.between(struct_span))
1003 {
1004 Some(struct_span)
1005 } else {
1006 None
1007 },
1008 field_names,
1009 variant_descr: def.variant_descr(),
1010 def_path_str: self.tcx.def_path_str(def.did()),
1011 labels,
1012 len: fields.len(),
1013 });
1014 }
1015
1016 fn check_expanded_fields(
1017 &self,
1018 adt: ty::AdtDef<'tcx>,
1019 variant: &'tcx ty::VariantDef,
1020 fields: &[hir::ExprField<'tcx>],
1021 hir_id: hir::HirId,
1022 span: Span,
1023 struct_span: Span,
1024 ) {
1025 let mut failed_fields = ::alloc::vec::Vec::new()vec![];
1026 for (vf_index, variant_field) in variant.fields.iter_enumerated() {
1027 let field =
1028 fields.iter().find(|f| self.typeck_results().field_index(f.hir_id) == vf_index);
1029 let (hir_id, use_ctxt, span) = match field {
1030 Some(field) => (field.hir_id, field.ident.span, field.span),
1031 None => (hir_id, span, span),
1032 };
1033 if self.check_field(hir_id, use_ctxt, adt, variant_field) {
1034 let name = match field {
1035 Some(field) => field.ident.name,
1036 None => variant_field.name,
1037 };
1038 failed_fields.push((name, span, field.is_some()));
1039 }
1040 }
1041 self.emit_unreachable_field_error(failed_fields, adt, Some(span), struct_span);
1042 }
1043}
1044
1045impl<'tcx> Visitor<'tcx> for NamePrivacyVisitor<'tcx> {
1046 fn visit_nested_body(&mut self, body_id: hir::BodyId) {
1047 let new_typeck_results = self.tcx.typeck_body(body_id);
1048 if new_typeck_results.tainted_by_errors.is_some() {
1050 return;
1051 }
1052 let old_maybe_typeck_results = self.maybe_typeck_results.replace(new_typeck_results);
1053 self.visit_body(self.tcx.hir_body(body_id));
1054 self.maybe_typeck_results = old_maybe_typeck_results;
1055 }
1056
1057 fn visit_expr(&mut self, expr: &'tcx hir::Expr<'tcx>) {
1058 if let hir::ExprKind::Struct(qpath, fields, ref base) = expr.kind {
1059 let res = self.typeck_results().qpath_res(qpath, expr.hir_id);
1060 let adt = self.typeck_results().expr_ty(expr).ty_adt_def().unwrap();
1061 let variant = adt.variant_of_res(res);
1062 match *base {
1063 hir::StructTailExpr::Base(base) => {
1064 self.check_expanded_fields(
1068 adt,
1069 variant,
1070 fields,
1071 base.hir_id,
1072 base.span,
1073 qpath.span(),
1074 );
1075 }
1076 hir::StructTailExpr::DefaultFields(span) => {
1077 self.check_expanded_fields(
1078 adt,
1079 variant,
1080 fields,
1081 expr.hir_id,
1082 span,
1083 qpath.span(),
1084 );
1085 }
1086 hir::StructTailExpr::None | hir::StructTailExpr::NoneWithError(_) => {
1087 let mut failed_fields = ::alloc::vec::Vec::new()vec![];
1088 for field in fields {
1089 let (hir_id, use_ctxt) = (field.hir_id, field.ident.span);
1090 let index = self.typeck_results().field_index(field.hir_id);
1091 if self.check_field(hir_id, use_ctxt, adt, &variant.fields[index]) {
1092 failed_fields.push((field.ident.name, field.ident.span, true));
1093 }
1094 }
1095 self.emit_unreachable_field_error(failed_fields, adt, None, qpath.span());
1096 }
1097 }
1098 }
1099
1100 intravisit::walk_expr(self, expr);
1101 }
1102
1103 fn visit_pat(&mut self, pat: &'tcx hir::Pat<'tcx>) {
1104 if let PatKind::Struct(ref qpath, fields, _) = pat.kind {
1105 let res = self.typeck_results().qpath_res(qpath, pat.hir_id);
1106 let adt = self.typeck_results().pat_ty(pat).ty_adt_def().unwrap();
1107 let variant = adt.variant_of_res(res);
1108 let mut failed_fields = ::alloc::vec::Vec::new()vec![];
1109 for field in fields {
1110 let (hir_id, use_ctxt) = (field.hir_id, field.ident.span);
1111 let index = self.typeck_results().field_index(field.hir_id);
1112 if self.check_field(hir_id, use_ctxt, adt, &variant.fields[index]) {
1113 failed_fields.push((field.ident.name, field.ident.span, true));
1114 }
1115 }
1116 self.emit_unreachable_field_error(failed_fields, adt, None, qpath.span());
1117 }
1118
1119 intravisit::walk_pat(self, pat);
1120 }
1121}
1122
1123struct TypePrivacyVisitor<'tcx> {
1128 tcx: TyCtxt<'tcx>,
1129 module_def_id: LocalModDefId,
1130 maybe_typeck_results: Option<&'tcx ty::TypeckResults<'tcx>>,
1131 span: Span,
1132}
1133
1134impl<'tcx> TypePrivacyVisitor<'tcx> {
1135 fn item_is_accessible(&self, did: DefId) -> bool {
1136 self.tcx.visibility(did).is_accessible_from(self.module_def_id, self.tcx)
1137 }
1138
1139 fn check_expr_pat_type(&mut self, id: hir::HirId, span: Span) -> bool {
1141 self.span = span;
1142 let typeck_results = self
1143 .maybe_typeck_results
1144 .unwrap_or_else(|| ::rustc_middle::util::bug::span_bug_fmt(span,
format_args!("`hir::Expr` or `hir::Pat` outside of a body"))span_bug!(span, "`hir::Expr` or `hir::Pat` outside of a body"));
1145 try {
1146 self.visit(typeck_results.node_type(id))?;
1147 self.visit(typeck_results.node_args(id))?;
1148 if let Some(adjustments) = typeck_results.adjustments().get(id) {
1149 adjustments.iter().try_for_each(|adjustment| self.visit(adjustment.target))?;
1150 }
1151 }
1152 .is_break()
1153 }
1154
1155 fn check_def_id(&self, def_id: DefId, kind: &str, descr: &dyn fmt::Display) -> bool {
1156 let is_error = !self.item_is_accessible(def_id);
1157 if is_error {
1158 self.tcx.dcx().emit_err(ItemIsPrivate { span: self.span, kind, descr: descr.into() });
1159 }
1160 is_error
1161 }
1162}
1163
1164impl<'tcx> rustc_ty_utils::sig_types::SpannedTypeVisitor<'tcx> for TypePrivacyVisitor<'tcx> {
1165 type Result = ControlFlow<()>;
1166 fn visit(&mut self, span: Span, value: impl TypeVisitable<TyCtxt<'tcx>>) -> Self::Result {
1167 self.span = span;
1168 value.visit_with(&mut self.skeleton())
1169 }
1170}
1171
1172impl<'tcx> Visitor<'tcx> for TypePrivacyVisitor<'tcx> {
1173 fn visit_nested_body(&mut self, body_id: hir::BodyId) {
1174 let old_maybe_typeck_results =
1175 self.maybe_typeck_results.replace(self.tcx.typeck_body(body_id));
1176 self.visit_body(self.tcx.hir_body(body_id));
1177 self.maybe_typeck_results = old_maybe_typeck_results;
1178 }
1179
1180 fn visit_ty(&mut self, hir_ty: &'tcx hir::Ty<'tcx, AmbigArg>) {
1181 self.span = hir_ty.span;
1182 if self
1183 .visit(
1184 self.maybe_typeck_results
1185 .unwrap_or_else(|| ::rustc_middle::util::bug::span_bug_fmt(hir_ty.span,
format_args!("`hir::Ty` outside of a body"))span_bug!(hir_ty.span, "`hir::Ty` outside of a body"))
1186 .node_type(hir_ty.hir_id),
1187 )
1188 .is_break()
1189 {
1190 return;
1191 }
1192
1193 intravisit::walk_ty(self, hir_ty);
1194 }
1195
1196 fn visit_infer(
1197 &mut self,
1198 inf_id: rustc_hir::HirId,
1199 inf_span: Span,
1200 _kind: InferKind<'tcx>,
1201 ) -> Self::Result {
1202 self.span = inf_span;
1203 if let Some(ty) = self
1204 .maybe_typeck_results
1205 .unwrap_or_else(|| ::rustc_middle::util::bug::span_bug_fmt(inf_span,
format_args!("Inference variable outside of a body"))span_bug!(inf_span, "Inference variable outside of a body"))
1206 .node_type_opt(inf_id)
1207 {
1208 if self.visit(ty).is_break() {
1209 return;
1210 }
1211 } else {
1212 }
1214
1215 self.visit_id(inf_id)
1216 }
1217
1218 fn visit_expr(&mut self, expr: &'tcx hir::Expr<'tcx>) {
1220 if self.check_expr_pat_type(expr.hir_id, expr.span) {
1221 return;
1223 }
1224 match expr.kind {
1225 hir::ExprKind::Assign(_, rhs, _) | hir::ExprKind::Match(rhs, ..) => {
1226 if self.check_expr_pat_type(rhs.hir_id, rhs.span) {
1228 return;
1229 }
1230 }
1231 hir::ExprKind::MethodCall(segment, ..) => {
1232 self.span = segment.ident.span;
1234 let typeck_results = self
1235 .maybe_typeck_results
1236 .unwrap_or_else(|| ::rustc_middle::util::bug::span_bug_fmt(self.span,
format_args!("`hir::Expr` outside of a body"))span_bug!(self.span, "`hir::Expr` outside of a body"));
1237 if let Some(def_id) = typeck_results.type_dependent_def_id(expr.hir_id) {
1238 if self
1239 .visit(self.tcx.type_of(def_id).instantiate_identity().skip_norm_wip())
1240 .is_break()
1241 {
1242 return;
1243 }
1244 } else {
1245 self.tcx
1246 .dcx()
1247 .span_delayed_bug(expr.span, "no type-dependent def for method call");
1248 }
1249 }
1250 _ => {}
1251 }
1252
1253 intravisit::walk_expr(self, expr);
1254 }
1255
1256 fn visit_qpath(&mut self, qpath: &'tcx hir::QPath<'tcx>, id: hir::HirId, span: Span) {
1263 let def = match qpath {
1264 hir::QPath::Resolved(_, path) => match path.res {
1265 Res::Def(kind, def_id) => Some((kind, def_id)),
1266 _ => None,
1267 },
1268 hir::QPath::TypeRelative(..) => {
1269 match self.maybe_typeck_results {
1270 Some(typeck_results) => typeck_results.type_dependent_def(id),
1271 None => None,
1273 }
1274 }
1275 };
1276 let def = def.filter(|(kind, _)| {
1277 #[allow(non_exhaustive_omitted_patterns)] match kind {
DefKind::AssocFn | DefKind::AssocConst { .. } | DefKind::AssocTy |
DefKind::Static { .. } => true,
_ => false,
}matches!(
1278 kind,
1279 DefKind::AssocFn
1280 | DefKind::AssocConst { .. }
1281 | DefKind::AssocTy
1282 | DefKind::Static { .. }
1283 )
1284 });
1285 if let Some((kind, def_id)) = def {
1286 let is_local_static =
1287 if let DefKind::Static { .. } = kind { def_id.is_local() } else { false };
1288 if !self.item_is_accessible(def_id) && !is_local_static {
1289 let name = match *qpath {
1290 hir::QPath::Resolved(_, path) => Some(self.tcx.def_path_str(path.res.def_id())),
1291 hir::QPath::TypeRelative(_, segment) => Some(segment.ident.to_string()),
1292 };
1293 let kind = self.tcx.def_descr(def_id);
1294 let sess = self.tcx.sess;
1295 let _ = match name {
1296 Some(name) => {
1297 sess.dcx().emit_err(ItemIsPrivate { span, kind, descr: (&name).into() })
1298 }
1299 None => sess.dcx().emit_err(UnnamedItemIsPrivate { span, kind }),
1300 };
1301 return;
1302 }
1303 }
1304
1305 intravisit::walk_qpath(self, qpath, id);
1306 }
1307
1308 fn visit_pat(&mut self, pattern: &'tcx hir::Pat<'tcx>) {
1310 if self.check_expr_pat_type(pattern.hir_id, pattern.span) {
1311 return;
1313 }
1314
1315 intravisit::walk_pat(self, pattern);
1316 }
1317
1318 fn visit_local(&mut self, local: &'tcx hir::LetStmt<'tcx>) {
1319 if let Some(init) = local.init {
1320 if self.check_expr_pat_type(init.hir_id, init.span) {
1321 return;
1323 }
1324 }
1325
1326 intravisit::walk_local(self, local);
1327 }
1328}
1329
1330impl<'tcx> DefIdVisitor<'tcx> for TypePrivacyVisitor<'tcx> {
1331 type Result = ControlFlow<()>;
1332 fn tcx(&self) -> TyCtxt<'tcx> {
1333 self.tcx
1334 }
1335 fn visit_def_id(
1336 &mut self,
1337 def_id: DefId,
1338 kind: &str,
1339 descr: &dyn fmt::Display,
1340 ) -> Self::Result {
1341 if self.check_def_id(def_id, kind, descr) {
1342 ControlFlow::Break(())
1343 } else {
1344 ControlFlow::Continue(())
1345 }
1346 }
1347}
1348
1349struct SearchInterfaceForPrivateItemsVisitor<'tcx> {
1355 tcx: TyCtxt<'tcx>,
1356 item_def_id: LocalDefId,
1357 required_visibility: ty::Visibility,
1359 required_effective_vis: Option<EffectiveVisibility>,
1360 hard_error: bool = false,
1361 in_primary_interface: bool = true,
1362 skip_assoc_tys: bool = false,
1363}
1364
1365impl SearchInterfaceForPrivateItemsVisitor<'_> {
1366 fn generics(&mut self) -> &mut Self {
1367 self.in_primary_interface = true;
1368 for param in &self.tcx.generics_of(self.item_def_id).own_params {
1369 if let GenericParamDefKind::Const { .. } = param.kind {
1370 let _ = self
1371 .visit(self.tcx.type_of(param.def_id).instantiate_identity().skip_norm_wip());
1372 }
1373 if let Some(default) = param.default_value(self.tcx) {
1374 let _ = self.visit(default.instantiate_identity().skip_norm_wip());
1375 }
1376 }
1377 self
1378 }
1379
1380 fn predicates(&mut self) -> &mut Self {
1381 self.in_primary_interface = false;
1382 let _ = self.visit_predicates(self.tcx.explicit_predicates_of(self.item_def_id));
1389 self
1390 }
1391
1392 fn bounds(&mut self) -> &mut Self {
1393 self.in_primary_interface = false;
1394 let _ = self.visit_clauses(self.tcx.explicit_item_bounds(self.item_def_id).skip_binder());
1395 self
1396 }
1397
1398 fn ty(&mut self) -> &mut Self {
1399 self.in_primary_interface = true;
1400 let _ =
1401 self.visit(self.tcx.type_of(self.item_def_id).instantiate_identity().skip_norm_wip());
1402 self
1403 }
1404
1405 fn trait_ref(&mut self) -> &mut Self {
1406 self.in_primary_interface = true;
1407 let _ = self.visit_trait(
1408 self.tcx.impl_trait_ref(self.item_def_id).instantiate_identity().skip_norm_wip(),
1409 );
1410 self
1411 }
1412
1413 fn check_def_id(&self, def_id: DefId, kind: &str, descr: &dyn fmt::Display) -> bool {
1414 if self.leaks_private_dep(def_id) {
1415 self.tcx.emit_node_span_lint(
1416 lint::builtin::EXPORTED_PRIVATE_DEPENDENCIES,
1417 self.tcx.local_def_id_to_hir_id(self.item_def_id),
1418 self.tcx.def_span(self.item_def_id.to_def_id()),
1419 FromPrivateDependencyInPublicInterface {
1420 kind,
1421 descr: descr.into(),
1422 krate: self.tcx.crate_name(def_id.krate),
1423 },
1424 );
1425 }
1426
1427 let Some(local_def_id) = def_id.as_local() else {
1428 return false;
1429 };
1430
1431 let vis = self.tcx.local_visibility(local_def_id);
1432 if self.hard_error && self.required_visibility.greater_than(vis, self.tcx) {
1433 let vis_descr = match vis {
1434 ty::Visibility::Public => "public",
1435 ty::Visibility::Restricted(vis_def_id) => {
1436 if vis_def_id
1437 == self.tcx.parent_module_from_def_id(local_def_id).to_local_def_id()
1438 {
1439 "private"
1440 } else if vis_def_id.is_top_level_module() {
1441 "crate-private"
1442 } else {
1443 "restricted"
1444 }
1445 }
1446 };
1447
1448 let span = self.tcx.def_span(self.item_def_id.to_def_id());
1449 let vis_span = self.tcx.def_span(def_id);
1450 self.tcx.dcx().emit_err(InPublicInterface {
1451 span,
1452 vis_descr,
1453 kind,
1454 descr: descr.into(),
1455 vis_span,
1456 });
1457 return false;
1458 }
1459
1460 let Some(effective_vis) = self.required_effective_vis else {
1461 return false;
1462 };
1463
1464 let reachable_at_vis = *effective_vis.at_level(Level::Reachable);
1465
1466 if reachable_at_vis.greater_than(vis, self.tcx) {
1467 let lint = if self.in_primary_interface {
1468 lint::builtin::PRIVATE_INTERFACES
1469 } else {
1470 lint::builtin::PRIVATE_BOUNDS
1471 };
1472 let span = self.tcx.def_span(self.item_def_id.to_def_id());
1473 let vis_span = self.tcx.def_span(def_id);
1474 self.tcx.emit_node_span_lint(
1475 lint,
1476 self.tcx.local_def_id_to_hir_id(self.item_def_id),
1477 span,
1478 PrivateInterfacesOrBoundsLint {
1479 item_span: span,
1480 item_kind: self.tcx.def_descr(self.item_def_id.to_def_id()),
1481 item_descr: (&LazyDefPathStr {
1482 def_id: self.item_def_id.to_def_id(),
1483 tcx: self.tcx,
1484 })
1485 .into(),
1486 item_vis_descr: &reachable_at_vis.to_string(self.item_def_id, self.tcx),
1487 ty_span: vis_span,
1488 ty_kind: kind,
1489 ty_descr: descr.into(),
1490 ty_vis_descr: &vis.to_string(local_def_id, self.tcx),
1491 },
1492 );
1493 }
1494
1495 false
1496 }
1497
1498 fn leaks_private_dep(&self, item_id: DefId) -> bool {
1503 let ret = self.required_visibility.is_public() && self.tcx.is_private_dep(item_id.krate);
1504
1505 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_privacy/src/lib.rs:1505",
"rustc_privacy", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_privacy/src/lib.rs"),
::tracing_core::__macro_support::Option::Some(1505u32),
::tracing_core::__macro_support::Option::Some("rustc_privacy"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("leaks_private_dep(item_id={0:?})={1}",
item_id, ret) as &dyn Value))])
});
} else { ; }
};debug!("leaks_private_dep(item_id={:?})={}", item_id, ret);
1506 ret
1507 }
1508}
1509
1510impl<'tcx> DefIdVisitor<'tcx> for SearchInterfaceForPrivateItemsVisitor<'tcx> {
1511 type Result = ControlFlow<()>;
1512 fn skip_assoc_tys(&self) -> bool {
1513 self.skip_assoc_tys
1514 }
1515 fn tcx(&self) -> TyCtxt<'tcx> {
1516 self.tcx
1517 }
1518 fn visit_def_id(
1519 &mut self,
1520 def_id: DefId,
1521 kind: &str,
1522 descr: &dyn fmt::Display,
1523 ) -> Self::Result {
1524 if self.check_def_id(def_id, kind, descr) {
1525 ControlFlow::Break(())
1526 } else {
1527 ControlFlow::Continue(())
1528 }
1529 }
1530}
1531
1532struct PrivateItemsInPublicInterfacesChecker<'a, 'tcx> {
1533 tcx: TyCtxt<'tcx>,
1534 effective_visibilities: &'a EffectiveVisibilities,
1535}
1536
1537impl<'tcx> PrivateItemsInPublicInterfacesChecker<'_, 'tcx> {
1538 fn check(
1539 &self,
1540 def_id: LocalDefId,
1541 required_visibility: ty::Visibility,
1542 required_effective_vis: Option<EffectiveVisibility>,
1543 ) -> SearchInterfaceForPrivateItemsVisitor<'tcx> {
1544 SearchInterfaceForPrivateItemsVisitor {
1545 tcx: self.tcx,
1546 item_def_id: def_id,
1547 required_visibility,
1548 required_effective_vis,
1549 ..
1550 }
1551 }
1552
1553 fn check_unnameable(&self, def_id: LocalDefId, effective_vis: Option<EffectiveVisibility>) {
1554 let Some(effective_vis) = effective_vis else {
1555 return;
1556 };
1557
1558 let reexported_at_vis = effective_vis.at_level(Level::Reexported);
1559 let reachable_at_vis = effective_vis.at_level(Level::Reachable);
1560
1561 if reachable_at_vis.is_public() && reexported_at_vis != reachable_at_vis {
1562 let hir_id = self.tcx.local_def_id_to_hir_id(def_id);
1563 let span = self.tcx.def_span(def_id.to_def_id());
1564 self.tcx.emit_node_span_lint(
1565 lint::builtin::UNNAMEABLE_TYPES,
1566 hir_id,
1567 span,
1568 UnnameableTypesLint {
1569 span,
1570 kind: self.tcx.def_descr(def_id.to_def_id()),
1571 descr: (&LazyDefPathStr { def_id: def_id.to_def_id(), tcx: self.tcx }).into(),
1572 reachable_vis: &reachable_at_vis.to_string(def_id, self.tcx),
1573 reexported_vis: &reexported_at_vis.to_string(def_id, self.tcx),
1574 },
1575 );
1576 }
1577 }
1578
1579 fn check_assoc_item(
1580 &self,
1581 item: &ty::AssocItem,
1582 vis: ty::Visibility,
1583 effective_vis: Option<EffectiveVisibility>,
1584 ) {
1585 let mut check = self.check(item.def_id.expect_local(), vis, effective_vis);
1586
1587 let is_assoc_ty = item.is_type();
1588 check.hard_error = is_assoc_ty;
1589 check.generics().predicates();
1590 if assoc_has_type_of(self.tcx, item) {
1591 check.ty();
1592 }
1593 if is_assoc_ty && item.container == AssocContainer::Trait {
1594 check.hard_error = false;
1597 check.bounds();
1598 }
1599 }
1600
1601 fn get(&self, def_id: LocalDefId) -> Option<EffectiveVisibility> {
1602 self.effective_visibilities.effective_vis(def_id).copied()
1603 }
1604
1605 fn check_item(&self, id: ItemId) {
1606 let tcx = self.tcx;
1607 let def_id = id.owner_id.def_id;
1608 let item_visibility = tcx.local_visibility(def_id);
1609 let effective_vis = self.get(def_id);
1610 let def_kind = tcx.def_kind(def_id);
1611
1612 match def_kind {
1613 DefKind::Const { .. } | DefKind::Static { .. } | DefKind::Fn | DefKind::TyAlias => {
1614 if let DefKind::TyAlias = def_kind {
1615 self.check_unnameable(def_id, effective_vis);
1616 }
1617 self.check(def_id, item_visibility, effective_vis).generics().predicates().ty();
1618 }
1619 DefKind::OpaqueTy => {
1620 self.check(def_id, item_visibility, effective_vis).generics().bounds();
1623 }
1624 DefKind::Trait => {
1625 self.check_unnameable(def_id, effective_vis);
1626
1627 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1628
1629 for assoc_item in tcx.associated_items(id.owner_id).in_definition_order() {
1630 self.check_assoc_item(assoc_item, item_visibility, effective_vis);
1631 }
1632 }
1633 DefKind::TraitAlias => {
1634 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1635 }
1636 DefKind::Enum => {
1637 self.check_unnameable(def_id, effective_vis);
1638 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1639
1640 let adt = tcx.adt_def(id.owner_id);
1641 for field in adt.all_fields() {
1642 self.check(field.did.expect_local(), item_visibility, effective_vis).ty();
1643 }
1644 }
1645 DefKind::Struct | DefKind::Union => {
1647 self.check_unnameable(def_id, effective_vis);
1648 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1649
1650 let adt = tcx.adt_def(id.owner_id);
1651 for field in adt.all_fields() {
1652 let visibility = min(item_visibility, field.vis.expect_local(), tcx);
1653 let field_ev = self.get(field.did.expect_local());
1654
1655 self.check(field.did.expect_local(), visibility, field_ev).ty();
1656 }
1657 }
1658 DefKind::ForeignMod => {}
1660 DefKind::Impl { of_trait } => {
1665 let impl_vis =
1666 ty::Visibility::of_impl::<false>(def_id, of_trait, tcx, &Default::default());
1667
1668 let impl_ev = EffectiveVisibility::of_impl::<false>(
1680 def_id,
1681 of_trait,
1682 tcx,
1683 self.effective_visibilities,
1684 );
1685
1686 let mut check = self.check(def_id, impl_vis, Some(impl_ev));
1687
1688 if !of_trait {
1691 check.generics().predicates();
1692 }
1693
1694 check.skip_assoc_tys = true;
1698 check.ty();
1699 if of_trait {
1700 check.trait_ref();
1701 }
1702
1703 for assoc_item in tcx.associated_items(id.owner_id).in_definition_order() {
1704 let impl_item_vis = if !of_trait {
1705 min(tcx.local_visibility(assoc_item.def_id.expect_local()), impl_vis, tcx)
1706 } else {
1707 impl_vis
1708 };
1709
1710 let impl_item_ev = if !of_trait {
1711 self.get(assoc_item.def_id.expect_local())
1712 .map(|ev| ev.min(impl_ev, self.tcx))
1713 } else {
1714 Some(impl_ev)
1715 };
1716
1717 self.check_assoc_item(assoc_item, impl_item_vis, impl_item_ev);
1718 }
1719 }
1720 _ => {}
1721 }
1722 }
1723
1724 fn check_foreign_item(&self, id: ForeignItemId) {
1725 let tcx = self.tcx;
1726 let def_id = id.owner_id.def_id;
1727 let item_visibility = tcx.local_visibility(def_id);
1728 let effective_vis = self.get(def_id);
1729
1730 if let DefKind::ForeignTy = self.tcx.def_kind(def_id) {
1731 self.check_unnameable(def_id, effective_vis);
1732 }
1733
1734 self.check(def_id, item_visibility, effective_vis).generics().predicates().ty();
1735 }
1736}
1737
1738pub fn provide(providers: &mut Providers) {
1739 *providers = Providers {
1740 effective_visibilities,
1741 check_private_in_public,
1742 check_mod_privacy,
1743 ..*providers
1744 };
1745}
1746
1747fn check_mod_privacy(tcx: TyCtxt<'_>, module_def_id: LocalModDefId) {
1748 let mut visitor = NamePrivacyVisitor { tcx, maybe_typeck_results: None };
1750 tcx.hir_visit_item_likes_in_module(module_def_id, &mut visitor);
1751
1752 let span = tcx.def_span(module_def_id);
1755 let mut visitor = TypePrivacyVisitor { tcx, module_def_id, maybe_typeck_results: None, span };
1756
1757 let module = tcx.hir_module_items(module_def_id);
1758 for def_id in module.definitions() {
1759 let _ = rustc_ty_utils::sig_types::walk_types(tcx, def_id, &mut visitor);
1760
1761 if let Some(body_id) = tcx.hir_maybe_body_owned_by(def_id) {
1762 visitor.visit_nested_body(body_id.id());
1763 }
1764
1765 if let DefKind::Impl { of_trait: true } = tcx.def_kind(def_id) {
1766 let trait_ref = tcx.impl_trait_ref(def_id);
1767 let trait_ref = trait_ref.instantiate_identity().skip_norm_wip();
1768 visitor.span =
1769 tcx.hir_expect_item(def_id).expect_impl().of_trait.unwrap().trait_ref.path.span;
1770 let _ =
1771 visitor.visit_def_id(trait_ref.def_id, "trait", &trait_ref.print_only_trait_path());
1772 }
1773 }
1774}
1775
1776fn effective_visibilities(tcx: TyCtxt<'_>, (): ()) -> &EffectiveVisibilities {
1777 let def_ids_to_impls = DefIdsToImplsCollector::collect(tcx);
1778
1779 let mut visitor = EmbargoVisitor {
1782 tcx,
1783 effective_visibilities: tcx.resolutions(()).effective_visibilities.clone(),
1784 queue: Default::default(),
1785 def_ids_to_impls,
1786 };
1787
1788 visitor.effective_visibilities.check_invariants(tcx);
1789
1790 let impl_trait_pass = !tcx.sess.opts.actually_rustdoc;
1794 if impl_trait_pass {
1795 let krate = tcx.hir_crate_items(());
1798 for id in krate.opaques() {
1799 let opaque = tcx.hir_node_by_def_id(id).expect_opaque_ty();
1800 let should_visit = match opaque.origin {
1801 hir::OpaqueTyOrigin::FnReturn {
1802 parent,
1803 in_trait_or_impl: Some(hir::RpitContext::Trait),
1804 }
1805 | hir::OpaqueTyOrigin::AsyncFn {
1806 parent,
1807 in_trait_or_impl: Some(hir::RpitContext::Trait),
1808 } => match tcx.hir_node_by_def_id(parent).expect_trait_item().expect_fn().1 {
1809 hir::TraitFn::Required(_) => false,
1810 hir::TraitFn::Provided(..) => true,
1811 },
1812
1813 hir::OpaqueTyOrigin::FnReturn {
1816 in_trait_or_impl: None | Some(hir::RpitContext::TraitImpl),
1817 ..
1818 }
1819 | hir::OpaqueTyOrigin::AsyncFn {
1820 in_trait_or_impl: None | Some(hir::RpitContext::TraitImpl),
1821 ..
1822 }
1823 | hir::OpaqueTyOrigin::TyAlias { .. } => true,
1824 };
1825 if should_visit {
1826 let pub_ev = EffectiveVisibility::from_vis(ty::Visibility::Public);
1830 visitor
1831 .reach_through_impl_trait(opaque.def_id, pub_ev)
1832 .generics()
1833 .predicates()
1834 .ty();
1835 }
1836 }
1837
1838 visitor.queue.clear();
1839 }
1840
1841 for (&adt_def_id, macro_mods) in &tcx.resolutions(()).macro_reachable_adts {
1844 let struct_def = tcx.adt_def(adt_def_id);
1845 let Some(struct_ev) = visitor.effective_visibilities.effective_vis(adt_def_id).copied()
1846 else {
1847 continue;
1848 };
1849 for field in &struct_def.non_enum_variant().fields {
1850 let def_id = field.did.expect_local();
1851 let field_vis = tcx.local_visibility(def_id);
1852
1853 for ¯o_mod in macro_mods {
1854 if field_vis.is_accessible_from(macro_mod, tcx) {
1855 visitor.reach(def_id, struct_ev).ty();
1856 }
1857 }
1858 }
1859 }
1860
1861 let crate_items = tcx.hir_crate_items(());
1862 for id in crate_items.free_items() {
1863 visitor.check_def_id(id.owner_id.def_id);
1864 }
1865 for id in crate_items.foreign_items() {
1866 visitor.check_def_id(id.owner_id.def_id);
1867 }
1868 while let Some(def_id) = visitor.queue.pop() {
1869 visitor.check_def_id(def_id);
1870 }
1871 visitor.effective_visibilities.check_invariants(tcx);
1872
1873 let check_visitor =
1874 TestReachabilityVisitor { tcx, effective_visibilities: &visitor.effective_visibilities };
1875 for id in crate_items.owners() {
1876 check_visitor.check_def_id(id);
1877 }
1878
1879 tcx.arena.alloc(visitor.effective_visibilities)
1880}
1881
1882fn check_private_in_public(tcx: TyCtxt<'_>, module_def_id: LocalModDefId) {
1883 let effective_visibilities = tcx.effective_visibilities(());
1884 let checker = PrivateItemsInPublicInterfacesChecker { tcx, effective_visibilities };
1886
1887 let crate_items = tcx.hir_module_items(module_def_id);
1888 let _ = crate_items.par_items(|id| Ok(checker.check_item(id)));
1889 let _ = crate_items.par_foreign_items(|id| Ok(checker.check_foreign_item(id)));
1890}