Skip to content

Commit 4de845e

Browse files
committed
[missing_asserts_for_indexing]: accept len equality checks
1 parent edb720b commit 4de845e

4 files changed

+99
-6
lines changed

clippy_lints/src/missing_asserts_for_indexing.rs

+16-5
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,7 @@ declare_clippy_lint! {
5252
/// Use instead:
5353
/// ```no_run
5454
/// fn sum(v: &[u8]) -> u8 {
55-
/// assert!(v.len() > 4);
55+
/// assert!(v.len() > 3);
5656
/// // no bounds checks
5757
/// v[0] + v[1] + v[2] + v[3]
5858
/// }
@@ -87,6 +87,9 @@ enum LengthComparison {
8787
LengthLessThanOrEqualInt,
8888
/// `5 <= v.len()`
8989
IntLessThanOrEqualLength,
90+
/// `5 == v.len()`
91+
/// `v.len() == 5`
92+
LengthEqualInt,
9093
}
9194

9295
/// Extracts parts out of a length comparison expression.
@@ -114,6 +117,8 @@ fn len_comparison<'hir>(
114117
(Rel::Lt, _, int_lit_pat!(right)) => Some((LengthComparison::LengthLessThanInt, *right as usize, left)),
115118
(Rel::Le, int_lit_pat!(left), _) => Some((LengthComparison::IntLessThanOrEqualLength, *left as usize, right)),
116119
(Rel::Le, _, int_lit_pat!(right)) => Some((LengthComparison::LengthLessThanOrEqualInt, *right as usize, left)),
120+
(Rel::Eq, int_lit_pat!(left), _) => Some((LengthComparison::LengthEqualInt, *left as usize, right)),
121+
(Rel::Eq, _, int_lit_pat!(right)) => Some((LengthComparison::LengthEqualInt, *right as usize, left)),
117122
_ => None,
118123
}
119124
}
@@ -316,11 +321,11 @@ fn report_indexes(cx: &LateContext<'_>, map: &UnhashMap<u64, Vec<IndexEntry<'_>>
316321
continue;
317322
};
318323

319-
match entry {
324+
match *entry {
320325
IndexEntry::AssertWithIndex {
321326
highest_index,
322327
asserted_len,
323-
indexes,
328+
ref indexes,
324329
comparison,
325330
assert_span,
326331
slice,
@@ -343,6 +348,12 @@ fn report_indexes(cx: &LateContext<'_>, map: &UnhashMap<u64, Vec<IndexEntry<'_>>
343348
"assert!({}.len() > {highest_index})",
344349
snippet(cx, slice.span, "..")
345350
)),
351+
// `highest_index` here is rather a length, so we need to add 1 to it
352+
LengthComparison::LengthEqualInt if asserted_len < highest_index + 1 => Some(format!(
353+
"assert!({}.len() == {})",
354+
snippet(cx, slice.span, ".."),
355+
highest_index + 1
356+
)),
346357
_ => None,
347358
};
348359

@@ -354,7 +365,7 @@ fn report_indexes(cx: &LateContext<'_>, map: &UnhashMap<u64, Vec<IndexEntry<'_>>
354365
indexes,
355366
|diag| {
356367
diag.span_suggestion(
357-
*assert_span,
368+
assert_span,
358369
"provide the highest index that is indexed with",
359370
sugg,
360371
Applicability::MachineApplicable,
@@ -364,7 +375,7 @@ fn report_indexes(cx: &LateContext<'_>, map: &UnhashMap<u64, Vec<IndexEntry<'_>>
364375
}
365376
},
366377
IndexEntry::IndexWithoutAssert {
367-
indexes,
378+
ref indexes,
368379
highest_index,
369380
slice,
370381
} if indexes.len() > 1 => {

tests/ui/missing_asserts_for_indexing.fixed

+15
Original file line numberDiff line numberDiff line change
@@ -118,4 +118,19 @@ fn index_different_slice_in_same_expr(v1: &[u8], v2: &[u8]) {
118118
let _ = v1[0] + v2[1];
119119
}
120120

121+
fn issue11835(v1: &[u8], v2: &[u8], v3: &[u8], v4: &[u8]) {
122+
assert!(v1.len() == 3);
123+
assert!(v2.len() == 4);
124+
assert!(v3.len() == 3);
125+
assert!(4 == v4.len());
126+
127+
let _ = v1[0] + v1[1] + v1[2];
128+
//~^ ERROR: indexing into a slice multiple times with an `assert` that does not cover the
129+
let _ = v2[0] + v2[1] + v2[2];
130+
131+
let _ = v3[0] + v3[1] + v3[2];
132+
//~^ ERROR: indexing into a slice multiple times with an `assert` that does not cover the
133+
let _ = v4[0] + v4[1] + v4[2];
134+
}
135+
121136
fn main() {}

tests/ui/missing_asserts_for_indexing.rs

+15
Original file line numberDiff line numberDiff line change
@@ -118,4 +118,19 @@ fn index_different_slice_in_same_expr(v1: &[u8], v2: &[u8]) {
118118
let _ = v1[0] + v2[1];
119119
}
120120

121+
fn issue11835(v1: &[u8], v2: &[u8], v3: &[u8], v4: &[u8]) {
122+
assert!(v1.len() == 2);
123+
assert!(v2.len() == 4);
124+
assert!(2 == v3.len());
125+
assert!(4 == v4.len());
126+
127+
let _ = v1[0] + v1[1] + v1[2];
128+
//~^ ERROR: indexing into a slice multiple times with an `assert` that does not cover the
129+
let _ = v2[0] + v2[1] + v2[2];
130+
131+
let _ = v3[0] + v3[1] + v3[2];
132+
//~^ ERROR: indexing into a slice multiple times with an `assert` that does not cover the
133+
let _ = v4[0] + v4[1] + v4[2];
134+
}
135+
121136
fn main() {}

tests/ui/missing_asserts_for_indexing.stderr

+53-1
Original file line numberDiff line numberDiff line change
@@ -249,5 +249,57 @@ LL | let _ = v1[0] + v1[12];
249249
| ^^^^^^
250250
= note: asserting the length before indexing will elide bounds checks
251251

252-
error: aborting due to 9 previous errors
252+
error: indexing into a slice multiple times with an `assert` that does not cover the highest index
253+
--> $DIR/missing_asserts_for_indexing.rs:127:13
254+
|
255+
LL | assert!(v1.len() == 2);
256+
| ---------------------- help: provide the highest index that is indexed with: `assert!(v1.len() == 3)`
257+
...
258+
LL | let _ = v1[0] + v1[1] + v1[2];
259+
| ^^^^^^^^^^^^^^^^^^^^^
260+
|
261+
note: slice indexed here
262+
--> $DIR/missing_asserts_for_indexing.rs:127:13
263+
|
264+
LL | let _ = v1[0] + v1[1] + v1[2];
265+
| ^^^^^
266+
note: slice indexed here
267+
--> $DIR/missing_asserts_for_indexing.rs:127:21
268+
|
269+
LL | let _ = v1[0] + v1[1] + v1[2];
270+
| ^^^^^
271+
note: slice indexed here
272+
--> $DIR/missing_asserts_for_indexing.rs:127:29
273+
|
274+
LL | let _ = v1[0] + v1[1] + v1[2];
275+
| ^^^^^
276+
= note: asserting the length before indexing will elide bounds checks
277+
278+
error: indexing into a slice multiple times with an `assert` that does not cover the highest index
279+
--> $DIR/missing_asserts_for_indexing.rs:131:13
280+
|
281+
LL | assert!(2 == v3.len());
282+
| ---------------------- help: provide the highest index that is indexed with: `assert!(v3.len() == 3)`
283+
...
284+
LL | let _ = v3[0] + v3[1] + v3[2];
285+
| ^^^^^^^^^^^^^^^^^^^^^
286+
|
287+
note: slice indexed here
288+
--> $DIR/missing_asserts_for_indexing.rs:131:13
289+
|
290+
LL | let _ = v3[0] + v3[1] + v3[2];
291+
| ^^^^^
292+
note: slice indexed here
293+
--> $DIR/missing_asserts_for_indexing.rs:131:21
294+
|
295+
LL | let _ = v3[0] + v3[1] + v3[2];
296+
| ^^^^^
297+
note: slice indexed here
298+
--> $DIR/missing_asserts_for_indexing.rs:131:29
299+
|
300+
LL | let _ = v3[0] + v3[1] + v3[2];
301+
| ^^^^^
302+
= note: asserting the length before indexing will elide bounds checks
303+
304+
error: aborting due to 11 previous errors
253305

0 commit comments

Comments
 (0)