2019-08-21 09:07:27 +02:00
|
|
|
#![feature(never_type)]
|
2018-11-20 16:09:06 +01:00
|
|
|
|
2019-02-15 09:32:54 +01:00
|
|
|
use std::{future::Future, pin::Pin, task::Poll, ptr};
|
2019-04-10 17:20:54 +02:00
|
|
|
use std::task::{Waker, RawWaker, RawWakerVTable, Context};
|
2018-11-20 16:09:06 +01:00
|
|
|
|
|
|
|
// See if we can run a basic `async fn`
|
|
|
|
pub async fn foo(x: &u32, y: u32) -> u32 {
|
|
|
|
let y = &y;
|
|
|
|
let z = 9;
|
|
|
|
let z = &z;
|
2019-07-31 16:42:38 +09:00
|
|
|
let y = async { *y + *z }.await;
|
2018-11-20 16:09:06 +01:00
|
|
|
let a = 10;
|
|
|
|
let a = &a;
|
|
|
|
*x + y + *a
|
|
|
|
}
|
|
|
|
|
2019-08-07 16:39:54 -07:00
|
|
|
async fn add(x: u32, y: u32) -> u32 {
|
2019-09-11 17:13:32 +02:00
|
|
|
let a = async { x + y };
|
|
|
|
a.await
|
2019-08-07 16:39:54 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
async fn build_aggregate(a: u32, b: u32, c: u32, d: u32) -> u32 {
|
|
|
|
let x = (add(a, b).await, add(c, d).await);
|
|
|
|
x.0 + x.1
|
|
|
|
}
|
|
|
|
|
|
|
|
enum Never {}
|
|
|
|
fn never() -> Never {
|
|
|
|
panic!()
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn includes_never(crash: bool, x: u32) -> u32 {
|
|
|
|
let mut result = async { x * x }.await;
|
|
|
|
if !crash {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
#[allow(unused)]
|
|
|
|
let bad = never();
|
|
|
|
result *= async { x + x }.await;
|
|
|
|
drop(bad);
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
2019-08-09 11:01:11 +02:00
|
|
|
async fn partial_init(x: u32) -> u32 {
|
|
|
|
#[allow(unreachable_code)]
|
|
|
|
let _x: (String, !) = (String::new(), return async { x + x }.await);
|
|
|
|
}
|
|
|
|
|
2019-08-09 10:26:48 +02:00
|
|
|
fn run_fut(mut fut: impl Future<Output=u32>, output: u32) {
|
|
|
|
fn raw_waker_clone(_this: *const ()) -> RawWaker {
|
|
|
|
panic!("unimplemented");
|
|
|
|
}
|
|
|
|
fn raw_waker_wake(_this: *const ()) {
|
|
|
|
panic!("unimplemented");
|
|
|
|
}
|
|
|
|
fn raw_waker_wake_by_ref(_this: *const ()) {
|
|
|
|
panic!("unimplemented");
|
|
|
|
}
|
|
|
|
fn raw_waker_drop(_this: *const ()) {}
|
2018-11-20 16:09:06 +01:00
|
|
|
|
2019-08-09 10:26:48 +02:00
|
|
|
static RAW_WAKER: RawWakerVTable = RawWakerVTable::new(
|
|
|
|
raw_waker_clone,
|
|
|
|
raw_waker_wake,
|
|
|
|
raw_waker_wake_by_ref,
|
|
|
|
raw_waker_drop,
|
|
|
|
);
|
2018-11-20 16:09:06 +01:00
|
|
|
|
2019-04-12 22:15:55 +02:00
|
|
|
let waker = unsafe { Waker::from_raw(RawWaker::new(ptr::null(), &RAW_WAKER)) };
|
2019-04-10 17:20:54 +02:00
|
|
|
let mut context = Context::from_waker(&waker);
|
2019-08-09 10:26:48 +02:00
|
|
|
assert_eq!(unsafe { Pin::new_unchecked(&mut fut) }.poll(&mut context), Poll::Ready(output));
|
|
|
|
}
|
2019-08-07 16:39:54 -07:00
|
|
|
|
2019-08-09 10:26:48 +02:00
|
|
|
fn main() {
|
|
|
|
let x = 5;
|
|
|
|
run_fut(foo(&x, 7), 31);
|
2019-08-07 16:39:54 -07:00
|
|
|
|
2019-08-09 10:26:48 +02:00
|
|
|
run_fut(build_aggregate(1, 2, 3, 4), 10);
|
|
|
|
|
|
|
|
run_fut(includes_never(false, 4), 16);
|
2019-08-09 11:01:11 +02:00
|
|
|
|
|
|
|
run_fut(partial_init(4), 8);
|
2018-11-20 16:09:06 +01:00
|
|
|
}
|