44 lines
1.2 KiB
Rust
44 lines
1.2 KiB
Rust
|
use async_executors::JoinHandle;
|
||
|
use core::future::Future;
|
||
|
use core::pin::Pin;
|
||
|
use core::sync::atomic::{AtomicBool, Ordering};
|
||
|
use core::task::{Context, Poll};
|
||
|
|
||
|
#[derive(Debug)]
|
||
|
pub struct MustJoinHandle<T> {
|
||
|
join_handle: JoinHandle<T>,
|
||
|
completed: AtomicBool,
|
||
|
}
|
||
|
|
||
|
impl<T> MustJoinHandle<T> {
|
||
|
pub fn new(join_handle: JoinHandle<T>) -> Self {
|
||
|
Self {
|
||
|
join_handle,
|
||
|
completed: AtomicBool::new(false),
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<T> Drop for MustJoinHandle<T> {
|
||
|
fn drop(&mut self) {
|
||
|
// panic if we haven't completed
|
||
|
if !self.completed.load(Ordering::Relaxed) {
|
||
|
panic!("MustJoinHandle was not completed upon drop. Add cooperative cancellation where appropriate to ensure this is completed before drop.")
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl<T: 'static> Future for MustJoinHandle<T> {
|
||
|
type Output = T;
|
||
|
|
||
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
||
|
match Pin::new(&mut self.join_handle).poll(cx) {
|
||
|
Poll::Ready(t) => {
|
||
|
self.completed.store(true, Ordering::Relaxed);
|
||
|
Poll::Ready(t)
|
||
|
}
|
||
|
Poll::Pending => Poll::Pending,
|
||
|
}
|
||
|
}
|
||
|
}
|