use std::collections::VecDeque; use clippy_utils::diagnostics::span_lint_and_sugg; use itertools::{izip, Itertools}; use rustc_ast::{walk_list, Label, Mutability}; use rustc_data_structures::fx::{FxHashMap, FxHashSet}; use rustc_errors::Applicability; use rustc_hir::def::Res; use rustc_hir::intravisit::{walk_expr, FnKind, Visitor}; use rustc_hir::{ Arm, Block, Body, Expr, ExprKind, Guard, HirId, Let, Local, Pat, PatKind, Path, PathSegment, QPath, Stmt, StmtKind, UnOp, }; use rustc_lint::{LateContext, LateLintPass}; use rustc_middle::ty; use rustc_middle::ty::{Ty, TyCtxt, TypeckResults}; use rustc_session::{declare_lint_pass, declare_tool_lint}; use rustc_span::symbol::kw; use rustc_span::symbol::Ident; use rustc_span::Span; declare_clippy_lint! { /// ### What it does /// Checks for arguments that is only used in recursion with no side-effects. /// The arguments can be involved in calculations and assignments but as long as /// the calculations have no side-effects (function calls or mutating dereference) /// and the assigned variables are also only in recursion, it is useless. /// /// ### Why is this bad? /// The could contain a useless calculation and can make function simpler. /// /// ### Known Issues /// It could not catch the variable that has no side effects but only used in recursion. /// /// ### Example /// ```rust /// fn f(a: usize, b: usize) -> usize { /// if a == 0 { /// 1 /// } else { /// f(a - 1, b + 1) /// } /// } /// # fn main() { /// # print!("{}", f(1, 1)); /// # } /// ``` /// Use instead: /// ```rust /// fn f(a: usize) -> usize { /// if a == 0 { /// 1 /// } else { /// f(a - 1) /// } /// } /// # fn main() { /// # print!("{}", f(1)); /// # } /// ``` #[clippy::version = "1.60.0"] pub ONLY_USED_IN_RECURSION, complexity, "default lint description" } declare_lint_pass!(OnlyUsedInRecursion => [ONLY_USED_IN_RECURSION]); impl<'tcx> LateLintPass<'tcx> for OnlyUsedInRecursion { fn check_fn( &mut self, cx: &LateContext<'tcx>, kind: FnKind<'tcx>, _: &'tcx rustc_hir::FnDecl<'tcx>, body: &'tcx Body<'tcx>, _: Span, _: HirId, ) { if let FnKind::ItemFn(ident, ..) | FnKind::Method(ident, ..) = kind { let ty_res = cx.typeck_results(); let param_span = body .params .iter() .flat_map(|param| { let mut v = Vec::new(); param.pat.each_binding(|_, hir_id, span, ident| { v.push((hir_id, span, ident)); }); v }) .skip(match kind { FnKind::Method(..) => 1, _ => 0, }) .filter(|(_, _, ident)| !ident.name.as_str().starts_with('_')) .collect_vec(); let params = body.params.iter().map(|param| param.pat).collect(); let mut visitor = SideEffectVisit { graph: FxHashMap::default(), has_side_effect: FxHashSet::default(), ret_vars: Vec::new(), contains_side_effect: false, break_vars: FxHashMap::default(), params, fn_ident: ident, is_method: matches!(kind, FnKind::Method(..)), ty_res, ty_ctx: cx.tcx, }; visitor.visit_expr(&body.value); let vars = std::mem::take(&mut visitor.ret_vars); visitor.add_side_effect(vars); let mut queue = visitor.has_side_effect.iter().copied().collect::>(); while let Some(id) = queue.pop_front() { if let Some(next) = visitor.graph.get(&id) { for i in next { if !visitor.has_side_effect.contains(i) { visitor.has_side_effect.insert(*i); queue.push_back(*i); } } } } for (id, span, ident) in param_span { if !visitor.has_side_effect.contains(&id) { span_lint_and_sugg( cx, ONLY_USED_IN_RECURSION, span, "parameter is only used in recursion with no side-effects", "if this is intentional, prefix with an underscore", format!("_{}", ident.name.as_str()), Applicability::MaybeIncorrect, ); } } } } } pub fn is_primitive(ty: Ty<'_>) -> bool { match ty.kind() { ty::Bool | ty::Char | ty::Int(_) | ty::Uint(_) | ty::Float(_) | ty::Str => true, ty::Ref(_, t, _) => is_primitive(t), _ => false, } } pub fn is_array(ty: Ty<'_>) -> bool { match ty.kind() { ty::Array(..) | ty::Slice(..) => true, ty::Ref(_, t, _) => is_array(t), _ => false, } } pub struct SideEffectVisit<'tcx> { graph: FxHashMap>, has_side_effect: FxHashSet, // bool for if the variable was dereferenced from mutable reference ret_vars: Vec<(HirId, bool)>, contains_side_effect: bool, // break label break_vars: FxHashMap>, params: Vec<&'tcx Pat<'tcx>>, fn_ident: Ident, is_method: bool, ty_res: &'tcx TypeckResults<'tcx>, ty_ctx: TyCtxt<'tcx>, } impl<'tcx> Visitor<'tcx> for SideEffectVisit<'tcx> { fn visit_block(&mut self, b: &'tcx Block<'tcx>) { b.stmts.iter().for_each(|stmt| { self.visit_stmt(stmt); self.ret_vars.clear(); }); walk_list!(self, visit_expr, b.expr); } fn visit_stmt(&mut self, s: &'tcx Stmt<'tcx>) { match s.kind { StmtKind::Local(Local { pat, init: Some(init), .. }) => { self.visit_pat_expr(pat, init); }, StmtKind::Item(i) => { let item = self.ty_ctx.hir().item(i); self.visit_item(item); }, StmtKind::Expr(e) | StmtKind::Semi(e) => self.visit_expr(e), StmtKind::Local(_) => {}, } } fn visit_expr(&mut self, ex: &'tcx Expr<'tcx>) { debug_assert!(self.ret_vars.is_empty()); match ex.kind { ExprKind::Array(exprs) | ExprKind::Tup(exprs) => { self.ret_vars = exprs .iter() .flat_map(|expr| { self.visit_expr(expr); std::mem::take(&mut self.ret_vars) }) .collect(); }, ExprKind::Call(callee, args) => self.visit_fn(callee, args), ExprKind::MethodCall(path, args, _) => self.visit_method_call(path, args), ExprKind::Binary(_, lhs, rhs) => { self.visit_bin_op(lhs, rhs); }, ExprKind::Unary(op, expr) => self.visit_un_op(op, expr), ExprKind::Let(Let { pat, init, .. }) => self.visit_pat_expr(pat, init), ExprKind::If(bind, then_expr, else_expr) => { self.visit_if(bind, then_expr, else_expr); }, ExprKind::Match(expr, arms, _) => self.visit_match(expr, arms), ExprKind::Closure(_, _, body_id, _, _) => { let body = self.ty_ctx.hir().body(body_id); self.visit_body(body); let vars = std::mem::take(&mut self.ret_vars); self.add_side_effect(vars); }, ExprKind::Loop(block, label, _, _) | ExprKind::Block(block, label) => { self.visit_block_label(block, label); }, ExprKind::Assign(bind, expr, _) => { self.visit_assign(bind, expr); }, ExprKind::AssignOp(_, bind, expr) => { self.visit_assign(bind, expr); self.visit_bin_op(bind, expr); }, ExprKind::Field(expr, _) => { self.visit_expr(expr); if matches!(self.ty_res.expr_ty(expr).kind(), ty::Ref(_, _, Mutability::Mut)) { self.ret_vars.iter_mut().for_each(|(_, b)| *b = true); } }, ExprKind::Index(expr, index) => { self.visit_expr(expr); let mut vars = std::mem::take(&mut self.ret_vars); self.visit_expr(index); self.ret_vars.append(&mut vars); if !is_array(self.ty_res.expr_ty(expr)) { self.add_side_effect(self.ret_vars.clone()); } else if matches!(self.ty_res.expr_ty(expr).kind(), ty::Ref(_, _, Mutability::Mut)) { self.ret_vars.iter_mut().for_each(|(_, b)| *b = true); } }, ExprKind::Break(dest, Some(expr)) => { self.visit_expr(expr); if let Some(label) = dest.label { self.break_vars .entry(label.ident) .or_insert(Vec::new()) .append(&mut self.ret_vars); } self.contains_side_effect = true; }, ExprKind::Ret(Some(expr)) => { self.visit_expr(expr); let vars = std::mem::take(&mut self.ret_vars); self.add_side_effect(vars); self.contains_side_effect = true; }, ExprKind::Break(_, None) | ExprKind::Continue(_) | ExprKind::Ret(None) => { self.contains_side_effect = true; }, ExprKind::Struct(_, exprs, expr) => { let mut ret_vars = exprs .iter() .flat_map(|field| { self.visit_expr(field.expr); std::mem::take(&mut self.ret_vars) }) .collect(); walk_list!(self, visit_expr, expr); self.ret_vars.append(&mut ret_vars); }, _ => walk_expr(self, ex), } } fn visit_path(&mut self, path: &'tcx Path<'tcx>, _id: HirId) { if let Res::Local(id) = path.res { self.ret_vars.push((id, false)); } } } impl<'tcx> SideEffectVisit<'tcx> { fn visit_assign(&mut self, lhs: &'tcx Expr<'tcx>, rhs: &'tcx Expr<'tcx>) { // Just support array and tuple unwrapping for now. // // ex) `(a, b) = (c, d);` // The graph would look like this: // a -> c // b -> d // // This would minimize the connection of the side-effect graph. match (&lhs.kind, &rhs.kind) { (ExprKind::Array(lhs), ExprKind::Array(rhs)) | (ExprKind::Tup(lhs), ExprKind::Tup(rhs)) => { // if not, it is a compile error debug_assert!(lhs.len() == rhs.len()); izip!(*lhs, *rhs).for_each(|(lhs, rhs)| self.visit_assign(lhs, rhs)); }, // in other assigns, we have to connect all each other // because they can be connected somehow _ => { self.visit_expr(lhs); let lhs_vars = std::mem::take(&mut self.ret_vars); self.visit_expr(rhs); let rhs_vars = std::mem::take(&mut self.ret_vars); self.connect_assign(&lhs_vars, &rhs_vars); }, } } fn visit_block_label(&mut self, block: &'tcx Block<'tcx>, label: Option