1#![feature(associated_type_defaults)]
3#![feature(default_field_values)]
4#![feature(try_blocks)]
5mod diagnostics;
8
9use std::marker::PhantomData;
10use std::ops::ControlFlow;
11use std::{debug_assert_matches, fmt};
12
13use diagnostics::{
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 _,
219 data @ ty::AliasTy {
220 kind:
221 kind @ (ty::Inherent { def_id }
222 | ty::Free { def_id }
223 | ty::Projection { def_id }),
224 ..
225 },
226 ) => {
227 if self.def_id_visitor.skip_assoc_tys() {
228 return V::Result::output();
234 }
235 if !self.visited_tys.insert(ty) {
236 return V::Result::output();
240 }
241
242 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(
243 def_id,
244 match kind {
245 ty::Inherent { .. } | ty::Projection { .. } => "associated type",
246 ty::Free { .. } => "type alias",
247 ty::Opaque { .. } => unreachable!(),
248 },
249 &LazyDefPathStr { def_id, tcx },
250 ));
251
252 return if V::SHALLOW {
254 V::Result::output()
255 } else if #[allow(non_exhaustive_omitted_patterns)] match kind {
ty::Projection { .. } => true,
_ => false,
}matches!(kind, ty::Projection { .. }) {
256 self.visit_projection_term(data.into())
257 } else {
258 V::Result::from_branch(
259 data.args.iter().try_for_each(|arg| arg.visit_with(self).branch()),
260 )
261 };
262 }
263 ty::Dynamic(predicates, ..) => {
264 for predicate in predicates {
267 let trait_ref = match predicate.skip_binder() {
268 ty::ExistentialPredicate::Trait(trait_ref) => trait_ref,
269 ty::ExistentialPredicate::Projection(proj) => proj.trait_ref(tcx),
270 ty::ExistentialPredicate::AutoTrait(def_id) => {
271 ty::ExistentialTraitRef::new(tcx, def_id, ty::GenericArgs::empty())
272 }
273 };
274 let ty::ExistentialTraitRef { def_id, .. } = trait_ref;
275 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));
276 }
277 }
278 ty::Alias(_, ty::AliasTy { kind: ty::Opaque { def_id }, .. }) => {
279 if self.visited_tys.insert(ty) {
281 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()));
289 }
290 }
291 ty::Bool
294 | ty::Char
295 | ty::Int(..)
296 | ty::Uint(..)
297 | ty::Float(..)
298 | ty::Str
299 | ty::Never
300 | ty::Array(..)
301 | ty::Slice(..)
302 | ty::Tuple(..)
303 | ty::RawPtr(..)
304 | ty::Ref(..)
305 | ty::Pat(..)
306 | ty::FnPtr(..)
307 | ty::UnsafeBinder(_)
308 | ty::Param(..)
309 | ty::Bound(..)
310 | ty::Error(_)
311 | ty::CoroutineWitness(..) => {}
312 ty::Placeholder(..) | ty::Infer(..) => {
313 ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected type: {0:?}", ty))bug!("unexpected type: {:?}", ty)
314 }
315 }
316
317 if V::SHALLOW { V::Result::output() } else { ty.super_visit_with(self) }
318 }
319
320 fn visit_const(&mut self, c: Const<'tcx>) -> Self::Result {
321 let tcx = self.def_id_visitor.tcx();
322 tcx.expand_abstract_consts(c).super_visit_with(self)
323 }
324}
325
326fn assoc_has_type_of(tcx: TyCtxt<'_>, item: &ty::AssocItem) -> bool {
327 if let ty::AssocKind::Type { data: ty::AssocTypeData::Normal(..) } = item.kind
328 && let hir::Node::TraitItem(item) =
329 tcx.hir_node(tcx.local_def_id_to_hir_id(item.def_id.expect_local()))
330 && let hir::TraitItemKind::Type(_, None) = item.kind
331 {
332 false
333 } else {
334 true
335 }
336}
337
338fn min(vis1: ty::Visibility, vis2: ty::Visibility, tcx: TyCtxt<'_>) -> ty::Visibility {
339 if vis1.greater_than(vis2, tcx) { vis2 } else { vis1 }
340}
341
342struct FindMin<'a, 'tcx, VL: VisibilityLike, const SHALLOW: bool> {
344 tcx: TyCtxt<'tcx>,
345 effective_visibilities: &'a EffectiveVisibilities,
346 min: VL,
347}
348
349impl<'a, 'tcx, VL: VisibilityLike, const SHALLOW: bool> DefIdVisitor<'tcx>
350 for FindMin<'a, 'tcx, VL, SHALLOW>
351{
352 const SHALLOW: bool = SHALLOW;
353 fn skip_assoc_tys(&self) -> bool {
354 true
355 }
356 fn tcx(&self) -> TyCtxt<'tcx> {
357 self.tcx
358 }
359 fn visit_def_id(&mut self, def_id: DefId, _kind: &str, _descr: &dyn fmt::Display) {
360 if let Some(def_id) = def_id.as_local() {
361 self.min = VL::new_min(self, def_id);
362 }
363 }
364}
365
366trait VisibilityLike: Sized {
367 const MAX: Self;
368 fn new_min<const SHALLOW: bool>(
369 find: &FindMin<'_, '_, Self, SHALLOW>,
370 def_id: LocalDefId,
371 ) -> Self;
372
373 fn of_impl<const SHALLOW: bool>(
376 def_id: LocalDefId,
377 of_trait: bool,
378 tcx: TyCtxt<'_>,
379 effective_visibilities: &EffectiveVisibilities,
380 ) -> Self {
381 let mut find = FindMin::<_, SHALLOW> { tcx, effective_visibilities, min: Self::MAX };
382 find.visit(tcx.type_of(def_id).instantiate_identity().skip_norm_wip());
383 if of_trait {
384 find.visit_trait(tcx.impl_trait_ref(def_id).instantiate_identity().skip_norm_wip());
385 }
386 find.min
387 }
388}
389
390impl VisibilityLike for ty::Visibility {
391 const MAX: Self = ty::Visibility::Public;
392 fn new_min<const SHALLOW: bool>(
393 find: &FindMin<'_, '_, Self, SHALLOW>,
394 def_id: LocalDefId,
395 ) -> Self {
396 min(find.tcx.local_visibility(def_id), find.min, find.tcx)
397 }
398}
399
400impl VisibilityLike for EffectiveVisibility {
401 const MAX: Self = EffectiveVisibility::from_vis(ty::Visibility::Public);
402 fn new_min<const SHALLOW: bool>(
403 find: &FindMin<'_, '_, Self, SHALLOW>,
404 def_id: LocalDefId,
405 ) -> Self {
406 let effective_vis =
407 find.effective_visibilities.effective_vis(def_id).copied().unwrap_or_else(|| {
408 let private_vis = ty::Visibility::Restricted(
409 find.tcx.parent_module_from_def_id(def_id).to_local_def_id(),
410 );
411 EffectiveVisibility::from_vis(private_vis)
412 });
413
414 effective_vis.min(find.min, find.tcx)
415 }
416}
417
418type DefIdsToImpls = FxHashMap<LocalDefId, FxHashSet<LocalDefId>>;
419
420struct DefIdsToImplsCollector<'tcx, 'a> {
423 tcx: TyCtxt<'tcx>,
424 def_ids_to_impls: &'a mut DefIdsToImpls,
425 impl_def_id: LocalDefId,
426}
427
428impl<'tcx, 'a> DefIdsToImplsCollector<'tcx, 'a> {
429 fn collect(tcx: TyCtxt<'tcx>) -> DefIdsToImpls {
430 let mut def_ids_to_impls = Default::default();
431 for item in tcx.hir_free_items() {
432 let impl_def_id = item.owner_id.def_id;
433 let DefKind::Impl { of_trait } = tcx.def_kind(impl_def_id) else {
434 continue;
435 };
436
437 let mut visitor = DefIdsToImplsCollector {
439 tcx,
440 impl_def_id,
441 def_ids_to_impls: &mut def_ids_to_impls,
442 };
443
444 visitor.visit(tcx.type_of(impl_def_id).instantiate_identity().skip_norm_wip());
445 if of_trait {
446 visitor.visit_trait(
447 tcx.impl_trait_ref(impl_def_id).instantiate_identity().skip_norm_wip(),
448 );
449 }
450 }
451
452 def_ids_to_impls
453 }
454}
455
456impl<'tcx, 'a> DefIdVisitor<'tcx> for DefIdsToImplsCollector<'tcx, 'a> {
457 const SHALLOW: bool = true;
458 fn skip_assoc_tys(&self) -> bool {
459 true
460 }
461 fn tcx(&self) -> TyCtxt<'tcx> {
462 self.tcx
463 }
464 fn visit_def_id(&mut self, def_id: DefId, _kind: &str, _descr: &dyn fmt::Display) {
465 if let Some(def_id) = def_id.as_local() {
466 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!(
467 self.tcx.def_kind(def_id),
468 DefKind::Enum
469 | DefKind::Union
470 | DefKind::Struct
471 | DefKind::ForeignTy
472 | DefKind::Trait
473 );
474 self.def_ids_to_impls.entry(def_id).or_default().insert(self.impl_def_id);
475 }
476 }
477}
478
479struct EmbargoVisitor<'tcx> {
481 tcx: TyCtxt<'tcx>,
482 effective_visibilities: EffectiveVisibilities,
484 queue: IndexSet<LocalDefId>,
486 def_ids_to_impls: DefIdsToImpls,
488}
489
490struct ReachEverythingInTheInterfaceVisitor<'a, 'tcx> {
491 effective_vis: EffectiveVisibility,
492 item_def_id: LocalDefId,
493 ev: &'a mut EmbargoVisitor<'tcx>,
494 level: Level,
495}
496
497impl<'tcx> EmbargoVisitor<'tcx> {
498 fn get(&self, def_id: LocalDefId) -> Option<EffectiveVisibility> {
499 self.effective_visibilities.effective_vis(def_id).copied()
500 }
501
502 fn update(
504 &mut self,
505 def_id: LocalDefId,
506 inherited_effective_vis: EffectiveVisibility,
507 level: Level,
508 ) {
509 let nominal_vis = self.tcx.local_visibility(def_id);
510 self.update_eff_vis(def_id, inherited_effective_vis, Some(nominal_vis), level);
511 }
512
513 fn update_eff_vis(
514 &mut self,
515 def_id: LocalDefId,
516 inherited_effective_vis: EffectiveVisibility,
517 max_vis: Option<ty::Visibility>,
518 level: Level,
519 ) -> bool {
520 let private_vis =
522 ty::Visibility::Restricted(self.tcx.parent_module_from_def_id(def_id).into());
523 if max_vis != Some(private_vis) {
524 return self.effective_visibilities.update(
525 def_id,
526 max_vis,
527 private_vis,
528 inherited_effective_vis,
529 level,
530 self.tcx,
531 );
532 }
533 false
534 }
535
536 fn reach(
537 &mut self,
538 def_id: LocalDefId,
539 effective_vis: EffectiveVisibility,
540 ) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
541 ReachEverythingInTheInterfaceVisitor {
542 effective_vis,
543 item_def_id: def_id,
544 ev: self,
545 level: Level::Reachable,
546 }
547 }
548
549 fn reach_through_impl_trait(
550 &mut self,
551 def_id: LocalDefId,
552 effective_vis: EffectiveVisibility,
553 ) -> ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
554 ReachEverythingInTheInterfaceVisitor {
555 effective_vis,
556 item_def_id: def_id,
557 ev: self,
558 level: Level::ReachableThroughImplTrait,
559 }
560 }
561}
562
563impl<'tcx> EmbargoVisitor<'tcx> {
564 fn check_assoc_item(&mut self, item: &ty::AssocItem, item_ev: EffectiveVisibility) {
565 let def_id = item.def_id.expect_local();
566 let tcx = self.tcx;
567 let mut reach = self.reach(def_id, item_ev);
568 reach.generics().predicates();
569 if assoc_has_type_of(tcx, item) {
570 reach.ty();
571 }
572 if item.is_type() && item.container == AssocContainer::Trait {
573 reach.bounds();
574 }
575 }
576
577 fn check_def_id(&mut self, def_id: LocalDefId) {
578 let item_ev = self.get(def_id);
581 let def_kind = self.tcx.def_kind(def_id);
582 match def_kind {
583 DefKind::Use | DefKind::ExternCrate | DefKind::GlobalAsm => {}
585 DefKind::Mod => {}
587 DefKind::Macro { .. } => {}
589 DefKind::ForeignTy
590 | DefKind::Const { .. }
591 | DefKind::Static { .. }
592 | DefKind::Fn
593 | DefKind::TyAlias => {
594 if let Some(item_ev) = item_ev {
595 self.reach(def_id, item_ev).generics().predicates().ty();
596 }
597 }
598 DefKind::Trait => {
599 if let Some(item_ev) = item_ev {
600 self.reach(def_id, item_ev).generics().predicates();
601
602 for assoc_item in self.tcx.associated_items(def_id).in_definition_order() {
603 let def_id = assoc_item.def_id.expect_local();
604 self.update(def_id, item_ev, Level::Reachable);
605
606 self.check_assoc_item(assoc_item, item_ev);
607 }
608 }
609 }
610 DefKind::TraitAlias => {
611 if let Some(item_ev) = item_ev {
612 self.reach(def_id, item_ev).generics().predicates();
613 }
614 }
615 DefKind::Impl { of_trait } => {
616 let item_ev = EffectiveVisibility::of_impl::<true>(
627 def_id,
628 of_trait,
629 self.tcx,
630 &self.effective_visibilities,
631 );
632
633 self.update_eff_vis(def_id, item_ev, None, Level::Direct);
634
635 {
636 let mut reach = self.reach(def_id, item_ev);
637 reach.generics().predicates().ty();
638 if of_trait {
639 reach.trait_ref();
640 }
641 }
642
643 for assoc_item in self.tcx.associated_items(def_id).in_definition_order() {
644 let def_id = assoc_item.def_id.expect_local();
645 let max_vis =
646 if of_trait { None } else { Some(self.tcx.local_visibility(def_id)) };
647 self.update_eff_vis(def_id, item_ev, max_vis, Level::Direct);
648
649 if let Some(impl_item_ev) = self.get(def_id) {
650 self.check_assoc_item(assoc_item, impl_item_ev);
651 }
652 }
653 }
654 DefKind::Enum => {
655 if let Some(item_ev) = item_ev {
656 self.reach(def_id, item_ev).generics().predicates();
657 }
658 let def = self.tcx.adt_def(def_id);
659 for variant in def.variants() {
660 if let Some(item_ev) = item_ev {
661 self.update(variant.def_id.expect_local(), item_ev, Level::Reachable);
662 }
663
664 if let Some(variant_ev) = self.get(variant.def_id.expect_local()) {
665 if let Some(ctor_def_id) = variant.ctor_def_id() {
666 self.update(ctor_def_id.expect_local(), variant_ev, Level::Reachable);
667 }
668
669 for field in &variant.fields {
670 let field = field.did.expect_local();
671 self.update(field, variant_ev, Level::Reachable);
672 self.reach(field, variant_ev).ty();
673 }
674 self.reach(def_id, variant_ev).ty();
677 }
678 if let Some(ctor_def_id) = variant.ctor_def_id() {
679 if let Some(ctor_ev) = self.get(ctor_def_id.expect_local()) {
680 self.reach(def_id, ctor_ev).ty();
681 }
682 }
683 }
684 }
685 DefKind::Struct | DefKind::Union => {
686 let def = self.tcx.adt_def(def_id).non_enum_variant();
687 if let Some(item_ev) = item_ev {
688 self.reach(def_id, item_ev).generics().predicates();
689 for field in &def.fields {
690 let field = field.did.expect_local();
691 self.update(field, item_ev, Level::Reachable);
692 if let Some(field_ev) = self.get(field) {
693 self.reach(field, field_ev).ty();
694 }
695 }
696 }
697 if let Some(ctor_def_id) = def.ctor_def_id() {
698 if let Some(item_ev) = item_ev {
699 self.update(ctor_def_id.expect_local(), item_ev, Level::Reachable);
700 }
701 if let Some(ctor_ev) = self.get(ctor_def_id.expect_local()) {
702 self.reach(def_id, ctor_ev).ty();
703 }
704 }
705 }
706 DefKind::ForeignMod => {}
708 DefKind::Field
709 | DefKind::Variant
710 | DefKind::AssocFn
711 | DefKind::AssocTy
712 | DefKind::AssocConst { .. }
713 | DefKind::TyParam
714 | DefKind::AnonConst
715 | DefKind::InlineConst
716 | DefKind::OpaqueTy
717 | DefKind::Closure
718 | DefKind::SyntheticCoroutineBody
719 | DefKind::ConstParam
720 | DefKind::LifetimeParam
721 | DefKind::Ctor(..) => {
722 ::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!(
723 self.tcx.def_span(def_id),
724 "{def_kind:?} should be checked while checking parent"
725 )
726 }
727 }
728 }
729}
730
731impl ReachEverythingInTheInterfaceVisitor<'_, '_> {
732 fn generics(&mut self) -> &mut Self {
733 for param in &self.ev.tcx.generics_of(self.item_def_id).own_params {
734 if let GenericParamDefKind::Const { .. } = param.kind {
735 self.visit(
736 self.ev.tcx.type_of(param.def_id).instantiate_identity().skip_norm_wip(),
737 );
738 }
739 if let Some(default) = param.default_value(self.ev.tcx) {
740 self.visit(default.instantiate_identity().skip_norm_wip());
741 }
742 }
743 self
744 }
745
746 fn predicates(&mut self) -> &mut Self {
747 self.visit_predicates(self.ev.tcx.explicit_predicates_of(self.item_def_id));
748 self
749 }
750
751 fn bounds(&mut self) -> &mut Self {
752 self.visit_clauses(self.ev.tcx.explicit_item_bounds(self.item_def_id).skip_binder());
753 self
754 }
755
756 fn ty(&mut self) -> &mut Self {
757 self.visit(self.ev.tcx.type_of(self.item_def_id).instantiate_identity().skip_norm_wip());
758 self
759 }
760
761 fn trait_ref(&mut self) -> &mut Self {
762 self.visit_trait(
763 self.ev.tcx.impl_trait_ref(self.item_def_id).instantiate_identity().skip_norm_wip(),
764 );
765 self
766 }
767
768 fn enqueue_def_id(&mut self, def_id: LocalDefId) {
771 let def_kind = self.ev.tcx.def_kind(def_id);
772 match def_kind {
773 DefKind::Enum
774 | DefKind::Union
775 | DefKind::Struct
776 | DefKind::ForeignTy
777 | DefKind::Trait => {
778 self.ev.queue.insert(def_id);
779 if let Some(impls) = self.ev.def_ids_to_impls.get(&def_id) {
781 #[allow(rustc::potential_query_instability)]
783 self.ev.queue.extend(impls);
784 }
785 }
786
787 DefKind::TraitAlias | DefKind::Fn | DefKind::TyAlias => {
788 self.ev.queue.insert(def_id);
789 }
790
791 DefKind::AssocConst { .. } | DefKind::AssocFn | DefKind::AssocTy => {
792 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::Field
812 | DefKind::Variant
813 | DefKind::Static { .. }
814 | DefKind::Macro(_)
815 | DefKind::TyParam
816 | DefKind::AnonConst
817 | DefKind::InlineConst
818 | DefKind::OpaqueTy
819 | DefKind::SyntheticCoroutineBody
820 | DefKind::ConstParam
821 | DefKind::LifetimeParam
822 | DefKind::Mod
823 | DefKind::Use
824 | DefKind::ExternCrate
825 | DefKind::GlobalAsm
826 | DefKind::ForeignMod
827 | DefKind::Const { .. } => {
828 ::rustc_middle::util::bug::span_bug_fmt(self.tcx().def_span(def_id),
format_args!("{0:?} unexpectedly reached by `ReachEverythingInTheInterfaceVisitor`",
def_kind))span_bug!(
829 self.tcx().def_span(def_id),
830 "{def_kind:?} unexpectedly reached by `ReachEverythingInTheInterfaceVisitor`"
831 )
832 }
833 }
834 }
835}
836
837impl<'tcx> DefIdVisitor<'tcx> for ReachEverythingInTheInterfaceVisitor<'_, 'tcx> {
838 fn tcx(&self) -> TyCtxt<'tcx> {
839 self.ev.tcx
840 }
841 fn visit_def_id(&mut self, def_id: DefId, _kind: &str, _descr: &dyn fmt::Display) {
842 if let Some(def_id) = def_id.as_local() {
843 let max_vis = (self.level != Level::ReachableThroughImplTrait)
851 .then(|| self.ev.tcx.local_visibility(def_id));
852 if self.ev.update_eff_vis(def_id, self.effective_vis, max_vis, self.level) {
853 self.enqueue_def_id(def_id);
854 }
855 }
856 }
857}
858
859pub struct TestReachabilityVisitor<'a, 'tcx> {
861 tcx: TyCtxt<'tcx>,
862 effective_visibilities: &'a EffectiveVisibilities,
863}
864
865impl<'a, 'tcx> TestReachabilityVisitor<'a, 'tcx> {
866 fn effective_visibility_diagnostic(&self, def_id: LocalDefId) {
867 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) {
868 let mut error_msg = String::new();
869 let span = self.tcx.def_span(def_id.to_def_id());
870 if let Some(effective_vis) = self.effective_visibilities.effective_vis(def_id) {
871 for level in Level::all_levels() {
872 let vis_str = effective_vis.at_level(level).to_string(def_id, self.tcx);
873 if level != Level::Direct {
874 error_msg.push_str(", ");
875 }
876 error_msg.push_str(&::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0:?}: {1}", level, vis_str))
})format!("{level:?}: {vis_str}"));
877 }
878 } else {
879 error_msg.push_str("not in the table");
880 }
881 self.tcx.dcx().emit_err(ReportEffectiveVisibility { span, descr: error_msg });
882 }
883 }
884}
885
886impl<'a, 'tcx> TestReachabilityVisitor<'a, 'tcx> {
887 fn check_def_id(&self, owner_id: OwnerId) {
888 self.effective_visibility_diagnostic(owner_id.def_id);
889
890 match self.tcx.def_kind(owner_id) {
891 DefKind::Enum => {
892 let def = self.tcx.adt_def(owner_id.def_id);
893 for variant in def.variants() {
894 self.effective_visibility_diagnostic(variant.def_id.expect_local());
895 if let Some(ctor_def_id) = variant.ctor_def_id() {
896 self.effective_visibility_diagnostic(ctor_def_id.expect_local());
897 }
898 for field in &variant.fields {
899 self.effective_visibility_diagnostic(field.did.expect_local());
900 }
901 }
902 }
903 DefKind::Struct | DefKind::Union => {
904 let def = self.tcx.adt_def(owner_id.def_id).non_enum_variant();
905 if let Some(ctor_def_id) = def.ctor_def_id() {
906 self.effective_visibility_diagnostic(ctor_def_id.expect_local());
907 }
908 for field in &def.fields {
909 self.effective_visibility_diagnostic(field.did.expect_local());
910 }
911 }
912 _ => {}
913 }
914 }
915}
916
917struct NamePrivacyVisitor<'tcx> {
923 tcx: TyCtxt<'tcx>,
924 maybe_typeck_results: Option<&'tcx ty::TypeckResults<'tcx>>,
925}
926
927impl<'tcx> NamePrivacyVisitor<'tcx> {
928 #[track_caller]
932 fn typeck_results(&self) -> &'tcx ty::TypeckResults<'tcx> {
933 self.maybe_typeck_results
934 .expect("`NamePrivacyVisitor::typeck_results` called outside of body")
935 }
936
937 fn check_field(
939 &self,
940 hir_id: hir::HirId, use_ctxt: Span, def: ty::AdtDef<'tcx>, field: &'tcx ty::FieldDef,
944 ) -> bool {
945 if def.is_enum() {
946 return true;
947 }
948
949 let ident = Ident::new(sym::dummy, use_ctxt);
951 let (_, def_id) = self.tcx.adjust_ident_and_get_scope(ident, def.did(), hir_id);
952 !field.vis.is_accessible_from(def_id, self.tcx)
953 }
954
955 fn emit_unreachable_field_error(
957 &self,
958 fields: Vec<(Symbol, Span, bool )>,
959 def: ty::AdtDef<'tcx>, update_syntax: Option<Span>,
961 struct_span: Span,
962 ) {
963 if def.is_enum() || fields.is_empty() {
964 return;
965 }
966
967 let Some(field_names) = listify(&fields[..], |(n, _, _)| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", n))
})format!("`{n}`")) else { return };
979 let span: MultiSpan = fields.iter().map(|(_, span, _)| *span).collect::<Vec<Span>>().into();
980
981 let rest_field_names: Vec<_> =
983 fields.iter().filter(|(_, _, is_present)| !is_present).map(|(n, _, _)| n).collect();
984 let rest_len = rest_field_names.len();
985 let rest_field_names =
986 listify(&rest_field_names[..], |n| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", n))
})format!("`{n}`")).unwrap_or_default();
987 let labels = fields
989 .iter()
990 .filter(|(_, _, is_present)| *is_present)
991 .map(|(_, span, _)| FieldIsPrivateLabel::Other { span: *span })
992 .chain(update_syntax.iter().map(|span| FieldIsPrivateLabel::IsUpdateSyntax {
993 span: *span,
994 rest_field_names: rest_field_names.clone(),
995 rest_len,
996 }))
997 .collect();
998
999 self.tcx.dcx().emit_err(FieldIsPrivate {
1000 span,
1001 struct_span: if self
1002 .tcx
1003 .sess
1004 .source_map()
1005 .is_multiline(fields[0].1.between(struct_span))
1006 {
1007 Some(struct_span)
1008 } else {
1009 None
1010 },
1011 field_names,
1012 variant_descr: def.variant_descr(),
1013 def_path_str: self.tcx.def_path_str(def.did()),
1014 labels,
1015 len: fields.len(),
1016 });
1017 }
1018
1019 fn check_expanded_fields(
1020 &self,
1021 adt: ty::AdtDef<'tcx>,
1022 variant: &'tcx ty::VariantDef,
1023 fields: &[hir::ExprField<'tcx>],
1024 hir_id: hir::HirId,
1025 span: Span,
1026 struct_span: Span,
1027 ) {
1028 let mut failed_fields = ::alloc::vec::Vec::new()vec![];
1029 for (vf_index, variant_field) in variant.fields.iter_enumerated() {
1030 let field =
1031 fields.iter().find(|f| self.typeck_results().field_index(f.hir_id) == vf_index);
1032 let (hir_id, use_ctxt, span) = match field {
1033 Some(field) => (field.hir_id, field.ident.span, field.span),
1034 None => (hir_id, span, span),
1035 };
1036 if self.check_field(hir_id, use_ctxt, adt, variant_field) {
1037 let name = match field {
1038 Some(field) => field.ident.name,
1039 None => variant_field.name,
1040 };
1041 failed_fields.push((name, span, field.is_some()));
1042 }
1043 }
1044 self.emit_unreachable_field_error(failed_fields, adt, Some(span), struct_span);
1045 }
1046}
1047
1048impl<'tcx> Visitor<'tcx> for NamePrivacyVisitor<'tcx> {
1049 fn visit_nested_body(&mut self, body_id: hir::BodyId) {
1050 let new_typeck_results = self.tcx.typeck_body(body_id);
1051 if new_typeck_results.tainted_by_errors.is_some() {
1053 return;
1054 }
1055 let old_maybe_typeck_results = self.maybe_typeck_results.replace(new_typeck_results);
1056 self.visit_body(self.tcx.hir_body(body_id));
1057 self.maybe_typeck_results = old_maybe_typeck_results;
1058 }
1059
1060 fn visit_expr(&mut self, expr: &'tcx hir::Expr<'tcx>) {
1061 if let hir::ExprKind::Struct(qpath, fields, ref base) = expr.kind {
1062 let res = self.typeck_results().qpath_res(qpath, expr.hir_id);
1063 let adt = self.typeck_results().expr_ty(expr).ty_adt_def().unwrap();
1064 let variant = adt.variant_of_res(res);
1065 match *base {
1066 hir::StructTailExpr::Base(base) => {
1067 self.check_expanded_fields(
1071 adt,
1072 variant,
1073 fields,
1074 base.hir_id,
1075 base.span,
1076 qpath.span(),
1077 );
1078 }
1079 hir::StructTailExpr::DefaultFields(span) => {
1080 self.check_expanded_fields(
1081 adt,
1082 variant,
1083 fields,
1084 expr.hir_id,
1085 span,
1086 qpath.span(),
1087 );
1088 }
1089 hir::StructTailExpr::None | hir::StructTailExpr::NoneWithError(_) => {
1090 let mut failed_fields = ::alloc::vec::Vec::new()vec![];
1091 for field in fields {
1092 let (hir_id, use_ctxt) = (field.hir_id, field.ident.span);
1093 let index = self.typeck_results().field_index(field.hir_id);
1094 if self.check_field(hir_id, use_ctxt, adt, &variant.fields[index]) {
1095 failed_fields.push((field.ident.name, field.ident.span, true));
1096 }
1097 }
1098 self.emit_unreachable_field_error(failed_fields, adt, None, qpath.span());
1099 }
1100 }
1101 }
1102
1103 intravisit::walk_expr(self, expr);
1104 }
1105
1106 fn visit_pat(&mut self, pat: &'tcx hir::Pat<'tcx>) {
1107 if let PatKind::Struct(ref qpath, fields, _) = pat.kind {
1108 let res = self.typeck_results().qpath_res(qpath, pat.hir_id);
1109 let adt = self.typeck_results().pat_ty(pat).ty_adt_def().unwrap();
1110 let variant = adt.variant_of_res(res);
1111 let mut failed_fields = ::alloc::vec::Vec::new()vec![];
1112 for field in fields {
1113 let (hir_id, use_ctxt) = (field.hir_id, field.ident.span);
1114 let index = self.typeck_results().field_index(field.hir_id);
1115 if self.check_field(hir_id, use_ctxt, adt, &variant.fields[index]) {
1116 failed_fields.push((field.ident.name, field.ident.span, true));
1117 }
1118 }
1119 self.emit_unreachable_field_error(failed_fields, adt, None, qpath.span());
1120 }
1121
1122 intravisit::walk_pat(self, pat);
1123 }
1124}
1125
1126struct TypePrivacyVisitor<'tcx> {
1131 tcx: TyCtxt<'tcx>,
1132 module_def_id: LocalModDefId,
1133 maybe_typeck_results: Option<&'tcx ty::TypeckResults<'tcx>>,
1134 span: Span,
1135}
1136
1137impl<'tcx> TypePrivacyVisitor<'tcx> {
1138 fn item_is_accessible(&self, did: DefId) -> bool {
1139 self.tcx.visibility(did).is_accessible_from(self.module_def_id, self.tcx)
1140 }
1141
1142 fn check_expr_pat_type(&mut self, id: hir::HirId, span: Span) -> bool {
1144 self.span = span;
1145 let typeck_results = self
1146 .maybe_typeck_results
1147 .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"));
1148 try {
1149 self.visit(typeck_results.node_type(id))?;
1150 self.visit(typeck_results.node_args(id))?;
1151 if let Some(adjustments) = typeck_results.adjustments().get(id) {
1152 adjustments.iter().try_for_each(|adjustment| self.visit(adjustment.target))?;
1153 }
1154 }
1155 .is_break()
1156 }
1157
1158 fn check_def_id(&self, def_id: DefId, kind: &str, descr: &dyn fmt::Display) -> bool {
1159 let is_error = !self.item_is_accessible(def_id);
1160 if is_error {
1161 self.tcx.dcx().emit_err(ItemIsPrivate { span: self.span, kind, descr: descr.into() });
1162 }
1163 is_error
1164 }
1165}
1166
1167impl<'tcx> rustc_ty_utils::sig_types::SpannedTypeVisitor<'tcx> for TypePrivacyVisitor<'tcx> {
1168 type Result = ControlFlow<()>;
1169 fn visit(&mut self, span: Span, value: impl TypeVisitable<TyCtxt<'tcx>>) -> Self::Result {
1170 self.span = span;
1171 value.visit_with(&mut self.skeleton())
1172 }
1173}
1174
1175impl<'tcx> Visitor<'tcx> for TypePrivacyVisitor<'tcx> {
1176 fn visit_nested_body(&mut self, body_id: hir::BodyId) {
1177 let old_maybe_typeck_results =
1178 self.maybe_typeck_results.replace(self.tcx.typeck_body(body_id));
1179 self.visit_body(self.tcx.hir_body(body_id));
1180 self.maybe_typeck_results = old_maybe_typeck_results;
1181 }
1182
1183 fn visit_ty(&mut self, hir_ty: &'tcx hir::Ty<'tcx, AmbigArg>) {
1184 self.span = hir_ty.span;
1185 if self
1186 .visit(
1187 self.maybe_typeck_results
1188 .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"))
1189 .node_type(hir_ty.hir_id),
1190 )
1191 .is_break()
1192 {
1193 return;
1194 }
1195
1196 intravisit::walk_ty(self, hir_ty);
1197 }
1198
1199 fn visit_infer(
1200 &mut self,
1201 inf_id: rustc_hir::HirId,
1202 inf_span: Span,
1203 _kind: InferKind<'tcx>,
1204 ) -> Self::Result {
1205 self.span = inf_span;
1206 if let Some(ty) = self
1207 .maybe_typeck_results
1208 .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"))
1209 .node_type_opt(inf_id)
1210 {
1211 if self.visit(ty).is_break() {
1212 return;
1213 }
1214 } else {
1215 }
1217
1218 self.visit_id(inf_id)
1219 }
1220
1221 fn visit_expr(&mut self, expr: &'tcx hir::Expr<'tcx>) {
1223 if self.check_expr_pat_type(expr.hir_id, expr.span) {
1224 return;
1226 }
1227 match expr.kind {
1228 hir::ExprKind::Assign(_, rhs, _) | hir::ExprKind::Match(rhs, ..) => {
1229 if self.check_expr_pat_type(rhs.hir_id, rhs.span) {
1231 return;
1232 }
1233 }
1234 hir::ExprKind::MethodCall(segment, ..) => {
1235 self.span = segment.ident.span;
1237 let typeck_results = self
1238 .maybe_typeck_results
1239 .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"));
1240 if let Some(def_id) = typeck_results.type_dependent_def_id(expr.hir_id) {
1241 if self
1242 .visit(self.tcx.type_of(def_id).instantiate_identity().skip_norm_wip())
1243 .is_break()
1244 {
1245 return;
1246 }
1247 } else {
1248 self.tcx
1249 .dcx()
1250 .span_delayed_bug(expr.span, "no type-dependent def for method call");
1251 }
1252 }
1253 _ => {}
1254 }
1255
1256 intravisit::walk_expr(self, expr);
1257 }
1258
1259 fn visit_qpath(&mut self, qpath: &'tcx hir::QPath<'tcx>, id: hir::HirId, span: Span) {
1266 let def = match qpath {
1267 hir::QPath::Resolved(_, path) => match path.res {
1268 Res::Def(kind, def_id) => Some((kind, def_id)),
1269 _ => None,
1270 },
1271 hir::QPath::TypeRelative(..) => {
1272 match self.maybe_typeck_results {
1273 Some(typeck_results) => typeck_results.type_dependent_def(id),
1274 None => None,
1276 }
1277 }
1278 };
1279 let def = def.filter(|(kind, _)| {
1280 #[allow(non_exhaustive_omitted_patterns)] match kind {
DefKind::AssocFn | DefKind::AssocConst { .. } | DefKind::AssocTy |
DefKind::Static { .. } => true,
_ => false,
}matches!(
1281 kind,
1282 DefKind::AssocFn
1283 | DefKind::AssocConst { .. }
1284 | DefKind::AssocTy
1285 | DefKind::Static { .. }
1286 )
1287 });
1288 if let Some((kind, def_id)) = def {
1289 let is_local_static =
1290 if let DefKind::Static { .. } = kind { def_id.is_local() } else { false };
1291 if !self.item_is_accessible(def_id) && !is_local_static {
1292 let name = match *qpath {
1293 hir::QPath::Resolved(_, path) => Some(self.tcx.def_path_str(path.res.def_id())),
1294 hir::QPath::TypeRelative(_, segment) => Some(segment.ident.to_string()),
1295 };
1296 let kind = self.tcx.def_descr(def_id);
1297 let sess = self.tcx.sess;
1298 let _ = match name {
1299 Some(name) => {
1300 sess.dcx().emit_err(ItemIsPrivate { span, kind, descr: (&name).into() })
1301 }
1302 None => sess.dcx().emit_err(UnnamedItemIsPrivate { span, kind }),
1303 };
1304 return;
1305 }
1306 }
1307
1308 intravisit::walk_qpath(self, qpath, id);
1309 }
1310
1311 fn visit_pat(&mut self, pattern: &'tcx hir::Pat<'tcx>) {
1313 if self.check_expr_pat_type(pattern.hir_id, pattern.span) {
1314 return;
1316 }
1317
1318 intravisit::walk_pat(self, pattern);
1319 }
1320
1321 fn visit_local(&mut self, local: &'tcx hir::LetStmt<'tcx>) {
1322 if let Some(init) = local.init {
1323 if self.check_expr_pat_type(init.hir_id, init.span) {
1324 return;
1326 }
1327 }
1328
1329 intravisit::walk_local(self, local);
1330 }
1331}
1332
1333impl<'tcx> DefIdVisitor<'tcx> for TypePrivacyVisitor<'tcx> {
1334 type Result = ControlFlow<()>;
1335 fn tcx(&self) -> TyCtxt<'tcx> {
1336 self.tcx
1337 }
1338 fn visit_def_id(
1339 &mut self,
1340 def_id: DefId,
1341 kind: &str,
1342 descr: &dyn fmt::Display,
1343 ) -> Self::Result {
1344 if self.check_def_id(def_id, kind, descr) {
1345 ControlFlow::Break(())
1346 } else {
1347 ControlFlow::Continue(())
1348 }
1349 }
1350}
1351
1352struct SearchInterfaceForPrivateItemsVisitor<'tcx> {
1358 tcx: TyCtxt<'tcx>,
1359 item_def_id: LocalDefId,
1360 required_visibility: ty::Visibility,
1362 required_effective_vis: Option<EffectiveVisibility>,
1363 hard_error: bool = false,
1364 in_primary_interface: bool = true,
1365 skip_assoc_tys: bool = false,
1366}
1367
1368impl SearchInterfaceForPrivateItemsVisitor<'_> {
1369 fn generics(&mut self) -> &mut Self {
1370 self.in_primary_interface = true;
1371 for param in &self.tcx.generics_of(self.item_def_id).own_params {
1372 if let GenericParamDefKind::Const { .. } = param.kind {
1373 let _ = self
1374 .visit(self.tcx.type_of(param.def_id).instantiate_identity().skip_norm_wip());
1375 }
1376 if let Some(default) = param.default_value(self.tcx) {
1377 let _ = self.visit(default.instantiate_identity().skip_norm_wip());
1378 }
1379 }
1380 self
1381 }
1382
1383 fn predicates(&mut self) -> &mut Self {
1384 self.in_primary_interface = false;
1385 let _ = self.visit_predicates(self.tcx.explicit_predicates_of(self.item_def_id));
1392 self
1393 }
1394
1395 fn bounds(&mut self) -> &mut Self {
1396 self.in_primary_interface = false;
1397 let _ = self.visit_clauses(self.tcx.explicit_item_bounds(self.item_def_id).skip_binder());
1398 self
1399 }
1400
1401 fn ty(&mut self) -> &mut Self {
1402 self.in_primary_interface = true;
1403 let _ =
1404 self.visit(self.tcx.type_of(self.item_def_id).instantiate_identity().skip_norm_wip());
1405 self
1406 }
1407
1408 fn trait_ref(&mut self) -> &mut Self {
1409 self.in_primary_interface = true;
1410 let _ = self.visit_trait(
1411 self.tcx.impl_trait_ref(self.item_def_id).instantiate_identity().skip_norm_wip(),
1412 );
1413 self
1414 }
1415
1416 fn check_def_id(&self, def_id: DefId, kind: &str, descr: &dyn fmt::Display) -> bool {
1417 if self.leaks_private_dep(def_id) {
1418 self.tcx.emit_node_span_lint(
1419 lint::builtin::EXPORTED_PRIVATE_DEPENDENCIES,
1420 self.tcx.local_def_id_to_hir_id(self.item_def_id),
1421 self.tcx.def_span(self.item_def_id.to_def_id()),
1422 FromPrivateDependencyInPublicInterface {
1423 kind,
1424 descr: descr.into(),
1425 krate: self.tcx.crate_name(def_id.krate),
1426 },
1427 );
1428 }
1429
1430 let Some(local_def_id) = def_id.as_local() else {
1431 return false;
1432 };
1433
1434 let vis = self.tcx.local_visibility(local_def_id);
1435 if self.hard_error && self.required_visibility.greater_than(vis, self.tcx) {
1436 let vis_descr = match vis {
1437 ty::Visibility::Public => "public",
1438 ty::Visibility::Restricted(vis_def_id) => {
1439 if vis_def_id
1440 == self.tcx.parent_module_from_def_id(local_def_id).to_local_def_id()
1441 {
1442 "private"
1443 } else if vis_def_id.is_top_level_module() {
1444 "crate-private"
1445 } else {
1446 "restricted"
1447 }
1448 }
1449 };
1450
1451 let span = self.tcx.def_span(self.item_def_id.to_def_id());
1452 let vis_span = self.tcx.def_span(def_id);
1453 self.tcx.dcx().emit_err(InPublicInterface {
1454 span,
1455 vis_descr,
1456 kind,
1457 descr: descr.into(),
1458 vis_span,
1459 });
1460 return false;
1461 }
1462
1463 let Some(effective_vis) = self.required_effective_vis else {
1464 return false;
1465 };
1466
1467 let reachable_at_vis = *effective_vis.at_level(Level::Reachable);
1468
1469 if reachable_at_vis.greater_than(vis, self.tcx) {
1470 let lint = if self.in_primary_interface {
1471 lint::builtin::PRIVATE_INTERFACES
1472 } else {
1473 lint::builtin::PRIVATE_BOUNDS
1474 };
1475 let span = self.tcx.def_span(self.item_def_id.to_def_id());
1476 let vis_span = self.tcx.def_span(def_id);
1477 self.tcx.emit_node_span_lint(
1478 lint,
1479 self.tcx.local_def_id_to_hir_id(self.item_def_id),
1480 span,
1481 PrivateInterfacesOrBoundsLint {
1482 item_span: span,
1483 item_kind: self.tcx.def_descr(self.item_def_id.to_def_id()),
1484 item_descr: (&LazyDefPathStr {
1485 def_id: self.item_def_id.to_def_id(),
1486 tcx: self.tcx,
1487 })
1488 .into(),
1489 item_vis_descr: &reachable_at_vis.to_string(self.item_def_id, self.tcx),
1490 ty_span: vis_span,
1491 ty_kind: kind,
1492 ty_descr: descr.into(),
1493 ty_vis_descr: &vis.to_string(local_def_id, self.tcx),
1494 },
1495 );
1496 }
1497
1498 false
1499 }
1500
1501 fn leaks_private_dep(&self, item_id: DefId) -> bool {
1506 let ret = self.required_visibility.is_public() && self.tcx.is_private_dep(item_id.krate);
1507
1508 {
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:1508",
"rustc_privacy", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_privacy/src/lib.rs"),
::tracing_core::__macro_support::Option::Some(1508u32),
::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);
1509 ret
1510 }
1511}
1512
1513impl<'tcx> DefIdVisitor<'tcx> for SearchInterfaceForPrivateItemsVisitor<'tcx> {
1514 type Result = ControlFlow<()>;
1515 fn skip_assoc_tys(&self) -> bool {
1516 self.skip_assoc_tys
1517 }
1518 fn tcx(&self) -> TyCtxt<'tcx> {
1519 self.tcx
1520 }
1521 fn visit_def_id(
1522 &mut self,
1523 def_id: DefId,
1524 kind: &str,
1525 descr: &dyn fmt::Display,
1526 ) -> Self::Result {
1527 if self.check_def_id(def_id, kind, descr) {
1528 ControlFlow::Break(())
1529 } else {
1530 ControlFlow::Continue(())
1531 }
1532 }
1533}
1534
1535struct PrivateItemsInPublicInterfacesChecker<'a, 'tcx> {
1536 tcx: TyCtxt<'tcx>,
1537 effective_visibilities: &'a EffectiveVisibilities,
1538}
1539
1540impl<'tcx> PrivateItemsInPublicInterfacesChecker<'_, 'tcx> {
1541 fn check(
1542 &self,
1543 def_id: LocalDefId,
1544 required_visibility: ty::Visibility,
1545 required_effective_vis: Option<EffectiveVisibility>,
1546 ) -> SearchInterfaceForPrivateItemsVisitor<'tcx> {
1547 SearchInterfaceForPrivateItemsVisitor {
1548 tcx: self.tcx,
1549 item_def_id: def_id,
1550 required_visibility,
1551 required_effective_vis,
1552 ..
1553 }
1554 }
1555
1556 fn check_unnameable(&self, def_id: LocalDefId, effective_vis: Option<EffectiveVisibility>) {
1557 let Some(effective_vis) = effective_vis else {
1558 return;
1559 };
1560
1561 let reexported_at_vis = effective_vis.at_level(Level::Reexported);
1562 let reachable_at_vis = effective_vis.at_level(Level::Reachable);
1563
1564 if reachable_at_vis.is_public() && reexported_at_vis != reachable_at_vis {
1565 let hir_id = self.tcx.local_def_id_to_hir_id(def_id);
1566 let span = self.tcx.def_span(def_id.to_def_id());
1567 self.tcx.emit_node_span_lint(
1568 lint::builtin::UNNAMEABLE_TYPES,
1569 hir_id,
1570 span,
1571 UnnameableTypesLint {
1572 span,
1573 kind: self.tcx.def_descr(def_id.to_def_id()),
1574 descr: (&LazyDefPathStr { def_id: def_id.to_def_id(), tcx: self.tcx }).into(),
1575 reachable_vis: &reachable_at_vis.to_string(def_id, self.tcx),
1576 reexported_vis: &reexported_at_vis.to_string(def_id, self.tcx),
1577 },
1578 );
1579 }
1580 }
1581
1582 fn check_assoc_item(
1583 &self,
1584 item: &ty::AssocItem,
1585 vis: ty::Visibility,
1586 effective_vis: Option<EffectiveVisibility>,
1587 ) {
1588 let mut check = self.check(item.def_id.expect_local(), vis, effective_vis);
1589
1590 let is_assoc_ty = item.is_type();
1591 check.hard_error = is_assoc_ty;
1592 check.generics().predicates();
1593 if assoc_has_type_of(self.tcx, item) {
1594 check.ty();
1595 }
1596 if is_assoc_ty && item.container == AssocContainer::Trait {
1597 check.hard_error = false;
1600 check.bounds();
1601 }
1602 }
1603
1604 fn get(&self, def_id: LocalDefId) -> Option<EffectiveVisibility> {
1605 self.effective_visibilities.effective_vis(def_id).copied()
1606 }
1607
1608 fn check_item(&self, id: ItemId) {
1609 let tcx = self.tcx;
1610 let def_id = id.owner_id.def_id;
1611 let item_visibility = tcx.local_visibility(def_id);
1612 let effective_vis = self.get(def_id);
1613 let def_kind = tcx.def_kind(def_id);
1614
1615 match def_kind {
1616 DefKind::Const { .. } | DefKind::Static { .. } | DefKind::Fn | DefKind::TyAlias => {
1617 if let DefKind::TyAlias = def_kind {
1618 self.check_unnameable(def_id, effective_vis);
1619 }
1620 self.check(def_id, item_visibility, effective_vis).generics().predicates().ty();
1621 }
1622 DefKind::OpaqueTy => {
1623 self.check(def_id, item_visibility, effective_vis).generics().bounds();
1626 }
1627 DefKind::Trait => {
1628 self.check_unnameable(def_id, effective_vis);
1629
1630 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1631
1632 for assoc_item in tcx.associated_items(id.owner_id).in_definition_order() {
1633 self.check_assoc_item(assoc_item, item_visibility, effective_vis);
1634 }
1635 }
1636 DefKind::TraitAlias => {
1637 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1638 }
1639 DefKind::Enum => {
1640 self.check_unnameable(def_id, effective_vis);
1641 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1642
1643 let adt = tcx.adt_def(id.owner_id);
1644 for field in adt.all_fields() {
1645 self.check(field.did.expect_local(), item_visibility, effective_vis).ty();
1646 }
1647 }
1648 DefKind::Struct | DefKind::Union => {
1650 self.check_unnameable(def_id, effective_vis);
1651 self.check(def_id, item_visibility, effective_vis).generics().predicates();
1652
1653 let adt = tcx.adt_def(id.owner_id);
1654 for field in adt.all_fields() {
1655 let visibility = min(item_visibility, field.vis.expect_local(), tcx);
1656 let field_ev = self.get(field.did.expect_local());
1657
1658 self.check(field.did.expect_local(), visibility, field_ev).ty();
1659 }
1660 }
1661 DefKind::ForeignMod => {}
1663 DefKind::Impl { of_trait } => {
1668 let impl_vis =
1669 ty::Visibility::of_impl::<false>(def_id, of_trait, tcx, &Default::default());
1670
1671 let impl_ev = EffectiveVisibility::of_impl::<false>(
1683 def_id,
1684 of_trait,
1685 tcx,
1686 self.effective_visibilities,
1687 );
1688
1689 let mut check = self.check(def_id, impl_vis, Some(impl_ev));
1690
1691 if !of_trait {
1694 check.generics().predicates();
1695 }
1696
1697 check.skip_assoc_tys = true;
1701 check.ty();
1702 if of_trait {
1703 check.trait_ref();
1704 }
1705
1706 for assoc_item in tcx.associated_items(id.owner_id).in_definition_order() {
1707 let impl_item_vis = if !of_trait {
1708 min(tcx.local_visibility(assoc_item.def_id.expect_local()), impl_vis, tcx)
1709 } else {
1710 impl_vis
1711 };
1712
1713 let impl_item_ev = if !of_trait {
1714 self.get(assoc_item.def_id.expect_local())
1715 .map(|ev| ev.min(impl_ev, self.tcx))
1716 } else {
1717 Some(impl_ev)
1718 };
1719
1720 self.check_assoc_item(assoc_item, impl_item_vis, impl_item_ev);
1721 }
1722 }
1723 _ => {}
1724 }
1725 }
1726
1727 fn check_foreign_item(&self, id: ForeignItemId) {
1728 let tcx = self.tcx;
1729 let def_id = id.owner_id.def_id;
1730 let item_visibility = tcx.local_visibility(def_id);
1731 let effective_vis = self.get(def_id);
1732
1733 if let DefKind::ForeignTy = self.tcx.def_kind(def_id) {
1734 self.check_unnameable(def_id, effective_vis);
1735 }
1736
1737 self.check(def_id, item_visibility, effective_vis).generics().predicates().ty();
1738 }
1739}
1740
1741pub fn provide(providers: &mut Providers) {
1742 *providers = Providers {
1743 effective_visibilities,
1744 check_private_in_public,
1745 check_mod_privacy,
1746 ..*providers
1747 };
1748}
1749
1750fn check_mod_privacy(tcx: TyCtxt<'_>, module_def_id: LocalModDefId) {
1751 let mut visitor = NamePrivacyVisitor { tcx, maybe_typeck_results: None };
1753 tcx.hir_visit_item_likes_in_module(module_def_id, &mut visitor);
1754
1755 let span = tcx.def_span(module_def_id);
1758 let mut visitor = TypePrivacyVisitor { tcx, module_def_id, maybe_typeck_results: None, span };
1759
1760 let module = tcx.hir_module_items(module_def_id);
1761 for def_id in module.definitions() {
1762 let _ = rustc_ty_utils::sig_types::walk_types(tcx, def_id, &mut visitor);
1763
1764 if let Some(body_id) = tcx.hir_maybe_body_owned_by(def_id) {
1765 visitor.visit_nested_body(body_id.id());
1766 }
1767
1768 if let DefKind::Impl { of_trait: true } = tcx.def_kind(def_id) {
1769 let trait_ref = tcx.impl_trait_ref(def_id);
1770 let trait_ref = trait_ref.instantiate_identity().skip_norm_wip();
1771 visitor.span =
1772 tcx.hir_expect_item(def_id).expect_impl().of_trait.unwrap().trait_ref.path.span;
1773 let _ =
1774 visitor.visit_def_id(trait_ref.def_id, "trait", &trait_ref.print_only_trait_path());
1775 }
1776 }
1777}
1778
1779fn effective_visibilities(tcx: TyCtxt<'_>, (): ()) -> &EffectiveVisibilities {
1780 let def_ids_to_impls = DefIdsToImplsCollector::collect(tcx);
1781
1782 let mut visitor = EmbargoVisitor {
1785 tcx,
1786 effective_visibilities: tcx.resolutions(()).effective_visibilities.clone(),
1787 queue: Default::default(),
1788 def_ids_to_impls,
1789 };
1790
1791 visitor.effective_visibilities.check_invariants(tcx);
1792
1793 let impl_trait_pass = !tcx.sess.opts.actually_rustdoc;
1797 if impl_trait_pass {
1798 let krate = tcx.hir_crate_items(());
1801 for id in krate.opaques() {
1802 let opaque = tcx.hir_node_by_def_id(id).expect_opaque_ty();
1803 let should_visit = match opaque.origin {
1804 hir::OpaqueTyOrigin::FnReturn {
1805 parent,
1806 in_trait_or_impl: Some(hir::RpitContext::Trait),
1807 }
1808 | hir::OpaqueTyOrigin::AsyncFn {
1809 parent,
1810 in_trait_or_impl: Some(hir::RpitContext::Trait),
1811 } => match tcx.hir_node_by_def_id(parent).expect_trait_item().expect_fn().1 {
1812 hir::TraitFn::Required(_) => false,
1813 hir::TraitFn::Provided(..) => true,
1814 },
1815
1816 hir::OpaqueTyOrigin::FnReturn {
1819 in_trait_or_impl: None | Some(hir::RpitContext::TraitImpl),
1820 ..
1821 }
1822 | hir::OpaqueTyOrigin::AsyncFn {
1823 in_trait_or_impl: None | Some(hir::RpitContext::TraitImpl),
1824 ..
1825 }
1826 | hir::OpaqueTyOrigin::TyAlias { .. } => true,
1827 };
1828 if should_visit {
1829 let pub_ev = EffectiveVisibility::from_vis(ty::Visibility::Public);
1833 visitor
1834 .reach_through_impl_trait(opaque.def_id, pub_ev)
1835 .generics()
1836 .predicates()
1837 .ty();
1838 }
1839 }
1840
1841 visitor.queue.clear();
1842 }
1843
1844 for (&adt_def_id, macro_mods) in &tcx.resolutions(()).macro_reachable_adts {
1847 let struct_def = tcx.adt_def(adt_def_id);
1848 let Some(struct_ev) = visitor.effective_visibilities.effective_vis(adt_def_id).copied()
1849 else {
1850 continue;
1851 };
1852 for field in &struct_def.non_enum_variant().fields {
1853 let def_id = field.did.expect_local();
1854 let field_vis = tcx.local_visibility(def_id);
1855
1856 for ¯o_mod in macro_mods {
1857 if field_vis.is_accessible_from(macro_mod, tcx) {
1858 visitor.reach(def_id, struct_ev).ty();
1859 }
1860 }
1861 }
1862 }
1863
1864 let crate_items = tcx.hir_crate_items(());
1865 for id in crate_items.free_items() {
1866 visitor.check_def_id(id.owner_id.def_id);
1867 }
1868 for id in crate_items.foreign_items() {
1869 visitor.check_def_id(id.owner_id.def_id);
1870 }
1871 while let Some(def_id) = visitor.queue.pop() {
1872 visitor.check_def_id(def_id);
1873 }
1874 visitor.effective_visibilities.check_invariants(tcx);
1875
1876 let check_visitor =
1877 TestReachabilityVisitor { tcx, effective_visibilities: &visitor.effective_visibilities };
1878 for id in crate_items.owners() {
1879 check_visitor.check_def_id(id);
1880 }
1881
1882 tcx.arena.alloc(visitor.effective_visibilities)
1883}
1884
1885fn check_private_in_public(tcx: TyCtxt<'_>, module_def_id: LocalModDefId) {
1886 let effective_visibilities = tcx.effective_visibilities(());
1887 let checker = PrivateItemsInPublicInterfacesChecker { tcx, effective_visibilities };
1889
1890 let crate_items = tcx.hir_module_items(module_def_id);
1891 let _ = crate_items.par_items(|id| Ok(checker.check_item(id)));
1892 let _ = crate_items.par_foreign_items(|id| Ok(checker.check_foreign_item(id)));
1893}