// We want to control preemption here. //@compile-flags: -Zmiri-preemption-rate=0 use std::thread::spawn; #[derive(Copy, Clone)] struct EvilSend(pub T); unsafe impl Send for EvilSend {} unsafe impl Sync for EvilSend {} extern "Rust" { fn __rust_dealloc(ptr: *mut u8, size: usize, align: usize); } pub fn main() { // Shared atomic pointer let pointer: *mut usize = Box::into_raw(Box::new(0usize)); let ptr = EvilSend(pointer); unsafe { let j1 = spawn(move || { __rust_dealloc( ptr.0 as *mut _, std::mem::size_of::(), std::mem::align_of::(), ); }); let j2 = spawn(move || { // Also an error of the form: Data race detected between Write on thread `` and Deallocate on thread `` // but the invalid allocation is detected first. *ptr.0 = 2; //~ ERROR: dereferenced after this allocation got freed }); j1.join().unwrap(); j2.join().unwrap(); } }