2023-10-19 14:59:44 +00:00
|
|
|
use clippy_config::msrvs::{self, Msrv};
|
2022-04-07 18:39:59 +01:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
|
|
|
use clippy_utils::sugg::Sugg;
|
|
|
|
use rustc_errors::Applicability;
|
|
|
|
use rustc_hir::{Expr, ExprKind};
|
|
|
|
use rustc_lint::LateContext;
|
2022-06-04 13:34:07 +02:00
|
|
|
use rustc_middle::ty::{self, Ty};
|
2022-04-07 18:39:59 +01:00
|
|
|
|
|
|
|
use super::CAST_ABS_TO_UNSIGNED;
|
|
|
|
|
|
|
|
pub(super) fn check(
|
|
|
|
cx: &LateContext<'_>,
|
|
|
|
expr: &Expr<'_>,
|
|
|
|
cast_expr: &Expr<'_>,
|
|
|
|
cast_from: Ty<'_>,
|
|
|
|
cast_to: Ty<'_>,
|
2022-11-21 14:37:51 +00:00
|
|
|
msrv: &Msrv,
|
2022-04-07 18:39:59 +01:00
|
|
|
) {
|
2022-11-21 14:37:51 +00:00
|
|
|
if msrv.meets(msrvs::UNSIGNED_ABS)
|
2022-06-04 13:34:07 +02:00
|
|
|
&& let ty::Int(from) = cast_from.kind()
|
|
|
|
&& let ty::Uint(to) = cast_to.kind()
|
2022-09-01 18:43:35 +09:00
|
|
|
&& let ExprKind::MethodCall(method_path, receiver, ..) = cast_expr.kind
|
2022-06-04 13:34:07 +02:00
|
|
|
&& method_path.ident.name.as_str() == "abs"
|
|
|
|
{
|
|
|
|
let span = if from.bit_width() == to.bit_width() {
|
|
|
|
expr.span
|
|
|
|
} else {
|
|
|
|
// if the result of `.unsigned_abs` would be a different type, keep the cast
|
|
|
|
// e.g. `i64 -> usize`, `i16 -> u8`
|
|
|
|
cast_expr.span
|
|
|
|
};
|
|
|
|
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
CAST_ABS_TO_UNSIGNED,
|
|
|
|
span,
|
|
|
|
&format!("casting the result of `{cast_from}::abs()` to {cast_to}"),
|
|
|
|
"replace with",
|
2022-09-01 18:43:35 +09:00
|
|
|
format!("{}.unsigned_abs()", Sugg::hir(cx, receiver, "..").maybe_par()),
|
2022-06-04 13:34:07 +02:00
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
2022-04-07 18:39:59 +01:00
|
|
|
}
|
|
|
|
}
|