rust-clippy/clippy_lints/src/operators/arithmetic_side_effects.rs

183 lines
6.3 KiB
Rust
Raw Normal View History

2022-07-18 17:29:45 +00:00
#![allow(
// False positive
clippy::match_same_arms
)]
2022-09-08 15:04:55 +00:00
use super::ARITHMETIC_SIDE_EFFECTS;
2022-07-18 17:29:45 +00:00
use clippy_utils::{consts::constant_simple, diagnostics::span_lint};
2022-09-07 13:00:45 +00:00
use rustc_ast as ast;
2022-07-18 17:29:45 +00:00
use rustc_data_structures::fx::FxHashSet;
use rustc_hir as hir;
use rustc_lint::{LateContext, LateLintPass};
2022-09-07 13:00:45 +00:00
use rustc_middle::ty::Ty;
2022-07-18 17:29:45 +00:00
use rustc_session::impl_lint_pass;
2022-09-07 13:00:45 +00:00
use rustc_span::source_map::{Span, Spanned};
2022-07-18 17:29:45 +00:00
2022-09-07 13:00:45 +00:00
const HARD_CODED_ALLOWED: &[&str] = &[
"f32",
"f64",
"std::num::Saturating",
"std::string::String",
"std::num::Wrapping",
];
2022-07-18 17:29:45 +00:00
#[derive(Debug)]
2022-09-08 15:04:55 +00:00
pub struct ArithmeticSideEffects {
2022-07-18 17:29:45 +00:00
allowed: FxHashSet<String>,
// Used to check whether expressions are constants, such as in enum discriminants and consts
const_span: Option<Span>,
expr_span: Option<Span>,
}
2022-09-08 15:04:55 +00:00
impl_lint_pass!(ArithmeticSideEffects => [ARITHMETIC_SIDE_EFFECTS]);
2022-07-18 17:29:45 +00:00
2022-09-08 15:04:55 +00:00
impl ArithmeticSideEffects {
2022-07-18 17:29:45 +00:00
#[must_use]
pub fn new(mut allowed: FxHashSet<String>) -> Self {
allowed.extend(HARD_CODED_ALLOWED.iter().copied().map(String::from));
Self {
allowed,
const_span: None,
expr_span: None,
}
}
2022-09-16 20:01:29 +00:00
/// Assuming that `expr` is a literal integer, checks operators (+=, -=, *, /) in a
/// non-constant environment that won't overflow.
2022-09-16 20:01:29 +00:00
fn has_valid_op(op: &Spanned<hir::BinOpKind>, expr: &hir::Expr<'_>) -> bool {
if let hir::BinOpKind::Add | hir::BinOpKind::Sub = op.node
&& let hir::ExprKind::Lit(ref lit) = expr.kind
&& let ast::LitKind::Int(0, _) = lit.node
{
return true;
}
if let hir::BinOpKind::Div | hir::BinOpKind::Rem = op.node
&& let hir::ExprKind::Lit(ref lit) = expr.kind
&& !matches!(lit.node, ast::LitKind::Int(0, _))
{
return true;
}
if let hir::BinOpKind::Mul = op.node
&& let hir::ExprKind::Lit(ref lit) = expr.kind
&& let ast::LitKind::Int(0 | 1, _) = lit.node
2022-09-07 13:00:45 +00:00
{
return true;
}
false
}
2022-07-18 17:29:45 +00:00
/// Checks if the given `expr` has any of the inner `allowed` elements.
fn is_allowed_ty(&self, cx: &LateContext<'_>, expr: &hir::Expr<'_>) -> bool {
self.allowed.contains(
cx.typeck_results()
.expr_ty(expr)
.to_string()
.split('<')
.next()
.unwrap_or_default(),
)
}
2022-09-07 13:00:45 +00:00
/// Explicit integers like `1` or `i32::MAX`. Does not take into consideration references.
fn is_literal_integer(expr: &hir::Expr<'_>, expr_refs: Ty<'_>) -> bool {
let is_integral = expr_refs.is_integral();
let is_literal = matches!(expr.kind, hir::ExprKind::Lit(_));
is_integral && is_literal
}
2022-07-18 17:29:45 +00:00
fn issue_lint(&mut self, cx: &LateContext<'_>, expr: &hir::Expr<'_>) {
let msg = "arithmetic operation that can potentially result in unexpected side-effects";
span_lint(cx, ARITHMETIC_SIDE_EFFECTS, expr.span, msg);
2022-07-18 17:29:45 +00:00
self.expr_span = Some(expr.span);
}
2022-09-07 13:00:45 +00:00
/// Manages when the lint should be triggered. Operations in constant environments, hard coded
/// types, custom allowed types and non-constant operations that won't overflow are ignored.
fn manage_bin_ops(
&mut self,
cx: &LateContext<'_>,
expr: &hir::Expr<'_>,
op: &Spanned<hir::BinOpKind>,
lhs: &hir::Expr<'_>,
rhs: &hir::Expr<'_>,
) {
if constant_simple(cx, cx.typeck_results(), expr).is_some() {
return;
}
if !matches!(
op.node,
hir::BinOpKind::Add
| hir::BinOpKind::Sub
| hir::BinOpKind::Mul
| hir::BinOpKind::Div
| hir::BinOpKind::Rem
| hir::BinOpKind::Shl
| hir::BinOpKind::Shr
) {
return;
};
if self.is_allowed_ty(cx, lhs) || self.is_allowed_ty(cx, rhs) {
return;
}
let has_valid_op = match (
2022-09-15 16:40:49 +00:00
Self::is_literal_integer(lhs, cx.typeck_results().expr_ty(lhs).peel_refs()),
Self::is_literal_integer(rhs, cx.typeck_results().expr_ty(rhs).peel_refs()),
) {
(true, true) => true,
2022-09-16 20:01:29 +00:00
(true, false) => Self::has_valid_op(op, lhs),
(false, true) => Self::has_valid_op(op, rhs),
(false, false) => false,
};
if !has_valid_op {
self.issue_lint(cx, expr);
2022-09-07 13:00:45 +00:00
}
}
2022-07-18 17:29:45 +00:00
}
2022-09-08 15:04:55 +00:00
impl<'tcx> LateLintPass<'tcx> for ArithmeticSideEffects {
2022-07-18 17:29:45 +00:00
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx hir::Expr<'_>) {
2022-09-07 13:00:45 +00:00
if self.expr_span.is_some() || self.const_span.map_or(false, |sp| sp.contains(expr.span)) {
2022-07-18 17:29:45 +00:00
return;
}
match &expr.kind {
hir::ExprKind::Binary(op, lhs, rhs) | hir::ExprKind::AssignOp(op, lhs, rhs) => {
2022-09-07 13:00:45 +00:00
self.manage_bin_ops(cx, expr, op, lhs, rhs);
2022-07-18 17:29:45 +00:00
},
hir::ExprKind::Unary(hir::UnOp::Neg, _) => {
if constant_simple(cx, cx.typeck_results(), expr).is_none() {
self.issue_lint(cx, expr);
}
},
_ => {},
}
}
fn check_body(&mut self, cx: &LateContext<'_>, body: &hir::Body<'_>) {
2022-09-07 13:00:45 +00:00
let body_owner = cx.tcx.hir().body_owner(body.id());
let body_owner_def_id = cx.tcx.hir().local_def_id(body_owner);
let body_owner_kind = cx.tcx.hir().body_owner_kind(body_owner_def_id);
if let hir::BodyOwnerKind::Const | hir::BodyOwnerKind::Static(_) = body_owner_kind {
let body_span = cx.tcx.hir().span_with_body(body_owner);
if let Some(span) = self.const_span && span.contains(body_span) {
return;
}
self.const_span = Some(body_span);
2022-07-18 17:29:45 +00:00
}
}
fn check_body_post(&mut self, cx: &LateContext<'_>, body: &hir::Body<'_>) {
let body_owner = cx.tcx.hir().body_owner(body.id());
let body_span = cx.tcx.hir().span(body_owner);
if let Some(span) = self.const_span && span.contains(body_span) {
return;
}
self.const_span = None;
}
fn check_expr_post(&mut self, _: &LateContext<'tcx>, expr: &'tcx hir::Expr<'_>) {
if Some(expr.span) == self.expr_span {
self.expr_span = None;
}
}
}