diff --git a/utils/src/constant_analyzer/mod.rs b/utils/src/constant_analyzer/mod.rs index b10ad0b8..a9e9457e 100644 --- a/utils/src/constant_analyzer/mod.rs +++ b/utils/src/constant_analyzer/mod.rs @@ -1,22 +1,24 @@ -use std::collections::HashSet; - extern crate rustc_hir; extern crate rustc_lint; -use clippy_utils::consts::constant; +use clippy_utils::consts::{constant, Constant}; +use if_chain::if_chain; use rustc_hir::{ def::{DefKind, Res}, - intravisit::{walk_expr, Visitor}, - Expr, ExprKind, HirId, QPath, StmtKind, + intravisit::{walk_local, Visitor}, + Expr, ExprKind, HirId, Node, QPath, }; use rustc_lint::LateContext; +use std::collections::HashSet; +/// Analyzes expressions to determine if they are constants or known at compile-time. pub struct ConstantAnalyzer<'a, 'tcx> { pub cx: &'a LateContext<'tcx>, pub constants: HashSet, } impl<'a, 'tcx> ConstantAnalyzer<'a, 'tcx> { + /// Checks if a QPath refers to a constant. fn is_qpath_constant(&self, path: &QPath) -> bool { if let QPath::Resolved(_, path) = path { match path.res { @@ -35,13 +37,12 @@ impl<'a, 'tcx> ConstantAnalyzer<'a, 'tcx> { } } + /// Determines if an expression is constant or known at compile-time. fn is_expr_constant(&self, expr: &Expr<'tcx>) -> bool { - // Evaluate the expression as a compile-time constant if constant(self.cx, self.cx.typeck_results(), expr).is_some() { return true; } - // If the expression is not a constant, verify if it is known at compile time match expr.kind { ExprKind::Array(expr_array) => expr_array .iter() @@ -51,9 +52,8 @@ impl<'a, 'tcx> ConstantAnalyzer<'a, 'tcx> { } ExprKind::Cast(cast_expr, _) => self.is_expr_constant(cast_expr), ExprKind::Field(field_expr, _) => self.is_expr_constant(field_expr), - // TODO: array with just index checking ExprKind::Index(array_expr, index_expr, _) => { - self.is_expr_constant(array_expr) && self.is_expr_constant(index_expr) + self.is_array_index_constant(array_expr, index_expr) } ExprKind::Lit(_) => true, ExprKind::Path(qpath_expr) => self.is_qpath_constant(&qpath_expr), @@ -65,24 +65,51 @@ impl<'a, 'tcx> ConstantAnalyzer<'a, 'tcx> { } } - pub fn is_compile_time_known(&self, expr: &Expr<'tcx>) -> bool { + /// Checks if an array index operation results in a constant value. + fn is_array_index_constant(&self, array_expr: &Expr<'tcx>, index_expr: &Expr<'tcx>) -> bool { + match ( + &array_expr.kind, + constant(self.cx, self.cx.typeck_results(), index_expr), + ) { + (ExprKind::Array(array_elements), Some(Constant::Int(index))) => { + self.is_array_element_constant(array_elements, index) + } + (ExprKind::Path(QPath::Resolved(_, path)), Some(Constant::Int(index))) => { + if_chain! { + if let Res::Local(hir_id) = path.res; + if let Node::LetStmt(let_stmt) = self.cx.tcx.parent_hir_node(hir_id); + if let Some(ExprKind::Array(array_elements)) = let_stmt.init.map(|init| &init.kind); + then { + self.is_array_element_constant(array_elements, index) + } else { + false + } + } + } + _ => false, + } + } + + /// Checks if a specific array element is constant. + fn is_array_element_constant(&self, elements: &[Expr<'tcx>], index: u128) -> bool { + elements + .get(index as usize) + .map_or(false, |element| self.is_expr_constant(element)) + } + + /// Public method to check if an expression is constant. + pub fn is_constant(&self, expr: &Expr<'tcx>) -> bool { self.is_expr_constant(expr) } } impl<'a, 'tcx> Visitor<'tcx> for ConstantAnalyzer<'a, 'tcx> { - fn visit_expr(&mut self, expr: &'tcx Expr<'tcx>) { - if let ExprKind::Block(block, _) = expr.kind { - for stmt in block.stmts { - if let StmtKind::Let(local) = stmt.kind { - if let Some(init) = local.init { - if self.is_expr_constant(init) { - self.constants.insert(local.pat.hir_id); - } - } - } + fn visit_local(&mut self, l: &'tcx rustc_hir::LetStmt<'tcx>) -> Self::Result { + if let Some(init) = l.init { + if self.is_expr_constant(init) { + self.constants.insert(l.pat.hir_id); } } - walk_expr(self, expr); + walk_local(self, l); } }