@@ -4,15 +4,17 @@ use clippy_utils::diagnostics::{span_lint, span_lint_and_sugg, span_lint_and_the
4
4
use clippy_utils:: msrvs:: { self , Msrv } ;
5
5
use clippy_utils:: source:: { SpanRangeExt , snippet, snippet_with_applicability} ;
6
6
use clippy_utils:: sugg:: Sugg ;
7
- use clippy_utils:: { get_parent_expr, higher, is_in_const_context, is_integer_const, path_to_local} ;
7
+ use clippy_utils:: {
8
+ fn_def_id, get_parent_expr, higher, is_in_const_context, is_integer_const, is_path_lang_item, path_to_local,
9
+ } ;
8
10
use rustc_ast:: ast:: RangeLimits ;
9
11
use rustc_errors:: Applicability ;
10
- use rustc_hir:: { BinOpKind , Expr , ExprKind , HirId } ;
12
+ use rustc_hir:: { BinOpKind , Expr , ExprKind , HirId , LangItem } ;
11
13
use rustc_lint:: { LateContext , LateLintPass } ;
12
- use rustc_middle:: ty;
14
+ use rustc_middle:: ty:: { self , ClauseKind , PredicatePolarity } ;
13
15
use rustc_session:: impl_lint_pass;
14
- use rustc_span:: Span ;
15
16
use rustc_span:: source_map:: Spanned ;
17
+ use rustc_span:: { Span , sym} ;
16
18
use std:: cmp:: Ordering ;
17
19
18
20
declare_clippy_lint ! {
@@ -24,6 +26,12 @@ declare_clippy_lint! {
24
26
/// The code is more readable with an inclusive range
25
27
/// like `x..=y`.
26
28
///
29
+ /// ### Limitations
30
+ /// The lint is conservative and will trigger only when switching
31
+ /// from an exclusive to an inclusive range is provably safe from
32
+ /// a typing point of view. This corresponds to situations where
33
+ /// the range is used as an iterator, or for indexing.
34
+ ///
27
35
/// ### Known problems
28
36
/// Will add unnecessary pair of parentheses when the
29
37
/// expression is not wrapped in a pair but starts with an opening parenthesis
@@ -34,11 +42,6 @@ declare_clippy_lint! {
34
42
/// exclusive ranges, because they essentially add an extra branch that
35
43
/// LLVM may fail to hoist out of the loop.
36
44
///
37
- /// This will cause a warning that cannot be fixed if the consumer of the
38
- /// range only accepts a specific range type, instead of the generic
39
- /// `RangeBounds` trait
40
- /// ([#3307](https://github.com/rust-lang/rust-clippy/issues/3307)).
41
- ///
42
45
/// ### Example
43
46
/// ```no_run
44
47
/// # let x = 0;
@@ -71,11 +74,11 @@ declare_clippy_lint! {
71
74
/// The code is more readable with an exclusive range
72
75
/// like `x..y`.
73
76
///
74
- /// ### Known problems
75
- /// This will cause a warning that cannot be fixed if
76
- /// the consumer of the range only accepts a specific range type, instead of
77
- /// the generic `RangeBounds` trait
78
- /// ([#3307](https://github.com/rust-lang/rust-clippy/issues/3307)) .
77
+ /// ### Limitations
78
+ /// The lint is conservative and will trigger only when switching
79
+ /// from an inclusive to an exclusive range is provably safe from
80
+ /// a typing point of view. This corresponds to situations where
81
+ /// the range is used as an iterator, or for indexing .
79
82
///
80
83
/// ### Example
81
84
/// ```no_run
@@ -344,6 +347,88 @@ fn check_range_bounds<'a, 'tcx>(cx: &'a LateContext<'tcx>, ex: &'a Expr<'_>) ->
344
347
None
345
348
}
346
349
350
+ /// Check whether `expr` could switch range types without breaking the typing requirements. This is
351
+ /// the case when `expr` is used as an iterator for example, or as a slice or `&str` index.
352
+ fn can_switch_ranges ( cx : & LateContext < ' _ > , expr : & Expr < ' _ > ) -> bool {
353
+ let Some ( parent_expr) = get_parent_expr ( cx, expr) else {
354
+ return false ;
355
+ } ;
356
+
357
+ // Check if `expr` is the argument of a compiler-generated `IntoIter::into_iter(expr)`
358
+ if let ExprKind :: Call ( func, [ arg] ) = parent_expr. kind
359
+ && arg. hir_id == expr. hir_id
360
+ && is_path_lang_item ( cx, func, LangItem :: IntoIterIntoIter )
361
+ {
362
+ return true ;
363
+ }
364
+
365
+ // Check if `expr` is used as the receiver of a method of the `Iterator`, `IntoIterator`,
366
+ // or `RangeBounds` traits.
367
+ if let ExprKind :: MethodCall ( _, receiver, _, _) = parent_expr. kind
368
+ && receiver. hir_id == expr. hir_id
369
+ && let Some ( method_did) = cx. typeck_results ( ) . type_dependent_def_id ( parent_expr. hir_id )
370
+ && let Some ( trait_did) = cx. tcx . trait_of_item ( method_did)
371
+ && matches ! (
372
+ cx. tcx. get_diagnostic_name( trait_did) ,
373
+ Some ( sym:: Iterator | sym:: IntoIterator | sym:: RangeBounds )
374
+ )
375
+ {
376
+ return true ;
377
+ }
378
+
379
+ // Check if `expr` is an argument of a call which requires an `Iterator`, `IntoIterator`,
380
+ // or `RangeBounds` trait.
381
+ if let ExprKind :: Call ( _, args) | ExprKind :: MethodCall ( _, _, args, _) = parent_expr. kind
382
+ && let Some ( id) = fn_def_id ( cx, parent_expr)
383
+ && let Some ( arg_idx) = args. iter ( ) . position ( |e| e. hir_id == expr. hir_id )
384
+ {
385
+ let input_idx = if matches ! ( parent_expr. kind, ExprKind :: MethodCall ( ..) ) {
386
+ arg_idx + 1
387
+ } else {
388
+ arg_idx
389
+ } ;
390
+ let inputs = cx
391
+ . tcx
392
+ . liberate_late_bound_regions ( id, cx. tcx . fn_sig ( id) . instantiate_identity ( ) )
393
+ . inputs ( ) ;
394
+ let expr_ty = inputs[ input_idx] ;
395
+ // Check that the `expr` type is present only one, otherwise modifying just one of them it might be
396
+ // risky if they are referenced using the same generic type for example.
397
+ if inputs. iter ( ) . enumerate ( ) . all ( |( n, ty) | n == input_idx || * ty != expr_ty)
398
+ // Look for a clause requiring `Iterator`, `IntoIterator`, or `RangeBounds`, and resolving to `expr_type`.
399
+ && cx
400
+ . tcx
401
+ . param_env ( id)
402
+ . caller_bounds ( )
403
+ . into_iter ( )
404
+ . any ( |p| {
405
+ if let ClauseKind :: Trait ( t) = p. kind ( ) . skip_binder ( )
406
+ && t. polarity == PredicatePolarity :: Positive
407
+ && matches ! (
408
+ cx. tcx. get_diagnostic_name( t. trait_ref. def_id) ,
409
+ Some ( sym:: Iterator | sym:: IntoIterator | sym:: RangeBounds )
410
+ )
411
+ {
412
+ t. self_ty ( ) == expr_ty
413
+ } else {
414
+ false
415
+ }
416
+ } )
417
+ {
418
+ return true ;
419
+ }
420
+ }
421
+
422
+ // Check if `expr` is used for indexing.
423
+ if let ExprKind :: Index ( _, index, _) = parent_expr. kind
424
+ && index. hir_id == expr. hir_id
425
+ {
426
+ return true ;
427
+ }
428
+
429
+ false
430
+ }
431
+
347
432
// exclusive range plus one: `x..(y+1)`
348
433
fn check_exclusive_range_plus_one ( cx : & LateContext < ' _ > , expr : & Expr < ' _ > ) {
349
434
if expr. span . can_be_used_for_suggestions ( )
@@ -353,6 +438,7 @@ fn check_exclusive_range_plus_one(cx: &LateContext<'_>, expr: &Expr<'_>) {
353
438
limits : RangeLimits :: HalfOpen ,
354
439
} ) = higher:: Range :: hir ( expr)
355
440
&& let Some ( y) = y_plus_one ( cx, end)
441
+ && can_switch_ranges ( cx, expr)
356
442
{
357
443
let span = expr. span ;
358
444
span_lint_and_then (
@@ -391,6 +477,7 @@ fn check_inclusive_range_minus_one(cx: &LateContext<'_>, expr: &Expr<'_>) {
391
477
limits : RangeLimits :: Closed ,
392
478
} ) = higher:: Range :: hir ( expr)
393
479
&& let Some ( y) = y_minus_one ( cx, end)
480
+ && can_switch_ranges ( cx, expr)
394
481
{
395
482
span_lint_and_then (
396
483
cx,
0 commit comments