rustc_type_ir/
elaborate.rs

1use std::marker::PhantomData;
2
3use smallvec::smallvec;
4
5use crate::data_structures::HashSet;
6use crate::inherent::*;
7use crate::lang_items::TraitSolverLangItem;
8use crate::outlives::{Component, push_outlives_components};
9use crate::{self as ty, Interner, Upcast as _};
10
11/// "Elaboration" is the process of identifying all the predicates that
12/// are implied by a source predicate. Currently, this basically means
13/// walking the "supertraits" and other similar assumptions. For example,
14/// if we know that `T: Ord`, the elaborator would deduce that `T: PartialOrd`
15/// holds as well. Similarly, if we have `trait Foo: 'static`, and we know that
16/// `T: Foo`, then we know that `T: 'static`.
17pub struct Elaborator<I: Interner, O> {
18    cx: I,
19    stack: Vec<O>,
20    visited: HashSet<ty::Binder<I, ty::PredicateKind<I>>>,
21    mode: Filter,
22    elaborate_sized: ElaborateSized,
23}
24
25enum Filter {
26    All,
27    OnlySelf,
28}
29
30#[derive(Eq, PartialEq)]
31enum ElaborateSized {
32    Yes,
33    No,
34}
35
36/// Describes how to elaborate an obligation into a sub-obligation.
37pub trait Elaboratable<I: Interner> {
38    fn predicate(&self) -> I::Predicate;
39
40    // Makes a new `Self` but with a different clause that comes from elaboration.
41    fn child(&self, clause: I::Clause) -> Self;
42
43    // Makes a new `Self` but with a different clause and a different cause
44    // code (if `Self` has one, such as [`PredicateObligation`]).
45    fn child_with_derived_cause(
46        &self,
47        clause: I::Clause,
48        span: I::Span,
49        parent_trait_pred: ty::Binder<I, ty::TraitPredicate<I>>,
50        index: usize,
51    ) -> Self;
52}
53
54pub struct ClauseWithSupertraitSpan<I: Interner> {
55    pub clause: I::Clause,
56    // Span of the supertrait predicatae that lead to this clause.
57    pub supertrait_span: I::Span,
58}
59impl<I: Interner> ClauseWithSupertraitSpan<I> {
60    pub fn new(clause: I::Clause, span: I::Span) -> Self {
61        ClauseWithSupertraitSpan { clause, supertrait_span: span }
62    }
63}
64impl<I: Interner> Elaboratable<I> for ClauseWithSupertraitSpan<I> {
65    fn predicate(&self) -> <I as Interner>::Predicate {
66        self.clause.as_predicate()
67    }
68
69    fn child(&self, clause: <I as Interner>::Clause) -> Self {
70        ClauseWithSupertraitSpan { clause, supertrait_span: self.supertrait_span }
71    }
72
73    fn child_with_derived_cause(
74        &self,
75        clause: <I as Interner>::Clause,
76        supertrait_span: <I as Interner>::Span,
77        _parent_trait_pred: crate::Binder<I, crate::TraitPredicate<I>>,
78        _index: usize,
79    ) -> Self {
80        ClauseWithSupertraitSpan { clause, supertrait_span }
81    }
82}
83
84pub fn elaborate<I: Interner, O: Elaboratable<I>>(
85    cx: I,
86    obligations: impl IntoIterator<Item = O>,
87) -> Elaborator<I, O> {
88    let mut elaborator = Elaborator {
89        cx,
90        stack: Vec::new(),
91        visited: HashSet::default(),
92        mode: Filter::All,
93        elaborate_sized: ElaborateSized::No,
94    };
95    elaborator.extend_deduped(obligations);
96    elaborator
97}
98
99impl<I: Interner, O: Elaboratable<I>> Elaborator<I, O> {
100    /// Adds `obligations` to the stack.
101    fn extend_deduped(&mut self, obligations: impl IntoIterator<Item = O>) {
102        // Only keep those bounds that we haven't already seen.
103        // This is necessary to prevent infinite recursion in some
104        // cases. One common case is when people define
105        // `trait Sized: Sized { }` rather than `trait Sized { }`.
106        self.stack.extend(
107            obligations.into_iter().filter(|o| {
108                self.visited.insert(self.cx.anonymize_bound_vars(o.predicate().kind()))
109            }),
110        );
111    }
112
113    /// Filter to only the supertraits of trait predicates, i.e. only the predicates
114    /// that have `Self` as their self type, instead of all implied predicates.
115    pub fn filter_only_self(mut self) -> Self {
116        self.mode = Filter::OnlySelf;
117        self
118    }
119
120    /// Start elaborating `Sized` - reqd during coherence checking, normally skipped to improve
121    /// compiler performance.
122    pub fn elaborate_sized(mut self) -> Self {
123        self.elaborate_sized = ElaborateSized::Yes;
124        self
125    }
126
127    fn elaborate(&mut self, elaboratable: &O) {
128        let cx = self.cx;
129
130        // We only elaborate clauses.
131        let Some(clause) = elaboratable.predicate().as_clause() else {
132            return;
133        };
134
135        // PERF(sized-hierarchy): To avoid iterating over sizedness supertraits in
136        // parameter environments, as an optimisation, sizedness supertraits aren't
137        // elaborated, so check if a `Sized` obligation is being elaborated to a
138        // `MetaSized` obligation and emit it. Candidate assembly and confirmation
139        // are modified to check for the `Sized` subtrait when a `MetaSized` obligation
140        // is present.
141        if self.elaborate_sized == ElaborateSized::No
142            && let Some(did) = clause.as_trait_clause().map(|c| c.def_id())
143            && self.cx.is_lang_item(did, TraitSolverLangItem::Sized)
144        {
145            return;
146        }
147
148        let bound_clause = clause.kind();
149        match bound_clause.skip_binder() {
150            ty::ClauseKind::Trait(data) => {
151                // Negative trait bounds do not imply any supertrait bounds
152                if data.polarity != ty::PredicatePolarity::Positive {
153                    return;
154                }
155
156                let map_to_child_clause =
157                    |(index, (clause, span)): (usize, (I::Clause, I::Span))| {
158                        elaboratable.child_with_derived_cause(
159                            clause.instantiate_supertrait(cx, bound_clause.rebind(data.trait_ref)),
160                            span,
161                            bound_clause.rebind(data),
162                            index,
163                        )
164                    };
165
166                // Get predicates implied by the trait, or only super predicates if we only care about self predicates.
167                match self.mode {
168                    Filter::All => self.extend_deduped(
169                        cx.explicit_implied_predicates_of(data.def_id())
170                            .iter_identity()
171                            .enumerate()
172                            .map(map_to_child_clause),
173                    ),
174                    Filter::OnlySelf => self.extend_deduped(
175                        cx.explicit_super_predicates_of(data.def_id())
176                            .iter_identity()
177                            .enumerate()
178                            .map(map_to_child_clause),
179                    ),
180                };
181            }
182            // `T: [const] Trait` implies `T: [const] Supertrait`.
183            ty::ClauseKind::HostEffect(data) => self.extend_deduped(
184                cx.explicit_implied_const_bounds(data.def_id()).iter_identity().map(|trait_ref| {
185                    elaboratable.child(
186                        trait_ref
187                            .to_host_effect_clause(cx, data.constness)
188                            .instantiate_supertrait(cx, bound_clause.rebind(data.trait_ref)),
189                    )
190                }),
191            ),
192            ty::ClauseKind::TypeOutlives(ty::OutlivesPredicate(ty_max, r_min)) => {
193                // We know that `T: 'a` for some type `T`. We can
194                // often elaborate this. For example, if we know that
195                // `[U]: 'a`, that implies that `U: 'a`. Similarly, if
196                // we know `&'a U: 'b`, then we know that `'a: 'b` and
197                // `U: 'b`.
198                //
199                // We can basically ignore bound regions here. So for
200                // example `for<'c> Foo<'a,'c>: 'b` can be elaborated to
201                // `'a: 'b`.
202
203                // Ignore `for<'a> T: 'a` -- we might in the future
204                // consider this as evidence that `T: 'static`, but
205                // I'm a bit wary of such constructions and so for now
206                // I want to be conservative. --nmatsakis
207                if r_min.is_bound() {
208                    return;
209                }
210
211                let mut components = smallvec![];
212                push_outlives_components(cx, ty_max, &mut components);
213                self.extend_deduped(
214                    components
215                        .into_iter()
216                        .filter_map(|component| elaborate_component_to_clause(cx, component, r_min))
217                        .map(|clause| elaboratable.child(bound_clause.rebind(clause).upcast(cx))),
218                );
219            }
220            ty::ClauseKind::RegionOutlives(..) => {
221                // Nothing to elaborate from `'a: 'b`.
222            }
223            ty::ClauseKind::WellFormed(..) => {
224                // Currently, we do not elaborate WF predicates,
225                // although we easily could.
226            }
227            ty::ClauseKind::Projection(..) => {
228                // Nothing to elaborate in a projection predicate.
229            }
230            ty::ClauseKind::ConstEvaluatable(..) => {
231                // Currently, we do not elaborate const-evaluatable
232                // predicates.
233            }
234            ty::ClauseKind::ConstArgHasType(..) => {
235                // Nothing to elaborate
236            }
237        }
238    }
239}
240
241fn elaborate_component_to_clause<I: Interner>(
242    cx: I,
243    component: Component<I>,
244    outlives_region: I::Region,
245) -> Option<ty::ClauseKind<I>> {
246    match component {
247        Component::Region(r) => {
248            if r.is_bound() {
249                None
250            } else {
251                Some(ty::ClauseKind::RegionOutlives(ty::OutlivesPredicate(r, outlives_region)))
252            }
253        }
254
255        Component::Param(p) => {
256            let ty = Ty::new_param(cx, p);
257            Some(ty::ClauseKind::TypeOutlives(ty::OutlivesPredicate(ty, outlives_region)))
258        }
259
260        Component::Placeholder(p) => {
261            let ty = Ty::new_placeholder(cx, p);
262            Some(ty::ClauseKind::TypeOutlives(ty::OutlivesPredicate(ty, outlives_region)))
263        }
264
265        Component::UnresolvedInferenceVariable(_) => None,
266
267        Component::Alias(alias_ty) => {
268            // We might end up here if we have `Foo<<Bar as Baz>::Assoc>: 'a`.
269            // With this, we can deduce that `<Bar as Baz>::Assoc: 'a`.
270            Some(ty::ClauseKind::TypeOutlives(ty::OutlivesPredicate(
271                alias_ty.to_ty(cx),
272                outlives_region,
273            )))
274        }
275
276        Component::EscapingAlias(_) => {
277            // We might be able to do more here, but we don't
278            // want to deal with escaping vars right now.
279            None
280        }
281    }
282}
283
284impl<I: Interner, O: Elaboratable<I>> Iterator for Elaborator<I, O> {
285    type Item = O;
286
287    fn size_hint(&self) -> (usize, Option<usize>) {
288        (self.stack.len(), None)
289    }
290
291    fn next(&mut self) -> Option<Self::Item> {
292        // Extract next item from top-most stack frame, if any.
293        if let Some(obligation) = self.stack.pop() {
294            self.elaborate(&obligation);
295            Some(obligation)
296        } else {
297            None
298        }
299    }
300}
301
302///////////////////////////////////////////////////////////////////////////
303// Supertrait iterator
304///////////////////////////////////////////////////////////////////////////
305
306/// Computes the def-ids of the transitive supertraits of `trait_def_id`. This (intentionally)
307/// does not compute the full elaborated super-predicates but just the set of def-ids. It is used
308/// to identify which traits may define a given associated type to help avoid cycle errors,
309/// and to make size estimates for vtable layout computation.
310pub fn supertrait_def_ids<I: Interner>(
311    cx: I,
312    trait_def_id: I::DefId,
313) -> impl Iterator<Item = I::DefId> {
314    let mut set = HashSet::default();
315    let mut stack = vec![trait_def_id];
316
317    set.insert(trait_def_id);
318
319    std::iter::from_fn(move || {
320        let trait_def_id = stack.pop()?;
321
322        for (predicate, _) in cx.explicit_super_predicates_of(trait_def_id).iter_identity() {
323            if let ty::ClauseKind::Trait(data) = predicate.kind().skip_binder()
324                && set.insert(data.def_id())
325            {
326                stack.push(data.def_id());
327            }
328        }
329
330        Some(trait_def_id)
331    })
332}
333
334pub fn supertraits<I: Interner>(
335    cx: I,
336    trait_ref: ty::Binder<I, ty::TraitRef<I>>,
337) -> FilterToTraits<I, Elaborator<I, I::Clause>> {
338    elaborate(cx, [trait_ref.upcast(cx)]).filter_only_self().filter_to_traits()
339}
340
341impl<I: Interner> Elaborator<I, I::Clause> {
342    fn filter_to_traits(self) -> FilterToTraits<I, Self> {
343        FilterToTraits { _cx: PhantomData, base_iterator: self }
344    }
345}
346
347/// A filter around an iterator of predicates that makes it yield up
348/// just trait references.
349pub struct FilterToTraits<I: Interner, It: Iterator<Item = I::Clause>> {
350    _cx: PhantomData<I>,
351    base_iterator: It,
352}
353
354impl<I: Interner, It: Iterator<Item = I::Clause>> Iterator for FilterToTraits<I, It> {
355    type Item = ty::Binder<I, ty::TraitRef<I>>;
356
357    fn next(&mut self) -> Option<ty::Binder<I, ty::TraitRef<I>>> {
358        while let Some(pred) = self.base_iterator.next() {
359            if let Some(data) = pred.as_trait_clause() {
360                return Some(data.map_bound(|t| t.trait_ref));
361            }
362        }
363        None
364    }
365
366    fn size_hint(&self) -> (usize, Option<usize>) {
367        let (_, upper) = self.base_iterator.size_hint();
368        (0, upper)
369    }
370}