@@ -3,6 +3,7 @@ use smallvec::smallvec;
3
3
use crate :: infer:: outlives:: components:: { push_outlives_components, Component } ;
4
4
use crate :: traits:: { self , Obligation , ObligationCauseCode , PredicateObligation } ;
5
5
use rustc_data_structures:: fx:: FxHashSet ;
6
+ use rustc_middle:: ty:: ToPolyTraitRef ;
6
7
use rustc_middle:: ty:: { self , Ty , TyCtxt , Upcast } ;
7
8
use rustc_span:: symbol:: Ident ;
8
9
use rustc_span:: Span ;
@@ -82,7 +83,6 @@ pub struct Elaborator<'tcx, O> {
82
83
enum Filter {
83
84
All ,
84
85
OnlySelf ,
85
- OnlySelfThatDefines ( Ident ) ,
86
86
}
87
87
88
88
/// Describes how to elaborate an obligation into a sub-obligation.
@@ -252,12 +252,6 @@ impl<'tcx, O: Elaboratable<'tcx>> Elaborator<'tcx, O> {
252
252
self
253
253
}
254
254
255
- /// Filter to only the supertraits of trait predicates that define the assoc_ty.
256
- pub fn filter_only_self_that_defines ( mut self , assoc_ty : Ident ) -> Self {
257
- self . mode = Filter :: OnlySelfThatDefines ( assoc_ty) ;
258
- self
259
- }
260
-
261
255
fn elaborate ( & mut self , elaboratable : & O ) {
262
256
let tcx = self . visited . tcx ;
263
257
@@ -277,9 +271,6 @@ impl<'tcx, O: Elaboratable<'tcx>> Elaborator<'tcx, O> {
277
271
let predicates = match self . mode {
278
272
Filter :: All => tcx. explicit_implied_predicates_of ( data. def_id ( ) ) ,
279
273
Filter :: OnlySelf => tcx. explicit_super_predicates_of ( data. def_id ( ) ) ,
280
- Filter :: OnlySelfThatDefines ( ident) => {
281
- tcx. explicit_supertraits_containing_assoc_item ( ( data. def_id ( ) , ident) )
282
- }
283
274
} ;
284
275
285
276
let obligations =
@@ -405,14 +396,14 @@ impl<'tcx, O: Elaboratable<'tcx>> Iterator for Elaborator<'tcx, O> {
405
396
pub fn supertraits < ' tcx > (
406
397
tcx : TyCtxt < ' tcx > ,
407
398
trait_ref : ty:: PolyTraitRef < ' tcx > ,
408
- ) -> FilterToTraits < Elaborator < ' tcx , ty:: Predicate < ' tcx > > > {
399
+ ) -> FilterToTraits < Elaborator < ' tcx , ty:: Clause < ' tcx > > > {
409
400
elaborate ( tcx, [ trait_ref. upcast ( tcx) ] ) . filter_only_self ( ) . filter_to_traits ( )
410
401
}
411
402
412
403
pub fn transitive_bounds < ' tcx > (
413
404
tcx : TyCtxt < ' tcx > ,
414
405
trait_refs : impl Iterator < Item = ty:: PolyTraitRef < ' tcx > > ,
415
- ) -> FilterToTraits < Elaborator < ' tcx , ty:: Predicate < ' tcx > > > {
406
+ ) -> FilterToTraits < Elaborator < ' tcx , ty:: Clause < ' tcx > > > {
416
407
elaborate ( tcx, trait_refs. map ( |trait_ref| trait_ref. upcast ( tcx) ) )
417
408
. filter_only_self ( )
418
409
. filter_to_traits ( )
@@ -427,17 +418,37 @@ pub fn transitive_bounds_that_define_assoc_item<'tcx>(
427
418
tcx : TyCtxt < ' tcx > ,
428
419
trait_refs : impl Iterator < Item = ty:: PolyTraitRef < ' tcx > > ,
429
420
assoc_name : Ident ,
430
- ) -> FilterToTraits < Elaborator < ' tcx , ty:: Predicate < ' tcx > > > {
431
- elaborate ( tcx, trait_refs. map ( |trait_ref| trait_ref. upcast ( tcx) ) )
432
- . filter_only_self_that_defines ( assoc_name)
433
- . filter_to_traits ( )
421
+ ) -> impl Iterator < Item = ty:: PolyTraitRef < ' tcx > > {
422
+ let mut seen = FxHashSet :: default ( ) ;
423
+ let mut stack: Vec < _ > = trait_refs. collect ( ) ;
424
+
425
+ std:: iter:: from_fn ( move || {
426
+ while let Some ( trait_ref) = stack. pop ( ) {
427
+ if !seen. insert ( tcx. anonymize_bound_vars ( trait_ref) ) {
428
+ continue ;
429
+ }
430
+
431
+ stack. extend (
432
+ tcx. explicit_supertraits_containing_assoc_item ( ( trait_ref. def_id ( ) , assoc_name) )
433
+ . instantiate_own_identity ( )
434
+ . map ( |( clause, _) | clause. instantiate_supertrait ( tcx, trait_ref) )
435
+ . filter_map ( |clause| clause. as_trait_clause ( ) )
436
+ // FIXME: Negative supertraits are elaborated here lol
437
+ . map ( |trait_pred| trait_pred. to_poly_trait_ref ( ) ) ,
438
+ ) ;
439
+
440
+ return Some ( trait_ref) ;
441
+ }
442
+
443
+ None
444
+ } )
434
445
}
435
446
436
447
///////////////////////////////////////////////////////////////////////////
437
448
// Other
438
449
///////////////////////////////////////////////////////////////////////////
439
450
440
- impl < ' tcx > Elaborator < ' tcx , ty:: Predicate < ' tcx > > {
451
+ impl < ' tcx > Elaborator < ' tcx , ty:: Clause < ' tcx > > {
441
452
fn filter_to_traits ( self ) -> FilterToTraits < Self > {
442
453
FilterToTraits { base_iterator : self }
443
454
}
@@ -449,7 +460,7 @@ pub struct FilterToTraits<I> {
449
460
base_iterator : I ,
450
461
}
451
462
452
- impl < ' tcx , I : Iterator < Item = ty:: Predicate < ' tcx > > > Iterator for FilterToTraits < I > {
463
+ impl < ' tcx , I : Iterator < Item = ty:: Clause < ' tcx > > > Iterator for FilterToTraits < I > {
453
464
type Item = ty:: PolyTraitRef < ' tcx > ;
454
465
455
466
fn next ( & mut self ) -> Option < ty:: PolyTraitRef < ' tcx > > {
0 commit comments