winit/src/platform_impl/web/async/channel.rs

124 lines
3.3 KiB
Rust
Raw Normal View History

2023-10-16 15:50:22 +02:00
use atomic_waker::AtomicWaker;
use std::future;
use std::rc::Rc;
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::mpsc::{self, Receiver, RecvError, SendError, Sender, TryRecvError};
use std::sync::{Arc, Mutex};
use std::task::Poll;
pub fn channel<T>() -> (AsyncSender<T>, AsyncReceiver<T>) {
let (sender, receiver) = mpsc::channel();
2023-12-22 22:20:41 +01:00
let shared = Arc::new(Shared {
2023-10-16 15:50:22 +02:00
closed: AtomicBool::new(false),
waker: AtomicWaker::new(),
});
2023-12-22 22:20:41 +01:00
let sender = AsyncSender(Arc::new(SenderInner {
sender: Mutex::new(sender),
shared: Arc::clone(&shared),
}));
2023-10-16 15:50:22 +02:00
let receiver = AsyncReceiver {
receiver: Rc::new(receiver),
2023-12-22 22:20:41 +01:00
shared,
2023-10-16 15:50:22 +02:00
};
(sender, receiver)
}
2023-12-22 22:20:41 +01:00
pub struct AsyncSender<T>(Arc<SenderInner<T>>);
struct SenderInner<T> {
2023-10-16 15:50:22 +02:00
// We need to wrap it into a `Mutex` to make it `Sync`. So the sender can't
// be accessed on the main thread, as it could block. Additionally we need
2023-12-22 22:20:41 +01:00
// to wrap `Sender` in an `Arc` to make it clonable on the main thread without
2023-10-16 15:50:22 +02:00
// having to block.
2023-12-22 22:20:41 +01:00
sender: Mutex<Sender<T>>,
shared: Arc<Shared>,
2023-10-16 15:50:22 +02:00
}
impl<T> AsyncSender<T> {
pub fn send(&self, event: T) -> Result<(), SendError<T>> {
2023-12-22 22:20:41 +01:00
self.0.sender.lock().unwrap().send(event)?;
self.0.shared.waker.wake();
2023-10-16 15:50:22 +02:00
Ok(())
}
2023-12-22 22:20:41 +01:00
}
2023-10-16 15:50:22 +02:00
2023-12-22 22:20:41 +01:00
impl<T> SenderInner<T> {
fn close(&self) {
self.shared.closed.store(true, Ordering::Relaxed);
self.shared.waker.wake();
2023-10-16 15:50:22 +02:00
}
}
impl<T> Clone for AsyncSender<T> {
fn clone(&self) -> Self {
2023-12-22 22:20:41 +01:00
Self(Arc::clone(&self.0))
}
}
impl<T> Drop for SenderInner<T> {
fn drop(&mut self) {
self.close();
2023-10-16 15:50:22 +02:00
}
}
pub struct AsyncReceiver<T> {
receiver: Rc<Receiver<T>>,
2023-12-22 22:20:41 +01:00
shared: Arc<Shared>,
2023-10-16 15:50:22 +02:00
}
impl<T> AsyncReceiver<T> {
pub async fn next(&self) -> Result<T, RecvError> {
future::poll_fn(|cx| match self.receiver.try_recv() {
Ok(event) => Poll::Ready(Ok(event)),
Err(TryRecvError::Empty) => {
2023-12-22 22:20:41 +01:00
self.shared.waker.register(cx.waker());
2023-10-16 15:50:22 +02:00
match self.receiver.try_recv() {
Ok(event) => Poll::Ready(Ok(event)),
Err(TryRecvError::Empty) => {
2023-12-22 22:20:41 +01:00
if self.shared.closed.load(Ordering::Relaxed) {
2023-10-16 15:50:22 +02:00
Poll::Ready(Err(RecvError))
} else {
Poll::Pending
}
}
Err(TryRecvError::Disconnected) => Poll::Ready(Err(RecvError)),
}
}
Err(TryRecvError::Disconnected) => Poll::Ready(Err(RecvError)),
})
.await
}
pub fn try_recv(&self) -> Result<Option<T>, RecvError> {
match self.receiver.try_recv() {
Ok(value) => Ok(Some(value)),
Err(TryRecvError::Empty) => Ok(None),
Err(TryRecvError::Disconnected) => Err(RecvError),
}
}
}
impl<T> Clone for AsyncReceiver<T> {
fn clone(&self) -> Self {
Self {
receiver: Rc::clone(&self.receiver),
2023-12-22 22:20:41 +01:00
shared: Arc::clone(&self.shared),
2023-10-16 15:50:22 +02:00
}
}
}
2023-12-22 22:20:41 +01:00
impl<T> Drop for AsyncReceiver<T> {
fn drop(&mut self) {
self.shared.closed.store(true, Ordering::Relaxed);
}
}
struct Shared {
2023-10-16 15:50:22 +02:00
closed: AtomicBool,
waker: AtomicWaker,
}