2021-10-27 14:48:06 +00:00
|
|
|
use super::SINGLE_ELEMENT_LOOP;
|
2021-03-25 18:29:11 +00:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2022-01-06 15:31:38 +00:00
|
|
|
use clippy_utils::source::{indent_of, snippet_with_applicability};
|
2021-03-12 14:30:50 +00:00
|
|
|
use if_chain::if_chain;
|
2022-04-01 06:04:19 +00:00
|
|
|
use rustc_ast::util::parser::PREC_PREFIX;
|
|
|
|
use rustc_ast::Mutability;
|
2021-03-12 14:30:50 +00:00
|
|
|
use rustc_errors::Applicability;
|
2022-04-01 06:04:19 +00:00
|
|
|
use rustc_hir::{is_range_literal, BorrowKind, Expr, ExprKind, Pat};
|
2021-03-12 14:30:50 +00:00
|
|
|
use rustc_lint::LateContext;
|
2022-04-01 06:04:19 +00:00
|
|
|
use rustc_span::edition::Edition;
|
2021-03-12 14:30:50 +00:00
|
|
|
|
|
|
|
pub(super) fn check<'tcx>(
|
|
|
|
cx: &LateContext<'tcx>,
|
|
|
|
pat: &'tcx Pat<'_>,
|
|
|
|
arg: &'tcx Expr<'_>,
|
|
|
|
body: &'tcx Expr<'_>,
|
|
|
|
expr: &'tcx Expr<'_>,
|
|
|
|
) {
|
2022-04-01 06:04:19 +00:00
|
|
|
let (arg_expression, prefix) = match arg.kind {
|
|
|
|
ExprKind::AddrOf(
|
|
|
|
BorrowKind::Ref,
|
|
|
|
Mutability::Not,
|
|
|
|
Expr {
|
|
|
|
kind: ExprKind::Array([arg]),
|
|
|
|
..
|
|
|
|
},
|
|
|
|
) => (arg, "&"),
|
|
|
|
ExprKind::AddrOf(
|
|
|
|
BorrowKind::Ref,
|
|
|
|
Mutability::Mut,
|
|
|
|
Expr {
|
|
|
|
kind: ExprKind::Array([arg]),
|
|
|
|
..
|
|
|
|
},
|
|
|
|
) => (arg, "&mut "),
|
|
|
|
ExprKind::MethodCall(
|
|
|
|
method,
|
|
|
|
[
|
|
|
|
Expr {
|
|
|
|
kind: ExprKind::Array([arg]),
|
|
|
|
..
|
|
|
|
},
|
|
|
|
],
|
|
|
|
_,
|
|
|
|
) if method.ident.name == rustc_span::sym::iter => (arg, "&"),
|
|
|
|
ExprKind::MethodCall(
|
|
|
|
method,
|
|
|
|
[
|
|
|
|
Expr {
|
|
|
|
kind: ExprKind::Array([arg]),
|
|
|
|
..
|
|
|
|
},
|
|
|
|
],
|
|
|
|
_,
|
|
|
|
) if method.ident.name.as_str() == "iter_mut" => (arg, "&mut "),
|
|
|
|
ExprKind::MethodCall(
|
|
|
|
method,
|
|
|
|
[
|
|
|
|
Expr {
|
|
|
|
kind: ExprKind::Array([arg]),
|
|
|
|
..
|
|
|
|
},
|
|
|
|
],
|
|
|
|
_,
|
|
|
|
) if method.ident.name == rustc_span::sym::into_iter => (arg, ""),
|
|
|
|
// Only check for arrays edition 2021 or later, as this case will trigger a compiler error otherwise.
|
|
|
|
ExprKind::Array([arg]) if cx.tcx.sess.edition() >= Edition::Edition2021 => (arg, ""),
|
2021-04-27 14:55:11 +00:00
|
|
|
_ => return,
|
|
|
|
};
|
2021-03-12 14:30:50 +00:00
|
|
|
if_chain! {
|
2021-04-08 15:50:13 +00:00
|
|
|
if let ExprKind::Block(block, _) = body.kind;
|
2021-03-12 14:30:50 +00:00
|
|
|
if !block.stmts.is_empty();
|
|
|
|
then {
|
2022-01-06 15:31:38 +00:00
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
|
|
|
let pat_snip = snippet_with_applicability(cx, pat.span, "..", &mut applicability);
|
2022-04-01 06:04:19 +00:00
|
|
|
let mut arg_snip = snippet_with_applicability(cx, arg_expression.span, "..", &mut applicability);
|
2022-01-06 15:31:38 +00:00
|
|
|
let mut block_str = snippet_with_applicability(cx, block.span, "..", &mut applicability).into_owned();
|
2021-03-12 14:30:50 +00:00
|
|
|
block_str.remove(0);
|
|
|
|
block_str.pop();
|
2022-01-06 15:31:38 +00:00
|
|
|
let indent = " ".repeat(indent_of(cx, block.stmts[0].span).unwrap_or(0));
|
2021-03-12 14:30:50 +00:00
|
|
|
|
2022-04-01 06:04:19 +00:00
|
|
|
// Reference iterator from `&(mut) []` or `[].iter(_mut)()`.
|
|
|
|
if !prefix.is_empty() && (
|
|
|
|
// Precedence of internal expression is less than or equal to precedence of `&expr`.
|
|
|
|
arg_expression.precedence().order() <= PREC_PREFIX || is_range_literal(arg_expression)
|
|
|
|
) {
|
|
|
|
arg_snip = format!("({arg_snip})").into();
|
|
|
|
}
|
|
|
|
|
2021-03-12 14:30:50 +00:00
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
SINGLE_ELEMENT_LOOP,
|
2021-10-27 14:48:06 +00:00
|
|
|
expr.span,
|
2021-03-12 14:30:50 +00:00
|
|
|
"for loop over a single element",
|
|
|
|
"try",
|
2022-04-01 06:04:19 +00:00
|
|
|
format!("{{\n{indent}let {pat_snip} = {prefix}{arg_snip};{block_str}}}"),
|
2022-01-06 15:31:38 +00:00
|
|
|
applicability,
|
2021-03-12 14:30:50 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|