// run-pass // revisions: mirunsafeck thirunsafeck // [thirunsafeck]compile-flags: -Z thir-unsafeck #![allow(dead_code)] use std::mem::needs_drop; use std::mem::ManuallyDrop; struct NeedDrop; impl Drop for NeedDrop { fn drop(&mut self) {} } // Constant expressios allow `NoDrop` to go out of scope, // unlike a value of the interior type implementing `Drop`. static X: () = (NoDrop { inner: ManuallyDrop::new(NeedDrop) }, ()).1; const Y: () = (NoDrop { inner: ManuallyDrop::new(NeedDrop) }, ()).1; const fn _f() { (NoDrop { inner: ManuallyDrop::new(NeedDrop) }, ()).1 } // A union that scrubs the drop glue from its inner type union NoDrop { inner: ManuallyDrop } // Copy currently can't be implemented on drop-containing unions, // this may change later // https://github.com/rust-lang/rust/pull/38934#issuecomment-271219289 // // We should be able to implement Copy for NoDrop // impl Copy for NoDrop {} // impl Clone for NoDrop {fn clone(&self) -> Self { *self }} // // We should be able to implement Copy for things using NoDrop // #[derive(Copy, Clone)] struct Foo { x: NoDrop> } struct Baz { x: NoDrop>, y: Box, } union ActuallyDrop { inner: ManuallyDrop } impl Drop for ActuallyDrop { fn drop(&mut self) {} } fn main() { // NoDrop should not make needs_drop true assert!(!needs_drop::()); assert!(!needs_drop::>()); assert!(!needs_drop::>>()); // presence of other drop types should still work assert!(needs_drop::()); // drop impl on union itself should work assert!(needs_drop::>()); assert!(needs_drop::>>()); }