2021-06-03 01:41:37 -05:00
|
|
|
use clippy_utils::consts::{constant, Constant};
|
2021-03-25 13:29:11 -05:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_then;
|
|
|
|
use clippy_utils::sext;
|
2019-12-26 06:34:18 -06:00
|
|
|
use if_chain::if_chain;
|
2020-02-21 02:39:38 -06:00
|
|
|
use rustc_hir::{BinOpKind, Expr, ExprKind};
|
2020-01-12 00:08:41 -06:00
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
Overhaul `TyS` and `Ty`.
Specifically, change `Ty` from this:
```
pub type Ty<'tcx> = &'tcx TyS<'tcx>;
```
to this
```
pub struct Ty<'tcx>(Interned<'tcx, TyS<'tcx>>);
```
There are two benefits to this.
- It's now a first class type, so we can define methods on it. This
means we can move a lot of methods away from `TyS`, leaving `TyS` as a
barely-used type, which is appropriate given that it's not meant to
be used directly.
- The uniqueness requirement is now explicit, via the `Interned` type.
E.g. the pointer-based `Eq` and `Hash` comes from `Interned`, rather
than via `TyS`, which wasn't obvious at all.
Much of this commit is boring churn. The interesting changes are in
these files:
- compiler/rustc_middle/src/arena.rs
- compiler/rustc_middle/src/mir/visit.rs
- compiler/rustc_middle/src/ty/context.rs
- compiler/rustc_middle/src/ty/mod.rs
Specifically:
- Most mentions of `TyS` are removed. It's very much a dumb struct now;
`Ty` has all the smarts.
- `TyS` now has `crate` visibility instead of `pub`.
- `TyS::make_for_test` is removed in favour of the static `BOOL_TY`,
which just works better with the new structure.
- The `Eq`/`Ord`/`Hash` impls are removed from `TyS`. `Interned`s impls
of `Eq`/`Hash` now suffice. `Ord` is now partly on `Interned`
(pointer-based, for the `Equal` case) and partly on `TyS`
(contents-based, for the other cases).
- There are many tedious sigil adjustments, i.e. adding or removing `*`
or `&`. They seem to be unavoidable.
2022-01-24 21:13:38 -06:00
|
|
|
use rustc_middle::ty::{self, Ty};
|
2020-01-11 05:37:08 -06:00
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
2019-12-26 06:34:18 -06:00
|
|
|
use std::fmt::Display;
|
|
|
|
|
|
|
|
declare_clippy_lint! {
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for modulo arithmetic.
|
2019-12-26 06:34:18 -06:00
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// The results of modulo (%) operation might differ
|
2019-12-26 06:34:18 -06:00
|
|
|
/// depending on the language, when negative numbers are involved.
|
|
|
|
/// If you interop with different languages it might be beneficial
|
|
|
|
/// to double check all places that use modulo arithmetic.
|
|
|
|
///
|
|
|
|
/// For example, in Rust `17 % -3 = 2`, but in Python `17 % -3 = -1`.
|
|
|
|
///
|
2021-07-29 05:16:06 -05:00
|
|
|
/// ### Example
|
2019-12-26 06:34:18 -06:00
|
|
|
/// ```rust
|
|
|
|
/// let x = -17 % 3;
|
|
|
|
/// ```
|
2021-12-06 05:33:31 -06:00
|
|
|
#[clippy::version = "1.42.0"]
|
2019-12-26 06:34:18 -06:00
|
|
|
pub MODULO_ARITHMETIC,
|
|
|
|
restriction,
|
|
|
|
"any modulo arithmetic statement"
|
|
|
|
}
|
|
|
|
|
|
|
|
declare_lint_pass!(ModuloArithmetic => [MODULO_ARITHMETIC]);
|
|
|
|
|
|
|
|
struct OperandInfo {
|
|
|
|
string_representation: Option<String>,
|
|
|
|
is_negative: bool,
|
|
|
|
is_integral: bool,
|
|
|
|
}
|
|
|
|
|
2020-06-25 15:41:36 -05:00
|
|
|
fn analyze_operand(operand: &Expr<'_>, cx: &LateContext<'_>, expr: &Expr<'_>) -> Option<OperandInfo> {
|
2020-07-17 03:47:04 -05:00
|
|
|
match constant(cx, cx.typeck_results(), operand) {
|
2020-08-03 17:18:29 -05:00
|
|
|
Some((Constant::Int(v), _)) => match *cx.typeck_results().expr_ty(expr).kind() {
|
2019-12-26 06:34:18 -06:00
|
|
|
ty::Int(ity) => {
|
|
|
|
let value = sext(cx.tcx, v, ity);
|
|
|
|
return Some(OperandInfo {
|
|
|
|
string_representation: Some(value.to_string()),
|
|
|
|
is_negative: value < 0,
|
|
|
|
is_integral: true,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
ty::Uint(_) => {
|
|
|
|
return Some(OperandInfo {
|
|
|
|
string_representation: None,
|
|
|
|
is_negative: false,
|
|
|
|
is_integral: true,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
_ => {},
|
|
|
|
},
|
|
|
|
Some((Constant::F32(f), _)) => {
|
|
|
|
return Some(floating_point_operand_info(&f));
|
|
|
|
},
|
|
|
|
Some((Constant::F64(f), _)) => {
|
|
|
|
return Some(floating_point_operand_info(&f));
|
|
|
|
},
|
|
|
|
_ => {},
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
fn floating_point_operand_info<T: Display + PartialOrd + From<f32>>(f: &T) -> OperandInfo {
|
|
|
|
OperandInfo {
|
|
|
|
string_representation: Some(format!("{:.3}", *f)),
|
|
|
|
is_negative: *f < 0.0.into(),
|
|
|
|
is_integral: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Overhaul `TyS` and `Ty`.
Specifically, change `Ty` from this:
```
pub type Ty<'tcx> = &'tcx TyS<'tcx>;
```
to this
```
pub struct Ty<'tcx>(Interned<'tcx, TyS<'tcx>>);
```
There are two benefits to this.
- It's now a first class type, so we can define methods on it. This
means we can move a lot of methods away from `TyS`, leaving `TyS` as a
barely-used type, which is appropriate given that it's not meant to
be used directly.
- The uniqueness requirement is now explicit, via the `Interned` type.
E.g. the pointer-based `Eq` and `Hash` comes from `Interned`, rather
than via `TyS`, which wasn't obvious at all.
Much of this commit is boring churn. The interesting changes are in
these files:
- compiler/rustc_middle/src/arena.rs
- compiler/rustc_middle/src/mir/visit.rs
- compiler/rustc_middle/src/ty/context.rs
- compiler/rustc_middle/src/ty/mod.rs
Specifically:
- Most mentions of `TyS` are removed. It's very much a dumb struct now;
`Ty` has all the smarts.
- `TyS` now has `crate` visibility instead of `pub`.
- `TyS::make_for_test` is removed in favour of the static `BOOL_TY`,
which just works better with the new structure.
- The `Eq`/`Ord`/`Hash` impls are removed from `TyS`. `Interned`s impls
of `Eq`/`Hash` now suffice. `Ord` is now partly on `Interned`
(pointer-based, for the `Equal` case) and partly on `TyS`
(contents-based, for the other cases).
- There are many tedious sigil adjustments, i.e. adding or removing `*`
or `&`. They seem to be unavoidable.
2022-01-24 21:13:38 -06:00
|
|
|
fn might_have_negative_value(t: Ty<'_>) -> bool {
|
2019-12-26 06:34:18 -06:00
|
|
|
t.is_signed() || t.is_floating_point()
|
|
|
|
}
|
|
|
|
|
2020-06-25 15:41:36 -05:00
|
|
|
fn check_const_operands<'tcx>(
|
|
|
|
cx: &LateContext<'tcx>,
|
2019-12-28 10:14:19 -06:00
|
|
|
expr: &'tcx Expr<'_>,
|
2019-12-26 06:34:18 -06:00
|
|
|
lhs_operand: &OperandInfo,
|
|
|
|
rhs_operand: &OperandInfo,
|
|
|
|
) {
|
|
|
|
if lhs_operand.is_negative ^ rhs_operand.is_negative {
|
|
|
|
span_lint_and_then(
|
|
|
|
cx,
|
|
|
|
MODULO_ARITHMETIC,
|
|
|
|
expr.span,
|
|
|
|
&format!(
|
|
|
|
"you are using modulo operator on constants with different signs: `{} % {}`",
|
|
|
|
lhs_operand.string_representation.as_ref().unwrap(),
|
|
|
|
rhs_operand.string_representation.as_ref().unwrap()
|
|
|
|
),
|
2020-04-17 01:08:00 -05:00
|
|
|
|diag| {
|
|
|
|
diag.note("double check for expected result especially when interoperating with different languages");
|
2019-12-26 06:34:18 -06:00
|
|
|
if lhs_operand.is_integral {
|
2020-04-17 01:08:00 -05:00
|
|
|
diag.note("or consider using `rem_euclid` or similar function");
|
2019-12-26 06:34:18 -06:00
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-25 15:41:36 -05:00
|
|
|
fn check_non_const_operands<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>, operand: &Expr<'_>) {
|
2020-07-17 03:47:04 -05:00
|
|
|
let operand_type = cx.typeck_results().expr_ty(operand);
|
2019-12-26 06:34:18 -06:00
|
|
|
if might_have_negative_value(operand_type) {
|
|
|
|
span_lint_and_then(
|
|
|
|
cx,
|
|
|
|
MODULO_ARITHMETIC,
|
|
|
|
expr.span,
|
|
|
|
"you are using modulo operator on types that might have different signs",
|
2020-04-17 01:08:00 -05:00
|
|
|
|diag| {
|
|
|
|
diag.note("double check for expected result especially when interoperating with different languages");
|
2019-12-26 06:34:18 -06:00
|
|
|
if operand_type.is_integral() {
|
2020-04-17 01:08:00 -05:00
|
|
|
diag.note("or consider using `rem_euclid` or similar function");
|
2019-12-26 06:34:18 -06:00
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-25 15:41:36 -05:00
|
|
|
impl<'tcx> LateLintPass<'tcx> for ModuloArithmetic {
|
|
|
|
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>) {
|
2019-12-26 06:34:18 -06:00
|
|
|
match &expr.kind {
|
|
|
|
ExprKind::Binary(op, lhs, rhs) | ExprKind::AssignOp(op, lhs, rhs) => {
|
2021-10-07 04:21:30 -05:00
|
|
|
if op.node == BinOpKind::Rem {
|
2019-12-26 06:34:18 -06:00
|
|
|
let lhs_operand = analyze_operand(lhs, cx, expr);
|
|
|
|
let rhs_operand = analyze_operand(rhs, cx, expr);
|
|
|
|
if_chain! {
|
|
|
|
if let Some(lhs_operand) = lhs_operand;
|
|
|
|
if let Some(rhs_operand) = rhs_operand;
|
|
|
|
then {
|
|
|
|
check_const_operands(cx, expr, &lhs_operand, &rhs_operand);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
check_non_const_operands(cx, expr, lhs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
},
|
|
|
|
_ => {},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|