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
use std::{
cell::{Cell, RefCell},
future::Future,
pin::Pin,
rc::{Rc, Weak},
task::{
ready,
Poll::{self, Ready},
},
};
use derivative::Derivative;
use futures_core::Stream;
#[derive(Debug, Derivative)]
#[derivative(Default(bound = ""))]
struct Inner<E> {
event_handle: event_listener::Event,
state: RefCell<Option<E>>,
version: Cell<u64>,
}
#[derive(Debug, Derivative)]
#[derivative(Default(bound = ""))]
pub struct Sender<E> {
inner: Rc<Inner<E>>,
}
impl<E> Sender<E> {
pub fn new() -> Self {
Default::default()
}
pub fn send(&self, state: E) {
*self.inner.state.borrow_mut() = Some(state);
self.inner.version.set(self.inner.version.get() + 1);
self.inner.event_handle.notify(usize::MAX);
}
pub fn new_receiver(&self) -> Receiver<E> {
Receiver {
inner: Rc::downgrade(&self.inner),
version: self.inner.version.get(),
listener: None,
}
}
}
#[derive(Debug)]
pub struct Receiver<E> {
inner: Weak<Inner<E>>,
listener: Option<event_listener::EventListener>,
version: u64,
}
impl<E: Clone> Stream for Receiver<E> {
type Item = E;
fn poll_next(
self: std::pin::Pin<&mut Self>,
cx: &mut std::task::Context<'_>,
) -> Poll<Option<Self::Item>> {
let this = self.get_mut();
let Some(inner) = this.inner.upgrade() else {
return Ready(None)
};
loop {
if this.version != inner.version.get() {
this.listener = None;
this.version = inner.version.get();
return Ready(Some(inner.state.borrow().clone().unwrap()))
}
let listener = this
.listener
.get_or_insert_with(|| inner.event_handle.listen());
ready!(Pin::new(listener).poll(cx));
}
}
}