@@ -137,7 +137,7 @@ impl<'tcx> TraitEngine<'tcx> for FulfillmentCtxt<'tcx> {
137
137
. collect ( ) ;
138
138
139
139
errors. extend ( self . obligations . overflowed . drain ( ..) . map ( |obligation| FulfillmentError {
140
- obligation : find_best_leaf_obligation ( infcx, & obligation) ,
140
+ obligation : find_best_leaf_obligation ( infcx, & obligation, true ) ,
141
141
code : FulfillmentErrorCode :: Ambiguity { overflow : Some ( true ) } ,
142
142
root_obligation : obligation,
143
143
} ) ) ;
@@ -198,7 +198,7 @@ fn fulfillment_error_for_no_solution<'tcx>(
198
198
infcx : & InferCtxt < ' tcx > ,
199
199
root_obligation : PredicateObligation < ' tcx > ,
200
200
) -> FulfillmentError < ' tcx > {
201
- let obligation = find_best_leaf_obligation ( infcx, & root_obligation) ;
201
+ let obligation = find_best_leaf_obligation ( infcx, & root_obligation, false ) ;
202
202
203
203
let code = match obligation. predicate . kind ( ) . skip_binder ( ) {
204
204
ty:: PredicateKind :: Clause ( ty:: ClauseKind :: Projection ( _) ) => {
@@ -266,7 +266,7 @@ fn fulfillment_error_for_stalled<'tcx>(
266
266
} ) ;
267
267
268
268
FulfillmentError {
269
- obligation : find_best_leaf_obligation ( infcx, & obligation) ,
269
+ obligation : find_best_leaf_obligation ( infcx, & obligation, true ) ,
270
270
code,
271
271
root_obligation : obligation,
272
272
}
@@ -275,19 +275,21 @@ fn fulfillment_error_for_stalled<'tcx>(
275
275
fn find_best_leaf_obligation < ' tcx > (
276
276
infcx : & InferCtxt < ' tcx > ,
277
277
obligation : & PredicateObligation < ' tcx > ,
278
+ consider_ambiguities : bool ,
278
279
) -> PredicateObligation < ' tcx > {
279
280
let obligation = infcx. resolve_vars_if_possible ( obligation. clone ( ) ) ;
280
281
infcx
281
282
. visit_proof_tree (
282
283
obligation. clone ( ) . into ( ) ,
283
- & mut BestObligation { obligation : obligation. clone ( ) } ,
284
+ & mut BestObligation { obligation : obligation. clone ( ) , consider_ambiguities } ,
284
285
)
285
286
. break_value ( )
286
287
. unwrap_or ( obligation)
287
288
}
288
289
289
290
struct BestObligation < ' tcx > {
290
291
obligation : PredicateObligation < ' tcx > ,
292
+ consider_ambiguities : bool ,
291
293
}
292
294
293
295
impl < ' tcx > BestObligation < ' tcx > {
@@ -355,11 +357,11 @@ impl<'tcx> ProofTreeVisitor<'tcx> for BestObligation<'tcx> {
355
357
}
356
358
}
357
359
358
- // Skip nested goals that hold .
359
- //FIXME: We should change the max allowed certainty based on if we're
360
- // visiting an ambiguity or error obligation.
361
- if matches ! ( nested_goal. result( ) , Ok ( Certainty :: Yes ) ) {
362
- continue ;
360
+ // Skip nested goals that aren't the *reason* for our goal's failure .
361
+ match self . consider_ambiguities {
362
+ true if matches ! ( nested_goal . result ( ) , Ok ( Certainty :: Maybe ( _ ) ) ) => { }
363
+ false if matches ! ( nested_goal. result( ) , Err ( _ ) ) => { }
364
+ _ => continue ,
363
365
}
364
366
365
367
self . with_derived_obligation ( obligation, |this| nested_goal. visit_with ( this) ) ?;
0 commit comments