blob: bfd4d4bcd5644397c68a83a160782047dbd4d6af [file] [log] [blame]
use std::borrow::Cow;
use super::{EXPLICIT_COUNTER_LOOP, IncrementVisitor, InitializeVisitor, make_iterator_snippet};
use clippy_utils::diagnostics::span_lint_and_then;
use clippy_utils::source::snippet_with_applicability;
use clippy_utils::sugg::{EMPTY, Sugg};
use clippy_utils::{get_enclosing_block, is_integer_const, is_integer_literal_untyped};
use rustc_ast::{Label, RangeLimits};
use rustc_errors::Applicability;
use rustc_hir::intravisit::{walk_block, walk_expr};
use rustc_hir::{Expr, Pat};
use rustc_lint::LateContext;
use rustc_middle::ty::{self, Ty, UintTy};
// To trigger the EXPLICIT_COUNTER_LOOP lint, a variable must be incremented exactly once in the
// loop body.
pub(super) fn check<'tcx>(
cx: &LateContext<'tcx>,
pat: &'tcx Pat<'_>,
arg: &'tcx Expr<'_>,
body: &'tcx Expr<'_>,
expr: &'tcx Expr<'_>,
label: Option<Label>,
) {
// Look for variables that are incremented once per loop iteration.
let mut increment_visitor = IncrementVisitor::new(cx);
walk_expr(&mut increment_visitor, body);
// For each candidate, check the parent block to see if
// it's initialized to zero at the start of the loop.
if let Some(block) = get_enclosing_block(cx, expr.hir_id) {
for id in increment_visitor.into_results() {
let mut initialize_visitor = InitializeVisitor::new(cx, expr, id);
walk_block(&mut initialize_visitor, block);
if let Some((name, ty, initializer)) = initialize_visitor.get_result() {
let is_zero = is_integer_const(cx, initializer, 0);
let mut applicability = Applicability::MaybeIncorrect;
let span = expr.span.with_hi(arg.span.hi());
let loop_label = label.map_or(String::new(), |l| format!("{}: ", l.ident.name));
span_lint_and_then(
cx,
EXPLICIT_COUNTER_LOOP,
span,
format!("the variable `{name}` is used as a loop counter"),
|diag| {
let pat_snippet = snippet_with_applicability(cx, pat.span, "item", &mut applicability);
let iter_snippet = make_iterator_snippet(cx, arg, &mut applicability);
let int_name = match ty.map(Ty::kind) {
Some(ty::Uint(UintTy::Usize)) | None => {
if is_zero {
diag.span_suggestion(
span,
"consider using",
format!(
"{loop_label}for ({name}, {pat_snippet}) in {iter_snippet}.enumerate()",
),
applicability,
);
return;
}
None
},
Some(ty::Int(int_ty)) => Some(int_ty.name_str()),
Some(ty::Uint(uint_ty)) => Some(uint_ty.name_str()),
_ => None,
}
.filter(|_| is_integer_literal_untyped(initializer));
let initializer = Sugg::hir_from_snippet(cx, initializer, |span| {
let snippet = snippet_with_applicability(cx, span, "..", &mut applicability);
if let Some(int_name) = int_name {
return Cow::Owned(format!("{snippet}_{int_name}"));
}
snippet
});
diag.span_suggestion(
span,
"consider using",
format!(
"{loop_label}for ({name}, {pat_snippet}) in ({}).zip({iter_snippet})",
initializer.range(&EMPTY, RangeLimits::HalfOpen)
),
applicability,
);
if is_zero && let Some(int_name) = int_name {
diag.note(format!(
"`{name}` is of type `{int_name}`, making it ineligible for `Iterator::enumerate`"
));
}
},
);
}
}
}
}