no_std version of futures::future::select_all

This commit is contained in:
alexmoon 2022-04-04 19:30:16 -04:00
parent b0de865e0b
commit 04a263c700
2 changed files with 58 additions and 0 deletions

View File

@ -1,9 +1,11 @@
//! Misc utilities
mod forever;
mod select_all;
mod yield_now;
pub use forever::*;
pub use select_all::*;
pub use yield_now::*;
/// Unsafely unborrow an owned singleton out of a `&mut`.

View File

@ -0,0 +1,56 @@
#![allow(dead_code)]
use core::future::Future;
use core::pin::Pin;
use core::task::{Context, Poll};
use futures::future::FutureExt;
/// Future for the [`select_all`] function.
#[derive(Debug)]
#[must_use = "futures do nothing unless you `.await` or poll them"]
pub struct SelectAll<Fut, const N: usize> {
inner: [Fut; N],
}
impl<Fut: Unpin, const N: usize> Unpin for SelectAll<Fut, N> {}
/// Creates a new future which will select over a list of futures.
///
/// The returned future will wait for any future within `iter` to be ready. Upon
/// completion the item resolved will be returned, along with the index of the
/// future that was ready.
///
/// # Panics
///
/// This function will panic if the array specified contains no items.
pub fn select_all<Fut: Future + Unpin, const N: usize>(arr: [Fut; N]) -> Option<SelectAll<Fut, N>> {
assert!(N > 0);
Some(SelectAll { inner: arr })
}
impl<Fut, const N: usize> SelectAll<Fut, N> {
/// Consumes this combinator, returning the underlying futures.
pub fn into_inner(self) -> [Fut; N] {
self.inner
}
}
impl<Fut: Future + Unpin, const N: usize> Future for SelectAll<Fut, N> {
type Output = (Fut::Output, usize);
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let item = self
.inner
.iter_mut()
.enumerate()
.find_map(|(i, f)| match f.poll_unpin(cx) {
Poll::Pending => None,
Poll::Ready(e) => Some((i, e)),
});
match item {
Some((idx, res)) => Poll::Ready((res, idx)),
None => Poll::Pending,
}
}
}