@@ -23,14 +23,15 @@ use rustc_middle::ty::{
23
23
use rustc_session:: config:: DumpSolverProofTree ;
24
24
use rustc_span:: DUMMY_SP ;
25
25
use std:: io:: Write ;
26
+ use std:: iter;
26
27
use std:: ops:: ControlFlow ;
27
28
28
29
use crate :: traits:: vtable:: { count_own_vtable_entries, prepare_vtable_segments, VtblSegment } ;
29
30
30
31
use super :: inspect:: ProofTreeBuilder ;
31
- use super :: SolverMode ;
32
32
use super :: { search_graph, GoalEvaluationKind } ;
33
33
use super :: { search_graph:: SearchGraph , Goal } ;
34
+ use super :: { GoalSource , SolverMode } ;
34
35
pub use select:: InferCtxtSelectExt ;
35
36
36
37
mod canonical;
@@ -105,7 +106,7 @@ pub(super) struct NestedGoals<'tcx> {
105
106
/// can be unsound with more powerful coinduction in the future.
106
107
pub ( super ) normalizes_to_hack_goal : Option < Goal < ' tcx , ty:: NormalizesTo < ' tcx > > > ,
107
108
/// The rest of the goals which have not yet processed or remain ambiguous.
108
- pub ( super ) goals : Vec < Goal < ' tcx , ty:: Predicate < ' tcx > > > ,
109
+ pub ( super ) goals : Vec < ( GoalSource , Goal < ' tcx , ty:: Predicate < ' tcx > > ) > ,
109
110
}
110
111
111
112
impl < ' tcx > NestedGoals < ' tcx > {
@@ -156,7 +157,7 @@ impl<'tcx> InferCtxtEvalExt<'tcx> for InferCtxt<'tcx> {
156
157
Option < inspect:: GoalEvaluation < ' tcx > > ,
157
158
) {
158
159
EvalCtxt :: enter_root ( self , generate_proof_tree, |ecx| {
159
- ecx. evaluate_goal ( GoalEvaluationKind :: Root , goal)
160
+ ecx. evaluate_goal ( GoalEvaluationKind :: Root , GoalSource :: Misc , goal)
160
161
} )
161
162
}
162
163
}
@@ -334,6 +335,7 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
334
335
fn evaluate_goal (
335
336
& mut self ,
336
337
goal_evaluation_kind : GoalEvaluationKind ,
338
+ source : GoalSource ,
337
339
goal : Goal < ' tcx , ty:: Predicate < ' tcx > > ,
338
340
) -> Result < ( bool , Certainty , Vec < Goal < ' tcx , ty:: Predicate < ' tcx > > > ) , NoSolution > {
339
341
let ( orig_values, canonical_goal) = self . canonicalize_goal ( goal) ;
@@ -353,13 +355,13 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
353
355
Ok ( response) => response,
354
356
} ;
355
357
356
- let has_changed = !canonical_response . value . var_values . is_identity_modulo_regions ( )
357
- || !canonical_response . value . external_constraints . opaque_types . is_empty ( ) ;
358
- let ( certainty , nested_goals ) = match self . instantiate_and_apply_query_response (
359
- goal . param_env ,
360
- orig_values,
361
- canonical_response,
362
- ) {
358
+ let ( certainty , has_changed, nested_goals ) = match self
359
+ . instantiate_response_discarding_overflow (
360
+ goal . param_env ,
361
+ source ,
362
+ orig_values,
363
+ canonical_response,
364
+ ) {
363
365
Err ( e) => {
364
366
self . inspect . goal_evaluation ( goal_evaluation) ;
365
367
return Err ( e) ;
@@ -386,6 +388,44 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
386
388
Ok ( ( has_changed, certainty, nested_goals) )
387
389
}
388
390
391
+ fn instantiate_response_discarding_overflow (
392
+ & mut self ,
393
+ param_env : ty:: ParamEnv < ' tcx > ,
394
+ source : GoalSource ,
395
+ original_values : Vec < ty:: GenericArg < ' tcx > > ,
396
+ response : CanonicalResponse < ' tcx > ,
397
+ ) -> Result < ( Certainty , bool , Vec < Goal < ' tcx , ty:: Predicate < ' tcx > > > ) , NoSolution > {
398
+ // The old solver did not evaluate nested goals when normalizing.
399
+ // It returned the selection constraints allowing a `Projection`
400
+ // obligation to not hold in coherence while avoiding the fatal error
401
+ // from overflow.
402
+ //
403
+ // We match this behavior here by considering all constraints
404
+ // from nested goals which are not from where-bounds. We will already
405
+ // need to track which nested goals are required by impl where-bounds
406
+ // for coinductive cycles, so we simply reuse that here.
407
+ //
408
+ // While we could consider overflow constraints in more cases, this should
409
+ // not be necessary for backcompat and results in better perf. It also
410
+ // avoids a potential inconsistency which would otherwise require some
411
+ // tracking for root goals as well. See #119071 for an example.
412
+ let keep_overflow_constraints = || {
413
+ self . search_graph . current_goal_is_normalizes_to ( )
414
+ && source != GoalSource :: ImplWhereBound
415
+ } ;
416
+
417
+ if response. value . certainty == Certainty :: OVERFLOW && !keep_overflow_constraints ( ) {
418
+ Ok ( ( Certainty :: OVERFLOW , false , Vec :: new ( ) ) )
419
+ } else {
420
+ let has_changed = !response. value . var_values . is_identity_modulo_regions ( )
421
+ || !response. value . external_constraints . opaque_types . is_empty ( ) ;
422
+
423
+ let ( certainty, nested_goals) =
424
+ self . instantiate_and_apply_query_response ( param_env, original_values, response) ?;
425
+ Ok ( ( certainty, has_changed, nested_goals) )
426
+ }
427
+ }
428
+
389
429
fn compute_goal ( & mut self , goal : Goal < ' tcx , ty:: Predicate < ' tcx > > ) -> QueryResult < ' tcx > {
390
430
let Goal { param_env, predicate } = goal;
391
431
let kind = predicate. kind ( ) ;
@@ -439,7 +479,7 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
439
479
} else {
440
480
let kind = self . infcx . instantiate_binder_with_placeholders ( kind) ;
441
481
let goal = goal. with ( self . tcx ( ) , ty:: Binder :: dummy ( kind) ) ;
442
- self . add_goal ( goal) ;
482
+ self . add_goal ( GoalSource :: Misc , goal) ;
443
483
self . evaluate_added_goals_and_make_canonical_response ( Certainty :: Yes )
444
484
}
445
485
}
@@ -488,6 +528,13 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
488
528
let mut goals = core:: mem:: replace ( & mut self . nested_goals , NestedGoals :: new ( ) ) ;
489
529
490
530
self . inspect . evaluate_added_goals_loop_start ( ) ;
531
+
532
+ fn with_misc_source < ' tcx > (
533
+ it : impl IntoIterator < Item = Goal < ' tcx , ty:: Predicate < ' tcx > > > ,
534
+ ) -> impl Iterator < Item = ( GoalSource , Goal < ' tcx , ty:: Predicate < ' tcx > > ) > {
535
+ iter:: zip ( iter:: repeat ( GoalSource :: Misc ) , it)
536
+ }
537
+
491
538
// If this loop did not result in any progress, what's our final certainty.
492
539
let mut unchanged_certainty = Some ( Certainty :: Yes ) ;
493
540
if let Some ( goal) = goals. normalizes_to_hack_goal . take ( ) {
@@ -501,9 +548,10 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
501
548
502
549
let ( _, certainty, instantiate_goals) = self . evaluate_goal (
503
550
GoalEvaluationKind :: Nested { is_normalizes_to_hack : IsNormalizesToHack :: Yes } ,
551
+ GoalSource :: Misc ,
504
552
unconstrained_goal,
505
553
) ?;
506
- self . nested_goals . goals . extend ( instantiate_goals) ;
554
+ self . nested_goals . goals . extend ( with_misc_source ( instantiate_goals) ) ;
507
555
508
556
// Finally, equate the goal's RHS with the unconstrained var.
509
557
// We put the nested goals from this into goals instead of
@@ -512,7 +560,7 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
512
560
// matters in practice, though.
513
561
let eq_goals =
514
562
self . eq_and_get_goals ( goal. param_env , goal. predicate . term , unconstrained_rhs) ?;
515
- goals. goals . extend ( eq_goals) ;
563
+ goals. goals . extend ( with_misc_source ( eq_goals) ) ;
516
564
517
565
// We only look at the `projection_ty` part here rather than
518
566
// looking at the "has changed" return from evaluate_goal,
@@ -533,20 +581,21 @@ impl<'a, 'tcx> EvalCtxt<'a, 'tcx> {
533
581
}
534
582
}
535
583
536
- for goal in goals. goals . drain ( ..) {
584
+ for ( source , goal) in goals. goals . drain ( ..) {
537
585
let ( has_changed, certainty, instantiate_goals) = self . evaluate_goal (
538
586
GoalEvaluationKind :: Nested { is_normalizes_to_hack : IsNormalizesToHack :: No } ,
587
+ source,
539
588
goal,
540
589
) ?;
541
- self . nested_goals . goals . extend ( instantiate_goals) ;
590
+ self . nested_goals . goals . extend ( with_misc_source ( instantiate_goals) ) ;
542
591
if has_changed {
543
592
unchanged_certainty = None ;
544
593
}
545
594
546
595
match certainty {
547
596
Certainty :: Yes => { }
548
597
Certainty :: Maybe ( _) => {
549
- self . nested_goals . goals . push ( goal) ;
598
+ self . nested_goals . goals . push ( ( source , goal) ) ;
550
599
unchanged_certainty = unchanged_certainty. map ( |c| c. unify_with ( certainty) ) ;
551
600
}
552
601
}
@@ -670,7 +719,7 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
670
719
. at ( & ObligationCause :: dummy ( ) , param_env)
671
720
. eq ( DefineOpaqueTypes :: No , lhs, rhs)
672
721
. map ( |InferOk { value : ( ) , obligations } | {
673
- self . add_goals ( obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
722
+ self . add_goals ( GoalSource :: Misc , obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
674
723
} )
675
724
. map_err ( |e| {
676
725
debug ! ( ?e, "failed to equate" ) ;
@@ -689,7 +738,7 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
689
738
. at ( & ObligationCause :: dummy ( ) , param_env)
690
739
. sub ( DefineOpaqueTypes :: No , sub, sup)
691
740
. map ( |InferOk { value : ( ) , obligations } | {
692
- self . add_goals ( obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
741
+ self . add_goals ( GoalSource :: Misc , obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
693
742
} )
694
743
. map_err ( |e| {
695
744
debug ! ( ?e, "failed to subtype" ) ;
@@ -709,7 +758,7 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
709
758
. at ( & ObligationCause :: dummy ( ) , param_env)
710
759
. relate ( DefineOpaqueTypes :: No , lhs, variance, rhs)
711
760
. map ( |InferOk { value : ( ) , obligations } | {
712
- self . add_goals ( obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
761
+ self . add_goals ( GoalSource :: Misc , obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
713
762
} )
714
763
. map_err ( |e| {
715
764
debug ! ( ?e, "failed to relate" ) ;
@@ -842,7 +891,7 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
842
891
true ,
843
892
& mut obligations,
844
893
) ?;
845
- self . add_goals ( obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
894
+ self . add_goals ( GoalSource :: Misc , obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
846
895
Ok ( ( ) )
847
896
}
848
897
@@ -862,7 +911,7 @@ impl<'tcx> EvalCtxt<'_, 'tcx> {
862
911
hidden_ty,
863
912
& mut obligations,
864
913
) ;
865
- self . add_goals ( obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
914
+ self . add_goals ( GoalSource :: Misc , obligations. into_iter ( ) . map ( |o| o. into ( ) ) ) ;
866
915
}
867
916
868
917
// Do something for each opaque/hidden pair defined with `def_id` in the
0 commit comments