// run-pass // needs-unwind // revisions: mir thir strict // [thir]compile-flags: -Zthir-unsafeck // [strict]compile-flags: -Zstrict-init-checks // ignore-tidy-linelength // This test checks panic emitted from `mem::{uninitialized,zeroed}`. #![feature(never_type)] #![allow(deprecated, invalid_value)] use std::{ mem::{self, MaybeUninit, ManuallyDrop}, panic, ptr::NonNull, num, }; #[allow(dead_code)] struct Foo { x: u8, y: !, } enum Bar {} #[allow(dead_code)] enum OneVariant { Variant(i32) } #[allow(dead_code, non_camel_case_types)] enum OneVariant_NonZero { Variant(i32, i32, num::NonZeroI32), DeadVariant(Bar), } #[allow(dead_code, non_camel_case_types)] enum OneVariant_Ref { Variant(&'static i32), DeadVariant(Bar), } // An `Aggregate` abi enum where 0 is not a valid discriminant. #[allow(dead_code)] #[repr(i32)] enum NoNullVariant { Variant1(i32, i32) = 1, Variant2(i32, i32) = 2, } // An enum with ScalarPair layout #[allow(dead_code)] enum LR { Left(i64), Right(i64), } #[allow(dead_code, non_camel_case_types)] enum LR_NonZero { Left(num::NonZeroI64), Right(num::NonZeroI64), } struct ZeroSized; #[allow(dead_code)] #[repr(i32)] enum ZeroIsValid { Zero(u8) = 0, One(NonNull<()>) = 1, } #[track_caller] fn test_panic_msg(op: impl (FnOnce() -> T) + panic::UnwindSafe, msg: &str) { let err = panic::catch_unwind(op).err(); assert_eq!( err.as_ref().and_then(|a| a.downcast_ref::<&str>()), Some(&msg) ); } #[track_caller] fn test_panic_msg_only_if_strict(op: impl (FnOnce() -> T) + panic::UnwindSafe, msg: &str) { let err = panic::catch_unwind(op).err(); assert_eq!( err.as_ref().and_then(|a| a.downcast_ref::<&str>()), if cfg!(strict) { Some(&msg) } else { None }, ); } fn main() { unsafe { // Uninhabited types test_panic_msg( || mem::uninitialized::(), "attempted to instantiate uninhabited type `!`" ); test_panic_msg( || mem::zeroed::(), "attempted to instantiate uninhabited type `!`" ); test_panic_msg( || MaybeUninit::::uninit().assume_init(), "attempted to instantiate uninhabited type `!`" ); test_panic_msg( || mem::uninitialized::(), "attempted to instantiate uninhabited type `Foo`" ); test_panic_msg( || mem::zeroed::(), "attempted to instantiate uninhabited type `Foo`" ); test_panic_msg( || MaybeUninit::::uninit().assume_init(), "attempted to instantiate uninhabited type `Foo`" ); test_panic_msg( || mem::uninitialized::(), "attempted to instantiate uninhabited type `Bar`" ); test_panic_msg( || mem::zeroed::(), "attempted to instantiate uninhabited type `Bar`" ); test_panic_msg( || MaybeUninit::::uninit().assume_init(), "attempted to instantiate uninhabited type `Bar`" ); test_panic_msg( || mem::uninitialized::<[Foo; 2]>(), "attempted to instantiate uninhabited type `[Foo; 2]`" ); test_panic_msg( || mem::zeroed::<[Foo; 2]>(), "attempted to instantiate uninhabited type `[Foo; 2]`" ); test_panic_msg( || MaybeUninit::<[Foo; 2]>::uninit().assume_init(), "attempted to instantiate uninhabited type `[Foo; 2]`" ); test_panic_msg( || mem::uninitialized::<[Bar; 2]>(), "attempted to instantiate uninhabited type `[Bar; 2]`" ); test_panic_msg( || mem::zeroed::<[Bar; 2]>(), "attempted to instantiate uninhabited type `[Bar; 2]`" ); test_panic_msg( || MaybeUninit::<[Bar; 2]>::uninit().assume_init(), "attempted to instantiate uninhabited type `[Bar; 2]`" ); // Types that don't allow either. test_panic_msg( || mem::zeroed::<&i32>(), "attempted to zero-initialize type `&i32`, which is invalid" ); test_panic_msg( || mem::uninitialized::<&i32>(), "attempted to leave type `&i32` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::>(), "attempted to zero-initialize type `alloc::boxed::Box<[i32; 0]>`, which is invalid" ); test_panic_msg( || mem::uninitialized::>(), "attempted to leave type `alloc::boxed::Box<[i32; 0]>` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::>(), "attempted to zero-initialize type `alloc::boxed::Box`, which is invalid" ); test_panic_msg( || mem::uninitialized::>(), "attempted to leave type `alloc::boxed::Box` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<&[i32]>(), "attempted to zero-initialize type `&[i32]`, which is invalid" ); test_panic_msg( || mem::uninitialized::<&[i32]>(), "attempted to leave type `&[i32]` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<&(u8, [u8])>(), "attempted to zero-initialize type `&(u8, [u8])`, which is invalid" ); test_panic_msg( || mem::uninitialized::<&(u8, [u8])>(), "attempted to leave type `&(u8, [u8])` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<&dyn Send>(), "attempted to zero-initialize type `&dyn core::marker::Send`, which is invalid" ); test_panic_msg( || mem::uninitialized::<&dyn Send>(), "attempted to leave type `&dyn core::marker::Send` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<*const dyn Send>(), "attempted to zero-initialize type `*const dyn core::marker::Send`, which is invalid" ); test_panic_msg( || mem::uninitialized::<*const dyn Send>(), "attempted to leave type `*const dyn core::marker::Send` uninitialized, which is invalid" ); test_panic_msg( || mem::uninitialized::(), "attempted to leave type `NoNullVariant` uninitialized, \ which is invalid" ); test_panic_msg( || mem::zeroed::(), "attempted to zero-initialize type `NoNullVariant`, \ which is invalid" ); test_panic_msg( || mem::zeroed::(), "attempted to zero-initialize type `OneVariant_Ref`, \ which is invalid" ); test_panic_msg( || mem::uninitialized::(), "attempted to leave type `OneVariant_Ref` uninitialized, which is invalid" ); // Types where both are invalid, but we allow uninit since the 0x01-filling is not LLVM UB. test_panic_msg( || mem::zeroed::(), "attempted to zero-initialize type `fn()`, which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::(), "attempted to leave type `fn()` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<&()>(), "attempted to zero-initialize type `&()`, which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::<&()>(), "attempted to leave type `&()` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<&[u8]>(), "attempted to zero-initialize type `&[u8]`, which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::<&[u8]>(), "attempted to leave type `&[u8]` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<&str>(), "attempted to zero-initialize type `&str`, which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::<&str>(), "attempted to leave type `&str` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::<(NonNull, u32, u32)>(), "attempted to zero-initialize type `(core::ptr::non_null::NonNull, u32, u32)`, \ which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::<(NonNull, u32, u32)>(), "attempted to leave type `(core::ptr::non_null::NonNull, u32, u32)` uninitialized, which is invalid" ); test_panic_msg( || mem::zeroed::(), "attempted to zero-initialize type `OneVariant_NonZero`, \ which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::(), "attempted to leave type `OneVariant_NonZero` uninitialized, which is invalid" ); // Types where both are invalid but we allow the zeroed form since it is not LLVM UB. test_panic_msg_only_if_strict( || mem::zeroed::(), "attempted to zero-initialize type `LR_NonZero`, which is invalid" ); test_panic_msg( || mem::uninitialized::(), "attempted to leave type `LR_NonZero` uninitialized, which is invalid" ); test_panic_msg_only_if_strict( || mem::zeroed::>(), "attempted to zero-initialize type `core::mem::manually_drop::ManuallyDrop`, \ which is invalid" ); test_panic_msg( || mem::uninitialized::>(), "attempted to leave type `core::mem::manually_drop::ManuallyDrop` uninitialized, \ which is invalid" ); // Some strict-only things test_panic_msg_only_if_strict( || mem::uninitialized::(), "attempted to leave type `i32` uninitialized, which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::<*const ()>(), "attempted to leave type `*const ()` uninitialized, which is invalid" ); test_panic_msg_only_if_strict( || mem::uninitialized::<[i32; 1]>(), "attempted to leave type `[i32; 1]` uninitialized, which is invalid" ); test_panic_msg_only_if_strict( || mem::zeroed::<[NonNull<()>; 1]>(), "attempted to zero-initialize type `[core::ptr::non_null::NonNull<()>; 1]`, which is invalid" ); // Types that can be zero, but not uninit (though some are mitigated). let _val = mem::zeroed::(); test_panic_msg( || mem::uninitialized::(), "attempted to leave type `LR` uninitialized, which is invalid" ); let _val = mem::zeroed::>(); test_panic_msg( || mem::uninitialized::>(), "attempted to leave type `core::mem::manually_drop::ManuallyDrop` uninitialized, which is invalid" ); let _val = mem::zeroed::(); test_panic_msg_only_if_strict( || mem::uninitialized::(), "attempted to leave type `bool` uninitialized, which is invalid" ); let _val = mem::zeroed::(); test_panic_msg_only_if_strict( || mem::uninitialized::(), "attempted to leave type `OneVariant` uninitialized, which is invalid" ); // Some things that are actually allowed. let _val = mem::zeroed::>(); let _val = mem::zeroed::>>(); let _val = mem::zeroed::<[!; 0]>(); let _val = mem::zeroed::(); let _val = mem::uninitialized::>(); let _val = mem::uninitialized::<[!; 0]>(); let _val = mem::uninitialized::<()>(); let _val = mem::uninitialized::(); } }