@@ -686,17 +686,36 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
686
686
return false ;
687
687
}
688
688
689
+ // Blacklist traits for which it would be nonsensical to suggest borrowing.
690
+ // For instance, immutable references are always Copy, so suggesting to
691
+ // borrow would always succeed, but it's probably not what the user wanted.
692
+ let blacklist: Vec < _ > =
693
+ [ LangItem :: Copy , LangItem :: Clone , LangItem :: Unpin , LangItem :: Sized , LangItem :: Send ]
694
+ . iter ( )
695
+ . filter_map ( |lang_item| self . tcx . lang_items ( ) . require ( * lang_item) . ok ( ) )
696
+ . collect ( ) ;
697
+
689
698
let span = obligation. cause . span ;
690
699
let param_env = obligation. param_env ;
691
700
let trait_ref = trait_ref. skip_binder ( ) ;
692
701
693
- if let ObligationCauseCode :: ImplDerivedObligation ( obligation) = & obligation. cause . code {
694
- // Try to apply the original trait binding obligation by borrowing.
695
- let self_ty = trait_ref. self_ty ( ) ;
696
- let found = self_ty. to_string ( ) ;
697
- let new_self_ty = self . tcx . mk_imm_ref ( self . tcx . lifetimes . re_static , self_ty) ;
698
- let substs = self . tcx . mk_substs_trait ( new_self_ty, & [ ] ) ;
699
- let new_trait_ref = ty:: TraitRef :: new ( obligation. parent_trait_ref . def_id ( ) , substs) ;
702
+ let found_ty = trait_ref. self_ty ( ) ;
703
+ let found_ty_str = found_ty. to_string ( ) ;
704
+ let imm_borrowed_found_ty = self . tcx . mk_imm_ref ( self . tcx . lifetimes . re_static , found_ty) ;
705
+ let imm_substs = self . tcx . mk_substs_trait ( imm_borrowed_found_ty, & [ ] ) ;
706
+ let mut_borrowed_found_ty = self . tcx . mk_mut_ref ( self . tcx . lifetimes . re_static , found_ty) ;
707
+ let mut_substs = self . tcx . mk_substs_trait ( mut_borrowed_found_ty, & [ ] ) ;
708
+
709
+ // Try to apply the original trait binding obligation by borrowing.
710
+ let mut try_borrowing = |new_trait_ref : ty:: TraitRef < ' tcx > ,
711
+ expected_trait_ref : ty:: TraitRef < ' tcx > ,
712
+ mtbl : bool ,
713
+ blacklist : & [ DefId ] |
714
+ -> bool {
715
+ if blacklist. contains ( & expected_trait_ref. def_id ) {
716
+ return false ;
717
+ }
718
+
700
719
let new_obligation = Obligation :: new (
701
720
ObligationCause :: dummy ( ) ,
702
721
param_env,
@@ -713,8 +732,8 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
713
732
714
733
let msg = format ! (
715
734
"the trait bound `{}: {}` is not satisfied" ,
716
- found ,
717
- obligation . parent_trait_ref . skip_binder ( ) . print_only_trait_path( ) ,
735
+ found_ty_str ,
736
+ expected_trait_ref . print_only_trait_path( ) ,
718
737
) ;
719
738
if has_custom_message {
720
739
err. note ( & msg) ;
@@ -730,7 +749,7 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
730
749
span,
731
750
& format ! (
732
751
"expected an implementor of trait `{}`" ,
733
- obligation . parent_trait_ref . skip_binder ( ) . print_only_trait_path( ) ,
752
+ expected_trait_ref . print_only_trait_path( ) ,
734
753
) ,
735
754
) ;
736
755
@@ -745,16 +764,52 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
745
764
746
765
err. span_suggestion (
747
766
span,
748
- "consider borrowing here" ,
749
- format ! ( "&{}" , snippet) ,
767
+ & format ! (
768
+ "consider{} borrowing here" ,
769
+ if mtbl { " mutably" } else { "" }
770
+ ) ,
771
+ format ! ( "&{}{}" , if mtbl { "mut " } else { "" } , snippet) ,
750
772
Applicability :: MaybeIncorrect ,
751
773
) ;
752
774
}
753
775
return true ;
754
776
}
755
777
}
778
+ return false ;
779
+ } ;
780
+
781
+ if let ObligationCauseCode :: ImplDerivedObligation ( obligation) = & obligation. cause . code {
782
+ let expected_trait_ref = obligation. parent_trait_ref . skip_binder ( ) ;
783
+ let new_imm_trait_ref =
784
+ ty:: TraitRef :: new ( obligation. parent_trait_ref . def_id ( ) , imm_substs) ;
785
+ let new_mut_trait_ref =
786
+ ty:: TraitRef :: new ( obligation. parent_trait_ref . def_id ( ) , mut_substs) ;
787
+ if try_borrowing ( new_imm_trait_ref, expected_trait_ref, false , & [ ] ) {
788
+ return true ;
789
+ } else {
790
+ return try_borrowing ( new_mut_trait_ref, expected_trait_ref, true , & [ ] ) ;
791
+ }
792
+ } else if let ObligationCauseCode :: BindingObligation ( _, _)
793
+ | ObligationCauseCode :: ItemObligation ( _) = & obligation. cause . code
794
+ {
795
+ if try_borrowing (
796
+ ty:: TraitRef :: new ( trait_ref. def_id , imm_substs) ,
797
+ trait_ref,
798
+ false ,
799
+ & blacklist[ ..] ,
800
+ ) {
801
+ return true ;
802
+ } else {
803
+ return try_borrowing (
804
+ ty:: TraitRef :: new ( trait_ref. def_id , mut_substs) ,
805
+ trait_ref,
806
+ true ,
807
+ & blacklist[ ..] ,
808
+ ) ;
809
+ }
810
+ } else {
811
+ false
756
812
}
757
- false
758
813
}
759
814
760
815
/// Whenever references are used by mistake, like `for (i, e) in &vec.iter().enumerate()`,
0 commit comments