1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
use std::future::Future;
use std::io;
use std::mem::{self, ManuallyDrop};
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Arc;
use std::task::{Context, RawWaker, RawWakerVTable, Waker};
use std::thread;
use std::thread::Thread;
use std::time::Instant;
use pin_utils::pin_mut;
use crate::{Timer, TimerHandle};
pub struct HelperThread {
thread: Option<thread::JoinHandle<()>>,
timer: TimerHandle,
done: Arc<AtomicBool>,
}
impl HelperThread {
pub fn new() -> io::Result<HelperThread> {
let timer = Timer::new();
let timer_handle = timer.handle();
let done = Arc::new(AtomicBool::new(false));
let done2 = done.clone();
let thread = thread::Builder::new().spawn(move || run(timer, done2))?;
Ok(HelperThread {
thread: Some(thread),
done,
timer: timer_handle,
})
}
pub fn handle(&self) -> TimerHandle {
self.timer.clone()
}
pub fn forget(mut self) {
self.thread.take();
}
}
impl Drop for HelperThread {
fn drop(&mut self) {
let thread = match self.thread.take() {
Some(thread) => thread,
None => return,
};
self.done.store(true, Ordering::SeqCst);
thread.thread().unpark();
drop(thread.join());
}
}
fn run(timer: Timer, done: Arc<AtomicBool>) {
let mut waker = current_thread_waker();
let mut cx = Context::from_waker(&mut waker);
pin_mut!(timer);
while !done.load(Ordering::SeqCst) {
drop(timer.as_mut().poll(&mut cx));
timer.advance();
match timer.next_event() {
Some(when) => {
let now = Instant::now();
if now < when {
thread::park_timeout(when - now)
} else {
}
}
None => thread::park(),
}
}
}
static VTABLE: RawWakerVTable = RawWakerVTable::new(raw_clone, raw_wake, raw_wake_by_ref, raw_drop);
fn raw_clone(ptr: *const ()) -> RawWaker {
let me = ManuallyDrop::new(unsafe { Arc::from_raw(ptr as *const Thread) });
mem::forget(me.clone());
RawWaker::new(ptr, &VTABLE)
}
fn raw_wake(ptr: *const ()) {
unsafe { Arc::from_raw(ptr as *const Thread) }.unpark()
}
fn raw_wake_by_ref(ptr: *const ()) {
ManuallyDrop::new(unsafe { Arc::from_raw(ptr as *const Thread) }).unpark()
}
fn raw_drop(ptr: *const ()) {
unsafe { Arc::from_raw(ptr as *const Thread) };
}
fn current_thread_waker() -> Waker {
let thread = Arc::new(thread::current());
unsafe { Waker::from_raw(raw_clone(Arc::into_raw(thread) as *const ())) }
}