2024-03-09 10:09:17 -06:00
|
|
|
use clippy_utils::diagnostics::{span_lint_and_sugg, span_lint_hir_and_then};
|
2022-06-04 06:34:07 -05:00
|
|
|
use clippy_utils::source::{snippet_opt, snippet_with_context};
|
2023-11-02 11:35:56 -05:00
|
|
|
use clippy_utils::sugg::has_enclosing_paren;
|
2023-07-17 03:19:29 -05:00
|
|
|
use clippy_utils::visitors::{for_each_expr_with_closures, Descend};
|
2024-01-28 15:43:40 -06:00
|
|
|
use clippy_utils::{
|
|
|
|
fn_def_id, is_from_proc_macro, is_inside_let_else, is_res_lang_ctor, path_res, path_to_local_id,
|
|
|
|
span_find_starting_semi,
|
|
|
|
};
|
2022-10-06 02:44:38 -05:00
|
|
|
use core::ops::ControlFlow;
|
2018-12-29 09:04:45 -06:00
|
|
|
use rustc_errors::Applicability;
|
2022-10-06 02:44:38 -05:00
|
|
|
use rustc_hir::intravisit::FnKind;
|
2024-01-28 15:43:40 -06:00
|
|
|
use rustc_hir::LangItem::ResultErr;
|
2023-07-31 16:53:53 -05:00
|
|
|
use rustc_hir::{
|
2023-12-01 11:21:58 -06:00
|
|
|
Block, Body, Expr, ExprKind, FnDecl, HirId, ItemKind, LangItem, MatchSource, Node, OwnerNode, PatKind, QPath, Stmt,
|
|
|
|
StmtKind,
|
2023-07-31 16:53:53 -05:00
|
|
|
};
|
2020-08-28 09:10:16 -05:00
|
|
|
use rustc_lint::{LateContext, LateLintPass, LintContext};
|
2020-03-30 04:02:14 -05:00
|
|
|
use rustc_middle::lint::in_external_macro;
|
2023-12-01 11:21:58 -06:00
|
|
|
use rustc_middle::ty::adjustment::Adjust;
|
2023-07-11 16:35:29 -05:00
|
|
|
use rustc_middle::ty::{self, GenericArgKind, Ty};
|
2023-12-01 11:21:58 -06:00
|
|
|
use rustc_session::declare_lint_pass;
|
2023-01-22 12:00:33 -06:00
|
|
|
use rustc_span::def_id::LocalDefId;
|
2023-11-01 22:10:12 -05:00
|
|
|
use rustc_span::{BytePos, Pos, Span};
|
2023-02-25 18:08:29 -06:00
|
|
|
use std::borrow::Cow;
|
2024-01-10 17:52:20 -06:00
|
|
|
use std::fmt::Display;
|
2015-08-11 14:47:34 -05:00
|
|
|
|
2018-03-28 08:24:26 -05:00
|
|
|
declare_clippy_lint! {
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for `let`-bindings, which are subsequently
|
2020-08-28 09:10:16 -05:00
|
|
|
/// returned.
|
2019-03-05 10:50:33 -06:00
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// It is just extraneous code. Remove it to make your code
|
2019-03-05 10:50:33 -06:00
|
|
|
/// more rusty.
|
|
|
|
///
|
2023-07-02 07:35:19 -05:00
|
|
|
/// ### Known problems
|
|
|
|
/// In the case of some temporaries, e.g. locks, eliding the variable binding could lead
|
|
|
|
/// to deadlocks. See [this issue](https://github.com/rust-lang/rust/issues/37612).
|
|
|
|
/// This could become relevant if the code is later changed to use the code that would have been
|
|
|
|
/// bound without first assigning it to a let-binding.
|
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Example
|
2023-11-02 11:35:56 -05:00
|
|
|
/// ```no_run
|
2020-08-28 09:10:16 -05:00
|
|
|
/// fn foo() -> String {
|
|
|
|
/// let x = String::new();
|
|
|
|
/// x
|
2019-03-05 10:50:33 -06:00
|
|
|
/// }
|
|
|
|
/// ```
|
2020-08-28 09:10:16 -05:00
|
|
|
/// instead, use
|
2023-11-02 11:35:56 -05:00
|
|
|
/// ```no_run
|
2020-08-28 09:10:16 -05:00
|
|
|
/// fn foo() -> String {
|
|
|
|
/// String::new()
|
2019-03-05 10:50:33 -06:00
|
|
|
/// }
|
|
|
|
/// ```
|
2021-12-06 05:33:31 -06:00
|
|
|
#[clippy::version = "pre 1.29.0"]
|
2020-08-28 09:10:16 -05:00
|
|
|
pub LET_AND_RETURN,
|
2018-03-28 08:24:26 -05:00
|
|
|
style,
|
2020-08-28 09:10:16 -05:00
|
|
|
"creating a let-binding and then immediately returning it like `let x = expr; x` at the end of a block"
|
2016-02-05 17:13:29 -06:00
|
|
|
}
|
|
|
|
|
2018-09-27 12:10:20 -05:00
|
|
|
declare_clippy_lint! {
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for return statements at the end of a block.
|
2019-03-05 10:50:33 -06:00
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// Removing the `return` and semicolon will make the code
|
2020-08-28 09:10:16 -05:00
|
|
|
/// more rusty.
|
2019-03-05 10:50:33 -06:00
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Example
|
2023-11-02 11:35:56 -05:00
|
|
|
/// ```no_run
|
2020-08-28 09:10:16 -05:00
|
|
|
/// fn foo(x: usize) -> usize {
|
|
|
|
/// return x;
|
2019-03-05 10:50:33 -06:00
|
|
|
/// }
|
|
|
|
/// ```
|
2020-08-28 09:10:16 -05:00
|
|
|
/// simplify to
|
2023-11-02 11:35:56 -05:00
|
|
|
/// ```no_run
|
2020-08-28 09:10:16 -05:00
|
|
|
/// fn foo(x: usize) -> usize {
|
|
|
|
/// x
|
|
|
|
/// }
|
|
|
|
/// ```
|
2021-12-06 05:33:31 -06:00
|
|
|
#[clippy::version = "pre 1.29.0"]
|
2020-08-28 09:10:16 -05:00
|
|
|
pub NEEDLESS_RETURN,
|
2018-09-27 12:10:20 -05:00
|
|
|
style,
|
2020-08-28 09:10:16 -05:00
|
|
|
"using a return statement like `return expr;` where an expression would suffice"
|
2018-09-27 12:10:20 -05:00
|
|
|
}
|
|
|
|
|
2023-07-31 16:53:53 -05:00
|
|
|
declare_clippy_lint! {
|
|
|
|
/// ### What it does
|
|
|
|
/// Checks for return statements on `Err` paired with the `?` operator.
|
|
|
|
///
|
|
|
|
/// ### Why is this bad?
|
|
|
|
/// The `return` is unnecessary.
|
|
|
|
///
|
|
|
|
/// ### Example
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// fn foo(x: usize) -> Result<(), Box<dyn Error>> {
|
|
|
|
/// if x == 0 {
|
|
|
|
/// return Err(...)?;
|
|
|
|
/// }
|
|
|
|
/// Ok(())
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
/// simplify to
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// fn foo(x: usize) -> Result<(), Box<dyn Error>> {
|
|
|
|
/// if x == 0 {
|
|
|
|
/// Err(...)?;
|
|
|
|
/// }
|
|
|
|
/// Ok(())
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
/// if paired with `try_err`, use instead:
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// fn foo(x: usize) -> Result<(), Box<dyn Error>> {
|
|
|
|
/// if x == 0 {
|
|
|
|
/// return Err(...);
|
|
|
|
/// }
|
|
|
|
/// Ok(())
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
#[clippy::version = "1.73.0"]
|
|
|
|
pub NEEDLESS_RETURN_WITH_QUESTION_MARK,
|
|
|
|
style,
|
|
|
|
"using a return statement like `return Err(expr)?;` where removing it would suffice"
|
|
|
|
}
|
|
|
|
|
2023-06-02 04:41:57 -05:00
|
|
|
#[derive(PartialEq, Eq)]
|
2023-02-25 18:08:29 -06:00
|
|
|
enum RetReplacement<'tcx> {
|
2019-06-19 07:56:02 -05:00
|
|
|
Empty,
|
2019-07-08 13:45:51 -05:00
|
|
|
Block,
|
2021-12-30 08:10:43 -06:00
|
|
|
Unit,
|
2023-02-25 18:08:29 -06:00
|
|
|
IfSequence(Cow<'tcx, str>, Applicability),
|
|
|
|
Expr(Cow<'tcx, str>, Applicability),
|
2019-06-19 07:56:02 -05:00
|
|
|
}
|
|
|
|
|
2023-02-25 18:08:29 -06:00
|
|
|
impl<'tcx> RetReplacement<'tcx> {
|
2023-06-02 04:41:57 -05:00
|
|
|
fn sugg_help(&self) -> &'static str {
|
2022-10-06 02:44:38 -05:00
|
|
|
match self {
|
2023-02-25 18:08:29 -06:00
|
|
|
Self::Empty | Self::Expr(..) => "remove `return`",
|
2022-10-06 02:44:38 -05:00
|
|
|
Self::Block => "replace `return` with an empty block",
|
|
|
|
Self::Unit => "replace `return` with a unit value",
|
2023-02-25 18:08:29 -06:00
|
|
|
Self::IfSequence(..) => "remove `return` and wrap the sequence with parentheses",
|
|
|
|
}
|
|
|
|
}
|
2023-06-02 04:41:57 -05:00
|
|
|
|
|
|
|
fn applicability(&self) -> Applicability {
|
2023-02-25 18:08:29 -06:00
|
|
|
match self {
|
2023-06-02 04:41:57 -05:00
|
|
|
Self::Expr(_, ap) | Self::IfSequence(_, ap) => *ap,
|
|
|
|
_ => Applicability::MachineApplicable,
|
2022-10-06 02:44:38 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-10 17:52:20 -06:00
|
|
|
impl<'tcx> Display for RetReplacement<'tcx> {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
2023-02-25 18:08:29 -06:00
|
|
|
match self {
|
2024-01-10 17:52:20 -06:00
|
|
|
Self::Empty => write!(f, ""),
|
|
|
|
Self::Block => write!(f, "{{}}"),
|
|
|
|
Self::Unit => write!(f, "()"),
|
|
|
|
Self::IfSequence(inner, _) => write!(f, "({inner})"),
|
|
|
|
Self::Expr(inner, _) => write!(f, "{inner}"),
|
2022-10-06 02:44:38 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-31 16:53:53 -05:00
|
|
|
declare_lint_pass!(Return => [LET_AND_RETURN, NEEDLESS_RETURN, NEEDLESS_RETURN_WITH_QUESTION_MARK]);
|
2015-08-11 14:47:34 -05:00
|
|
|
|
2023-12-01 11:21:58 -06:00
|
|
|
/// Checks if a return statement is "needed" in the middle of a block, or if it can be removed. This
|
|
|
|
/// is the case when the enclosing block expression is coerced to some other type, which only works
|
|
|
|
/// because of the never-ness of `return` expressions
|
|
|
|
fn stmt_needs_never_type(cx: &LateContext<'_>, stmt_hir_id: HirId) -> bool {
|
|
|
|
cx.tcx
|
|
|
|
.hir()
|
|
|
|
.parent_iter(stmt_hir_id)
|
|
|
|
.find_map(|(_, node)| if let Node::Expr(expr) = node { Some(expr) } else { None })
|
|
|
|
.is_some_and(|e| {
|
|
|
|
cx.typeck_results()
|
|
|
|
.expr_adjustments(e)
|
|
|
|
.iter()
|
|
|
|
.any(|adjust| adjust.target != cx.tcx.types.unit && matches!(adjust.kind, Adjust::NeverToAny))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-08-28 09:10:16 -05:00
|
|
|
impl<'tcx> LateLintPass<'tcx> for Return {
|
2023-07-31 16:53:53 -05:00
|
|
|
fn check_stmt(&mut self, cx: &LateContext<'tcx>, stmt: &'tcx Stmt<'_>) {
|
|
|
|
if !in_external_macro(cx.sess(), stmt.span)
|
|
|
|
&& let StmtKind::Semi(expr) = stmt.kind
|
|
|
|
&& let ExprKind::Ret(Some(ret)) = expr.kind
|
2024-01-28 15:43:40 -06:00
|
|
|
// return Err(...)? desugars to a match
|
|
|
|
// over a Err(...).branch()
|
|
|
|
// which breaks down to a branch call, with the callee being
|
|
|
|
// the constructor of the Err variant
|
|
|
|
&& let ExprKind::Match(maybe_cons, _, MatchSource::TryDesugar(_)) = ret.kind
|
|
|
|
&& let ExprKind::Call(_, [maybe_result_err]) = maybe_cons.kind
|
|
|
|
&& let ExprKind::Call(maybe_constr, _) = maybe_result_err.kind
|
|
|
|
&& is_res_lang_ctor(cx, path_res(cx, maybe_constr), ResultErr)
|
|
|
|
|
2023-07-31 16:53:53 -05:00
|
|
|
// Ensure this is not the final stmt, otherwise removing it would cause a compile error
|
2024-01-25 11:47:03 -06:00
|
|
|
&& let OwnerNode::Item(item) = cx.tcx.hir_owner_node(cx.tcx.hir().get_parent_item(expr.hir_id))
|
2023-07-31 16:53:53 -05:00
|
|
|
&& let ItemKind::Fn(_, _, body) = item.kind
|
|
|
|
&& let block = cx.tcx.hir().body(body).value
|
|
|
|
&& let ExprKind::Block(block, _) = block.kind
|
2023-11-16 12:13:24 -06:00
|
|
|
&& !is_inside_let_else(cx.tcx, expr)
|
2023-07-31 16:53:53 -05:00
|
|
|
&& let [.., final_stmt] = block.stmts
|
|
|
|
&& final_stmt.hir_id != stmt.hir_id
|
|
|
|
&& !is_from_proc_macro(cx, expr)
|
2023-12-01 11:21:58 -06:00
|
|
|
&& !stmt_needs_never_type(cx, stmt.hir_id)
|
2023-07-31 16:53:53 -05:00
|
|
|
{
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
NEEDLESS_RETURN_WITH_QUESTION_MARK,
|
|
|
|
expr.span.until(ret.span),
|
|
|
|
"unneeded `return` statement with `?` operator",
|
|
|
|
"remove it",
|
|
|
|
String::new(),
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-28 09:10:16 -05:00
|
|
|
fn check_block(&mut self, cx: &LateContext<'tcx>, block: &'tcx Block<'_>) {
|
|
|
|
// we need both a let-binding stmt and an expr
|
2023-11-16 12:13:24 -06:00
|
|
|
if let Some(retexpr) = block.expr
|
|
|
|
&& let Some(stmt) = block.stmts.iter().last()
|
|
|
|
&& let StmtKind::Local(local) = &stmt.kind
|
|
|
|
&& local.ty.is_none()
|
|
|
|
&& cx.tcx.hir().attrs(local.hir_id).is_empty()
|
|
|
|
&& let Some(initexpr) = &local.init
|
|
|
|
&& let PatKind::Binding(_, local_id, _, _) = local.pat.kind
|
|
|
|
&& path_to_local_id(retexpr, local_id)
|
|
|
|
&& !last_statement_borrows(cx, initexpr)
|
|
|
|
&& !in_external_macro(cx.sess(), initexpr.span)
|
|
|
|
&& !in_external_macro(cx.sess(), retexpr.span)
|
|
|
|
&& !local.span.from_expansion()
|
|
|
|
{
|
|
|
|
span_lint_hir_and_then(
|
|
|
|
cx,
|
|
|
|
LET_AND_RETURN,
|
|
|
|
retexpr.hir_id,
|
|
|
|
retexpr.span,
|
|
|
|
"returning the result of a `let` binding from a block",
|
|
|
|
|err| {
|
|
|
|
err.span_label(local.span, "unnecessary `let` binding");
|
2020-08-28 09:10:16 -05:00
|
|
|
|
2023-11-16 12:13:24 -06:00
|
|
|
if let Some(mut snippet) = snippet_opt(cx, initexpr.span) {
|
|
|
|
if !cx.typeck_results().expr_adjustments(retexpr).is_empty() {
|
|
|
|
if !has_enclosing_paren(&snippet) {
|
|
|
|
snippet = format!("({snippet})");
|
2020-08-28 09:10:16 -05:00
|
|
|
}
|
2023-11-16 12:13:24 -06:00
|
|
|
snippet.push_str(" as _");
|
2020-08-28 09:10:16 -05:00
|
|
|
}
|
2023-11-16 12:13:24 -06:00
|
|
|
err.multipart_suggestion(
|
|
|
|
"return the expression directly",
|
|
|
|
vec![(local.span, String::new()), (retexpr.span, snippet)],
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
err.span_help(initexpr.span, "this expression can be directly returned");
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2015-08-11 14:47:34 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-28 09:10:16 -05:00
|
|
|
fn check_fn(
|
2019-06-21 07:46:34 -05:00
|
|
|
&mut self,
|
2020-08-28 09:10:16 -05:00
|
|
|
cx: &LateContext<'tcx>,
|
|
|
|
kind: FnKind<'tcx>,
|
|
|
|
_: &'tcx FnDecl<'tcx>,
|
|
|
|
body: &'tcx Body<'tcx>,
|
2023-01-27 14:09:08 -06:00
|
|
|
sp: Span,
|
2023-01-22 12:00:33 -06:00
|
|
|
_: LocalDefId,
|
2019-06-21 07:46:34 -05:00
|
|
|
) {
|
2023-07-31 16:53:53 -05:00
|
|
|
if sp.from_expansion() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-08-28 09:10:16 -05:00
|
|
|
match kind {
|
2020-11-27 02:24:42 -06:00
|
|
|
FnKind::Closure => {
|
2021-01-30 11:06:34 -06:00
|
|
|
// when returning without value in closure, replace this `return`
|
|
|
|
// with an empty block to prevent invalid suggestion (see #6501)
|
|
|
|
let replacement = if let ExprKind::Ret(None) = &body.value.kind {
|
|
|
|
RetReplacement::Block
|
|
|
|
} else {
|
|
|
|
RetReplacement::Empty
|
|
|
|
};
|
2023-04-11 08:31:08 -05:00
|
|
|
check_final_expr(cx, body.value, vec![], replacement, None);
|
2021-01-30 11:06:34 -06:00
|
|
|
},
|
2020-08-28 09:10:16 -05:00
|
|
|
FnKind::ItemFn(..) | FnKind::Method(..) => {
|
2023-01-27 14:09:08 -06:00
|
|
|
check_block_return(cx, &body.value.kind, sp, vec![]);
|
2016-12-20 11:21:30 -06:00
|
|
|
},
|
2015-08-11 14:47:34 -05:00
|
|
|
}
|
|
|
|
}
|
2020-08-28 09:10:16 -05:00
|
|
|
}
|
2015-08-11 14:47:34 -05:00
|
|
|
|
2022-10-06 02:44:38 -05:00
|
|
|
// if `expr` is a block, check if there are needless returns in it
|
2023-01-27 14:09:08 -06:00
|
|
|
fn check_block_return<'tcx>(cx: &LateContext<'tcx>, expr_kind: &ExprKind<'tcx>, sp: Span, mut semi_spans: Vec<Span>) {
|
2022-10-06 02:44:38 -05:00
|
|
|
if let ExprKind::Block(block, _) = expr_kind {
|
|
|
|
if let Some(block_expr) = block.expr {
|
2023-04-11 08:31:08 -05:00
|
|
|
check_final_expr(cx, block_expr, semi_spans, RetReplacement::Empty, None);
|
2022-10-06 02:44:38 -05:00
|
|
|
} else if let Some(stmt) = block.stmts.iter().last() {
|
|
|
|
match stmt.kind {
|
|
|
|
StmtKind::Expr(expr) => {
|
2023-04-11 08:31:08 -05:00
|
|
|
check_final_expr(cx, expr, semi_spans, RetReplacement::Empty, None);
|
2022-10-06 02:44:38 -05:00
|
|
|
},
|
|
|
|
StmtKind::Semi(semi_expr) => {
|
2023-01-27 14:09:08 -06:00
|
|
|
// Remove ending semicolons and any whitespace ' ' in between.
|
|
|
|
// Without `return`, the suggestion might not compile if the semicolon is retained
|
|
|
|
if let Some(semi_span) = stmt.span.trim_start(semi_expr.span) {
|
|
|
|
let semi_span_to_remove =
|
|
|
|
span_find_starting_semi(cx.sess().source_map(), semi_span.with_hi(sp.hi()));
|
|
|
|
semi_spans.push(semi_span_to_remove);
|
2022-10-06 02:44:38 -05:00
|
|
|
}
|
2023-04-11 08:31:08 -05:00
|
|
|
check_final_expr(cx, semi_expr, semi_spans, RetReplacement::Empty, None);
|
2022-10-06 02:44:38 -05:00
|
|
|
},
|
|
|
|
_ => (),
|
|
|
|
}
|
2020-08-28 09:10:16 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_final_expr<'tcx>(
|
|
|
|
cx: &LateContext<'tcx>,
|
|
|
|
expr: &'tcx Expr<'tcx>,
|
2022-10-06 02:44:38 -05:00
|
|
|
semi_spans: Vec<Span>, /* containing all the places where we would need to remove semicolons if finding an
|
|
|
|
* needless return */
|
2023-02-25 18:08:29 -06:00
|
|
|
replacement: RetReplacement<'tcx>,
|
2023-04-11 08:31:08 -05:00
|
|
|
match_ty_opt: Option<Ty<'_>>,
|
2020-08-28 09:10:16 -05:00
|
|
|
) {
|
2022-10-06 02:44:38 -05:00
|
|
|
let peeled_drop_expr = expr.peel_drop_temps();
|
|
|
|
match &peeled_drop_expr.kind {
|
2020-08-28 09:10:16 -05:00
|
|
|
// simple return is always "bad"
|
|
|
|
ExprKind::Ret(ref inner) => {
|
2023-02-25 18:08:29 -06:00
|
|
|
// check if expr return nothing
|
|
|
|
let ret_span = if inner.is_none() && replacement == RetReplacement::Empty {
|
|
|
|
extend_span_to_previous_non_ws(cx, peeled_drop_expr.span)
|
|
|
|
} else {
|
|
|
|
peeled_drop_expr.span
|
|
|
|
};
|
|
|
|
|
|
|
|
let replacement = if let Some(inner_expr) = inner {
|
|
|
|
// if desugar of `do yeet`, don't lint
|
|
|
|
if let ExprKind::Call(path_expr, _) = inner_expr.kind
|
2023-11-23 00:17:43 -06:00
|
|
|
&& let ExprKind::Path(QPath::LangItem(LangItem::TryTraitFromYeet, ..)) = path_expr.kind
|
2023-02-25 18:08:29 -06:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
|
|
|
let (snippet, _) = snippet_with_context(cx, inner_expr.span, ret_span.ctxt(), "..", &mut applicability);
|
|
|
|
if expr_contains_conjunctive_ifs(inner_expr) {
|
|
|
|
RetReplacement::IfSequence(snippet, applicability)
|
|
|
|
} else {
|
|
|
|
RetReplacement::Expr(snippet, applicability)
|
|
|
|
}
|
|
|
|
} else {
|
2023-04-11 08:31:08 -05:00
|
|
|
match match_ty_opt {
|
|
|
|
Some(match_ty) => {
|
|
|
|
match match_ty.kind() {
|
|
|
|
// If the code got till here with
|
|
|
|
// tuple not getting detected before it,
|
|
|
|
// then we are sure it's going to be Unit
|
|
|
|
// type
|
|
|
|
ty::Tuple(_) => RetReplacement::Unit,
|
|
|
|
// We don't want to anything in this case
|
|
|
|
// cause we can't predict what the user would
|
|
|
|
// want here
|
|
|
|
_ => return,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => replacement,
|
|
|
|
}
|
2023-02-25 18:08:29 -06:00
|
|
|
};
|
|
|
|
|
2023-01-12 12:48:13 -06:00
|
|
|
if !cx.tcx.hir().attrs(expr.hir_id).is_empty() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let borrows = inner.map_or(false, |inner| last_statement_borrows(cx, inner));
|
|
|
|
if borrows {
|
|
|
|
return;
|
2020-08-28 09:10:16 -05:00
|
|
|
}
|
2023-01-12 12:48:13 -06:00
|
|
|
|
2024-03-09 10:09:17 -06:00
|
|
|
emit_return_lint(cx, ret_span, semi_spans, &replacement, expr.hir_id);
|
2020-08-28 09:10:16 -05:00
|
|
|
},
|
2021-01-01 12:38:11 -06:00
|
|
|
ExprKind::If(_, then, else_clause_opt) => {
|
2023-01-27 14:09:08 -06:00
|
|
|
check_block_return(cx, &then.kind, peeled_drop_expr.span, semi_spans.clone());
|
2021-01-01 12:38:11 -06:00
|
|
|
if let Some(else_clause) = else_clause_opt {
|
2023-01-27 14:09:08 -06:00
|
|
|
check_block_return(cx, &else_clause.kind, peeled_drop_expr.span, semi_spans);
|
2021-01-01 12:38:11 -06:00
|
|
|
}
|
|
|
|
},
|
2020-08-28 09:10:16 -05:00
|
|
|
// a match expr, check all arms
|
|
|
|
// an if/if let expr, check both exprs
|
|
|
|
// note, if without else is going to be a type checking error anyways
|
|
|
|
// (except for unit type functions) so we don't match it
|
2021-09-08 09:31:47 -05:00
|
|
|
ExprKind::Match(_, arms, MatchSource::Normal) => {
|
2023-04-11 08:31:08 -05:00
|
|
|
let match_ty = cx.typeck_results().expr_ty(peeled_drop_expr);
|
2023-07-02 07:35:19 -05:00
|
|
|
for arm in *arms {
|
2023-04-11 08:31:08 -05:00
|
|
|
check_final_expr(cx, arm.body, semi_spans.clone(), RetReplacement::Unit, Some(match_ty));
|
2021-09-08 09:31:47 -05:00
|
|
|
}
|
2020-08-28 09:10:16 -05:00
|
|
|
},
|
2022-10-06 02:44:38 -05:00
|
|
|
// if it's a whole block, check it
|
2023-01-27 14:09:08 -06:00
|
|
|
other_expr_kind => check_block_return(cx, other_expr_kind, peeled_drop_expr.span, semi_spans),
|
2015-08-11 14:47:34 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-25 18:08:29 -06:00
|
|
|
fn expr_contains_conjunctive_ifs<'tcx>(expr: &'tcx Expr<'tcx>) -> bool {
|
|
|
|
fn contains_if(expr: &Expr<'_>, on_if: bool) -> bool {
|
|
|
|
match expr.kind {
|
|
|
|
ExprKind::If(..) => on_if,
|
|
|
|
ExprKind::Binary(_, left, right) => contains_if(left, true) || contains_if(right, true),
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
contains_if(expr, false)
|
|
|
|
}
|
|
|
|
|
2024-03-09 10:09:17 -06:00
|
|
|
fn emit_return_lint(
|
|
|
|
cx: &LateContext<'_>,
|
|
|
|
ret_span: Span,
|
|
|
|
semi_spans: Vec<Span>,
|
|
|
|
replacement: &RetReplacement<'_>,
|
|
|
|
at: HirId,
|
|
|
|
) {
|
2021-01-15 03:56:44 -06:00
|
|
|
if ret_span.from_expansion() {
|
|
|
|
return;
|
|
|
|
}
|
2023-04-11 08:31:08 -05:00
|
|
|
|
2024-03-09 10:09:17 -06:00
|
|
|
span_lint_hir_and_then(
|
|
|
|
cx,
|
|
|
|
NEEDLESS_RETURN,
|
|
|
|
at,
|
|
|
|
ret_span,
|
|
|
|
"unneeded `return` statement",
|
|
|
|
|diag| {
|
|
|
|
let suggestions = std::iter::once((ret_span, replacement.to_string()))
|
|
|
|
.chain(semi_spans.into_iter().map(|span| (span, String::new())))
|
|
|
|
.collect();
|
2023-06-02 04:41:57 -05:00
|
|
|
|
2024-03-09 10:09:17 -06:00
|
|
|
diag.multipart_suggestion_verbose(replacement.sugg_help(), suggestions, replacement.applicability());
|
|
|
|
},
|
|
|
|
);
|
2015-08-11 14:47:34 -05:00
|
|
|
}
|
2016-11-19 19:15:40 -06:00
|
|
|
|
2020-08-28 09:10:16 -05:00
|
|
|
fn last_statement_borrows<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) -> bool {
|
2023-07-17 03:19:29 -05:00
|
|
|
for_each_expr_with_closures(cx, expr, |e| {
|
2022-10-06 02:44:38 -05:00
|
|
|
if let Some(def_id) = fn_def_id(cx, e)
|
|
|
|
&& cx
|
2020-08-28 09:10:16 -05:00
|
|
|
.tcx
|
2023-01-18 17:52:47 -06:00
|
|
|
.fn_sig(def_id)
|
2023-07-11 16:35:29 -05:00
|
|
|
.instantiate_identity()
|
2020-08-28 09:10:16 -05:00
|
|
|
.skip_binder()
|
2022-10-06 02:44:38 -05:00
|
|
|
.output()
|
2022-01-11 21:19:52 -06:00
|
|
|
.walk()
|
2023-07-17 03:19:29 -05:00
|
|
|
.any(|arg| matches!(arg.unpack(), GenericArgKind::Lifetime(re) if !re.is_static()))
|
2022-10-06 02:44:38 -05:00
|
|
|
{
|
|
|
|
ControlFlow::Break(())
|
|
|
|
} else {
|
2023-01-12 12:48:13 -06:00
|
|
|
ControlFlow::Continue(Descend::from(!e.span.from_expansion()))
|
2020-08-28 09:10:16 -05:00
|
|
|
}
|
2022-10-06 02:44:38 -05:00
|
|
|
})
|
|
|
|
.is_some()
|
2020-05-17 10:36:26 -05:00
|
|
|
}
|
2022-12-01 11:29:38 -06:00
|
|
|
|
|
|
|
// Go backwards while encountering whitespace and extend the given Span to that point.
|
|
|
|
fn extend_span_to_previous_non_ws(cx: &LateContext<'_>, sp: Span) -> Span {
|
|
|
|
if let Ok(prev_source) = cx.sess().source_map().span_to_prev_source(sp) {
|
2024-03-15 09:57:22 -05:00
|
|
|
let ws = [b' ', b'\t', b'\n'];
|
|
|
|
if let Some(non_ws_pos) = prev_source.bytes().rposition(|c| !ws.contains(&c)) {
|
2022-12-01 11:29:38 -06:00
|
|
|
let len = prev_source.len() - non_ws_pos - 1;
|
|
|
|
return sp.with_lo(sp.lo() - BytePos::from_usize(len));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sp
|
|
|
|
}
|