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
11pub 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
36pub trait Elaboratable<I: Interner> {
38 fn predicate(&self) -> I::Predicate;
39
40 fn child(&self, clause: I::Clause) -> Self;
42
43 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 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 fn extend_deduped(&mut self, obligations: impl IntoIterator<Item = O>) {
102 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 pub fn filter_only_self(mut self) -> Self {
116 self.mode = Filter::OnlySelf;
117 self
118 }
119
120 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 let Some(clause) = elaboratable.predicate().as_clause() else {
132 return;
133 };
134
135 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 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 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 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 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 }
223 ty::ClauseKind::WellFormed(..) => {
224 }
227 ty::ClauseKind::Projection(..) => {
228 }
230 ty::ClauseKind::ConstEvaluatable(..) => {
231 }
234 ty::ClauseKind::ConstArgHasType(..) => {
235 }
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 Some(ty::ClauseKind::TypeOutlives(ty::OutlivesPredicate(
271 alias_ty.to_ty(cx),
272 outlives_region,
273 )))
274 }
275
276 Component::EscapingAlias(_) => {
277 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 if let Some(obligation) = self.stack.pop() {
294 self.elaborate(&obligation);
295 Some(obligation)
296 } else {
297 None
298 }
299 }
300}
301
302pub 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
347pub 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}