2021-03-15 19:55:45 -05:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2022-11-21 08:37:51 -06:00
|
|
|
use clippy_utils::in_constant;
|
|
|
|
use clippy_utils::msrvs::{self, Msrv};
|
2021-03-14 18:17:44 -05:00
|
|
|
use clippy_utils::source::snippet_opt;
|
2021-03-13 17:01:03 -06:00
|
|
|
use clippy_utils::ty::is_isize_or_usize;
|
2021-03-08 20:44:52 -06:00
|
|
|
use rustc_errors::Applicability;
|
|
|
|
use rustc_hir::{Expr, ExprKind};
|
|
|
|
use rustc_lint::LateContext;
|
|
|
|
use rustc_middle::ty::{self, FloatTy, Ty};
|
|
|
|
|
|
|
|
use super::{utils, CAST_LOSSLESS};
|
|
|
|
|
2021-11-09 11:35:35 -06:00
|
|
|
pub(super) fn check(
|
|
|
|
cx: &LateContext<'_>,
|
|
|
|
expr: &Expr<'_>,
|
|
|
|
cast_op: &Expr<'_>,
|
|
|
|
cast_from: Ty<'_>,
|
|
|
|
cast_to: Ty<'_>,
|
2022-11-21 08:37:51 -06:00
|
|
|
msrv: &Msrv,
|
2021-11-09 11:35:35 -06:00
|
|
|
) {
|
|
|
|
if !should_lint(cx, expr, cast_from, cast_to, msrv) {
|
2021-03-08 20:44:52 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The suggestion is to use a function call, so if the original expression
|
|
|
|
// has parens on the outside, they are no longer needed.
|
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
2023-09-16 13:42:12 -05:00
|
|
|
let opt = snippet_opt(cx, cast_op.span.source_callsite());
|
2021-03-08 20:44:52 -06:00
|
|
|
let sugg = opt.as_ref().map_or_else(
|
|
|
|
|| {
|
|
|
|
applicability = Applicability::HasPlaceholders;
|
|
|
|
".."
|
|
|
|
},
|
|
|
|
|snip| {
|
|
|
|
if should_strip_parens(cast_op, snip) {
|
|
|
|
&snip[1..snip.len() - 1]
|
|
|
|
} else {
|
|
|
|
snip.as_str()
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2021-11-12 11:01:35 -06:00
|
|
|
let message = if cast_from.is_bool() {
|
2022-09-23 12:42:59 -05:00
|
|
|
format!("casting `{cast_from:}` to `{cast_to:}` is more cleanly stated with `{cast_to:}::from(_)`")
|
2021-11-12 11:01:35 -06:00
|
|
|
} else {
|
2022-09-23 12:42:59 -05:00
|
|
|
format!("casting `{cast_from}` to `{cast_to}` may become silently lossy if you later change the type")
|
2021-11-12 11:01:35 -06:00
|
|
|
};
|
|
|
|
|
2021-03-08 20:44:52 -06:00
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
CAST_LOSSLESS,
|
|
|
|
expr.span,
|
2021-11-12 11:01:35 -06:00
|
|
|
&message,
|
2021-03-08 20:44:52 -06:00
|
|
|
"try",
|
2022-09-23 12:42:59 -05:00
|
|
|
format!("{cast_to}::from({sugg})"),
|
2021-03-08 20:44:52 -06:00
|
|
|
applicability,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-11-21 08:37:51 -06:00
|
|
|
fn should_lint(cx: &LateContext<'_>, expr: &Expr<'_>, cast_from: Ty<'_>, cast_to: Ty<'_>, msrv: &Msrv) -> bool {
|
2021-03-08 20:44:52 -06:00
|
|
|
// Do not suggest using From in consts/statics until it is valid to do so (see #2267).
|
|
|
|
if in_constant(cx, expr.hir_id) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
match (cast_from.is_integral(), cast_to.is_integral()) {
|
|
|
|
(true, true) => {
|
|
|
|
let cast_signed_to_unsigned = cast_from.is_signed() && !cast_to.is_signed();
|
|
|
|
let from_nbits = utils::int_ty_to_nbits(cast_from, cx.tcx);
|
|
|
|
let to_nbits = utils::int_ty_to_nbits(cast_to, cx.tcx);
|
|
|
|
!is_isize_or_usize(cast_from)
|
|
|
|
&& !is_isize_or_usize(cast_to)
|
|
|
|
&& from_nbits < to_nbits
|
|
|
|
&& !cast_signed_to_unsigned
|
|
|
|
},
|
|
|
|
|
|
|
|
(true, false) => {
|
|
|
|
let from_nbits = utils::int_ty_to_nbits(cast_from, cx.tcx);
|
|
|
|
let to_nbits = if let ty::Float(FloatTy::F32) = cast_to.kind() {
|
|
|
|
32
|
|
|
|
} else {
|
|
|
|
64
|
|
|
|
};
|
2022-05-02 15:46:29 -05:00
|
|
|
!is_isize_or_usize(cast_from) && from_nbits < to_nbits
|
2021-03-08 20:44:52 -06:00
|
|
|
},
|
2022-11-21 08:37:51 -06:00
|
|
|
(false, true) if matches!(cast_from.kind(), ty::Bool) && msrv.meets(msrvs::FROM_BOOL) => true,
|
2021-03-08 20:44:52 -06:00
|
|
|
(_, _) => {
|
|
|
|
matches!(cast_from.kind(), ty::Float(FloatTy::F32)) && matches!(cast_to.kind(), ty::Float(FloatTy::F64))
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn should_strip_parens(cast_expr: &Expr<'_>, snip: &str) -> bool {
|
|
|
|
if let ExprKind::Binary(_, _, _) = cast_expr.kind {
|
|
|
|
if snip.starts_with('(') && snip.ends_with(')') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|