2022-08-31 09:24:45 -04:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2022-12-01 18:29:38 +01:00
|
|
|
use clippy_utils::msrvs::{self, Msrv};
|
2022-08-31 09:24:45 -04:00
|
|
|
use clippy_utils::source::snippet_with_applicability;
|
2022-12-01 18:29:38 +01:00
|
|
|
use clippy_utils::{match_def_path, paths};
|
2022-08-31 09:24:45 -04:00
|
|
|
use if_chain::if_chain;
|
|
|
|
use rustc_errors::Applicability;
|
|
|
|
use rustc_hir::{def_id::DefId, Expr, ExprKind};
|
|
|
|
use rustc_lint::LateContext;
|
|
|
|
use rustc_middle::ty::{self, Ty};
|
|
|
|
|
|
|
|
use super::CAST_SLICE_FROM_RAW_PARTS;
|
|
|
|
|
|
|
|
enum RawPartsKind {
|
|
|
|
Immutable,
|
|
|
|
Mutable,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn raw_parts_kind(cx: &LateContext<'_>, did: DefId) -> Option<RawPartsKind> {
|
|
|
|
if match_def_path(cx, did, &paths::SLICE_FROM_RAW_PARTS) {
|
|
|
|
Some(RawPartsKind::Immutable)
|
|
|
|
} else if match_def_path(cx, did, &paths::SLICE_FROM_RAW_PARTS_MUT) {
|
|
|
|
Some(RawPartsKind::Mutable)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-01 18:29:38 +01:00
|
|
|
pub(super) fn check(cx: &LateContext<'_>, expr: &Expr<'_>, cast_expr: &Expr<'_>, cast_to: Ty<'_>, msrv: &Msrv) {
|
2022-08-31 09:24:45 -04:00
|
|
|
if_chain! {
|
2022-12-01 18:29:38 +01:00
|
|
|
if msrv.meets(msrvs::PTR_SLICE_RAW_PARTS);
|
2022-08-31 09:24:45 -04:00
|
|
|
if let ty::RawPtr(ptrty) = cast_to.kind();
|
|
|
|
if let ty::Slice(_) = ptrty.ty.kind();
|
|
|
|
if let ExprKind::Call(fun, [ptr_arg, len_arg]) = cast_expr.peel_blocks().kind;
|
|
|
|
if let ExprKind::Path(ref qpath) = fun.kind;
|
|
|
|
if let Some(fun_def_id) = cx.qpath_res(qpath, fun.hir_id).opt_def_id();
|
|
|
|
if let Some(rpk) = raw_parts_kind(cx, fun_def_id);
|
|
|
|
then {
|
|
|
|
let func = match rpk {
|
|
|
|
RawPartsKind::Immutable => "from_raw_parts",
|
|
|
|
RawPartsKind::Mutable => "from_raw_parts_mut"
|
|
|
|
};
|
|
|
|
let span = expr.span;
|
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
|
|
|
let ptr = snippet_with_applicability(cx, ptr_arg.span, "ptr", &mut applicability);
|
|
|
|
let len = snippet_with_applicability(cx, len_arg.span, "len", &mut applicability);
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
CAST_SLICE_FROM_RAW_PARTS,
|
|
|
|
span,
|
|
|
|
&format!("casting the result of `{func}` to {cast_to}"),
|
|
|
|
"replace with",
|
|
|
|
format!("core::ptr::slice_{func}({ptr}, {len})"),
|
|
|
|
applicability
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|