2020-12-26 22:44:53 +00:00
|
|
|
use core::future::Future;
|
|
|
|
use core::pin::Pin;
|
|
|
|
use core::task::{Context, Poll};
|
2021-03-29 20:48:57 +00:00
|
|
|
use futures::{future::select, future::Either, pin_mut, Stream};
|
2020-12-26 22:44:53 +00:00
|
|
|
|
2021-07-12 01:10:01 +00:00
|
|
|
use crate::executor::raw;
|
2020-12-26 22:44:53 +00:00
|
|
|
use crate::time::{Duration, Instant};
|
|
|
|
|
2021-03-29 20:48:57 +00:00
|
|
|
pub struct TimeoutError;
|
|
|
|
pub async fn with_timeout<F: Future>(timeout: Duration, fut: F) -> Result<F::Output, TimeoutError> {
|
|
|
|
let timeout_fut = Timer::after(timeout);
|
|
|
|
pin_mut!(fut);
|
|
|
|
match select(fut, timeout_fut).await {
|
|
|
|
Either::Left((r, _)) => Ok(r),
|
|
|
|
Either::Right(_) => Err(TimeoutError),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-22 01:30:03 +00:00
|
|
|
/// A future that completes at a specified [Instant](struct.Instant.html).
|
2020-12-26 22:44:53 +00:00
|
|
|
pub struct Timer {
|
|
|
|
expires_at: Instant,
|
2020-12-29 23:56:32 +00:00
|
|
|
yielded_once: bool,
|
2020-12-26 22:44:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Timer {
|
2021-03-22 01:30:03 +00:00
|
|
|
/// Expire at specified [Instant](struct.Instant.html)
|
2020-12-26 22:44:53 +00:00
|
|
|
pub fn at(expires_at: Instant) -> Self {
|
2020-12-29 23:56:32 +00:00
|
|
|
Self {
|
|
|
|
expires_at,
|
|
|
|
yielded_once: false,
|
|
|
|
}
|
2020-12-26 22:44:53 +00:00
|
|
|
}
|
|
|
|
|
2021-03-22 01:30:03 +00:00
|
|
|
/// Expire after specified [Duration](struct.Duration.html).
|
|
|
|
/// This can be used as a `sleep` abstraction.
|
|
|
|
///
|
|
|
|
/// Example:
|
|
|
|
/// ``` no_run
|
|
|
|
/// # #![feature(trait_alias)]
|
|
|
|
/// # #![feature(type_alias_impl_trait)]
|
|
|
|
/// #
|
|
|
|
/// # fn foo() {}
|
|
|
|
/// use embassy::time::{Duration, Timer};
|
|
|
|
///
|
2021-03-29 01:00:48 +00:00
|
|
|
/// #[embassy::task]
|
2021-03-22 01:30:03 +00:00
|
|
|
/// async fn demo_sleep_seconds() {
|
|
|
|
/// // suspend this task for one second.
|
|
|
|
/// Timer::after(Duration::from_secs(1)).await;
|
|
|
|
/// }
|
|
|
|
/// ```
|
2020-12-26 22:44:53 +00:00
|
|
|
pub fn after(duration: Duration) -> Self {
|
|
|
|
Self {
|
|
|
|
expires_at: Instant::now() + duration,
|
2020-12-29 23:56:32 +00:00
|
|
|
yielded_once: false,
|
2020-12-26 22:44:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Unpin for Timer {}
|
|
|
|
|
|
|
|
impl Future for Timer {
|
|
|
|
type Output = ();
|
2020-12-29 23:56:32 +00:00
|
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
if self.yielded_once && self.expires_at <= Instant::now() {
|
2020-12-26 22:44:53 +00:00
|
|
|
Poll::Ready(())
|
|
|
|
} else {
|
2021-02-02 04:14:52 +00:00
|
|
|
unsafe { raw::register_timer(self.expires_at, cx.waker()) };
|
2020-12-29 23:56:32 +00:00
|
|
|
self.yielded_once = true;
|
2020-12-26 22:44:53 +00:00
|
|
|
Poll::Pending
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-22 01:30:03 +00:00
|
|
|
/// Asynchronous stream that yields every Duration, indefinitely.
|
|
|
|
///
|
|
|
|
/// This stream will tick at uniform intervals, even if blocking work is performed between ticks.
|
|
|
|
///
|
|
|
|
/// For instance, consider the following code fragment.
|
|
|
|
/// ``` no_run
|
|
|
|
/// # #![feature(trait_alias)]
|
|
|
|
/// # #![feature(type_alias_impl_trait)]
|
|
|
|
/// #
|
|
|
|
/// use embassy::time::{Duration, Timer};
|
|
|
|
/// # fn foo() {}
|
|
|
|
///
|
2021-03-29 01:00:48 +00:00
|
|
|
/// #[embassy::task]
|
2021-03-22 01:30:03 +00:00
|
|
|
/// async fn ticker_example_0() {
|
|
|
|
/// loop {
|
|
|
|
/// foo();
|
|
|
|
/// Timer::after(Duration::from_secs(1)).await;
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// This fragment will not call `foo` every second.
|
|
|
|
/// Instead, it will call it every second + the time it took to previously call `foo`.
|
|
|
|
///
|
|
|
|
/// Example using ticker, which will consistently call `foo` once a second.
|
|
|
|
///
|
|
|
|
/// ``` no_run
|
|
|
|
/// # #![feature(trait_alias)]
|
|
|
|
/// # #![feature(type_alias_impl_trait)]
|
|
|
|
/// #
|
|
|
|
/// use embassy::time::{Duration, Ticker};
|
|
|
|
/// use futures::StreamExt;
|
|
|
|
/// # fn foo(){}
|
|
|
|
///
|
2021-03-29 01:00:48 +00:00
|
|
|
/// #[embassy::task]
|
2021-03-22 01:30:03 +00:00
|
|
|
/// async fn ticker_example_1() {
|
|
|
|
/// let mut ticker = Ticker::every(Duration::from_secs(1));
|
|
|
|
/// loop {
|
|
|
|
/// foo();
|
|
|
|
/// ticker.next().await;
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
2020-12-26 22:44:53 +00:00
|
|
|
pub struct Ticker {
|
|
|
|
expires_at: Instant,
|
|
|
|
duration: Duration,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Ticker {
|
2021-03-22 01:30:03 +00:00
|
|
|
/// Creates a new ticker that ticks at the specified duration interval.
|
2020-12-26 22:44:53 +00:00
|
|
|
pub fn every(duration: Duration) -> Self {
|
|
|
|
let expires_at = Instant::now() + duration;
|
|
|
|
Self {
|
|
|
|
expires_at,
|
|
|
|
duration,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Unpin for Ticker {}
|
|
|
|
|
|
|
|
impl Stream for Ticker {
|
|
|
|
type Item = ();
|
|
|
|
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
|
|
|
if self.expires_at <= Instant::now() {
|
|
|
|
let dur = self.duration;
|
|
|
|
self.expires_at += dur;
|
|
|
|
Poll::Ready(Some(()))
|
|
|
|
} else {
|
2021-02-02 04:14:52 +00:00
|
|
|
unsafe { raw::register_timer(self.expires_at, cx.waker()) };
|
2020-12-26 22:44:53 +00:00
|
|
|
Poll::Pending
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|