use crate::sync::AtomicWaker;
use tokio_test::task;
use std::task::Waker;
#[allow(unused)]
trait AssertSend: Send {}
#[allow(unused)]
trait AssertSync: Sync {}
impl AssertSend for AtomicWaker {}
impl AssertSync for AtomicWaker {}
impl AssertSend for Waker {}
impl AssertSync for Waker {}
#[cfg(all(target_family = "wasm", not(target_os = "wasi")))]
use wasm_bindgen_test::wasm_bindgen_test as test;
#[test]
fn basic_usage() {
let mut waker = task::spawn(AtomicWaker::new());
waker.enter(|cx, waker| waker.register_by_ref(cx.waker()));
waker.wake();
assert!(waker.is_woken());
}
#[test]
fn wake_without_register() {
let mut waker = task::spawn(AtomicWaker::new());
waker.wake();
waker.enter(|cx, waker| waker.register_by_ref(cx.waker()));
assert!(!waker.is_woken());
}
#[test]
#[cfg_attr(target_family = "wasm", ignore)] fn failed_wake_synchronizes() {
for _ in 0..1000 {
failed_wake_synchronizes_inner();
}
}
fn failed_wake_synchronizes_inner() {
use futures::task::noop_waker_ref;
use std::sync::atomic::{AtomicBool, Ordering};
static DID_SYNCHRONIZE: AtomicBool = AtomicBool::new(false);
DID_SYNCHRONIZE.store(false, Ordering::Relaxed);
let waker = AtomicWaker::new();
waker.register_by_ref(noop_waker_ref());
std::thread::scope(|s| {
let jh = s.spawn(|| {
DID_SYNCHRONIZE.store(true, Ordering::Relaxed);
waker.take_waker()
});
waker.take_waker();
waker.register_by_ref(noop_waker_ref());
let did_synchronize = DID_SYNCHRONIZE.load(Ordering::Relaxed);
let did_take = jh.join().unwrap().is_some();
assert!(did_synchronize || did_take);
});
}
#[cfg(panic = "unwind")]
#[test]
#[cfg(not(target_family = "wasm"))] fn atomic_waker_panic_safe() {
use std::panic;
use std::ptr;
use std::task::{RawWaker, RawWakerVTable, Waker};
static PANICKING_VTABLE: RawWakerVTable = RawWakerVTable::new(
|_| panic!("clone"),
|_| unimplemented!("wake"),
|_| unimplemented!("wake_by_ref"),
|_| (),
);
static NONPANICKING_VTABLE: RawWakerVTable = RawWakerVTable::new(
|_| RawWaker::new(ptr::null(), &NONPANICKING_VTABLE),
|_| unimplemented!("wake"),
|_| unimplemented!("wake_by_ref"),
|_| (),
);
let panicking = unsafe { Waker::from_raw(RawWaker::new(ptr::null(), &PANICKING_VTABLE)) };
let nonpanicking = unsafe { Waker::from_raw(RawWaker::new(ptr::null(), &NONPANICKING_VTABLE)) };
let atomic_waker = AtomicWaker::new();
let panicking = panic::AssertUnwindSafe(&panicking);
let result = panic::catch_unwind(|| {
let panic::AssertUnwindSafe(panicking) = panicking;
atomic_waker.register_by_ref(panicking);
});
assert!(result.is_err());
assert!(atomic_waker.take_waker().is_none());
atomic_waker.register_by_ref(&nonpanicking);
assert!(atomic_waker.take_waker().is_some());
}