1use core::num::{NonZeroU32, NonZeroU64};
8use core::time::Duration;
9
10#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Hash)]
13pub struct NonZeroDuration(Duration);
14
15impl NonZeroDuration {
16    pub const unsafe fn new_unchecked(d: Duration) -> NonZeroDuration {
23        NonZeroDuration(d)
24    }
25
26    pub const fn from_secs(secs: u64) -> Option<NonZeroDuration> {
29        NonZeroDuration::new(Duration::from_secs(secs))
30    }
31
32    pub const fn from_nonzero_secs(secs: NonZeroU64) -> NonZeroDuration {
35        NonZeroDuration(Duration::from_secs(secs.get()))
36    }
37
38    pub const fn from_secs_nanos(secs: u64, nanos: u32) -> Option<NonZeroDuration> {
51        NonZeroDuration::new(Duration::new(secs, nanos))
52    }
53
54    pub const fn from_nonzero_secs_nanos(secs: NonZeroU64, nanos: NonZeroU32) -> NonZeroDuration {
66        NonZeroDuration(Duration::new(secs.get(), nanos.get()))
67    }
68
69    pub const fn from_millis(millis: u64) -> Option<NonZeroDuration> {
72        NonZeroDuration::new(Duration::from_millis(millis))
73    }
74
75    pub const fn from_nonzero_millis(millis: NonZeroU64) -> NonZeroDuration {
78        NonZeroDuration(Duration::from_millis(millis.get()))
79    }
80
81    pub const fn from_micros(micros: u64) -> Option<NonZeroDuration> {
84        NonZeroDuration::new(Duration::from_micros(micros))
85    }
86
87    pub const fn from_nonzero_micros(micros: NonZeroU64) -> NonZeroDuration {
90        NonZeroDuration(Duration::from_micros(micros.get()))
91    }
92
93    pub const fn from_nanos(nanos: u64) -> Option<NonZeroDuration> {
96        NonZeroDuration::new(Duration::from_nanos(nanos))
97    }
98
99    pub const fn from_nonzero_nanos(nanos: NonZeroU64) -> NonZeroDuration {
102        NonZeroDuration(Duration::from_nanos(nanos.get()))
103    }
104
105    pub const fn new(d: Duration) -> Option<NonZeroDuration> {
107        if d.as_nanos() == 0 {
110            return None;
111        }
112
113        Some(NonZeroDuration(d))
114    }
115
116    pub fn saturating_add(self, d: Duration) -> Self {
118        Self(self.0.saturating_add(d))
121    }
122
123    pub fn saturating_mul(self, m: NonZeroU32) -> Self {
125        Self(self.0.saturating_mul(m.into()))
127    }
128
129    pub const fn get(self) -> Duration {
131        self.0
132    }
133}
134
135impl From<NonZeroDuration> for Duration {
136    fn from(NonZeroDuration(d): NonZeroDuration) -> Duration {
137        d
138    }
139}
140
141impl core::ops::Add<Duration> for NonZeroDuration {
142    type Output = Self;
143
144    fn add(self, rhs: Duration) -> Self::Output {
145        let Self(d) = self;
146        Self(d + rhs)
147    }
148}
149
150impl core::ops::Add<NonZeroDuration> for NonZeroDuration {
151    type Output = Self;
152
153    fn add(self, rhs: NonZeroDuration) -> Self::Output {
154        self + rhs.get()
155    }
156}
157
158impl core::ops::Mul<NonZeroU32> for NonZeroDuration {
159    type Output = Self;
160
161    fn mul(self, rhs: NonZeroU32) -> Self::Output {
162        let Self(d) = self;
163        Self(d * rhs.get())
164    }
165}
166
167pub(crate) fn round_to_next_multiple_of_four(x: usize) -> usize {
170    (x + 3) & !3
171}
172
173#[cfg(test)]
174mod tests {
175    use super::*;
176
177    #[test]
178    fn non_zero_duration() {
179        assert_eq!(NonZeroDuration::new(Duration::from_secs(0)), None);
181
182        let d = Duration::from_secs(1);
184        assert_eq!(unsafe { NonZeroDuration::new_unchecked(d) }, NonZeroDuration(d));
185
186        let non_zero = NonZeroDuration::new(d);
188        assert_eq!(non_zero, Some(NonZeroDuration(d)));
189
190        let one_u64 = NonZeroU64::new(1).unwrap();
191        let expect = NonZeroDuration(Duration::from_secs(1));
192        assert_eq!(NonZeroDuration::from_secs(1), Some(expect));
193        assert_eq!(NonZeroDuration::from_nonzero_secs(one_u64), expect);
194
195        let expect = NonZeroDuration(Duration::new(1, 1));
196        assert_eq!(NonZeroDuration::from_secs_nanos(1, 1), Some(expect));
197        assert_eq!(
198            NonZeroDuration::from_nonzero_secs_nanos(one_u64, NonZeroU32::new(1).unwrap()),
199            expect
200        );
201
202        let expect = NonZeroDuration(Duration::from_millis(1));
203        assert_eq!(NonZeroDuration::from_millis(1), Some(expect));
204        assert_eq!(NonZeroDuration::from_nonzero_millis(one_u64), expect);
205
206        let expect = NonZeroDuration(Duration::from_micros(1));
207        assert_eq!(NonZeroDuration::from_micros(1), Some(expect));
208        assert_eq!(NonZeroDuration::from_nonzero_micros(one_u64), expect);
209
210        let expect = NonZeroDuration(Duration::from_nanos(1));
211        assert_eq!(NonZeroDuration::from_nanos(1), Some(expect));
212        assert_eq!(NonZeroDuration::from_nonzero_nanos(one_u64), expect);
213
214        let non_zero = non_zero.unwrap();
216        assert_eq!(d, non_zero.get());
217        assert_eq!(d, non_zero.into());
218    }
219
220    #[test]
221    fn test_next_multiple_of_four() {
222        for x in 0usize..=(core::u16::MAX - 3) as usize {
223            let y = round_to_next_multiple_of_four(x);
224            assert_eq!(y % 4, 0);
225            assert!(y >= x);
226            if x % 4 == 0 {
227                assert_eq!(x, y);
228            } else {
229                assert_eq!(x + (4 - x % 4), y);
230            }
231        }
232    }
233
234    #[test]
235    fn add_duration() {
236        let a = NonZeroDuration::new(Duration::from_secs(48291)).unwrap();
237        let b = Duration::from_secs(195811);
238        assert_eq!(Some(a + b), NonZeroDuration::new(a.get() + b));
239
240        assert_eq!(a + Duration::from_secs(0), a);
241    }
242
243    #[test]
244    fn add_nonzero_duration() {
245        let a = NonZeroDuration::new(Duration::from_secs(48291)).unwrap();
246        let b = NonZeroDuration::new(Duration::from_secs(195811)).unwrap();
247        assert_eq!(Some(a + b), NonZeroDuration::new(a.get() + b.get()));
248    }
249}