use rustc_abi::{Align, Size}; use rustc_middle::mir::interpret::{InterpResult, Pointer}; use rustc_middle::ty::{self, ExistentialPredicateStableCmpExt, Ty, TyCtxt, VtblEntry}; use tracing::trace; use super::util::ensure_monomorphic_enough; use super::{ InterpCx, MPlaceTy, Machine, MemPlaceMeta, OffsetMode, Projectable, interp_ok, throw_ub, }; impl<'tcx, M: Machine<'tcx>> InterpCx<'tcx, M> { /// Creates a dynamic vtable for the given type and vtable origin. This is used only for /// objects. /// /// The `dyn_ty` encodes the erased self type. Hence, if we are making an object /// `Foo + Send>` from a value of type `Foo`, then `dyn_ty` /// would be `Trait + Send`. If this list doesn't have a principal trait ref, /// we only need the basic vtable prefix (drop, size, align). pub fn get_vtable_ptr( &self, ty: Ty<'tcx>, dyn_ty: &'tcx ty::List>, ) -> InterpResult<'tcx, Pointer>> { trace!("get_vtable(ty={ty:?}, dyn_ty={dyn_ty:?})"); let (ty, dyn_ty) = self.tcx.erase_regions((ty, dyn_ty)); // All vtables must be monomorphic, bail out otherwise. ensure_monomorphic_enough(*self.tcx, ty)?; ensure_monomorphic_enough(*self.tcx, dyn_ty)?; let salt = M::get_global_alloc_salt(self, None); let vtable_symbolic_allocation = self.tcx.reserve_and_set_vtable_alloc(ty, dyn_ty, salt); let vtable_ptr = self.global_root_pointer(Pointer::from(vtable_symbolic_allocation))?; interp_ok(vtable_ptr.into()) } pub fn get_vtable_size_and_align( &self, vtable: Pointer>, expected_trait: Option<&'tcx ty::List>>, ) -> InterpResult<'tcx, (Size, Align)> { let ty = self.get_ptr_vtable_ty(vtable, expected_trait)?; let layout = self.layout_of(ty)?; assert!(layout.is_sized(), "there are no vtables for unsized types"); interp_ok((layout.size, layout.align.abi)) } pub(super) fn vtable_entries( &self, trait_: Option>, dyn_ty: Ty<'tcx>, ) -> &'tcx [VtblEntry<'tcx>] { if let Some(trait_) = trait_ { let trait_ref = trait_.with_self_ty(*self.tcx, dyn_ty); let trait_ref = self.tcx.erase_regions(self.tcx.instantiate_bound_regions_with_erased(trait_ref)); self.tcx.vtable_entries(trait_ref) } else { TyCtxt::COMMON_VTABLE_ENTRIES } } /// Check that the given vtable trait is valid for a pointer/reference/place with the given /// expected trait type. pub(super) fn check_vtable_for_type( &self, vtable_dyn_type: &'tcx ty::List>, expected_dyn_type: &'tcx ty::List>, ) -> InterpResult<'tcx> { // We check validity by comparing the lists of predicates for equality. We *could* instead // check that the dynamic type to which the vtable belongs satisfies all the expected // predicates, but that would likely be a lot slower and seems unnecessarily permissive. // FIXME: we are skipping auto traits for now, but might revisit this in the future. let mut sorted_vtable: Vec<_> = vtable_dyn_type.without_auto_traits().collect(); let mut sorted_expected: Vec<_> = expected_dyn_type.without_auto_traits().collect(); // `skip_binder` here is okay because `stable_cmp` doesn't look at binders sorted_vtable.sort_by(|a, b| a.skip_binder().stable_cmp(*self.tcx, &b.skip_binder())); sorted_vtable.dedup(); sorted_expected.sort_by(|a, b| a.skip_binder().stable_cmp(*self.tcx, &b.skip_binder())); sorted_expected.dedup(); if sorted_vtable.len() != sorted_expected.len() { throw_ub!(InvalidVTableTrait { vtable_dyn_type, expected_dyn_type }); } // This checks whether there is a subtyping relation between the predicates in either direction. // For example: // - casting between `dyn for<'a> Trait` and `dyn Trait` is OK // - casting between `dyn Trait fn(&'a u8)>` and either of the above is UB for (a_pred, b_pred) in std::iter::zip(sorted_vtable, sorted_expected) { let a_pred = self.tcx.normalize_erasing_late_bound_regions(self.typing_env, a_pred); let b_pred = self.tcx.normalize_erasing_late_bound_regions(self.typing_env, b_pred); if a_pred != b_pred { throw_ub!(InvalidVTableTrait { vtable_dyn_type, expected_dyn_type }); } } interp_ok(()) } /// Turn a place with a `dyn Trait` type into a place with the actual dynamic type. pub(super) fn unpack_dyn_trait( &self, mplace: &MPlaceTy<'tcx, M::Provenance>, expected_trait: &'tcx ty::List>, ) -> InterpResult<'tcx, MPlaceTy<'tcx, M::Provenance>> { assert!( matches!(mplace.layout.ty.kind(), ty::Dynamic(_, _, ty::Dyn)), "`unpack_dyn_trait` only makes sense on `dyn*` types" ); let vtable = mplace.meta().unwrap_meta().to_pointer(self)?; let ty = self.get_ptr_vtable_ty(vtable, Some(expected_trait))?; // This is a kind of transmute, from a place with unsized type and metadata to // a place with sized type and no metadata. let layout = self.layout_of(ty)?; let mplace = mplace.offset_with_meta( Size::ZERO, OffsetMode::Wrapping, MemPlaceMeta::None, layout, self, )?; interp_ok(mplace) } }