-
Notifications
You must be signed in to change notification settings - Fork 352
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Auto merge of #2713 - RalfJung:not-unpin-fake-read, r=RalfJung
for now, do not do fake reads on non-Unpin mutable references Work-around for rust-lang/unsafe-code-guidelines#381, needed to make the new test pass. Undoes parts of #2694.
- Loading branch information
Showing
4 changed files
with
116 additions
and
54 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
17 changes: 0 additions & 17 deletions
17
tests/fail/stacked_borrows/notunpin_dereferenceable_fakeread.rs
This file was deleted.
Oops, something went wrong.
28 changes: 0 additions & 28 deletions
28
tests/fail/stacked_borrows/notunpin_dereferenceable_fakeread.stderr
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,102 @@ | ||
#![feature(pin_macro)] | ||
|
||
use std::future::*; | ||
use std::marker::PhantomPinned; | ||
use std::pin::*; | ||
use std::ptr; | ||
use std::task::*; | ||
|
||
struct Delay { | ||
delay: usize, | ||
} | ||
|
||
impl Delay { | ||
fn new(delay: usize) -> Self { | ||
Delay { delay } | ||
} | ||
} | ||
|
||
impl Future for Delay { | ||
type Output = (); | ||
fn poll(mut self: Pin<&mut Self>, _cx: &mut Context<'_>) -> Poll<()> { | ||
if self.delay > 0 { | ||
self.delay -= 1; | ||
Poll::Pending | ||
} else { | ||
Poll::Ready(()) | ||
} | ||
} | ||
} | ||
|
||
async fn do_stuff() { | ||
(&mut Delay::new(1)).await; | ||
} | ||
|
||
// Same thing implemented by hand | ||
struct DoStuff { | ||
state: usize, | ||
delay: Delay, | ||
delay_ref: *mut Delay, | ||
_marker: PhantomPinned, | ||
} | ||
|
||
impl DoStuff { | ||
fn new() -> Self { | ||
DoStuff { | ||
state: 0, | ||
delay: Delay::new(1), | ||
delay_ref: ptr::null_mut(), | ||
_marker: PhantomPinned, | ||
} | ||
} | ||
} | ||
|
||
impl Future for DoStuff { | ||
type Output = (); | ||
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<()> { | ||
unsafe { | ||
let this = self.get_unchecked_mut(); | ||
match this.state { | ||
0 => { | ||
// Set up self-ref. | ||
this.delay_ref = &mut this.delay; | ||
// Move to next state. | ||
this.state = 1; | ||
Poll::Pending | ||
} | ||
1 => { | ||
let delay = &mut *this.delay_ref; | ||
Pin::new_unchecked(delay).poll(cx) | ||
} | ||
_ => unreachable!(), | ||
} | ||
} | ||
} | ||
} | ||
|
||
fn run_fut<T>(fut: impl Future<Output = T>) -> T { | ||
use std::sync::Arc; | ||
|
||
struct MyWaker; | ||
impl Wake for MyWaker { | ||
fn wake(self: Arc<Self>) { | ||
unimplemented!() | ||
} | ||
} | ||
|
||
let waker = Waker::from(Arc::new(MyWaker)); | ||
let mut context = Context::from_waker(&waker); | ||
|
||
let mut pinned = pin!(fut); | ||
loop { | ||
match pinned.as_mut().poll(&mut context) { | ||
Poll::Pending => continue, | ||
Poll::Ready(v) => return v, | ||
} | ||
} | ||
} | ||
|
||
fn main() { | ||
run_fut(do_stuff()); | ||
run_fut(DoStuff::new()); | ||
} |