use super::utils::is_layout_incompatible; use super::UNSOUND_COLLECTION_TRANSMUTE; use crate::utils::{match_def_path, paths}; use clippy_utils::diagnostics::span_lint; use rustc_hir::Expr; use rustc_lint::LateContext; use rustc_middle::ty::{self, Ty}; // used to check for UNSOUND_COLLECTION_TRANSMUTE static COLLECTIONS: &[&[&str]] = &[ &paths::VEC, &paths::VEC_DEQUE, &paths::BINARY_HEAP, &paths::BTREESET, &paths::BTREEMAP, &paths::HASHSET, &paths::HASHMAP, ]; /// Checks for `unsound_collection_transmute` lint. /// Returns `true` if it's triggered, otherwise returns `false`. pub(super) fn check<'tcx>(cx: &LateContext<'tcx>, e: &'tcx Expr<'_>, from_ty: Ty<'tcx>, to_ty: Ty<'tcx>) -> bool { match (&from_ty.kind(), &to_ty.kind()) { (ty::Adt(from_adt, from_substs), ty::Adt(to_adt, to_substs)) => { if from_adt.did != to_adt.did || !COLLECTIONS.iter().any(|path| match_def_path(cx, to_adt.did, path)) { return false; } if from_substs .types() .zip(to_substs.types()) .any(|(from_ty, to_ty)| is_layout_incompatible(cx, from_ty, to_ty)) { span_lint( cx, UNSOUND_COLLECTION_TRANSMUTE, e.span, &format!( "transmute from `{}` to `{}` with mismatched layout is unsound", from_ty, to_ty ), ); true } else { false } }, _ => false, } }