rustdoc/clean/
blanket_impl.rs
1use rustc_hir as hir;
2use rustc_infer::infer::{DefineOpaqueTypes, InferOk, TyCtxtInferExt};
3use rustc_infer::traits;
4use rustc_middle::ty::{self, TypingMode, Upcast};
5use rustc_span::DUMMY_SP;
6use rustc_span::def_id::DefId;
7use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt;
8use thin_vec::ThinVec;
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.all_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).unwrap();
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);
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);
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 .chain(Some(impl_trait_ref.upcast(tcx)));
68 for predicate in predicates {
69 let obligation = traits::Obligation::new(
70 tcx,
71 traits::ObligationCause::dummy(),
72 param_env,
73 predicate,
74 );
75 match infcx.evaluate_obligation(&obligation) {
76 Ok(eval_result) if eval_result.may_apply() => {}
77 Err(traits::OverflowError::Canonical) => {}
78 _ => continue 'blanket_impls,
79 }
80 }
81 debug!("found applicable impl for trait ref {trait_ref:?}");
82
83 cx.generated_synthetics.insert((ty.skip_binder(), trait_def_id));
84
85 blanket_impls.push(clean::Item {
86 name: None,
87 item_id: clean::ItemId::Blanket { impl_id: impl_def_id, for_: item_def_id },
88 inner: Box::new(clean::ItemInner {
89 attrs: Default::default(),
90 stability: None,
91 kind: clean::ImplItem(Box::new(clean::Impl {
92 safety: hir::Safety::Safe,
93 generics: clean_ty_generics(
94 cx,
95 tcx.generics_of(impl_def_id),
96 tcx.explicit_predicates_of(impl_def_id),
97 ),
98 trait_: Some(clean_trait_ref_with_constraints(
101 cx,
102 ty::Binder::dummy(trait_ref.instantiate_identity()),
103 ThinVec::new(),
104 )),
105 for_: clean_middle_ty(
106 ty::Binder::dummy(ty.instantiate_identity()),
107 cx,
108 None,
109 None,
110 ),
111 items: tcx
112 .associated_items(impl_def_id)
113 .in_definition_order()
114 .filter(|item| !item.is_impl_trait_in_trait())
115 .map(|item| clean_middle_assoc_item(item, cx))
116 .collect(),
117 polarity: ty::ImplPolarity::Positive,
118 kind: clean::ImplKind::Blanket(Box::new(clean_middle_ty(
119 ty::Binder::dummy(trait_ref.instantiate_identity().self_ty()),
120 cx,
121 None,
122 None,
123 ))),
124 })),
125 }),
126 cfg: None,
127 inline_stmt_id: None,
128 });
129 }
130 }
131
132 blanket_impls
133}