@@ -2,10 +2,11 @@ pub mod on_unimplemented;
2
2
pub mod suggestions;
3
3
4
4
use super :: {
5
- EvaluationResult , FulfillmentContext , FulfillmentError , FulfillmentErrorCode ,
6
- MismatchedProjectionTypes , Obligation , ObligationCause , ObligationCauseCode ,
7
- OnUnimplementedDirective , OnUnimplementedNote , OutputTypeParameterMismatch , Overflow ,
8
- PredicateObligation , SelectionContext , SelectionError , TraitNotObjectSafe ,
5
+ DerivedObligationCause , EvaluationResult , FulfillmentContext , FulfillmentError ,
6
+ FulfillmentErrorCode , ImplDerivedObligationCause , MismatchedProjectionTypes , Obligation ,
7
+ ObligationCause , ObligationCauseCode , OnUnimplementedDirective , OnUnimplementedNote ,
8
+ OutputTypeParameterMismatch , Overflow , PredicateObligation , SelectionContext , SelectionError ,
9
+ TraitNotObjectSafe ,
9
10
} ;
10
11
11
12
use crate :: infer:: error_reporting:: { TyCategory , TypeAnnotationNeeded as ErrorCode } ;
@@ -654,11 +655,77 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
654
655
} else if !suggested {
655
656
// Can't show anything else useful, try to find similar impls.
656
657
let impl_candidates = self . find_similar_impl_candidates ( trait_ref) ;
657
- self . report_similar_impl_candidates (
658
+ if ! self . report_similar_impl_candidates (
658
659
impl_candidates,
659
660
trait_ref,
660
661
& mut err,
661
- ) ;
662
+ ) {
663
+ // This is *almost* equivalent to
664
+ // `obligation.cause.code().peel_derives()`, but it gives us the
665
+ // trait predicate for that corresponding root obligation. This
666
+ // lets us get a derived obligation from a type parameter, like
667
+ // when calling `string.strip_suffix(p)` where `p` is *not* an
668
+ // implementer of `Pattern<'_>`.
669
+ let mut code = obligation. cause . code ( ) ;
670
+ let mut trait_pred = trait_predicate;
671
+ let mut peeled = false ;
672
+ loop {
673
+ match & * code {
674
+ ObligationCauseCode :: FunctionArgumentObligation {
675
+ parent_code,
676
+ ..
677
+ } => {
678
+ code = & parent_code;
679
+ }
680
+ ObligationCauseCode :: ImplDerivedObligation (
681
+ box ImplDerivedObligationCause {
682
+ derived :
683
+ DerivedObligationCause {
684
+ parent_code,
685
+ parent_trait_pred,
686
+ } ,
687
+ ..
688
+ } ,
689
+ )
690
+ | ObligationCauseCode :: BuiltinDerivedObligation (
691
+ DerivedObligationCause {
692
+ parent_code,
693
+ parent_trait_pred,
694
+ } ,
695
+ )
696
+ | ObligationCauseCode :: DerivedObligation (
697
+ DerivedObligationCause {
698
+ parent_code,
699
+ parent_trait_pred,
700
+ } ,
701
+ ) => {
702
+ peeled = true ;
703
+ code = & parent_code;
704
+ trait_pred = * parent_trait_pred;
705
+ }
706
+ _ => break ,
707
+ } ;
708
+ }
709
+ let def_id = trait_pred. def_id ( ) ;
710
+ // Mention *all* the `impl`s for the *top most* obligation, the
711
+ // user might have meant to use one of them, if any found. We skip
712
+ // auto-traits or fundamental traits that might not be exactly what
713
+ // the user might expect to be presented with. Instead this is
714
+ // useful for less general traits.
715
+ if peeled
716
+ && !self . tcx . trait_is_auto ( def_id)
717
+ && !self . tcx . lang_items ( ) . items ( ) . contains ( & Some ( def_id) )
718
+ {
719
+ let trait_ref = trait_pred. to_poly_trait_ref ( ) ;
720
+ let impl_candidates =
721
+ self . find_similar_impl_candidates ( trait_ref) ;
722
+ self . report_similar_impl_candidates (
723
+ impl_candidates,
724
+ trait_ref,
725
+ & mut err,
726
+ ) ;
727
+ }
728
+ }
662
729
}
663
730
664
731
// Changing mutability doesn't make a difference to whether we have
0 commit comments