2021-03-25 13:29:11 -05:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_then;
|
2018-12-29 09:04:45 -06:00
|
|
|
use rustc_errors::Applicability;
|
2020-01-06 10:39:50 -06:00
|
|
|
use rustc_hir::{BindingAnnotation, Mutability, Node, Pat, PatKind};
|
2020-01-12 00:08:41 -06:00
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
2020-01-11 05:37:08 -06:00
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
2017-06-24 05:04:56 -05:00
|
|
|
|
2018-03-28 08:24:26 -05:00
|
|
|
declare_clippy_lint! {
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### What it does
|
2022-10-06 02:44:38 -05:00
|
|
|
/// Checks for bindings that needlessly destructure a reference and borrow the inner
|
2021-03-12 08:30:50 -06:00
|
|
|
/// value with `&ref`.
|
2019-03-05 10:50:33 -06:00
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// This pattern has no effect in almost all cases.
|
2019-03-05 10:50:33 -06:00
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Example
|
2019-03-05 10:50:33 -06:00
|
|
|
/// ```rust
|
|
|
|
/// let mut v = Vec::<String>::new();
|
2022-06-16 10:39:06 -05:00
|
|
|
/// v.iter_mut().filter(|&ref a| a.is_empty());
|
2022-10-06 02:44:38 -05:00
|
|
|
///
|
|
|
|
/// if let &[ref first, ref second] = v.as_slice() {}
|
2019-03-05 10:50:33 -06:00
|
|
|
/// ```
|
2021-03-12 08:30:50 -06:00
|
|
|
///
|
2022-06-16 10:39:06 -05:00
|
|
|
/// Use instead:
|
2021-03-12 08:30:50 -06:00
|
|
|
/// ```rust
|
|
|
|
/// let mut v = Vec::<String>::new();
|
2022-06-16 10:39:06 -05:00
|
|
|
/// v.iter_mut().filter(|a| a.is_empty());
|
2022-10-06 02:44:38 -05:00
|
|
|
///
|
|
|
|
/// if let [first, second] = v.as_slice() {}
|
2021-03-12 08:30:50 -06:00
|
|
|
/// ```
|
2021-12-06 05:33:31 -06:00
|
|
|
#[clippy::version = "pre 1.29.0"]
|
2017-06-24 05:04:56 -05:00
|
|
|
pub NEEDLESS_BORROWED_REFERENCE,
|
2018-03-28 08:24:26 -05:00
|
|
|
complexity,
|
2021-03-12 08:30:50 -06:00
|
|
|
"destructuring a reference and borrowing the inner value"
|
2017-06-24 05:04:56 -05:00
|
|
|
}
|
|
|
|
|
2019-04-08 15:43:55 -05:00
|
|
|
declare_lint_pass!(NeedlessBorrowedRef => [NEEDLESS_BORROWED_REFERENCE]);
|
2017-06-24 05:04:56 -05:00
|
|
|
|
2020-06-25 15:41:36 -05:00
|
|
|
impl<'tcx> LateLintPass<'tcx> for NeedlessBorrowedRef {
|
|
|
|
fn check_pat(&mut self, cx: &LateContext<'tcx>, pat: &'tcx Pat<'_>) {
|
2019-08-19 11:30:32 -05:00
|
|
|
if pat.span.from_expansion() {
|
2017-06-24 05:04:56 -05:00
|
|
|
// OK, simple enough, lints doesn't check in macro.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-10-06 02:44:38 -05:00
|
|
|
// Do not lint patterns that are part of an OR `|` pattern, the binding mode must match in all arms
|
|
|
|
for (_, node) in cx.tcx.hir().parent_iter(pat.hir_id) {
|
|
|
|
let Node::Pat(pat) = node else { break };
|
|
|
|
|
|
|
|
if matches!(pat.kind, PatKind::Or(_)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only lint immutable refs, because `&mut ref T` may be useful.
|
|
|
|
let PatKind::Ref(sub_pat, Mutability::Not) = pat.kind else { return };
|
2017-07-01 05:01:39 -05:00
|
|
|
|
2022-10-06 02:44:38 -05:00
|
|
|
match sub_pat.kind {
|
2017-07-01 09:56:19 -05:00
|
|
|
// Check sub_pat got a `ref` keyword (excluding `ref mut`).
|
2022-10-06 02:44:38 -05:00
|
|
|
PatKind::Binding(BindingAnnotation::REF, _, ident, None) => {
|
|
|
|
span_lint_and_then(
|
|
|
|
cx,
|
|
|
|
NEEDLESS_BORROWED_REFERENCE,
|
|
|
|
pat.span,
|
|
|
|
"this pattern takes a reference on something that is being dereferenced",
|
|
|
|
|diag| {
|
|
|
|
// `&ref ident`
|
|
|
|
// ^^^^^
|
|
|
|
let span = pat.span.until(ident.span);
|
|
|
|
diag.span_suggestion_verbose(
|
|
|
|
span,
|
|
|
|
"try removing the `&ref` part",
|
|
|
|
String::new(),
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
// Slices where each element is `ref`: `&[ref a, ref b, ..., ref z]`
|
|
|
|
PatKind::Slice(
|
|
|
|
before,
|
|
|
|
None
|
|
|
|
| Some(Pat {
|
|
|
|
kind: PatKind::Wild, ..
|
|
|
|
}),
|
|
|
|
after,
|
|
|
|
) => {
|
|
|
|
let mut suggestions = Vec::new();
|
|
|
|
|
|
|
|
for element_pat in itertools::chain(before, after) {
|
|
|
|
if let PatKind::Binding(BindingAnnotation::REF, _, ident, None) = element_pat.kind {
|
|
|
|
// `&[..., ref ident, ...]`
|
|
|
|
// ^^^^
|
|
|
|
let span = element_pat.span.until(ident.span);
|
|
|
|
suggestions.push((span, String::new()));
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-25 07:19:09 -05:00
|
|
|
}
|
2022-10-06 02:44:38 -05:00
|
|
|
|
|
|
|
if !suggestions.is_empty() {
|
|
|
|
span_lint_and_then(
|
|
|
|
cx,
|
|
|
|
NEEDLESS_BORROWED_REFERENCE,
|
|
|
|
pat.span,
|
|
|
|
"dereferencing a slice pattern where every element takes a reference",
|
|
|
|
|diag| {
|
|
|
|
// `&[...]`
|
|
|
|
// ^
|
|
|
|
let span = pat.span.until(sub_pat.span);
|
|
|
|
suggestions.push((span, String::new()));
|
|
|
|
|
|
|
|
diag.multipart_suggestion(
|
|
|
|
"try removing the `&` and `ref` parts",
|
|
|
|
suggestions,
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {},
|
2017-10-23 14:18:02 -05:00
|
|
|
}
|
2017-06-24 05:04:56 -05:00
|
|
|
}
|
|
|
|
}
|