From c884d333e05c6911e4544a9830e29a18d4156eed Mon Sep 17 00:00:00 2001 From: Vara Prasad Bandaru Date: Tue, 30 Jan 2024 19:04:33 +0530 Subject: [PATCH] Upd missing_owner_check to not report local variables, duplicate warnings when to_account_info is called on Anchor's AccountInfo. --- lints/missing_owner_check/src/lib.rs | 24 +++++++++++++++++++++++- 1 file changed, 23 insertions(+), 1 deletion(-) diff --git a/lints/missing_owner_check/src/lib.rs b/lints/missing_owner_check/src/lib.rs index 65de085..904a6a6 100644 --- a/lints/missing_owner_check/src/lib.rs +++ b/lints/missing_owner_check/src/lib.rs @@ -11,7 +11,7 @@ use clippy_utils::{ use if_chain::if_chain; use rustc_hir::{ intravisit::{walk_expr, FnKind, Visitor}, - Body, Expr, ExprKind, FnDecl, HirId, + Body, Expr, ExprKind, FnDecl, HirId, QPath, }; use rustc_lint::{LateContext, LateLintPass}; use rustc_middle::ty; @@ -107,6 +107,7 @@ impl<'cx, 'tcx> Visitor<'tcx> for AccountUses<'cx, 'tcx> { if !is_call_to_clone(self.cx, expr); let ty = self.cx.typeck_results().expr_ty(expr); if match_type(self.cx, ty, &paths::SOLANA_PROGRAM_ACCOUNT_INFO); + if !is_expr_local_variable(expr); if !is_safe_to_account_info(self.cx, expr); let mut spanless_eq = SpanlessEq::new(self.cx); if !self.uses.iter().any(|e| spanless_eq.eq_expr(e, expr)); @@ -118,6 +119,7 @@ impl<'cx, 'tcx> Visitor<'tcx> for AccountUses<'cx, 'tcx> { } } +// s3v3ru5: the following check removes duplicate warnings where lint would report both `x` and `x.clone()` expressions. /// Return true if the expr is a method call to clone else false fn is_call_to_clone<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) -> bool { if_chain! { @@ -132,6 +134,24 @@ fn is_call_to_clone<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) -> boo } } +// s3v3ru5: if a local variable is of type AccountInfo, the rhs of the let statement assigning to variable +// will be of type AccountInfo. The lint would check that expression and there is no need for checking the +// local variable as well. +// This removes the false positives of following pattern: +// `let x = {Account, Program, ...verified structs}.to_account_info()`, +// the lint reports uses of `x`. Having this check would remove such false positives. +fn is_expr_local_variable<'tcx>(expr: &'tcx Expr<'tcx>) -> bool { + if_chain! { + if let ExprKind::Path(QPath::Resolved(None, path)) = expr.kind; + if path.segments.len() == 1; + then { + true + } else { + false + } + } +} + // smoelius: See: https://github.com/crytic/solana-lints/issues/31 fn is_safe_to_account_info<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) -> bool { if_chain! { @@ -157,6 +177,8 @@ fn is_safe_to_account_info<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>) &paths::ANCHOR_LANG_ACCOUNT_LOADER, &paths::ANCHOR_LANG_SIGNER, &paths::ANCHOR_LANG_SYSVAR, + // s3v3ru5: The following line will remove duplicate warnings where lint reports both `x` and `x.to_account_info()` when x is of type Anchor's AccountInfo. + &paths::SOLANA_PROGRAM_ACCOUNT_INFO, ], ) .is_some();