2019-11-16 08:55:00 -06:00
|
|
|
use if_chain::if_chain;
|
2020-01-23 09:20:50 -06:00
|
|
|
use rustc::lint::in_external_macro;
|
2020-01-06 10:39:50 -06:00
|
|
|
use rustc_hir::*;
|
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};
|
2019-11-16 08:55:00 -06:00
|
|
|
|
2020-01-26 19:56:22 -06:00
|
|
|
use crate::utils::{is_must_use_func_call, is_must_use_ty, span_lint_and_help};
|
2019-11-16 08:55:00 -06:00
|
|
|
|
|
|
|
declare_clippy_lint! {
|
|
|
|
/// **What it does:** Checks for `let _ = <expr>`
|
|
|
|
/// where expr is #[must_use]
|
|
|
|
///
|
|
|
|
/// **Why is this bad?** It's better to explicitly
|
|
|
|
/// handle the value of a #[must_use] expr
|
|
|
|
///
|
|
|
|
/// **Known problems:** None.
|
|
|
|
///
|
|
|
|
/// **Example:**
|
|
|
|
/// ```rust
|
|
|
|
/// fn f() -> Result<u32, u32> {
|
|
|
|
/// Ok(0)
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// let _ = f();
|
|
|
|
/// // is_ok() is marked #[must_use]
|
|
|
|
/// let _ = f().is_ok();
|
|
|
|
/// ```
|
|
|
|
pub LET_UNDERSCORE_MUST_USE,
|
|
|
|
restriction,
|
2020-01-06 00:30:43 -06:00
|
|
|
"non-binding let on a `#[must_use]` expression"
|
2019-11-16 08:55:00 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
declare_lint_pass!(LetUnderscore => [LET_UNDERSCORE_MUST_USE]);
|
|
|
|
|
|
|
|
impl<'a, 'tcx> LateLintPass<'a, 'tcx> for LetUnderscore {
|
2019-12-27 01:12:26 -06:00
|
|
|
fn check_stmt(&mut self, cx: &LateContext<'_, '_>, stmt: &Stmt<'_>) {
|
2020-01-23 09:20:50 -06:00
|
|
|
if in_external_macro(cx.tcx.sess, stmt.span) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-16 08:55:00 -06:00
|
|
|
if_chain! {
|
|
|
|
if let StmtKind::Local(ref local) = stmt.kind;
|
|
|
|
if let PatKind::Wild = local.pat.kind;
|
|
|
|
if let Some(ref init) = local.init;
|
|
|
|
then {
|
|
|
|
if is_must_use_ty(cx, cx.tables.expr_ty(init)) {
|
2020-01-26 19:56:22 -06:00
|
|
|
span_lint_and_help(
|
2019-11-16 08:55:00 -06:00
|
|
|
cx,
|
|
|
|
LET_UNDERSCORE_MUST_USE,
|
|
|
|
stmt.span,
|
2020-01-06 00:30:43 -06:00
|
|
|
"non-binding let on an expression with `#[must_use]` type",
|
2019-11-16 08:55:00 -06:00
|
|
|
"consider explicitly using expression value"
|
|
|
|
)
|
|
|
|
} else if is_must_use_func_call(cx, init) {
|
2020-01-26 19:56:22 -06:00
|
|
|
span_lint_and_help(
|
2019-11-16 08:55:00 -06:00
|
|
|
cx,
|
|
|
|
LET_UNDERSCORE_MUST_USE,
|
|
|
|
stmt.span,
|
2020-01-06 00:30:43 -06:00
|
|
|
"non-binding let on a result of a `#[must_use]` function",
|
2019-11-16 08:55:00 -06:00
|
|
|
"consider explicitly using function result"
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|