Skip to content

Commit d14500b

Browse files
committed
Auto merge of #2784 - bryangarza:future-sizes-tests, r=RalfJung
Add tests for moving data across await point This patch adds a few tests to assert the current behavior when passing data across an await point. This will help to test out an upcoming fix for the issue of arguments in async functions growing in size because of the generator upvar that is generated when we desugar the async function. See rust-lang/rust#62958 Also relates to rust-lang/rust#107500 FYI `@oli-obk` `@pnkfelix`
2 parents 30b2a03 + dc6be57 commit d14500b

File tree

1 file changed

+81
-0
lines changed

1 file changed

+81
-0
lines changed
+81
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
use std::future::Future;
2+
use std::ptr;
3+
4+
// This test:
5+
// - Compares addresses of non-Copy data before and after moving it
6+
// - Writes to the pointer after it has moved across the await point
7+
//
8+
// This is only meant to assert current behavior, not guarantee that this is
9+
// how it should work in the future. In fact, upcoming changes to rustc
10+
// *should* break these tests.
11+
// See: https://github.com/rust-lang/rust/issues/62958
12+
async fn data_moved_async() {
13+
async fn helper(mut data: Vec<u8>, raw_pointer: *mut Vec<u8>) {
14+
let raw_pointer2 = ptr::addr_of_mut!(data);
15+
// `raw_pointer` points to the original location where the Vec was stored in the caller.
16+
// `data` is where that Vec (to be precise, its ptr+capacity+len on-stack data)
17+
// got moved to. Those will usually not be the same since the Vec got moved twice
18+
// (into the function call, and then into the generator upvar).
19+
assert_ne!(raw_pointer, raw_pointer2);
20+
unsafe {
21+
// This writes into the `x` in `data_moved_async`, re-initializing it.
22+
std::ptr::write(raw_pointer, vec![3]);
23+
}
24+
}
25+
// Vec<T> is not Copy
26+
let mut x: Vec<u8> = vec![2];
27+
let raw_pointer = ptr::addr_of_mut!(x);
28+
helper(x, raw_pointer).await;
29+
unsafe {
30+
assert_eq!(*raw_pointer, vec![3]);
31+
// Drop to prevent leak.
32+
std::ptr::drop_in_place(raw_pointer);
33+
}
34+
}
35+
36+
// Same thing as above, but non-async.
37+
fn data_moved() {
38+
fn helper(mut data: Vec<u8>, raw_pointer: *mut Vec<u8>) {
39+
let raw_pointer2 = ptr::addr_of_mut!(data);
40+
assert_ne!(raw_pointer, raw_pointer2);
41+
unsafe {
42+
std::ptr::write(raw_pointer, vec![3]);
43+
}
44+
}
45+
46+
let mut x: Vec<u8> = vec![2];
47+
let raw_pointer = ptr::addr_of_mut!(x);
48+
helper(x, raw_pointer);
49+
unsafe {
50+
assert_eq!(*raw_pointer, vec![3]);
51+
std::ptr::drop_in_place(raw_pointer);
52+
}
53+
}
54+
55+
fn run_fut<T>(fut: impl Future<Output = T>) -> T {
56+
use std::sync::Arc;
57+
use std::task::{Context, Poll, Wake, Waker};
58+
59+
struct MyWaker;
60+
impl Wake for MyWaker {
61+
fn wake(self: Arc<Self>) {
62+
unimplemented!()
63+
}
64+
}
65+
66+
let waker = Waker::from(Arc::new(MyWaker));
67+
let mut context = Context::from_waker(&waker);
68+
69+
let mut pinned = Box::pin(fut);
70+
loop {
71+
match pinned.as_mut().poll(&mut context) {
72+
Poll::Pending => continue,
73+
Poll::Ready(v) => return v,
74+
}
75+
}
76+
}
77+
78+
fn main() {
79+
run_fut(data_moved_async());
80+
data_moved();
81+
}

0 commit comments

Comments
 (0)