rust/clippy_lints/src/dereference.rs

108 lines
3.8 KiB
Rust
Raw Normal View History

use crate::utils::{get_parent_expr, implements_trait, snippet, span_lint_and_sugg};
2020-01-26 12:48:30 -06:00
use if_chain::if_chain;
use rustc_ast::util::parser::{ExprPrecedence, PREC_POSTFIX, PREC_PREFIX};
2020-01-23 09:28:01 -06:00
use rustc_errors::Applicability;
use rustc_hir::{Expr, ExprKind};
2020-01-23 09:28:01 -06:00
use rustc_lint::{LateContext, LateLintPass};
2020-01-26 12:48:30 -06:00
use rustc_session::{declare_lint_pass, declare_tool_lint};
use rustc_span::source_map::Span;
2018-10-03 11:53:39 -05:00
declare_clippy_lint! {
2020-01-23 09:28:01 -06:00
/// **What it does:** Checks for explicit `deref()` or `deref_mut()` method calls.
///
/// **Why is this bad?** Derefencing by `&*x` or `&mut *x` is clearer and more concise,
/// when not part of a method chain.
///
/// **Example:**
/// ```rust
/// use std::ops::Deref;
/// let a: &mut String = &mut String::from("foo");
/// let b: &str = a.deref();
2020-01-23 09:28:01 -06:00
/// ```
/// Could be written as:
/// ```rust
/// let a: &mut String = &mut String::from("foo");
2020-01-23 09:28:01 -06:00
/// let b = &*a;
/// ```
2020-01-26 12:48:30 -06:00
///
2020-01-23 09:28:01 -06:00
/// This lint excludes
/// ```rust,ignore
/// let _ = d.unwrap().deref();
2020-01-23 09:28:01 -06:00
/// ```
pub EXPLICIT_DEREF_METHODS,
2018-10-03 11:53:39 -05:00
pedantic,
"Explicit use of deref or deref_mut method while not in a method chain."
}
2020-01-23 09:28:01 -06:00
declare_lint_pass!(Dereferencing => [
EXPLICIT_DEREF_METHODS
2020-01-23 09:28:01 -06:00
]);
2018-10-03 11:53:39 -05:00
impl<'tcx> LateLintPass<'tcx> for Dereferencing {
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>) {
2018-10-03 11:53:39 -05:00
if_chain! {
if !expr.span.from_expansion();
2020-06-09 16:44:04 -05:00
if let ExprKind::MethodCall(ref method_name, _, ref args, _) = &expr.kind;
2020-01-26 12:48:30 -06:00
if args.len() == 1;
2018-10-03 11:53:39 -05:00
then {
if let Some(parent_expr) = get_parent_expr(cx, expr) {
// Check if we have the whole call chain here
if let ExprKind::MethodCall(..) = parent_expr.kind {
return;
}
// Check for Expr that we don't want to be linted
let precedence = parent_expr.precedence();
match precedence {
// Lint a Call is ok though
ExprPrecedence::Call | ExprPrecedence::AddrOf => (),
_ => {
if precedence.order() >= PREC_PREFIX && precedence.order() <= PREC_POSTFIX {
return;
}
}
2020-01-26 12:48:30 -06:00
}
}
let name = method_name.ident.as_str();
lint_deref(cx, &*name, &args[0], args[0].span, expr.span);
2018-10-03 11:53:39 -05:00
}
}
}
}
2020-01-26 12:48:30 -06:00
fn lint_deref(cx: &LateContext<'_>, method_name: &str, call_expr: &Expr<'_>, var_span: Span, expr_span: Span) {
match method_name {
2020-01-26 12:48:30 -06:00
"deref" => {
if cx.tcx.lang_items().deref_trait().map_or(false, |id| {
implements_trait(cx, cx.tables().expr_ty(&call_expr), id, &[])
}) {
span_lint_and_sugg(
cx,
EXPLICIT_DEREF_METHODS,
expr_span,
"explicit deref method call",
"try this",
format!("&*{}", &snippet(cx, var_span, "..")),
Applicability::MachineApplicable,
);
}
2020-01-26 12:48:30 -06:00
},
"deref_mut" => {
if cx.tcx.lang_items().deref_mut_trait().map_or(false, |id| {
implements_trait(cx, cx.tables().expr_ty(&call_expr), id, &[])
}) {
span_lint_and_sugg(
cx,
EXPLICIT_DEREF_METHODS,
expr_span,
"explicit deref_mut method call",
"try this",
format!("&mut *{}", &snippet(cx, var_span, "..")),
Applicability::MachineApplicable,
);
}
2020-01-26 12:48:30 -06:00
},
_ => (),
}
}