-
Notifications
You must be signed in to change notification settings - Fork 11.2k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[Linter] Meaningless Math Operation (#16626)
## Description The lint identifies operations that have no effect on the result, such as multiplying by 0 or 1, adding or subtracting 0, or shifting by 0. These operations are considered redundant and can be simplified to improve code clarity. Main Logic: For binary operations, it checks the operator and the right-hand side operand. It identifies specific patterns of meaningless operations. Detected Patterns: The lint checks for the following meaningless operations: Multiplication or division by 0 Multiplication by 1 Addition or subtraction of 0 Left or right shift by 0 ## Test plan Added more use case including true positive, true negative, false positive, false negative case ## Release notes - [ ] Protocol: - [ ] Nodes (Validators and Full nodes): - [ ] Indexer: - [ ] JSON-RPC: - [ ] GraphQL: - [X] CLI: Move will now lint against unnecessary math operations in many cases. - [ ] Rust SDK: --------- Co-authored-by: jamedzung <[email protected]> Co-authored-by: Todd Nowacki <[email protected]>
- Loading branch information
Showing
8 changed files
with
615 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
152 changes: 152 additions & 0 deletions
152
external-crates/move/crates/move-compiler/src/linters/meaningless_math_operation.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
// Copyright (c) The Move Contributors | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
//! Detects meaningless math operations like `x * 0`, `x << 0`, `x >> 0`, `x * 1`, `x + 0`, `x - 0` | ||
//! Aims to reduce code redundancy and improve clarity by flagging operations with no effect. | ||
use crate::{ | ||
cfgir::ast as G, | ||
cfgir::visitor::{CFGIRVisitorConstructor, CFGIRVisitorContext}, | ||
diag, | ||
diagnostics::{ | ||
codes::{custom, DiagnosticInfo, Severity}, | ||
WarningFilters, | ||
}, | ||
hlir::ast::{self as H, Value_}, | ||
parser::ast::BinOp_, | ||
shared::CompilationEnv, | ||
}; | ||
use move_core_types::u256::U256; | ||
use move_ir_types::location::Loc; | ||
|
||
use super::{LinterDiagnosticCategory, LINT_WARNING_PREFIX, MEANINGLESS_MATH_DIAG_CODE}; | ||
|
||
const MEANINGLESS_MATH_OP_DIAG: DiagnosticInfo = custom( | ||
LINT_WARNING_PREFIX, | ||
Severity::Warning, | ||
LinterDiagnosticCategory::Complexity as u8, | ||
MEANINGLESS_MATH_DIAG_CODE, | ||
"math operation can be simplified", | ||
); | ||
|
||
pub struct MeaninglessMathOperation; | ||
|
||
pub struct Context<'a> { | ||
env: &'a mut CompilationEnv, | ||
} | ||
|
||
impl CFGIRVisitorConstructor for MeaninglessMathOperation { | ||
type Context<'a> = Context<'a>; | ||
|
||
fn context<'a>(env: &'a mut CompilationEnv, _program: &G::Program) -> Self::Context<'a> { | ||
Context { env } | ||
} | ||
} | ||
|
||
impl CFGIRVisitorContext for Context<'_> { | ||
fn add_warning_filter_scope(&mut self, filter: WarningFilters) { | ||
self.env.add_warning_filter_scope(filter) | ||
} | ||
fn pop_warning_filter_scope(&mut self) { | ||
self.env.pop_warning_filter_scope() | ||
} | ||
|
||
fn visit_exp_custom(&mut self, exp: &mut H::Exp) -> bool { | ||
let H::UnannotatedExp_::BinopExp(lhs, op, rhs) = &exp.exp.value else { | ||
return false; | ||
}; | ||
|
||
// unchanged operations | ||
let is_unchanged = match op.value { | ||
BinOp_::Mul => is_one(lhs).or(is_one(rhs)), | ||
BinOp_::Div => is_one(rhs), | ||
BinOp_::Add => is_zero(lhs).or(is_zero(rhs)), | ||
BinOp_::Sub => is_zero(rhs), | ||
BinOp_::Shl | BinOp_::Shr => is_zero(rhs), | ||
_ => None, | ||
}; | ||
if let Some(meaningless_operand) = is_unchanged { | ||
let msg = "This operation has no effect and can be removed"; | ||
self.env.add_diag(diag!( | ||
MEANINGLESS_MATH_OP_DIAG, | ||
(exp.exp.loc, msg), | ||
(meaningless_operand, "Because of this operand"), | ||
)); | ||
} | ||
|
||
// always zero | ||
let is_always_zero = match op.value { | ||
BinOp_::Mul => is_zero(lhs).or(is_zero(rhs)), | ||
BinOp_::Div => is_zero(lhs), | ||
BinOp_::Mod => is_zero(lhs).or(is_one(rhs)), | ||
_ => None, | ||
}; | ||
if let Some(zero_operand) = is_always_zero { | ||
let msg = "This operation is always zero and can be replaced with '0'"; | ||
self.env.add_diag(diag!( | ||
MEANINGLESS_MATH_OP_DIAG, | ||
(exp.exp.loc, msg), | ||
(zero_operand, "Because of this operand"), | ||
)); | ||
} | ||
|
||
// always one | ||
let is_always_one = match op.value { | ||
BinOp_::Mod => is_one(lhs), | ||
_ => None, | ||
}; | ||
if let Some(one_operand) = is_always_one { | ||
let msg = "This operation is always one and can be replaced with '1'"; | ||
self.env.add_diag(diag!( | ||
MEANINGLESS_MATH_OP_DIAG, | ||
(exp.exp.loc, msg), | ||
(one_operand, "Because of this operand"), | ||
)); | ||
} | ||
|
||
// for aborts, e.g. x / 0, we will let the optimizer give a warning | ||
|
||
false | ||
} | ||
} | ||
|
||
fn is_zero(exp: &H::Exp) -> Option<Loc> { | ||
let H::UnannotatedExp_::Value(sp!(loc, value_)) = &exp.exp.value else { | ||
return None; | ||
}; | ||
match value_ { | ||
Value_::U8(0) | Value_::U16(0) | Value_::U32(0) | Value_::U64(0) | Value_::U128(0) => { | ||
Some(*loc) | ||
} | ||
Value_::U256(u) if u == &U256::zero() => Some(*loc), | ||
Value_::U8(_) | ||
| Value_::U16(_) | ||
| Value_::U32(_) | ||
| Value_::U64(_) | ||
| Value_::U128(_) | ||
| Value_::U256(_) | ||
| Value_::Address(_) | ||
| Value_::Bool(_) | ||
| Value_::Vector(_, _) => None, | ||
} | ||
} | ||
|
||
fn is_one(exp: &H::Exp) -> Option<Loc> { | ||
let H::UnannotatedExp_::Value(sp!(loc, value_)) = &exp.exp.value else { | ||
return None; | ||
}; | ||
match value_ { | ||
Value_::U8(1) | Value_::U16(1) | Value_::U32(1) | Value_::U64(1) | Value_::U128(1) => { | ||
Some(*loc) | ||
} | ||
Value_::U256(u) if u == &U256::one() => Some(*loc), | ||
Value_::U8(_) | ||
| Value_::U16(_) | ||
| Value_::U32(_) | ||
| Value_::U64(_) | ||
| Value_::U128(_) | ||
| Value_::U256(_) | ||
| Value_::Address(_) | ||
| Value_::Bool(_) | ||
| Value_::Vector(_, _) => None, | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
12 changes: 12 additions & 0 deletions
12
...tes/move/crates/move-compiler/tests/linter/false_negative_meaningless_math_operation.move
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
module 0x42::M { | ||
public fun zero_shift_complex(x: u64, y: u64): u64 { | ||
x * (y - y) // This is effectively * 0, but is not currently caught | ||
} | ||
|
||
public fun ast_fold() { | ||
// we do not lint on these because they are folded to a single value | ||
let x = 0; | ||
x * 1; | ||
1 * 0; | ||
} | ||
} |
8 changes: 8 additions & 0 deletions
8
...al-crates/move/crates/move-compiler/tests/linter/suppress_meaningless_math_operation.move
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
// tests suppression of unnecessary math operations | ||
module 0x42::M { | ||
|
||
#[allow(lint(unnecessary_math))] | ||
public fun add_zero(x: u64): u64 { | ||
x + 0 | ||
} | ||
} |
17 changes: 17 additions & 0 deletions
17
...ates/move/crates/move-compiler/tests/linter/true_negative_meaningless_math_operation.move
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
module 0x42::M { | ||
public fun multiply_by_two(x: u64): u64 { | ||
x * 2 // Should not trigger the linter | ||
} | ||
|
||
public fun left_shift_by_one(x: u64): u64 { | ||
x << 1 // Should not trigger the linter | ||
} | ||
|
||
public fun add_one(x: u64): u64 { | ||
x + 1 // Should not trigger the linter | ||
} | ||
|
||
public fun divide_by_two(x: u64): u64 { | ||
x / 2 // Should not trigger the linter | ||
} | ||
} |
Oops, something went wrong.