@@ -13,7 +13,9 @@ use rustc_hir_analysis::astconv::AstConv;
13
13
use rustc_infer:: infer;
14
14
use rustc_infer:: traits:: { self , StatementAsExpression } ;
15
15
use rustc_middle:: lint:: in_external_macro;
16
- use rustc_middle:: ty:: { self , Binder , DefIdTree , IsSuggestable , Ty } ;
16
+ use rustc_middle:: ty:: {
17
+ self , suggest_constraining_type_params, Binder , DefIdTree , IsSuggestable , ToPredicate , Ty ,
18
+ } ;
17
19
use rustc_session:: errors:: ExprParenthesesNeeded ;
18
20
use rustc_span:: symbol:: sym;
19
21
use rustc_span:: Span ;
@@ -1276,15 +1278,13 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
1276
1278
&& !results. expr_adjustments ( callee_expr) . iter ( ) . any ( |adj| matches ! ( adj. kind, ty:: adjustment:: Adjust :: Deref ( ..) ) )
1277
1279
// Check that we're in fact trying to clone into the expected type
1278
1280
&& self . can_coerce ( * pointee_ty, expected_ty)
1281
+ && let trait_ref = ty:: Binder :: dummy ( self . tcx . mk_trait_ref ( clone_trait_did, [ expected_ty] ) )
1279
1282
// And the expected type doesn't implement `Clone`
1280
1283
&& !self . predicate_must_hold_considering_regions ( & traits:: Obligation :: new (
1281
1284
self . tcx ,
1282
1285
traits:: ObligationCause :: dummy ( ) ,
1283
1286
self . param_env ,
1284
- ty:: Binder :: dummy ( self . tcx . mk_trait_ref (
1285
- clone_trait_did,
1286
- [ expected_ty] ,
1287
- ) ) ,
1287
+ trait_ref,
1288
1288
) )
1289
1289
{
1290
1290
diag. span_note (
@@ -1293,6 +1293,19 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
1293
1293
"`{expected_ty}` does not implement `Clone`, so `{found_ty}` was cloned instead"
1294
1294
) ,
1295
1295
) ;
1296
+ let owner = self . tcx . hir ( ) . enclosing_body_owner ( expr. hir_id ) ;
1297
+ if let ty:: Param ( param) = expected_ty. kind ( )
1298
+ && let Some ( generics) = self . tcx . hir ( ) . get_generics ( owner)
1299
+ {
1300
+ suggest_constraining_type_params (
1301
+ self . tcx ,
1302
+ generics,
1303
+ diag,
1304
+ vec ! [ ( param. name. as_str( ) , "Clone" , Some ( clone_trait_did) ) ] . into_iter ( ) ,
1305
+ ) ;
1306
+ } else {
1307
+ self . suggest_derive ( diag, & [ ( trait_ref. to_predicate ( self . tcx ) , None , None ) ] ) ;
1308
+ }
1296
1309
}
1297
1310
}
1298
1311
0 commit comments