75 lines
2.8 KiB
Rust
75 lines
2.8 KiB
Rust
|
use super::EXPLICIT_ITER_LOOP;
|
||
|
use crate::utils::{match_trait_method, snippet_with_applicability, span_lint_and_sugg};
|
||
|
use rustc_errors::Applicability;
|
||
|
use rustc_hir::{Expr, Mutability};
|
||
|
use rustc_lint::LateContext;
|
||
|
use rustc_middle::ty::{self, Ty, TyS};
|
||
|
use rustc_span::sym;
|
||
|
|
||
|
use crate::utils::{is_type_diagnostic_item, match_type, paths};
|
||
|
|
||
|
pub(super) fn check(cx: &LateContext<'_>, args: &[Expr<'_>], arg: &Expr<'_>, method_name: &str) {
|
||
|
let should_lint = match method_name {
|
||
|
"iter" | "iter_mut" => is_ref_iterable_type(cx, &args[0]),
|
||
|
"into_iter" if match_trait_method(cx, arg, &paths::INTO_ITERATOR) => {
|
||
|
let receiver_ty = cx.typeck_results().expr_ty(&args[0]);
|
||
|
let receiver_ty_adjusted = cx.typeck_results().expr_ty_adjusted(&args[0]);
|
||
|
let ref_receiver_ty = cx.tcx.mk_ref(
|
||
|
cx.tcx.lifetimes.re_erased,
|
||
|
ty::TypeAndMut {
|
||
|
ty: receiver_ty,
|
||
|
mutbl: Mutability::Not,
|
||
|
},
|
||
|
);
|
||
|
TyS::same_type(receiver_ty_adjusted, ref_receiver_ty)
|
||
|
},
|
||
|
_ => false,
|
||
|
};
|
||
|
|
||
|
if !should_lint {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
let mut applicability = Applicability::MachineApplicable;
|
||
|
let object = snippet_with_applicability(cx, args[0].span, "_", &mut applicability);
|
||
|
let muta = if method_name == "iter_mut" { "mut " } else { "" };
|
||
|
span_lint_and_sugg(
|
||
|
cx,
|
||
|
EXPLICIT_ITER_LOOP,
|
||
|
arg.span,
|
||
|
"it is more concise to loop over references to containers instead of using explicit \
|
||
|
iteration methods",
|
||
|
"to write this more concisely, try",
|
||
|
format!("&{}{}", muta, object),
|
||
|
applicability,
|
||
|
)
|
||
|
}
|
||
|
|
||
|
/// Returns `true` if the type of expr is one that provides `IntoIterator` impls
|
||
|
/// for `&T` and `&mut T`, such as `Vec`.
|
||
|
#[rustfmt::skip]
|
||
|
fn is_ref_iterable_type(cx: &LateContext<'_>, e: &Expr<'_>) -> bool {
|
||
|
// no walk_ptrs_ty: calling iter() on a reference can make sense because it
|
||
|
// will allow further borrows afterwards
|
||
|
let ty = cx.typeck_results().expr_ty(e);
|
||
|
is_iterable_array(ty, cx) ||
|
||
|
is_type_diagnostic_item(cx, ty, sym::vec_type) ||
|
||
|
match_type(cx, ty, &paths::LINKED_LIST) ||
|
||
|
is_type_diagnostic_item(cx, ty, sym::hashmap_type) ||
|
||
|
is_type_diagnostic_item(cx, ty, sym::hashset_type) ||
|
||
|
is_type_diagnostic_item(cx, ty, sym::vecdeque_type) ||
|
||
|
match_type(cx, ty, &paths::BINARY_HEAP) ||
|
||
|
match_type(cx, ty, &paths::BTREEMAP) ||
|
||
|
match_type(cx, ty, &paths::BTREESET)
|
||
|
}
|
||
|
|
||
|
fn is_iterable_array<'tcx>(ty: Ty<'tcx>, cx: &LateContext<'tcx>) -> bool {
|
||
|
// IntoIterator is currently only implemented for array sizes <= 32 in rustc
|
||
|
match ty.kind() {
|
||
|
ty::Array(_, n) => n
|
||
|
.try_eval_usize(cx.tcx, cx.param_env)
|
||
|
.map_or(false, |val| (0..=32).contains(&val)),
|
||
|
_ => false,
|
||
|
}
|
||
|
}
|