1
1
use clippy_config:: msrvs:: { self , Msrv } ;
2
2
use clippy_utils:: diagnostics:: span_lint_and_then;
3
- use clippy_utils:: mir:: PossibleBorrowerMap ;
3
+ use clippy_utils:: mir:: { enclosing_mir , expr_local , local_assignments , used_exactly_once , PossibleBorrowerMap } ;
4
4
use clippy_utils:: source:: snippet_with_context;
5
5
use clippy_utils:: ty:: { implements_trait, is_copy} ;
6
6
use clippy_utils:: { expr_use_ctxt, peel_n_hir_expr_refs, DefinedTy , ExprUseNode } ;
@@ -11,6 +11,7 @@ use rustc_hir::{Body, Expr, ExprKind, Mutability, Path, QPath};
11
11
use rustc_index:: bit_set:: BitSet ;
12
12
use rustc_infer:: infer:: TyCtxtInferExt ;
13
13
use rustc_lint:: { LateContext , LateLintPass } ;
14
+ use rustc_middle:: mir:: { Rvalue , StatementKind } ;
14
15
use rustc_middle:: ty:: {
15
16
self , ClauseKind , EarlyBinder , FnSig , GenericArg , GenericArgKind , ParamTy , ProjectionPredicate , Ty ,
16
17
} ;
@@ -107,6 +108,7 @@ impl<'tcx> LateLintPass<'tcx> for NeedlessBorrowsForGenericArgs<'tcx> {
107
108
}
108
109
&& let count = needless_borrow_count (
109
110
cx,
111
+ & mut self . possible_borrowers ,
110
112
fn_id,
111
113
cx. typeck_results ( ) . node_args ( hir_id) ,
112
114
i,
@@ -155,9 +157,14 @@ fn path_has_args(p: &QPath<'_>) -> bool {
155
157
/// The following constraints will be checked:
156
158
/// * The borrowed expression meets all the generic type's constraints.
157
159
/// * The generic type appears only once in the functions signature.
158
- /// * The borrowed value is Copy itself OR not a variable (created by a function call)
160
+ /// * The borrowed value is:
161
+ /// - `Copy` itself, or
162
+ /// - the only use of a mutable reference, or
163
+ /// - not a variable (created by a function call)
164
+ #[ expect( clippy:: too_many_arguments) ]
159
165
fn needless_borrow_count < ' tcx > (
160
166
cx : & LateContext < ' tcx > ,
167
+ possible_borrowers : & mut Vec < ( LocalDefId , PossibleBorrowerMap < ' tcx , ' tcx > ) > ,
161
168
fn_id : DefId ,
162
169
callee_args : ty:: GenericArgsRef < ' tcx > ,
163
170
arg_index : usize ,
@@ -232,9 +239,9 @@ fn needless_borrow_count<'tcx>(
232
239
233
240
let referent_ty = cx. typeck_results ( ) . expr_ty ( referent) ;
234
241
235
- if ( ! is_copy ( cx, referent_ty) && !referent_ty . is_ref ( ) )
236
- && let ExprKind :: AddrOf ( _ , _ , inner ) = reference. kind
237
- && ! matches ! ( inner . kind, ExprKind :: Call ( ..) | ExprKind :: MethodCall ( ..) )
242
+ if ! ( is_copy ( cx, referent_ty)
243
+ || referent_ty . is_ref ( ) && referent_used_exactly_once ( cx , possible_borrowers , reference)
244
+ || matches ! ( referent . kind, ExprKind :: Call ( ..) | ExprKind :: MethodCall ( ..) ) )
238
245
{
239
246
return false ;
240
247
}
@@ -337,6 +344,37 @@ fn is_mixed_projection_predicate<'tcx>(
337
344
}
338
345
}
339
346
347
+ fn referent_used_exactly_once < ' tcx > (
348
+ cx : & LateContext < ' tcx > ,
349
+ possible_borrowers : & mut Vec < ( LocalDefId , PossibleBorrowerMap < ' tcx , ' tcx > ) > ,
350
+ reference : & Expr < ' tcx > ,
351
+ ) -> bool {
352
+ if let Some ( mir) = enclosing_mir ( cx. tcx , reference. hir_id )
353
+ && let Some ( local) = expr_local ( cx. tcx , reference)
354
+ && let [ location] = * local_assignments ( mir, local) . as_slice ( )
355
+ && let block_data = & mir. basic_blocks [ location. block ]
356
+ && let Some ( statement) = block_data. statements . get ( location. statement_index )
357
+ && let StatementKind :: Assign ( box ( _, Rvalue :: Ref ( _, _, place) ) ) = statement. kind
358
+ && !place. is_indirect_first_projection ( )
359
+ {
360
+ let body_owner_local_def_id = cx. tcx . hir ( ) . enclosing_body_owner ( reference. hir_id ) ;
361
+ if possible_borrowers
362
+ . last ( )
363
+ . map_or ( true , |& ( local_def_id, _) | local_def_id != body_owner_local_def_id)
364
+ {
365
+ possible_borrowers. push ( ( body_owner_local_def_id, PossibleBorrowerMap :: new ( cx, mir) ) ) ;
366
+ }
367
+ let possible_borrower = & mut possible_borrowers. last_mut ( ) . unwrap ( ) . 1 ;
368
+ // If `only_borrowers` were used here, the `copyable_iterator::warn` test would fail. The reason is
369
+ // that `PossibleBorrowerVisitor::visit_terminator` considers `place.local` a possible borrower of
370
+ // itself. See the comment in that method for an explanation as to why.
371
+ possible_borrower. bounded_borrowers ( & [ local] , & [ local, place. local ] , place. local , location)
372
+ && used_exactly_once ( mir, place. local ) . unwrap_or ( false )
373
+ } else {
374
+ false
375
+ }
376
+ }
377
+
340
378
// Iteratively replaces `param_ty` with `new_ty` in `args`, and similarly for each resulting
341
379
// projected type that is a type parameter. Returns `false` if replacing the types would have an
342
380
// effect on the function signature beyond substituting `new_ty` for `param_ty`.
0 commit comments