2021-03-12 15:30:50 +01:00
|
|
|
use super::WHILE_LET_ON_ITERATOR;
|
2021-03-25 19:29:11 +01:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
|
|
|
use clippy_utils::source::snippet_with_applicability;
|
2023-07-17 10:19:29 +02:00
|
|
|
use clippy_utils::visitors::is_res_used;
|
|
|
|
use clippy_utils::{get_enclosing_loop_or_multi_call_closure, higher, is_refutable, is_res_lang_ctor, is_trait_method};
|
2021-03-12 15:30:50 +01:00
|
|
|
use if_chain::if_chain;
|
|
|
|
use rustc_errors::Applicability;
|
2023-07-17 10:19:29 +02:00
|
|
|
use rustc_hir::def::Res;
|
2022-01-15 16:07:52 -06:00
|
|
|
use rustc_hir::intravisit::{walk_expr, Visitor};
|
2023-07-17 10:19:29 +02:00
|
|
|
use rustc_hir::{Closure, Expr, ExprKind, HirId, LangItem, Local, Mutability, PatKind, UnOp};
|
2021-03-12 15:30:50 +01:00
|
|
|
use rustc_lint::LateContext;
|
2022-07-18 09:39:37 +02:00
|
|
|
use rustc_middle::hir::nested_filter::OnlyBodies;
|
2022-01-13 13:18:19 +01:00
|
|
|
use rustc_middle::ty::adjustment::Adjust;
|
2023-07-17 10:19:29 +02:00
|
|
|
use rustc_span::symbol::sym;
|
|
|
|
use rustc_span::Symbol;
|
2021-03-12 15:30:50 +01:00
|
|
|
|
2022-01-13 13:18:19 +01:00
|
|
|
pub(super) fn check<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>) {
|
|
|
|
let (scrutinee_expr, iter_expr_struct, iter_expr, some_pat, loop_expr) = if_chain! {
|
2021-09-08 16:31:47 +02:00
|
|
|
if let Some(higher::WhileLet { if_then, let_pat, let_expr }) = higher::WhileLet::hir(expr);
|
2021-05-20 12:30:31 +02:00
|
|
|
// check for `Some(..)` pattern
|
2022-10-06 09:44:38 +02:00
|
|
|
if let PatKind::TupleStruct(ref pat_path, some_pat, _) = let_pat.kind;
|
|
|
|
if is_res_lang_ctor(cx, cx.qpath_res(pat_path, let_pat.hir_id), LangItem::OptionSome);
|
2021-05-20 12:30:31 +02:00
|
|
|
// check for call to `Iterator::next`
|
2022-09-01 18:43:35 +09:00
|
|
|
if let ExprKind::MethodCall(method_name, iter_expr, [], _) = let_expr.kind;
|
2021-05-20 12:30:31 +02:00
|
|
|
if method_name.ident.name == sym::next;
|
2021-08-08 11:49:13 -03:00
|
|
|
if is_trait_method(cx, let_expr, sym::Iterator);
|
|
|
|
if let Some(iter_expr_struct) = try_parse_iter_expr(cx, iter_expr);
|
2021-05-20 12:30:31 +02:00
|
|
|
// get the loop containing the match expression
|
2021-08-08 11:49:13 -03:00
|
|
|
if !uses_iter(cx, &iter_expr_struct, if_then);
|
2021-05-20 12:30:31 +02:00
|
|
|
then {
|
2022-01-13 13:18:19 +01:00
|
|
|
(let_expr, iter_expr_struct, iter_expr, some_pat, expr)
|
2021-05-20 12:30:31 +02:00
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
};
|
2021-03-12 15:30:50 +01:00
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
|
|
|
let loop_var = if let Some(some_pat) = some_pat.first() {
|
|
|
|
if is_refutable(cx, some_pat) {
|
|
|
|
// Refutable patterns don't work with for loops.
|
|
|
|
return;
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
2021-05-20 12:30:31 +02:00
|
|
|
snippet_with_applicability(cx, some_pat.span, "..", &mut applicability)
|
|
|
|
} else {
|
|
|
|
"_".into()
|
|
|
|
};
|
2021-03-12 15:30:50 +01:00
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
// If the iterator is a field or the iterator is accessed after the loop is complete it needs to be
|
|
|
|
// borrowed mutably. TODO: If the struct can be partially moved from and the struct isn't used
|
|
|
|
// afterwards a mutable borrow of a field isn't necessary.
|
2022-01-13 13:18:19 +01:00
|
|
|
let by_ref = if cx.typeck_results().expr_ty(iter_expr).ref_mutability() == Some(Mutability::Mut)
|
|
|
|
|| !iter_expr_struct.can_move
|
|
|
|
|| !iter_expr_struct.fields.is_empty()
|
|
|
|
|| needs_mutable_borrow(cx, &iter_expr_struct, loop_expr)
|
|
|
|
{
|
2021-09-28 18:03:12 +01:00
|
|
|
".by_ref()"
|
2021-05-20 12:30:31 +02:00
|
|
|
} else {
|
|
|
|
""
|
2021-03-12 15:30:50 +01:00
|
|
|
};
|
2021-05-20 12:30:31 +02:00
|
|
|
|
|
|
|
let iterator = snippet_with_applicability(cx, iter_expr.span, "_", &mut applicability);
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
WHILE_LET_ON_ITERATOR,
|
|
|
|
expr.span.with_hi(scrutinee_expr.span.hi()),
|
|
|
|
"this loop could be written as a `for` loop",
|
|
|
|
"try",
|
2022-10-06 09:44:38 +02:00
|
|
|
format!("for {loop_var} in {iterator}{by_ref}"),
|
2021-05-20 12:30:31 +02:00
|
|
|
applicability,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
struct IterExpr {
|
|
|
|
/// The fields used, in order of child to parent.
|
|
|
|
fields: Vec<Symbol>,
|
|
|
|
/// The path being used.
|
|
|
|
path: Res,
|
2022-01-13 13:18:19 +01:00
|
|
|
/// Whether or not the iterator can be moved.
|
|
|
|
can_move: bool,
|
2021-05-20 12:30:31 +02:00
|
|
|
}
|
2021-08-08 11:49:13 -03:00
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
/// Parses any expression to find out which field of which variable is used. Will return `None` if
|
|
|
|
/// the expression might have side effects.
|
|
|
|
fn try_parse_iter_expr(cx: &LateContext<'_>, mut e: &Expr<'_>) -> Option<IterExpr> {
|
|
|
|
let mut fields = Vec::new();
|
2022-01-13 13:18:19 +01:00
|
|
|
let mut can_move = true;
|
2021-05-20 12:30:31 +02:00
|
|
|
loop {
|
2022-01-13 13:18:19 +01:00
|
|
|
if cx
|
|
|
|
.typeck_results()
|
|
|
|
.expr_adjustments(e)
|
|
|
|
.iter()
|
|
|
|
.any(|a| matches!(a.kind, Adjust::Deref(Some(..))))
|
|
|
|
{
|
|
|
|
// Custom deref impls need to borrow the whole value as it's captured by reference
|
|
|
|
can_move = false;
|
|
|
|
fields.clear();
|
|
|
|
}
|
2021-05-20 12:30:31 +02:00
|
|
|
match e.kind {
|
|
|
|
ExprKind::Path(ref path) => {
|
|
|
|
break Some(IterExpr {
|
|
|
|
fields,
|
|
|
|
path: cx.qpath_res(path, e.hir_id),
|
2022-01-13 13:18:19 +01:00
|
|
|
can_move,
|
2021-05-20 12:30:31 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
ExprKind::Field(base, name) => {
|
|
|
|
fields.push(name.name);
|
|
|
|
e = base;
|
|
|
|
},
|
|
|
|
// Dereferencing a pointer has no side effects and doesn't affect which field is being used.
|
|
|
|
ExprKind::Unary(UnOp::Deref, base) if cx.typeck_results().expr_ty(base).is_ref() => e = base,
|
|
|
|
|
|
|
|
// Shouldn't have side effects, but there's no way to trace which field is used. So forget which fields have
|
|
|
|
// already been seen.
|
2023-08-03 21:43:17 +02:00
|
|
|
ExprKind::Index(base, idx, _) if !idx.can_have_side_effects() => {
|
2022-01-13 13:18:19 +01:00
|
|
|
can_move = false;
|
2021-05-20 12:30:31 +02:00
|
|
|
fields.clear();
|
|
|
|
e = base;
|
|
|
|
},
|
|
|
|
ExprKind::Unary(UnOp::Deref, base) => {
|
2022-01-13 13:18:19 +01:00
|
|
|
can_move = false;
|
2021-05-20 12:30:31 +02:00
|
|
|
fields.clear();
|
|
|
|
e = base;
|
|
|
|
},
|
|
|
|
|
|
|
|
// No effect and doesn't affect which field is being used.
|
|
|
|
ExprKind::DropTemps(base) | ExprKind::AddrOf(_, _, base) | ExprKind::Type(base, _) => e = base,
|
|
|
|
_ => break None,
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
fn is_expr_same_field(cx: &LateContext<'_>, mut e: &Expr<'_>, mut fields: &[Symbol], path_res: Res) -> bool {
|
|
|
|
loop {
|
|
|
|
match (&e.kind, fields) {
|
|
|
|
(&ExprKind::Field(base, name), [head_field, tail_fields @ ..]) if name.name == *head_field => {
|
|
|
|
e = base;
|
|
|
|
fields = tail_fields;
|
|
|
|
},
|
|
|
|
(ExprKind::Path(path), []) => {
|
|
|
|
break cx.qpath_res(path, e.hir_id) == path_res;
|
|
|
|
},
|
|
|
|
(&(ExprKind::DropTemps(base) | ExprKind::AddrOf(_, _, base) | ExprKind::Type(base, _)), _) => e = base,
|
|
|
|
_ => break false,
|
|
|
|
}
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
/// Checks if the given expression is the same field as, is a child of, or is the parent of the
|
|
|
|
/// given field. Used to check if the expression can be used while the given field is borrowed
|
|
|
|
/// mutably. e.g. if checking for `x.y`, then `x.y`, `x.y.z`, and `x` will all return true, but
|
|
|
|
/// `x.z`, and `y` will return false.
|
|
|
|
fn is_expr_same_child_or_parent_field(cx: &LateContext<'_>, expr: &Expr<'_>, fields: &[Symbol], path_res: Res) -> bool {
|
|
|
|
match expr.kind {
|
|
|
|
ExprKind::Field(base, name) => {
|
|
|
|
if let Some((head_field, tail_fields)) = fields.split_first() {
|
2021-08-12 11:16:25 +02:00
|
|
|
if name.name == *head_field && is_expr_same_field(cx, base, tail_fields, path_res) {
|
2021-05-20 12:30:31 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Check if the expression is a parent field
|
|
|
|
let mut fields_iter = tail_fields.iter();
|
|
|
|
while let Some(field) = fields_iter.next() {
|
|
|
|
if *field == name.name && is_expr_same_field(cx, base, fields_iter.as_slice(), path_res) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the expression is a child field.
|
|
|
|
let mut e = base;
|
|
|
|
loop {
|
|
|
|
match e.kind {
|
|
|
|
ExprKind::Field(..) if is_expr_same_field(cx, e, fields, path_res) => break true,
|
|
|
|
ExprKind::Field(base, _) | ExprKind::DropTemps(base) | ExprKind::Type(base, _) => e = base,
|
|
|
|
ExprKind::Path(ref path) if fields.is_empty() => {
|
|
|
|
break cx.qpath_res(path, e.hir_id) == path_res;
|
|
|
|
},
|
|
|
|
_ => break false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// If the path matches, this is either an exact match, or the expression is a parent of the field.
|
|
|
|
ExprKind::Path(ref path) => cx.qpath_res(path, expr.hir_id) == path_res,
|
|
|
|
ExprKind::DropTemps(base) | ExprKind::Type(base, _) | ExprKind::AddrOf(_, _, base) => {
|
|
|
|
is_expr_same_child_or_parent_field(cx, base, fields, path_res)
|
|
|
|
},
|
|
|
|
_ => false,
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
/// Strips off all field and path expressions. This will return true if a field or path has been
|
|
|
|
/// skipped. Used to skip them after failing to check for equality.
|
2022-01-13 13:18:19 +01:00
|
|
|
fn skip_fields_and_path<'tcx>(expr: &'tcx Expr<'_>) -> (Option<&'tcx Expr<'tcx>>, bool) {
|
2021-05-20 12:30:31 +02:00
|
|
|
let mut e = expr;
|
|
|
|
let e = loop {
|
|
|
|
match e.kind {
|
|
|
|
ExprKind::Field(base, _) | ExprKind::DropTemps(base) | ExprKind::Type(base, _) => e = base,
|
|
|
|
ExprKind::Path(_) => return (None, true),
|
|
|
|
_ => break e,
|
|
|
|
}
|
2021-03-12 15:30:50 +01:00
|
|
|
};
|
2021-05-20 12:30:31 +02:00
|
|
|
(Some(e), e.hir_id != expr.hir_id)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks if the given expression uses the iterator.
|
2022-01-13 13:18:19 +01:00
|
|
|
fn uses_iter<'tcx>(cx: &LateContext<'tcx>, iter_expr: &IterExpr, container: &'tcx Expr<'_>) -> bool {
|
2021-05-20 12:30:31 +02:00
|
|
|
struct V<'a, 'b, 'tcx> {
|
|
|
|
cx: &'a LateContext<'tcx>,
|
|
|
|
iter_expr: &'b IterExpr,
|
|
|
|
uses_iter: bool,
|
|
|
|
}
|
2022-01-13 13:18:19 +01:00
|
|
|
impl<'tcx> Visitor<'tcx> for V<'_, '_, 'tcx> {
|
2021-05-20 12:30:31 +02:00
|
|
|
fn visit_expr(&mut self, e: &'tcx Expr<'_>) {
|
|
|
|
if self.uses_iter {
|
|
|
|
// return
|
|
|
|
} else if is_expr_same_child_or_parent_field(self.cx, e, &self.iter_expr.fields, self.iter_expr.path) {
|
|
|
|
self.uses_iter = true;
|
|
|
|
} else if let (e, true) = skip_fields_and_path(e) {
|
|
|
|
if let Some(e) = e {
|
|
|
|
self.visit_expr(e);
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
2022-06-30 14:18:51 +04:00
|
|
|
} else if let ExprKind::Closure(&Closure { body: id, .. }) = e.kind {
|
2021-05-20 12:30:31 +02:00
|
|
|
if is_res_used(self.cx, self.iter_expr.path, id) {
|
|
|
|
self.uses_iter = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
walk_expr(self, e);
|
|
|
|
}
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
let mut v = V {
|
|
|
|
cx,
|
|
|
|
iter_expr,
|
|
|
|
uses_iter: false,
|
|
|
|
};
|
|
|
|
v.visit_expr(container);
|
|
|
|
v.uses_iter
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
|
2022-05-21 13:24:00 +02:00
|
|
|
#[expect(clippy::too_many_lines)]
|
2022-01-13 13:18:19 +01:00
|
|
|
fn needs_mutable_borrow(cx: &LateContext<'_>, iter_expr: &IterExpr, loop_expr: &Expr<'_>) -> bool {
|
2021-05-20 12:30:31 +02:00
|
|
|
struct AfterLoopVisitor<'a, 'b, 'tcx> {
|
|
|
|
cx: &'a LateContext<'tcx>,
|
|
|
|
iter_expr: &'b IterExpr,
|
|
|
|
loop_id: HirId,
|
|
|
|
after_loop: bool,
|
|
|
|
used_iter: bool,
|
|
|
|
}
|
2022-01-13 13:18:19 +01:00
|
|
|
impl<'tcx> Visitor<'tcx> for AfterLoopVisitor<'_, '_, 'tcx> {
|
2022-07-18 09:39:37 +02:00
|
|
|
type NestedFilter = OnlyBodies;
|
|
|
|
fn nested_visit_map(&mut self) -> Self::Map {
|
|
|
|
self.cx.tcx.hir()
|
|
|
|
}
|
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
fn visit_expr(&mut self, e: &'tcx Expr<'_>) {
|
|
|
|
if self.used_iter {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if self.after_loop {
|
|
|
|
if is_expr_same_child_or_parent_field(self.cx, e, &self.iter_expr.fields, self.iter_expr.path) {
|
|
|
|
self.used_iter = true;
|
|
|
|
} else if let (e, true) = skip_fields_and_path(e) {
|
|
|
|
if let Some(e) = e {
|
|
|
|
self.visit_expr(e);
|
|
|
|
}
|
2022-06-30 14:18:51 +04:00
|
|
|
} else if let ExprKind::Closure(&Closure { body: id, .. }) = e.kind {
|
2021-05-20 12:30:31 +02:00
|
|
|
self.used_iter = is_res_used(self.cx, self.iter_expr.path, id);
|
|
|
|
} else {
|
|
|
|
walk_expr(self, e);
|
|
|
|
}
|
|
|
|
} else if self.loop_id == e.hir_id {
|
|
|
|
self.after_loop = true;
|
|
|
|
} else {
|
|
|
|
walk_expr(self, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct NestedLoopVisitor<'a, 'b, 'tcx> {
|
|
|
|
cx: &'a LateContext<'tcx>,
|
|
|
|
iter_expr: &'b IterExpr,
|
|
|
|
local_id: HirId,
|
|
|
|
loop_id: HirId,
|
|
|
|
after_loop: bool,
|
|
|
|
found_local: bool,
|
|
|
|
used_after: bool,
|
|
|
|
}
|
2022-01-13 13:18:19 +01:00
|
|
|
impl<'a, 'b, 'tcx> Visitor<'tcx> for NestedLoopVisitor<'a, 'b, 'tcx> {
|
2022-07-18 09:39:37 +02:00
|
|
|
type NestedFilter = OnlyBodies;
|
|
|
|
fn nested_visit_map(&mut self) -> Self::Map {
|
|
|
|
self.cx.tcx.hir()
|
|
|
|
}
|
|
|
|
|
2021-05-20 12:30:31 +02:00
|
|
|
fn visit_local(&mut self, l: &'tcx Local<'_>) {
|
|
|
|
if !self.after_loop {
|
|
|
|
l.pat.each_binding_or_first(&mut |_, id, _, _| {
|
|
|
|
if id == self.local_id {
|
|
|
|
self.found_local = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if let Some(e) = l.init {
|
|
|
|
self.visit_expr(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn visit_expr(&mut self, e: &'tcx Expr<'_>) {
|
|
|
|
if self.used_after {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if self.after_loop {
|
|
|
|
if is_expr_same_child_or_parent_field(self.cx, e, &self.iter_expr.fields, self.iter_expr.path) {
|
|
|
|
self.used_after = true;
|
|
|
|
} else if let (e, true) = skip_fields_and_path(e) {
|
|
|
|
if let Some(e) = e {
|
|
|
|
self.visit_expr(e);
|
|
|
|
}
|
2022-06-30 14:18:51 +04:00
|
|
|
} else if let ExprKind::Closure(&Closure { body: id, .. }) = e.kind {
|
2021-05-20 12:30:31 +02:00
|
|
|
self.used_after = is_res_used(self.cx, self.iter_expr.path, id);
|
|
|
|
} else {
|
|
|
|
walk_expr(self, e);
|
|
|
|
}
|
|
|
|
} else if e.hir_id == self.loop_id {
|
|
|
|
self.after_loop = true;
|
|
|
|
} else {
|
|
|
|
walk_expr(self, e);
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-20 12:30:31 +02:00
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
if let Some(e) = get_enclosing_loop_or_multi_call_closure(cx, loop_expr) {
|
2022-10-23 15:18:45 +02:00
|
|
|
let Res::Local(local_id) = iter_expr.path else {
|
2023-07-17 10:19:29 +02:00
|
|
|
return true;
|
2021-05-20 12:30:31 +02:00
|
|
|
};
|
|
|
|
let mut v = NestedLoopVisitor {
|
|
|
|
cx,
|
|
|
|
iter_expr,
|
|
|
|
local_id,
|
|
|
|
loop_id: loop_expr.hir_id,
|
|
|
|
after_loop: false,
|
|
|
|
found_local: false,
|
|
|
|
used_after: false,
|
|
|
|
};
|
|
|
|
v.visit_expr(e);
|
|
|
|
v.used_after || !v.found_local
|
|
|
|
} else {
|
|
|
|
let mut v = AfterLoopVisitor {
|
|
|
|
cx,
|
|
|
|
iter_expr,
|
|
|
|
loop_id: loop_expr.hir_id,
|
|
|
|
after_loop: false,
|
|
|
|
used_iter: false,
|
|
|
|
};
|
2022-09-09 13:36:26 +02:00
|
|
|
v.visit_expr(cx.tcx.hir().body(cx.enclosing_body.unwrap()).value);
|
2021-05-20 12:30:31 +02:00
|
|
|
v.used_iter
|
2021-03-12 15:30:50 +01:00
|
|
|
}
|
|
|
|
}
|