rustdoc/clean/
blanket_impl.rs1use rustc_data_structures::thin_vec::ThinVec;
2use rustc_hir as hir;
3use rustc_infer::infer::{DefineOpaqueTypes, InferOk, TyCtxtInferExt};
4use rustc_infer::traits;
5use rustc_middle::ty::{self, TypingMode, Unnormalized, Upcast};
6use rustc_span::DUMMY_SP;
7use rustc_span::def_id::DefId;
8use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt;
9use tracing::{debug, instrument, trace};
10
11use crate::clean;
12use crate::clean::{
13 clean_middle_assoc_item, clean_middle_ty, clean_trait_ref_with_constraints, clean_ty_generics,
14};
15use crate::core::DocContext;
16
17#[instrument(level = "debug", skip(cx))]
18pub(crate) fn synthesize_blanket_impls(
19 cx: &mut DocContext<'_>,
20 item_def_id: DefId,
21) -> Vec<clean::Item> {
22 let tcx = cx.tcx;
23 let ty = tcx.type_of(item_def_id);
24
25 let mut blanket_impls = Vec::new();
26 for trait_def_id in tcx.visible_traits() {
27 if !cx.cache.effective_visibilities.is_reachable(tcx, trait_def_id)
28 || cx.generated_synthetics.contains(&(ty.skip_binder(), trait_def_id))
29 {
30 continue;
31 }
32 let trait_impls = tcx.trait_impls_of(trait_def_id);
34 'blanket_impls: for &impl_def_id in trait_impls.blanket_impls() {
35 trace!("considering impl `{impl_def_id:?}` for trait `{trait_def_id:?}`");
36
37 let trait_ref = tcx.impl_trait_ref(impl_def_id);
38 if !matches!(trait_ref.skip_binder().self_ty().kind(), ty::Param(_)) {
39 continue;
40 }
41 let infcx = tcx.infer_ctxt().build(TypingMode::non_body_analysis());
42 let args = infcx.fresh_args_for_item(DUMMY_SP, item_def_id);
43 let impl_ty = ty.instantiate(tcx, args).skip_norm_wip();
44 let param_env = ty::ParamEnv::empty();
45
46 let impl_args = infcx.fresh_args_for_item(DUMMY_SP, impl_def_id);
47 let impl_trait_ref = trait_ref.instantiate(tcx, impl_args).skip_norm_wip();
48
49 let Ok(eq_result) = infcx.at(&traits::ObligationCause::dummy(), param_env).eq(
52 DefineOpaqueTypes::Yes,
53 impl_trait_ref.self_ty(),
54 impl_ty,
55 ) else {
56 continue;
57 };
58 let InferOk { value: (), obligations } = eq_result;
59 drop(obligations);
61
62 let predicates = tcx
63 .predicates_of(impl_def_id)
64 .instantiate(tcx, impl_args)
65 .predicates
66 .into_iter()
67 .map(Unnormalized::skip_norm_wip)
68 .chain(Some(impl_trait_ref.upcast(tcx)));
69 for predicate in predicates {
70 let obligation = traits::Obligation::new(
71 tcx,
72 traits::ObligationCause::dummy(),
73 param_env,
74 predicate,
75 );
76 match infcx.evaluate_obligation(&obligation) {
77 Ok(eval_result) if eval_result.may_apply() => {}
78 Err(traits::OverflowError::Canonical) => {}
79 _ => continue 'blanket_impls,
80 }
81 }
82 debug!("found applicable impl for trait ref {trait_ref:?}");
83
84 cx.generated_synthetics.insert((ty.skip_binder(), trait_def_id));
85
86 blanket_impls.push(clean::Item {
87 inner: Box::new(clean::ItemInner {
88 name: None,
89 item_id: clean::ItemId::Blanket { impl_id: impl_def_id, for_: item_def_id },
90 attrs: Default::default(),
91 stability: None,
92 kind: clean::ImplItem(Box::new(clean::Impl {
93 safety: hir::Safety::Safe,
94 generics: clean_ty_generics(cx, impl_def_id),
95 trait_: Some(clean_trait_ref_with_constraints(
98 cx,
99 ty::Binder::dummy(trait_ref.instantiate_identity().skip_norm_wip()),
100 ThinVec::new(),
101 )),
102 for_: clean_middle_ty(
103 ty::Binder::dummy(ty.instantiate_identity().skip_norm_wip()),
104 cx,
105 None,
106 None,
107 ),
108 items: tcx
109 .associated_items(impl_def_id)
110 .in_definition_order()
111 .filter(|item| !item.is_impl_trait_in_trait())
112 .map(|item| clean_middle_assoc_item(item, cx))
113 .collect(),
114 polarity: ty::ImplPolarity::Positive,
115 kind: clean::ImplKind::Blanket(Box::new(clean_middle_ty(
116 ty::Binder::dummy(
117 trait_ref.instantiate_identity().skip_norm_wip().self_ty(),
118 ),
119 cx,
120 None,
121 None,
122 ))),
123 is_deprecated: tcx
124 .lookup_deprecation(impl_def_id)
125 .is_some_and(|deprecation| deprecation.is_in_effect()),
126 })),
127 cfg: None,
128 inline_stmt_id: None,
129 }),
130 });
131 }
132 }
133
134 blanket_impls
135}