mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-24 07:44:10 +00:00
clippy: Fix pattern_type_mismatch for loop
This commit is contained in:
parent
9d5ea8f709
commit
7508a9b22c
@ -1,16 +1,12 @@
|
|||||||
use clippy_utils::diagnostics::span_lint_and_help;
|
use clippy_utils::diagnostics::span_lint_and_help;
|
||||||
use clippy_utils::last_path_segment;
|
|
||||||
use rustc_hir::{
|
use rustc_hir::{
|
||||||
intravisit, Body, Expr, ExprKind, FnDecl, HirId, LocalSource, MatchSource, Mutability, Pat, PatField, PatKind,
|
intravisit, Body, Expr, ExprKind, FnDecl, HirId, LocalSource, Mutability, Pat, PatKind, Stmt, StmtKind,
|
||||||
QPath, Stmt, StmtKind,
|
|
||||||
};
|
};
|
||||||
use rustc_lint::{LateContext, LateLintPass, LintContext};
|
use rustc_lint::{LateContext, LateLintPass, LintContext};
|
||||||
use rustc_middle::lint::in_external_macro;
|
use rustc_middle::lint::in_external_macro;
|
||||||
use rustc_middle::ty::subst::SubstsRef;
|
use rustc_middle::ty;
|
||||||
use rustc_middle::ty::{AdtDef, FieldDef, Ty, TyKind, VariantDef};
|
|
||||||
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
||||||
use rustc_span::source_map::Span;
|
use rustc_span::source_map::Span;
|
||||||
use std::iter;
|
|
||||||
|
|
||||||
declare_clippy_lint! {
|
declare_clippy_lint! {
|
||||||
/// ### What it does
|
/// ### What it does
|
||||||
@ -87,43 +83,28 @@ declare_lint_pass!(PatternTypeMismatch => [PATTERN_TYPE_MISMATCH]);
|
|||||||
impl<'tcx> LateLintPass<'tcx> for PatternTypeMismatch {
|
impl<'tcx> LateLintPass<'tcx> for PatternTypeMismatch {
|
||||||
fn check_stmt(&mut self, cx: &LateContext<'tcx>, stmt: &'tcx Stmt<'_>) {
|
fn check_stmt(&mut self, cx: &LateContext<'tcx>, stmt: &'tcx Stmt<'_>) {
|
||||||
if let StmtKind::Local(local) = stmt.kind {
|
if let StmtKind::Local(local) = stmt.kind {
|
||||||
if let Some(init) = &local.init {
|
if in_external_macro(cx.sess(), local.pat.span) {
|
||||||
if let Some(init_ty) = cx.typeck_results().node_type_opt(init.hir_id) {
|
return;
|
||||||
let pat = &local.pat;
|
|
||||||
if in_external_macro(cx.sess(), pat.span) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
let deref_possible = match local.source {
|
|
||||||
LocalSource::Normal => DerefPossible::Possible,
|
|
||||||
_ => DerefPossible::Impossible,
|
|
||||||
};
|
|
||||||
apply_lint(cx, pat, init_ty, deref_possible);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
let deref_possible = match local.source {
|
||||||
|
LocalSource::Normal => DerefPossible::Possible,
|
||||||
|
_ => DerefPossible::Impossible,
|
||||||
|
};
|
||||||
|
apply_lint(cx, local.pat, deref_possible);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>) {
|
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>) {
|
||||||
if let ExprKind::Match(scrutinee, arms, MatchSource::Normal) = expr.kind {
|
if let ExprKind::Match(_, arms, _) = expr.kind {
|
||||||
if let Some(expr_ty) = cx.typeck_results().node_type_opt(scrutinee.hir_id) {
|
for arm in arms {
|
||||||
'pattern_checks: for arm in arms {
|
let pat = &arm.pat;
|
||||||
let pat = &arm.pat;
|
if apply_lint(cx, pat, DerefPossible::Possible) {
|
||||||
if in_external_macro(cx.sess(), pat.span) {
|
break;
|
||||||
continue 'pattern_checks;
|
|
||||||
}
|
|
||||||
if apply_lint(cx, pat, expr_ty, DerefPossible::Possible) {
|
|
||||||
break 'pattern_checks;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let ExprKind::Let(let_pat, let_expr, _) = expr.kind {
|
if let ExprKind::Let(let_pat, ..) = expr.kind {
|
||||||
if let Some(expr_ty) = cx.typeck_results().node_type_opt(let_expr.hir_id) {
|
apply_lint(cx, let_pat, DerefPossible::Possible);
|
||||||
if in_external_macro(cx.sess(), let_pat.span) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
apply_lint(cx, let_pat, expr_ty, DerefPossible::Possible);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -134,12 +115,10 @@ impl<'tcx> LateLintPass<'tcx> for PatternTypeMismatch {
|
|||||||
_: &'tcx FnDecl<'_>,
|
_: &'tcx FnDecl<'_>,
|
||||||
body: &'tcx Body<'_>,
|
body: &'tcx Body<'_>,
|
||||||
_: Span,
|
_: Span,
|
||||||
hir_id: HirId,
|
_: HirId,
|
||||||
) {
|
) {
|
||||||
if let Some(fn_sig) = cx.typeck_results().liberated_fn_sigs().get(hir_id) {
|
for param in body.params {
|
||||||
for (param, ty) in iter::zip(body.params, fn_sig.inputs()) {
|
apply_lint(cx, param.pat, DerefPossible::Impossible);
|
||||||
apply_lint(cx, param.pat, ty, DerefPossible::Impossible);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -150,8 +129,8 @@ enum DerefPossible {
|
|||||||
Impossible,
|
Impossible,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn apply_lint<'tcx>(cx: &LateContext<'tcx>, pat: &Pat<'_>, expr_ty: Ty<'tcx>, deref_possible: DerefPossible) -> bool {
|
fn apply_lint<'tcx>(cx: &LateContext<'tcx>, pat: &Pat<'_>, deref_possible: DerefPossible) -> bool {
|
||||||
let maybe_mismatch = find_first_mismatch(cx, pat, expr_ty, Level::Top);
|
let maybe_mismatch = find_first_mismatch(cx, pat);
|
||||||
if let Some((span, mutability, level)) = maybe_mismatch {
|
if let Some((span, mutability, level)) = maybe_mismatch {
|
||||||
span_lint_and_help(
|
span_lint_and_help(
|
||||||
cx,
|
cx,
|
||||||
@ -184,132 +163,32 @@ enum Level {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[allow(rustc::usage_of_ty_tykind)]
|
#[allow(rustc::usage_of_ty_tykind)]
|
||||||
fn find_first_mismatch<'tcx>(
|
fn find_first_mismatch<'tcx>(cx: &LateContext<'tcx>, pat: &Pat<'_>) -> Option<(Span, Mutability, Level)> {
|
||||||
cx: &LateContext<'tcx>,
|
let mut result = None;
|
||||||
pat: &Pat<'_>,
|
pat.walk(|p| {
|
||||||
ty: Ty<'tcx>,
|
if result.is_some() {
|
||||||
level: Level,
|
return false;
|
||||||
) -> Option<(Span, Mutability, Level)> {
|
|
||||||
if let PatKind::Ref(sub_pat, _) = pat.kind {
|
|
||||||
if let TyKind::Ref(_, sub_ty, _) = ty.kind() {
|
|
||||||
return find_first_mismatch(cx, sub_pat, sub_ty, Level::Lower);
|
|
||||||
}
|
}
|
||||||
}
|
if in_external_macro(cx.sess(), p.span) {
|
||||||
|
return true;
|
||||||
if let TyKind::Ref(_, _, mutability) = *ty.kind() {
|
|
||||||
if is_non_ref_pattern(&pat.kind) {
|
|
||||||
return Some((pat.span, mutability, level));
|
|
||||||
}
|
}
|
||||||
}
|
let adjust_pat = match p.kind {
|
||||||
|
PatKind::Or([p, ..]) => p,
|
||||||
if let PatKind::Struct(ref qpath, field_pats, _) = pat.kind {
|
_ => p,
|
||||||
if let TyKind::Adt(adt_def, substs_ref) = ty.kind() {
|
|
||||||
if let Some(variant) = get_variant(adt_def, qpath) {
|
|
||||||
let field_defs = &variant.fields;
|
|
||||||
return find_first_mismatch_in_struct(cx, field_pats, field_defs, substs_ref);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let PatKind::TupleStruct(ref qpath, pats, _) = pat.kind {
|
|
||||||
if let TyKind::Adt(adt_def, substs_ref) = ty.kind() {
|
|
||||||
if let Some(variant) = get_variant(adt_def, qpath) {
|
|
||||||
let field_defs = &variant.fields;
|
|
||||||
let ty_iter = field_defs.iter().map(|field_def| field_def.ty(cx.tcx, substs_ref));
|
|
||||||
return find_first_mismatch_in_tuple(cx, pats, ty_iter);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let PatKind::Tuple(pats, _) = pat.kind {
|
|
||||||
if let TyKind::Tuple(..) = ty.kind() {
|
|
||||||
return find_first_mismatch_in_tuple(cx, pats, ty.tuple_fields());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let PatKind::Or(sub_pats) = pat.kind {
|
|
||||||
for pat in sub_pats {
|
|
||||||
let maybe_mismatch = find_first_mismatch(cx, pat, ty, level);
|
|
||||||
if let Some(mismatch) = maybe_mismatch {
|
|
||||||
return Some(mismatch);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
None
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_variant<'a>(adt_def: &'a AdtDef, qpath: &QPath<'_>) -> Option<&'a VariantDef> {
|
|
||||||
if adt_def.is_struct() {
|
|
||||||
if let Some(variant) = adt_def.variants.iter().next() {
|
|
||||||
return Some(variant);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if adt_def.is_enum() {
|
|
||||||
let pat_ident = last_path_segment(qpath).ident;
|
|
||||||
for variant in &adt_def.variants {
|
|
||||||
if variant.ident == pat_ident {
|
|
||||||
return Some(variant);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
None
|
|
||||||
}
|
|
||||||
|
|
||||||
fn find_first_mismatch_in_tuple<'tcx, I>(
|
|
||||||
cx: &LateContext<'tcx>,
|
|
||||||
pats: &[Pat<'_>],
|
|
||||||
ty_iter_src: I,
|
|
||||||
) -> Option<(Span, Mutability, Level)>
|
|
||||||
where
|
|
||||||
I: IntoIterator<Item = Ty<'tcx>>,
|
|
||||||
{
|
|
||||||
let mut field_tys = ty_iter_src.into_iter();
|
|
||||||
'fields: for pat in pats {
|
|
||||||
let field_ty = if let Some(ty) = field_tys.next() {
|
|
||||||
ty
|
|
||||||
} else {
|
|
||||||
break 'fields;
|
|
||||||
};
|
};
|
||||||
|
if let Some(adjustments) = cx.typeck_results().pat_adjustments().get(adjust_pat.hir_id) {
|
||||||
let maybe_mismatch = find_first_mismatch(cx, pat, field_ty, Level::Lower);
|
if let [first, ..] = **adjustments {
|
||||||
if let Some(mismatch) = maybe_mismatch {
|
if let ty::Ref(.., mutability) = *first.kind() {
|
||||||
return Some(mismatch);
|
let level = if p.hir_id == pat.hir_id {
|
||||||
}
|
Level::Top
|
||||||
}
|
} else {
|
||||||
|
Level::Lower
|
||||||
None
|
};
|
||||||
}
|
result = Some((p.span, mutability, level));
|
||||||
|
|
||||||
fn find_first_mismatch_in_struct<'tcx>(
|
|
||||||
cx: &LateContext<'tcx>,
|
|
||||||
field_pats: &[PatField<'_>],
|
|
||||||
field_defs: &[FieldDef],
|
|
||||||
substs_ref: SubstsRef<'tcx>,
|
|
||||||
) -> Option<(Span, Mutability, Level)> {
|
|
||||||
for field_pat in field_pats {
|
|
||||||
'definitions: for field_def in field_defs {
|
|
||||||
if field_pat.ident == field_def.ident {
|
|
||||||
let field_ty = field_def.ty(cx.tcx, substs_ref);
|
|
||||||
let pat = &field_pat.pat;
|
|
||||||
let maybe_mismatch = find_first_mismatch(cx, pat, field_ty, Level::Lower);
|
|
||||||
if let Some(mismatch) = maybe_mismatch {
|
|
||||||
return Some(mismatch);
|
|
||||||
}
|
}
|
||||||
break 'definitions;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
result.is_none()
|
||||||
|
});
|
||||||
None
|
result
|
||||||
}
|
|
||||||
|
|
||||||
fn is_non_ref_pattern(pat_kind: &PatKind<'_>) -> bool {
|
|
||||||
match pat_kind {
|
|
||||||
PatKind::Struct(..) | PatKind::Tuple(..) | PatKind::TupleStruct(..) | PatKind::Path(..) => true,
|
|
||||||
PatKind::Or(sub_pats) => sub_pats.iter().any(|pat| is_non_ref_pattern(&pat.kind)),
|
|
||||||
_ => false,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user