Skip to content

Commit db80969

Browse files
committed
Propose to exchange ranges only when it is safe to do so
To avoid false positives, the `range_plus_one` and `range_minus_one` lints will restrict themselves to situations where the iterator types can be easily switched from exclusive to inclusive or vice-versa. This includes situations where the range is used as an iterator, or is used for indexing. Also, when the range is used as a function or method call argument and the parameter type is generic over traits implemented by both kind of ranges, the lint will trigger. On the other hand, assignments of the range to variables, including automatically typed ones or wildcards, will no longer trigger the lint. However, the cases where such an assignment would benefit from the lint are probably rare.
1 parent b96fecf commit db80969

7 files changed

+217
-49
lines changed

clippy_lints/src/cognitive_complexity.rs

-1
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,6 @@ impl CognitiveComplexity {
9797
FnKind::ItemFn(ident, _, _) | FnKind::Method(ident, _) => ident.span,
9898
FnKind::Closure => {
9999
let header_span = body_span.with_hi(decl.output.span().lo());
100-
#[expect(clippy::range_plus_one)]
101100
if let Some(range) = header_span.map_range(cx, |src, range| {
102101
let mut idxs = src.get(range.clone())?.match_indices('|');
103102
Some(range.start + idxs.next()?.0..range.start + idxs.next()?.0 + 1)

clippy_lints/src/doc/mod.rs

-1
Original file line numberDiff line numberDiff line change
@@ -1159,7 +1159,6 @@ fn check_doc<'a, Events: Iterator<Item = (pulldown_cmark::Event<'a>, Range<usize
11591159
headers
11601160
}
11611161

1162-
#[expect(clippy::range_plus_one)] // inclusive ranges aren't the same type
11631162
fn looks_like_refdef(doc: &str, range: Range<usize>) -> Option<Range<usize>> {
11641163
if range.end < range.start {
11651164
return None;

clippy_lints/src/methods/manual_inspect.rs

-1
Original file line numberDiff line numberDiff line change
@@ -100,7 +100,6 @@ pub(crate) fn check(cx: &LateContext<'_>, expr: &Expr<'_>, arg: &Expr<'_>, name:
100100
match x {
101101
UseKind::Return(s) => edits.push((s.with_leading_whitespace(cx).with_ctxt(s.ctxt()), String::new())),
102102
UseKind::Borrowed(s) => {
103-
#[expect(clippy::range_plus_one)]
104103
let range = s.map_range(cx, |src, range| {
105104
let src = src.get(range.clone())?;
106105
let trimmed = src.trim_start_matches([' ', '\t', '\n', '\r', '(']);

clippy_lints/src/ranges.rs

+101-14
Original file line numberDiff line numberDiff line change
@@ -4,15 +4,17 @@ use clippy_utils::diagnostics::{span_lint, span_lint_and_sugg, span_lint_and_the
44
use clippy_utils::msrvs::{self, Msrv};
55
use clippy_utils::source::{SpanRangeExt, snippet, snippet_with_applicability};
66
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+
};
810
use rustc_ast::ast::RangeLimits;
911
use rustc_errors::Applicability;
10-
use rustc_hir::{BinOpKind, Expr, ExprKind, HirId};
12+
use rustc_hir::{BinOpKind, Expr, ExprKind, HirId, LangItem};
1113
use rustc_lint::{LateContext, LateLintPass};
12-
use rustc_middle::ty;
14+
use rustc_middle::ty::{self, ClauseKind, PredicatePolarity};
1315
use rustc_session::impl_lint_pass;
14-
use rustc_span::Span;
1516
use rustc_span::source_map::Spanned;
17+
use rustc_span::{Span, sym};
1618
use std::cmp::Ordering;
1719

1820
declare_clippy_lint! {
@@ -24,6 +26,12 @@ declare_clippy_lint! {
2426
/// The code is more readable with an inclusive range
2527
/// like `x..=y`.
2628
///
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+
///
2735
/// ### Known problems
2836
/// Will add unnecessary pair of parentheses when the
2937
/// expression is not wrapped in a pair but starts with an opening parenthesis
@@ -34,11 +42,6 @@ declare_clippy_lint! {
3442
/// exclusive ranges, because they essentially add an extra branch that
3543
/// LLVM may fail to hoist out of the loop.
3644
///
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-
///
4245
/// ### Example
4346
/// ```no_run
4447
/// # let x = 0;
@@ -71,11 +74,11 @@ declare_clippy_lint! {
7174
/// The code is more readable with an exclusive range
7275
/// like `x..y`.
7376
///
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.
7982
///
8083
/// ### Example
8184
/// ```no_run
@@ -344,6 +347,88 @@ fn check_range_bounds<'a, 'tcx>(cx: &'a LateContext<'tcx>, ex: &'a Expr<'_>) ->
344347
None
345348
}
346349

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+
347432
// exclusive range plus one: `x..(y+1)`
348433
fn check_exclusive_range_plus_one(cx: &LateContext<'_>, expr: &Expr<'_>) {
349434
if expr.span.can_be_used_for_suggestions()
@@ -353,6 +438,7 @@ fn check_exclusive_range_plus_one(cx: &LateContext<'_>, expr: &Expr<'_>) {
353438
limits: RangeLimits::HalfOpen,
354439
}) = higher::Range::hir(expr)
355440
&& let Some(y) = y_plus_one(cx, end)
441+
&& can_switch_ranges(cx, expr)
356442
{
357443
let span = expr.span;
358444
span_lint_and_then(
@@ -391,6 +477,7 @@ fn check_inclusive_range_minus_one(cx: &LateContext<'_>, expr: &Expr<'_>) {
391477
limits: RangeLimits::Closed,
392478
}) = higher::Range::hir(expr)
393479
&& let Some(y) = y_minus_one(cx, end)
480+
&& can_switch_ranges(cx, expr)
394481
{
395482
span_lint_and_then(
396483
cx,

tests/ui/range_plus_minus_one.fixed

+41-8
Original file line numberDiff line numberDiff line change
@@ -45,15 +45,13 @@ fn main() {
4545
//~^ range_plus_one
4646
for _ in 0..=(1 + f()) {}
4747

48+
// Those are not linted, as in the general case we cannot be sure that the exact type won't be
49+
// important.
4850
let _ = ..11 - 1;
49-
let _ = ..11;
50-
//~^ range_minus_one
51-
let _ = ..11;
52-
//~^ range_minus_one
53-
let _ = (1..=11);
54-
//~^ range_plus_one
55-
let _ = ((f() + 1)..=f());
56-
//~^ range_plus_one
51+
let _ = ..=11 - 1;
52+
let _ = ..=(11 - 1);
53+
let _ = (1..11 + 1);
54+
let _ = (f() + 1)..(f() + 1);
5755

5856
const ONE: usize = 1;
5957
// integer consts are linted, too
@@ -65,4 +63,39 @@ fn main() {
6563

6664
macro_plus_one!(5);
6765
macro_minus_one!(5);
66+
67+
// As an instance of `Iterator`
68+
(1..=10).for_each(|_| {});
69+
//~^ range_plus_one
70+
71+
// As an instance of `IntoIterator`
72+
#[allow(clippy::useless_conversion)]
73+
(1..=10).into_iter().for_each(|_| {});
74+
//~^ range_plus_one
75+
76+
// As an instance of `RangeBounds`
77+
{
78+
use std::ops::RangeBounds;
79+
let _ = (1..=10).start_bound();
80+
//~^ range_plus_one
81+
}
82+
83+
// As a `SliceIndex`
84+
let a = [10, 20, 30];
85+
let _ = &a[1..=1];
86+
//~^ range_plus_one
87+
88+
vec.drain(2..=3);
89+
//~^ range_plus_one
90+
91+
take_arg(10..=20);
92+
//~^ range_plus_one
93+
take_arg(10..20);
94+
//~^ range_minus_one
95+
96+
// Do not lint, as changing the same type is used for both parameters
97+
take_args(10..20 + 1, 10..21);
6898
}
99+
100+
fn take_arg<T: Iterator<Item = u32>>(_: T) {}
101+
fn take_args<T: Iterator<Item = u32>>(_: T, _: T) {}

tests/ui/range_plus_minus_one.rs

+37-4
Original file line numberDiff line numberDiff line change
@@ -45,15 +45,13 @@ fn main() {
4545
//~^ range_plus_one
4646
for _ in 0..=(1 + f()) {}
4747

48+
// Those are not linted, as in the general case we cannot be sure that the exact type won't be
49+
// important.
4850
let _ = ..11 - 1;
4951
let _ = ..=11 - 1;
50-
//~^ range_minus_one
5152
let _ = ..=(11 - 1);
52-
//~^ range_minus_one
5353
let _ = (1..11 + 1);
54-
//~^ range_plus_one
5554
let _ = (f() + 1)..(f() + 1);
56-
//~^ range_plus_one
5755

5856
const ONE: usize = 1;
5957
// integer consts are linted, too
@@ -65,4 +63,39 @@ fn main() {
6563

6664
macro_plus_one!(5);
6765
macro_minus_one!(5);
66+
67+
// As an instance of `Iterator`
68+
(1..10 + 1).for_each(|_| {});
69+
//~^ range_plus_one
70+
71+
// As an instance of `IntoIterator`
72+
#[allow(clippy::useless_conversion)]
73+
(1..10 + 1).into_iter().for_each(|_| {});
74+
//~^ range_plus_one
75+
76+
// As an instance of `RangeBounds`
77+
{
78+
use std::ops::RangeBounds;
79+
let _ = (1..10 + 1).start_bound();
80+
//~^ range_plus_one
81+
}
82+
83+
// As a `SliceIndex`
84+
let a = [10, 20, 30];
85+
let _ = &a[1..1 + 1];
86+
//~^ range_plus_one
87+
88+
vec.drain(2..3 + 1);
89+
//~^ range_plus_one
90+
91+
take_arg(10..20 + 1);
92+
//~^ range_plus_one
93+
take_arg(10..=20 - 1);
94+
//~^ range_minus_one
95+
96+
// Do not lint, as changing the same type is used for both parameters
97+
take_args(10..20 + 1, 10..21);
6898
}
99+
100+
fn take_arg<T: Iterator<Item = u32>>(_: T) {}
101+
fn take_args<T: Iterator<Item = u32>>(_: T, _: T) {}

tests/ui/range_plus_minus_one.stderr

+38-20
Original file line numberDiff line numberDiff line change
@@ -25,38 +25,56 @@ error: an inclusive range would be more readable
2525
LL | for _ in 0..(1 + f()) {}
2626
| ^^^^^^^^^^^^ help: use: `0..=f()`
2727

28-
error: an exclusive range would be more readable
29-
--> tests/ui/range_plus_minus_one.rs:49:13
28+
error: an inclusive range would be more readable
29+
--> tests/ui/range_plus_minus_one.rs:58:14
3030
|
31-
LL | let _ = ..=11 - 1;
32-
| ^^^^^^^^^ help: use: `..11`
31+
LL | for _ in 1..ONE + ONE {}
32+
| ^^^^^^^^^^^^ help: use: `1..=ONE`
33+
34+
error: an inclusive range would be more readable
35+
--> tests/ui/range_plus_minus_one.rs:68:5
3336
|
34-
= note: `-D clippy::range-minus-one` implied by `-D warnings`
35-
= help: to override `-D warnings` add `#[allow(clippy::range_minus_one)]`
37+
LL | (1..10 + 1).for_each(|_| {});
38+
| ^^^^^^^^^^^ help: use: `(1..=10)`
3639

37-
error: an exclusive range would be more readable
38-
--> tests/ui/range_plus_minus_one.rs:51:13
40+
error: an inclusive range would be more readable
41+
--> tests/ui/range_plus_minus_one.rs:73:5
3942
|
40-
LL | let _ = ..=(11 - 1);
41-
| ^^^^^^^^^^^ help: use: `..11`
43+
LL | (1..10 + 1).into_iter().for_each(|_| {});
44+
| ^^^^^^^^^^^ help: use: `(1..=10)`
4245

4346
error: an inclusive range would be more readable
44-
--> tests/ui/range_plus_minus_one.rs:53:13
47+
--> tests/ui/range_plus_minus_one.rs:79:17
4548
|
46-
LL | let _ = (1..11 + 1);
47-
| ^^^^^^^^^^^ help: use: `(1..=11)`
49+
LL | let _ = (1..10 + 1).start_bound();
50+
| ^^^^^^^^^^^ help: use: `(1..=10)`
4851

4952
error: an inclusive range would be more readable
50-
--> tests/ui/range_plus_minus_one.rs:55:13
53+
--> tests/ui/range_plus_minus_one.rs:85:16
5154
|
52-
LL | let _ = (f() + 1)..(f() + 1);
53-
| ^^^^^^^^^^^^^^^^^^^^ help: use: `((f() + 1)..=f())`
55+
LL | let _ = &a[1..1 + 1];
56+
| ^^^^^^^^ help: use: `1..=1`
5457

5558
error: an inclusive range would be more readable
56-
--> tests/ui/range_plus_minus_one.rs:60:14
59+
--> tests/ui/range_plus_minus_one.rs:88:15
5760
|
58-
LL | for _ in 1..ONE + ONE {}
59-
| ^^^^^^^^^^^^ help: use: `1..=ONE`
61+
LL | vec.drain(2..3 + 1);
62+
| ^^^^^^^^ help: use: `2..=3`
63+
64+
error: an inclusive range would be more readable
65+
--> tests/ui/range_plus_minus_one.rs:91:14
66+
|
67+
LL | take_arg(10..20 + 1);
68+
| ^^^^^^^^^^ help: use: `10..=20`
69+
70+
error: an exclusive range would be more readable
71+
--> tests/ui/range_plus_minus_one.rs:93:14
72+
|
73+
LL | take_arg(10..=20 - 1);
74+
| ^^^^^^^^^^^ help: use: `10..20`
75+
|
76+
= note: `-D clippy::range-minus-one` implied by `-D warnings`
77+
= help: to override `-D warnings` add `#[allow(clippy::range_minus_one)]`
6078

61-
error: aborting due to 9 previous errors
79+
error: aborting due to 12 previous errors
6280

0 commit comments

Comments
 (0)