114 lines
3.9 KiB
Rust
114 lines
3.9 KiB
Rust
use crate::utils::{get_parent_expr, implements_trait, snippet, span_lint_and_sugg};
|
|
use if_chain::if_chain;
|
|
use rustc_ast::util::parser::{ExprPrecedence, PREC_POSTFIX, PREC_PREFIX};
|
|
use rustc_errors::Applicability;
|
|
use rustc_hir::{Expr, ExprKind};
|
|
use rustc_lint::{LateContext, LateLintPass};
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
|
use rustc_span::source_map::Span;
|
|
|
|
declare_clippy_lint! {
|
|
/// **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();
|
|
/// ```
|
|
/// Could be written as:
|
|
/// ```rust
|
|
/// let a: &mut String = &mut String::from("foo");
|
|
/// let b = &*a;
|
|
/// ```
|
|
///
|
|
/// This lint excludes
|
|
/// ```rust,ignore
|
|
/// let _ = d.unwrap().deref();
|
|
/// ```
|
|
pub EXPLICIT_DEREF_METHODS,
|
|
pedantic,
|
|
"Explicit use of deref or deref_mut method while not in a method chain."
|
|
}
|
|
|
|
declare_lint_pass!(Dereferencing => [
|
|
EXPLICIT_DEREF_METHODS
|
|
]);
|
|
|
|
impl<'a, 'tcx> LateLintPass<'a, 'tcx> for Dereferencing {
|
|
fn check_expr(&mut self, cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr<'_>) {
|
|
if_chain! {
|
|
if !expr.span.from_expansion();
|
|
if let ExprKind::MethodCall(ref method_name, _, ref args) = &expr.kind;
|
|
if args.len() == 1;
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
let name = method_name.ident.as_str();
|
|
lint_deref(cx, &*name, &args[0], args[0].span, expr.span);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn lint_deref(cx: &LateContext<'_, '_>, method_name: &str, call_expr: &Expr<'_>, var_span: Span, expr_span: Span) {
|
|
match method_name {
|
|
"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,
|
|
);
|
|
}
|
|
},
|
|
"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,
|
|
);
|
|
}
|
|
},
|
|
_ => (),
|
|
}
|
|
}
|