2020-02-25 22:06:24 +00:00
|
|
|
use crate::utils::{
|
|
|
|
get_parent_expr, get_trait_def_id, implements_trait, method_calls, paths, snippet, span_lint_and_sugg,
|
|
|
|
};
|
2020-01-26 18:48:30 +00:00
|
|
|
use if_chain::if_chain;
|
2020-01-23 15:28:01 +00:00
|
|
|
use rustc_errors::Applicability;
|
2020-01-26 18:48:30 +00:00
|
|
|
use rustc_hir as hir;
|
|
|
|
use rustc_hir::{Expr, ExprKind, QPath, StmtKind};
|
2020-01-23 15:28:01 +00:00
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
2020-01-26 18:48:30 +00:00
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
|
|
|
use rustc_span::source_map::Span;
|
2018-10-03 16:53:39 +00:00
|
|
|
|
|
|
|
declare_clippy_lint! {
|
2020-01-23 15:28:01 +00: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
|
2020-02-25 22:06:24 +00:00
|
|
|
/// use std::ops::Deref;
|
|
|
|
/// let a: &mut String = &mut String::from("foo");
|
|
|
|
/// let b: &str = a.deref();
|
2020-01-23 15:28:01 +00:00
|
|
|
/// ```
|
|
|
|
/// Could be written as:
|
|
|
|
/// ```rust
|
2020-02-25 22:06:24 +00:00
|
|
|
/// let a: &mut String = &mut String::from("foo");
|
2020-01-23 15:28:01 +00:00
|
|
|
/// let b = &*a;
|
|
|
|
/// ```
|
2020-01-26 18:48:30 +00:00
|
|
|
///
|
2020-01-23 15:28:01 +00:00
|
|
|
/// This lint excludes
|
2020-02-25 22:06:24 +00:00
|
|
|
/// ```rust,ignore
|
|
|
|
/// let _ = d.unwrap().deref();
|
2020-01-23 15:28:01 +00:00
|
|
|
/// ```
|
2018-10-03 16:53:39 +00:00
|
|
|
pub EXPLICIT_DEREF_METHOD,
|
|
|
|
pedantic,
|
|
|
|
"Explicit use of deref or deref_mut method while not in a method chain."
|
|
|
|
}
|
|
|
|
|
2020-01-23 15:28:01 +00:00
|
|
|
declare_lint_pass!(Dereferencing => [
|
|
|
|
EXPLICIT_DEREF_METHOD
|
|
|
|
]);
|
2018-10-03 16:53:39 +00:00
|
|
|
|
2020-01-23 15:28:01 +00:00
|
|
|
impl<'a, 'tcx> LateLintPass<'a, 'tcx> for Dereferencing {
|
2020-01-26 18:48:30 +00:00
|
|
|
fn check_stmt(&mut self, cx: &LateContext<'a, 'tcx>, stmt: &'tcx hir::Stmt<'_>) {
|
|
|
|
if_chain! {
|
|
|
|
if let StmtKind::Local(ref local) = stmt.kind;
|
|
|
|
if let Some(ref init) = local.init;
|
|
|
|
|
|
|
|
then {
|
|
|
|
match init.kind {
|
|
|
|
ExprKind::Call(ref _method, args) => {
|
|
|
|
for arg in args {
|
|
|
|
if_chain! {
|
|
|
|
// Caller must call only one other function (deref or deref_mut)
|
2020-02-25 22:06:24 +00:00
|
|
|
// otherwise it can lead to error prone suggestions (ie: &*a.len())
|
2020-01-26 18:48:30 +00:00
|
|
|
let (method_names, arg_list, _) = method_calls(arg, 2);
|
|
|
|
if method_names.len() == 1;
|
|
|
|
// Caller must be a variable
|
|
|
|
let variables = arg_list[0];
|
|
|
|
if variables.len() == 1;
|
|
|
|
if let ExprKind::Path(QPath::Resolved(None, _)) = variables[0].kind;
|
|
|
|
|
|
|
|
then {
|
|
|
|
let name = method_names[0].as_str();
|
2020-02-25 22:06:24 +00:00
|
|
|
lint_deref(cx, &*name, &variables[0], variables[0].span, arg.span);
|
2020-01-26 18:48:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ExprKind::MethodCall(ref method_name, _, ref args) => {
|
|
|
|
if init.span.from_expansion() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if_chain! {
|
|
|
|
if args.len() == 1;
|
|
|
|
if let ExprKind::Path(QPath::Resolved(None, _)) = args[0].kind;
|
|
|
|
// Caller must call only one other function (deref or deref_mut)
|
2020-02-25 22:06:24 +00:00
|
|
|
// otherwise it can lead to error prone suggestions (ie: &*a.len())
|
2020-01-26 18:48:30 +00:00
|
|
|
let (method_names, arg_list, _) = method_calls(init, 2);
|
|
|
|
if method_names.len() == 1;
|
|
|
|
// Caller must be a variable
|
|
|
|
let variables = arg_list[0];
|
|
|
|
if variables.len() == 1;
|
|
|
|
if let ExprKind::Path(QPath::Resolved(None, _)) = variables[0].kind;
|
|
|
|
|
|
|
|
then {
|
|
|
|
let name = method_name.ident.as_str();
|
2020-02-25 22:06:24 +00:00
|
|
|
lint_deref(cx, &*name, init, args[0].span, init.span);
|
2020-01-26 18:48:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => ()
|
|
|
|
}
|
|
|
|
}
|
2018-10-03 16:53:39 +00:00
|
|
|
}
|
2020-01-26 18:48:30 +00:00
|
|
|
}
|
2018-10-03 16:53:39 +00:00
|
|
|
|
2020-01-26 18:48:30 +00:00
|
|
|
fn check_expr(&mut self, cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr<'_>) {
|
2018-10-03 16:53:39 +00:00
|
|
|
if_chain! {
|
2020-01-23 15:28:01 +00:00
|
|
|
if let ExprKind::MethodCall(ref method_name, _, ref args) = &expr.kind;
|
2020-01-26 18:48:30 +00:00
|
|
|
if args.len() == 1;
|
2020-02-25 22:06:24 +00:00
|
|
|
if let ExprKind::Path(QPath::Resolved(None, _)) = args[0].kind;
|
2020-01-26 18:48:30 +00:00
|
|
|
if let Some(parent) = get_parent_expr(cx, &expr);
|
|
|
|
|
2018-10-03 16:53:39 +00:00
|
|
|
then {
|
2020-01-26 18:48:30 +00:00
|
|
|
match parent.kind {
|
2020-02-25 22:06:24 +00:00
|
|
|
// Already linted using statements
|
|
|
|
ExprKind::Call(_, _) | ExprKind::MethodCall(_, _, _) => (),
|
2020-01-26 18:48:30 +00:00
|
|
|
_ => {
|
|
|
|
let name = method_name.ident.as_str();
|
2020-02-25 22:06:24 +00:00
|
|
|
lint_deref(cx, &*name, &args[0], args[0].span, expr.span);
|
2020-01-26 18:48:30 +00:00
|
|
|
}
|
|
|
|
}
|
2018-10-03 16:53:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-26 18:48:30 +00:00
|
|
|
|
2020-02-25 22:06:24 +00:00
|
|
|
fn lint_deref(cx: &LateContext<'_, '_>, fn_name: &str, call_expr: &Expr<'_>, var_span: Span, expr_span: Span) {
|
2020-01-26 18:48:30 +00:00
|
|
|
match fn_name {
|
|
|
|
"deref" => {
|
2020-02-25 22:06:24 +00:00
|
|
|
if get_trait_def_id(cx, &paths::DEREF_TRAIT)
|
|
|
|
.map_or(false, |id| implements_trait(cx, cx.tables.expr_ty(&call_expr), id, &[]))
|
|
|
|
{
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
EXPLICIT_DEREF_METHOD,
|
|
|
|
expr_span,
|
|
|
|
"explicit deref method call",
|
|
|
|
"try this",
|
|
|
|
format!("&*{}", &snippet(cx, var_span, "..")),
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
}
|
2020-01-26 18:48:30 +00:00
|
|
|
},
|
|
|
|
"deref_mut" => {
|
2020-02-25 22:06:24 +00:00
|
|
|
if get_trait_def_id(cx, &paths::DEREF_MUT_TRAIT)
|
|
|
|
.map_or(false, |id| implements_trait(cx, cx.tables.expr_ty(&call_expr), id, &[]))
|
|
|
|
{
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
EXPLICIT_DEREF_METHOD,
|
|
|
|
expr_span,
|
|
|
|
"explicit deref_mut method call",
|
|
|
|
"try this",
|
|
|
|
format!("&mut *{}", &snippet(cx, var_span, "..")),
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
}
|
2020-01-26 18:48:30 +00:00
|
|
|
},
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
}
|