@@ -2419,6 +2419,10 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
2419
2419
return Ok ( None ) ;
2420
2420
}
2421
2421
2422
+ //
2423
+ // Select applicable inherent associated type candidates modulo regions.
2424
+ //
2425
+
2422
2426
// In contexts that have no inference context, just make a new one.
2423
2427
// We do need a local variable to store it, though.
2424
2428
let infcx_;
@@ -2431,14 +2435,18 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
2431
2435
}
2432
2436
} ;
2433
2437
2434
- let param_env = tcx. param_env ( block. owner . to_def_id ( ) ) ;
2438
+ // FIXME(inherent_associated_types): Acquiring the ParamEnv this early leads to cycle errors
2439
+ // when inside of an ADT (#108491) or where clause.
2440
+ let param_env = tcx. param_env ( block. owner ) ;
2435
2441
let cause = ObligationCause :: misc ( span, block. owner . def_id ) ;
2436
2442
2437
2443
let mut fulfillment_errors = Vec :: new ( ) ;
2438
2444
let mut applicable_candidates: Vec < _ > = infcx. probe ( |_| {
2439
2445
let universe = infcx. create_next_universe ( ) ;
2440
2446
2441
2447
// Regions are not considered during selection.
2448
+ // FIXME(non_lifetime_binders): Here we are "truncating" or "flattening" the universes
2449
+ // of type and const binders. Is that correct in the selection phase? See also #109505.
2442
2450
let self_ty = tcx. replace_escaping_bound_vars_uncached (
2443
2451
self_ty,
2444
2452
FnMutDelegate {
@@ -2454,41 +2462,40 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
2454
2462
2455
2463
candidates
2456
2464
. iter ( )
2457
- . filter_map ( |& ( impl_, ( assoc_item, def_scope) ) | {
2465
+ . copied ( )
2466
+ . filter ( |& ( impl_, _) | {
2458
2467
infcx. probe ( |_| {
2459
2468
let ocx = ObligationCtxt :: new_in_snapshot ( & infcx) ;
2460
2469
2461
- let impl_ty = tcx. type_of ( impl_) ;
2462
2470
let impl_substs = infcx. fresh_item_substs ( impl_) ;
2463
- let impl_ty = impl_ty . subst ( tcx, impl_substs) ;
2471
+ let impl_ty = tcx . type_of ( impl_ ) . subst ( tcx, impl_substs) ;
2464
2472
let impl_ty = ocx. normalize ( & cause, param_env, impl_ty) ;
2465
2473
2466
- // Check that the Self-types can be related.
2467
- // FIXME(fmease): Should we use `eq` here?
2468
- ocx. sup ( & ObligationCause :: dummy ( ) , param_env, impl_ty, self_ty) . ok ( ) ?;
2474
+ // Check that the self types can be related.
2475
+ // FIXME(inherent_associated_types): Should we use `eq` here? Method probing uses
2476
+ // `sup` for this situtation, too. What for? To constrain inference variables?
2477
+ if ocx. sup ( & ObligationCause :: dummy ( ) , param_env, impl_ty, self_ty) . is_err ( )
2478
+ {
2479
+ return false ;
2480
+ }
2469
2481
2470
2482
// Check whether the impl imposes obligations we have to worry about.
2471
- let impl_bounds = tcx. predicates_of ( impl_) ;
2472
- let impl_bounds = impl_bounds. instantiate ( tcx, impl_substs) ;
2473
-
2483
+ let impl_bounds = tcx. predicates_of ( impl_) . instantiate ( tcx, impl_substs) ;
2474
2484
let impl_bounds = ocx. normalize ( & cause, param_env, impl_bounds) ;
2475
-
2476
2485
let impl_obligations = traits:: predicates_for_generics (
2477
2486
|_, _| cause. clone ( ) ,
2478
2487
param_env,
2479
2488
impl_bounds,
2480
2489
) ;
2481
-
2482
2490
ocx. register_obligations ( impl_obligations) ;
2483
2491
2484
2492
let mut errors = ocx. select_where_possible ( ) ;
2485
2493
if !errors. is_empty ( ) {
2486
2494
fulfillment_errors. append ( & mut errors) ;
2487
- return None ;
2495
+ return false ;
2488
2496
}
2489
2497
2490
- // FIXME(fmease): Unsolved vars can escape this InferCtxt snapshot.
2491
- Some ( ( assoc_item, def_scope, infcx. resolve_vars_if_possible ( impl_substs) ) )
2498
+ true
2492
2499
} )
2493
2500
} )
2494
2501
. collect ( )
@@ -2497,24 +2504,26 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
2497
2504
if applicable_candidates. len ( ) > 1 {
2498
2505
return Err ( self . complain_about_ambiguous_inherent_assoc_type (
2499
2506
name,
2500
- applicable_candidates. into_iter ( ) . map ( |( candidate, .. ) | candidate) . collect ( ) ,
2507
+ applicable_candidates. into_iter ( ) . map ( |( _ , ( candidate, _ ) ) | candidate) . collect ( ) ,
2501
2508
span,
2502
2509
) ) ;
2503
2510
}
2504
2511
2505
- if let Some ( ( assoc_item , def_scope , impl_substs ) ) = applicable_candidates. pop ( ) {
2512
+ if let Some ( ( impl_ , ( assoc_item , def_scope ) ) ) = applicable_candidates. pop ( ) {
2506
2513
self . check_assoc_ty ( assoc_item, name, def_scope, block, span) ;
2507
2514
2508
- // FIXME(inherent_associated_types): To fully *confirm* the *probed* candidate, we still
2509
- // need to relate the Self-type with fresh item substs & register region obligations for
2510
- // regionck to prove/disprove.
2511
-
2512
- let item_substs =
2513
- self . create_substs_for_associated_item ( span, assoc_item, segment, impl_substs) ;
2515
+ // FIXME(fmease): Currently creating throwaway `parent_substs` to please
2516
+ // `create_substs_for_associated_item`. Modify the latter instead (or sth. similar) to
2517
+ // not require the parent substs logic.
2518
+ let parent_substs = InternalSubsts :: identity_for_item ( tcx, impl_) ;
2519
+ let substs =
2520
+ self . create_substs_for_associated_item ( span, assoc_item, segment, parent_substs) ;
2521
+ let substs = tcx. mk_substs_from_iter (
2522
+ std:: iter:: once ( ty:: GenericArg :: from ( self_ty) )
2523
+ . chain ( substs. into_iter ( ) . skip ( parent_substs. len ( ) ) ) ,
2524
+ ) ;
2514
2525
2515
- // FIXME(fmease, #106722): Check if the bounds on the parameters of the
2516
- // associated type hold, if any.
2517
- let ty = tcx. type_of ( assoc_item) . subst ( tcx, item_substs) ;
2526
+ let ty = tcx. mk_alias ( ty:: Inherent , tcx. mk_alias_ty ( assoc_item, substs) ) ;
2518
2527
2519
2528
return Ok ( Some ( ( ty, assoc_item) ) ) ;
2520
2529
}
0 commit comments