2019-11-23 21:35:21 +00:00
|
|
|
use crate::utils::{
|
|
|
|
get_trait_def_id, if_sequence, implements_trait, parent_node_is_if_expr, paths, span_help_and_lint, SpanlessEq,
|
|
|
|
};
|
2020-01-06 16:39:50 +00:00
|
|
|
use rustc_hir::*;
|
2020-01-12 06:08:41 +00:00
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
2020-01-11 11:37:08 +00:00
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
2019-09-24 21:55:05 +00:00
|
|
|
|
|
|
|
declare_clippy_lint! {
|
|
|
|
/// **What it does:** Checks comparison chains written with `if` that can be
|
|
|
|
/// rewritten with `match` and `cmp`.
|
|
|
|
///
|
|
|
|
/// **Why is this bad?** `if` is not guaranteed to be exhaustive and conditionals can get
|
|
|
|
/// repetitive
|
|
|
|
///
|
|
|
|
/// **Known problems:** None.
|
|
|
|
///
|
|
|
|
/// **Example:**
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// # fn a() {}
|
|
|
|
/// # fn b() {}
|
|
|
|
/// # fn c() {}
|
|
|
|
/// fn f(x: u8, y: u8) {
|
|
|
|
/// if x > y {
|
|
|
|
/// a()
|
|
|
|
/// } else if x < y {
|
|
|
|
/// b()
|
|
|
|
/// } else {
|
|
|
|
/// c()
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// Could be written:
|
|
|
|
///
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// use std::cmp::Ordering;
|
|
|
|
/// # fn a() {}
|
|
|
|
/// # fn b() {}
|
|
|
|
/// # fn c() {}
|
|
|
|
/// fn f(x: u8, y: u8) {
|
2019-09-24 22:05:43 +00:00
|
|
|
/// match x.cmp(&y) {
|
2019-09-24 21:55:05 +00:00
|
|
|
/// Ordering::Greater => a(),
|
|
|
|
/// Ordering::Less => b(),
|
|
|
|
/// Ordering::Equal => c()
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
pub COMPARISON_CHAIN,
|
|
|
|
style,
|
|
|
|
"`if`s that can be rewritten with `match` and `cmp`"
|
|
|
|
}
|
|
|
|
|
|
|
|
declare_lint_pass!(ComparisonChain => [COMPARISON_CHAIN]);
|
|
|
|
|
|
|
|
impl<'a, 'tcx> LateLintPass<'a, 'tcx> for ComparisonChain {
|
2019-12-27 07:12:26 +00:00
|
|
|
fn check_expr(&mut self, cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr<'_>) {
|
2019-09-24 21:55:05 +00:00
|
|
|
if expr.span.from_expansion() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We only care about the top-most `if` in the chain
|
|
|
|
if parent_node_is_if_expr(expr, cx) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that there exists at least one explicit else condition
|
|
|
|
let (conds, _) = if_sequence(expr);
|
|
|
|
if conds.len() < 2 {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for cond in conds.windows(2) {
|
|
|
|
if let (
|
|
|
|
&ExprKind::Binary(ref kind1, ref lhs1, ref rhs1),
|
|
|
|
&ExprKind::Binary(ref kind2, ref lhs2, ref rhs2),
|
2019-09-27 15:16:06 +00:00
|
|
|
) = (&cond[0].kind, &cond[1].kind)
|
2019-09-24 21:55:05 +00:00
|
|
|
{
|
|
|
|
if !kind_is_cmp(kind1.node) || !kind_is_cmp(kind2.node) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that both sets of operands are equal
|
|
|
|
let mut spanless_eq = SpanlessEq::new(cx);
|
|
|
|
if (!spanless_eq.eq_expr(lhs1, lhs2) || !spanless_eq.eq_expr(rhs1, rhs2))
|
|
|
|
&& (!spanless_eq.eq_expr(lhs1, rhs2) || !spanless_eq.eq_expr(rhs1, lhs2))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2019-11-23 21:35:21 +00:00
|
|
|
|
|
|
|
// Check that the type being compared implements `core::cmp::Ord`
|
|
|
|
let ty = cx.tables.expr_ty(lhs1);
|
|
|
|
let is_ord = get_trait_def_id(cx, &paths::ORD).map_or(false, |id| implements_trait(cx, ty, id, &[]));
|
|
|
|
|
|
|
|
if !is_ord {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-24 21:55:05 +00:00
|
|
|
} else {
|
|
|
|
// We only care about comparison chains
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
span_help_and_lint(
|
|
|
|
cx,
|
|
|
|
COMPARISON_CHAIN,
|
|
|
|
expr.span,
|
|
|
|
"`if` chain can be rewritten with `match`",
|
|
|
|
"Consider rewriting the `if` chain to use `cmp` and `match`.",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn kind_is_cmp(kind: BinOpKind) -> bool {
|
|
|
|
match kind {
|
|
|
|
BinOpKind::Lt | BinOpKind::Gt | BinOpKind::Eq => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|