2021-03-16 00:55:45 +00:00
|
|
|
use crate::utils::{match_trait_method, paths};
|
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2021-03-13 23:01:03 +00:00
|
|
|
use clippy_utils::ty::has_iter_method;
|
2021-03-06 08:28:14 +00:00
|
|
|
use rustc_errors::Applicability;
|
|
|
|
use rustc_hir as hir;
|
|
|
|
use rustc_lint::LateContext;
|
|
|
|
use rustc_middle::ty::{self, Ty};
|
|
|
|
use rustc_span::source_map::Span;
|
2021-03-11 11:02:29 +00:00
|
|
|
use rustc_span::symbol::Symbol;
|
2021-03-06 08:28:14 +00:00
|
|
|
|
|
|
|
use super::INTO_ITER_ON_REF;
|
|
|
|
|
|
|
|
pub(super) fn check(cx: &LateContext<'_>, expr: &hir::Expr<'_>, self_ref_ty: Ty<'_>, method_span: Span) {
|
|
|
|
if !match_trait_method(cx, expr, &paths::INTO_ITERATOR) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if let Some((kind, method_name)) = ty_has_iter_method(cx, self_ref_ty) {
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
INTO_ITER_ON_REF,
|
|
|
|
method_span,
|
|
|
|
&format!(
|
|
|
|
"this `.into_iter()` call is equivalent to `.{}()` and will not consume the `{}`",
|
|
|
|
method_name, kind,
|
|
|
|
),
|
|
|
|
"call directly",
|
|
|
|
method_name.to_string(),
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-11 11:02:29 +00:00
|
|
|
fn ty_has_iter_method(cx: &LateContext<'_>, self_ref_ty: Ty<'_>) -> Option<(Symbol, &'static str)> {
|
2021-03-06 08:28:14 +00:00
|
|
|
has_iter_method(cx, self_ref_ty).map(|ty_name| {
|
|
|
|
let mutbl = match self_ref_ty.kind() {
|
|
|
|
ty::Ref(_, _, mutbl) => mutbl,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
let method_name = match mutbl {
|
|
|
|
hir::Mutability::Not => "iter",
|
|
|
|
hir::Mutability::Mut => "iter_mut",
|
|
|
|
};
|
|
|
|
(ty_name, method_name)
|
|
|
|
})
|
|
|
|
}
|