use std::sync::atomic::{AtomicU64, Ordering};
#[derive(Default, Debug)]
pub(crate) struct OptTimestamp {
latest: AtomicU64,
}
impl OptTimestamp {
pub(crate) const fn new() -> Self {
OptTimestamp {
latest: AtomicU64::new(0),
}
}
pub(crate) fn update(&self) {
self.update_to(coarsetime::Instant::now());
}
pub(crate) fn update_if_none(&self) {
let now = coarsetime::Instant::now().as_ticks();
let _ignore = self
.latest
.compare_exchange(0, now, Ordering::Relaxed, Ordering::Relaxed);
}
pub(crate) fn clear(&self) {
self.latest.store(0, Ordering::Relaxed);
}
pub(crate) fn time_since_update(&self) -> Option<coarsetime::Duration> {
self.time_since_update_at(coarsetime::Instant::now())
}
#[inline]
pub(crate) fn time_since_update_at(
&self,
now: coarsetime::Instant,
) -> Option<coarsetime::Duration> {
let earlier = self.latest.load(Ordering::Relaxed);
let now = now.as_ticks();
if now >= earlier && earlier != 0 {
Some(coarsetime::Duration::from_ticks(now - earlier))
} else {
None
}
}
#[inline]
pub(crate) fn update_to(&self, now: coarsetime::Instant) {
self.latest.fetch_max(now.as_ticks(), Ordering::Relaxed);
}
}
#[cfg(test)]
#[allow(clippy::unwrap_used)]
mod test {
#![allow(clippy::bool_assert_comparison)]
#![allow(clippy::clone_on_copy)]
#![allow(clippy::dbg_macro)]
#![allow(clippy::print_stderr)]
#![allow(clippy::print_stdout)]
#![allow(clippy::single_char_pattern)]
#![allow(clippy::unwrap_used)]
#![allow(clippy::unchecked_duration_subtraction)]
use super::*;
#[test]
fn opt_timestamp() {
use coarsetime::{Duration, Instant};
let ts = OptTimestamp::new();
assert!(ts.time_since_update().is_none());
let zero = Duration::from_secs(0);
let one_sec = Duration::from_secs(1);
let first = Instant::now();
let in_a_bit = first + one_sec * 10;
let even_later = first + one_sec * 25;
assert!(ts.time_since_update_at(first).is_none());
ts.update_to(first);
assert_eq!(ts.time_since_update_at(first), Some(zero));
assert_eq!(ts.time_since_update_at(in_a_bit), Some(one_sec * 10));
ts.update_to(in_a_bit);
assert!(ts.time_since_update_at(first).is_none());
assert_eq!(ts.time_since_update_at(in_a_bit), Some(zero));
assert_eq!(ts.time_since_update_at(even_later), Some(one_sec * 15));
ts.update_to(first);
assert!(ts.time_since_update_at(first).is_none());
assert_eq!(ts.time_since_update_at(in_a_bit), Some(zero));
assert_eq!(ts.time_since_update_at(even_later), Some(one_sec * 15));
ts.clear();
assert!(ts.time_since_update_at(first).is_none());
assert!(ts.time_since_update_at(in_a_bit).is_none());
assert!(ts.time_since_update_at(even_later).is_none());
}
#[test]
fn update_if_none() {
let ts = OptTimestamp::new();
assert!(ts.time_since_update().is_none());
let time1 = coarsetime::Instant::now();
ts.update_if_none();
let d = ts.time_since_update();
let time2 = coarsetime::Instant::now();
assert!(d.is_some());
assert!(d.unwrap() <= time2 - time1);
std::thread::sleep(std::time::Duration::from_millis(100));
let time3 = coarsetime::Instant::now();
assert!(time3 > time2);
ts.update_if_none();
let d2 = ts.time_since_update();
assert!(d2.is_some());
assert!(d2.unwrap() > d.unwrap());
}
}