]> git.lizzy.rs Git - rust.git/blob - compiler/rustc_hir_analysis/src/collect/item_bounds.rs
Auto merge of #106948 - cuviper:ci-mingw-check, r=nikic
[rust.git] / compiler / rustc_hir_analysis / src / collect / item_bounds.rs
1 use super::ItemCtxt;
2 use crate::astconv::AstConv;
3 use rustc_hir as hir;
4 use rustc_infer::traits::util;
5 use rustc_middle::ty::subst::InternalSubsts;
6 use rustc_middle::ty::{self, DefIdTree, TyCtxt};
7 use rustc_span::def_id::DefId;
8 use rustc_span::Span;
9
10 /// For associated types we include both bounds written on the type
11 /// (`type X: Trait`) and predicates from the trait: `where Self::X: Trait`.
12 ///
13 /// Note that this filtering is done with the items identity substs to
14 /// simplify checking that these bounds are met in impls. This means that
15 /// a bound such as `for<'b> <Self as X<'b>>::U: Clone` can't be used, as in
16 /// `hr-associated-type-bound-1.rs`.
17 fn associated_type_bounds<'tcx>(
18     tcx: TyCtxt<'tcx>,
19     assoc_item_def_id: DefId,
20     ast_bounds: &'tcx [hir::GenericBound<'tcx>],
21     span: Span,
22 ) -> &'tcx [(ty::Predicate<'tcx>, Span)] {
23     let item_ty = tcx.mk_projection(
24         assoc_item_def_id,
25         InternalSubsts::identity_for_item(tcx, assoc_item_def_id),
26     );
27
28     let icx = ItemCtxt::new(tcx, assoc_item_def_id);
29     let mut bounds = icx.astconv().compute_bounds(item_ty, ast_bounds);
30     // Associated types are implicitly sized unless a `?Sized` bound is found
31     icx.astconv().add_implicitly_sized(&mut bounds, item_ty, ast_bounds, None, span);
32
33     let trait_def_id = tcx.parent(assoc_item_def_id);
34     let trait_predicates = tcx.trait_explicit_predicates_and_bounds(trait_def_id.expect_local());
35
36     let bounds_from_parent = trait_predicates.predicates.iter().copied().filter(|(pred, _)| {
37         match pred.kind().skip_binder() {
38             ty::PredicateKind::Clause(ty::Clause::Trait(tr)) => tr.self_ty() == item_ty,
39             ty::PredicateKind::Clause(ty::Clause::Projection(proj)) => {
40                 proj.projection_ty.self_ty() == item_ty
41             }
42             ty::PredicateKind::Clause(ty::Clause::TypeOutlives(outlives)) => outlives.0 == item_ty,
43             _ => false,
44         }
45     });
46
47     let all_bounds = tcx.arena.alloc_from_iter(bounds.predicates().chain(bounds_from_parent));
48     debug!("associated_type_bounds({}) = {:?}", tcx.def_path_str(assoc_item_def_id), all_bounds);
49     all_bounds
50 }
51
52 /// Opaque types don't inherit bounds from their parent: for return position
53 /// impl trait it isn't possible to write a suitable predicate on the
54 /// containing function and for type-alias impl trait we don't have a backwards
55 /// compatibility issue.
56 #[instrument(level = "trace", skip(tcx), ret)]
57 fn opaque_type_bounds<'tcx>(
58     tcx: TyCtxt<'tcx>,
59     opaque_def_id: DefId,
60     ast_bounds: &'tcx [hir::GenericBound<'tcx>],
61     span: Span,
62     in_trait: bool,
63 ) -> &'tcx [(ty::Predicate<'tcx>, Span)] {
64     ty::print::with_no_queries!({
65         let substs = InternalSubsts::identity_for_item(tcx, opaque_def_id);
66         let item_ty = if in_trait {
67             tcx.mk_projection(opaque_def_id, substs)
68         } else {
69             tcx.mk_opaque(opaque_def_id, substs)
70         };
71
72         let icx = ItemCtxt::new(tcx, opaque_def_id);
73         let mut bounds = icx.astconv().compute_bounds(item_ty, ast_bounds);
74         // Opaque types are implicitly sized unless a `?Sized` bound is found
75         icx.astconv().add_implicitly_sized(&mut bounds, item_ty, ast_bounds, None, span);
76         debug!(?bounds);
77
78         tcx.arena.alloc_from_iter(bounds.predicates())
79     })
80 }
81
82 pub(super) fn explicit_item_bounds(
83     tcx: TyCtxt<'_>,
84     def_id: DefId,
85 ) -> &'_ [(ty::Predicate<'_>, Span)] {
86     let hir_id = tcx.hir().local_def_id_to_hir_id(def_id.expect_local());
87     match tcx.hir().get(hir_id) {
88         hir::Node::TraitItem(hir::TraitItem {
89             kind: hir::TraitItemKind::Type(bounds, _),
90             span,
91             ..
92         }) => associated_type_bounds(tcx, def_id, bounds, *span),
93         hir::Node::Item(hir::Item {
94             kind: hir::ItemKind::OpaqueTy(hir::OpaqueTy { bounds, in_trait, .. }),
95             span,
96             ..
97         }) => opaque_type_bounds(tcx, def_id, bounds, *span, *in_trait),
98         _ => bug!("item_bounds called on {:?}", def_id),
99     }
100 }
101
102 pub(super) fn item_bounds(
103     tcx: TyCtxt<'_>,
104     def_id: DefId,
105 ) -> ty::EarlyBinder<&'_ ty::List<ty::Predicate<'_>>> {
106     let bounds = tcx.mk_predicates(
107         util::elaborate_predicates(
108             tcx,
109             tcx.explicit_item_bounds(def_id).iter().map(|&(bound, _span)| bound),
110         )
111         .map(|obligation| obligation.predicate),
112     );
113     ty::EarlyBinder(bounds)
114 }