]> git.lizzy.rs Git - rust.git/blob - src/test/ui/traits/negative-impls/pin-unsound-issue-66544-derefmut.rs
Auto merge of #84959 - camsteffen:lint-suggest-group, r=estebank
[rust.git] / src / test / ui / traits / negative-impls / pin-unsound-issue-66544-derefmut.rs
1 // Demonstrate that "rogue" `DerefMut` impls for `&T` are not allowed.
2 //
3 // https://github.com/rust-lang/rust/issues/66544
4
5 use std::cell::Cell;
6 use std::marker::PhantomPinned;
7 use std::ops::DerefMut;
8 use std::pin::Pin;
9
10 struct MyType<'a>(Cell<Option<&'a mut MyType<'a>>>, PhantomPinned);
11
12 impl<'a> DerefMut for &'a MyType<'a> {
13     //~^ ERROR E0751
14     fn deref_mut(&mut self) -> &mut MyType<'a> {
15         self.0.take().unwrap()
16     }
17 }
18
19 fn main() {
20     let mut unpinned = MyType(Cell::new(None), PhantomPinned);
21     let bad_addr = &unpinned as *const MyType<'_> as usize;
22     let p = Box::pin(MyType(Cell::new(Some(&mut unpinned)), PhantomPinned));
23
24     // p_ref is okay: it does not point to the bad_addr
25     let mut p_ref: Pin<&MyType<'_>> = p.as_ref();
26     assert_ne!(bad_addr, &*p_ref as *const _ as usize);
27
28     // but p_mut does point to bad_addr! this is unsound
29     let p_mut: Pin<&mut MyType<'_>> = p_ref.as_mut();
30     assert_eq!(bad_addr, &*p_mut as *const _ as usize);
31
32     println!("oh no!");
33 }