rtic_time/half_period_counter.rs
1//! Utility to implement a race condition free half-period based monotonic.
2//!
3//! # Background
4//!
5//! Monotonics are continuous and never wrap (in a reasonable amount of time), while
6//! the underlying hardware usually wraps frequently and has interrupts to indicate that
7//! a wrap happened.
8//!
9//! The biggest problem when implementing a monotonic from such hardware is that there exists
10//! a non-trivial race condition while reading data from the timer. Let's assume we increment
11//! a period counter every time an overflow interrupt happens.
12//! Which should we then read first when computing the current time? The period counter or
13//! the timer value?
14//! - When reading the timer value first, an overflow interrupt could happen before we read
15//! the period counter, causing the calculated time to be much too high
16//! - When reading the period counter first, the timer value could overflow before we
17//! read it, causing the calculated time to be much too low
18//!
19//! The reason this is non-trivil to solve is because even critical sections do not help
20//! much - the inherent problem here is that the timer value continues to change, and there
21//! is no way to read it together with the period counter in an atomic way.
22//!
23//! # Solution
24//!
25//! This module provides utilities to solve this problem in a reliable, race-condition free way.
26//! A second interrupt must be added at the half-period mark, which effectively converts the period counter
27//! to a half-period counter. This creates one bit of overlap between the
28//! timer value and the period counter, which makes it mathematically possible to solve the
29//! race condition.
30//!
31//! The following steps have to be fulfilled to make this reliable:
32//! - The period counter gets incremented twice per period; once when the timer overflow happens and once
33//! at the half-period mark. For example, a 16-bit timer would require the period counter to be
34//! incremented at the values `0x0000` and `0x8000`.
35//! - The timer value and the period counter must be in sync. After the overflow interrupt
36//! was processed, the period counter must be even, and after the half-way interrupt was
37//! processed, the period counter must be odd.
38//! - Both the overflow interrupt and the half-way interrupt must be processed within half a
39//! timer period. This means those interrupts should be the highest priority in the
40//! system - disabling them for more than half a period will cause the monotonic to misbehave.
41//!
42//! If those conditions are fulfilled, the [`calculate_now`] function will reliably
43//! return the correct time value.
44//!
45//! # Why does this work?
46//!
47//! It's complicated. In essence, this one bit of overlap gets used to make
48//! it irrelevant whether the period counter was already incremented or not.
49//! For example, during the second part of the timer period, it is irrelevant if the
50//! period counter is `2` (before the interrupt) or `3` (after the interrupt) - [`calculate_now`]
51//! will yield the same result. Then half a period later, in the first part of the next timer period,
52//! it is irrelevant if the period counter is `3` or `4` - they again will yield the same result.
53//!
54//! This means that as long as we read the period counter **before** the timer value, we will
55//! always get the correct result, given that the interrupts are not delayed by more than half a period.
56//!
57//! # Example
58//!
59//! This example takes a 16-bit timer and uses a 32-bit period counter
60//! to extend the timer to 47-bit. Note that one bit gets lost because
61//! this method requires the period counter to be increased twice per period.
62//!
63//! The resulting time value is returned as a `u64`.
64//!
65//! ```rust
66//! # fn timer_stop() {}
67//! # fn timer_reset() {}
68//! # fn timer_enable_overflow_interrupt() {}
69//! # fn timer_enable_compare_interrupt(_val: u16) {}
70//! # fn timer_start() {}
71//! # fn overflow_interrupt_happened() -> bool { false }
72//! # fn compare_interrupt_happened() -> bool { false }
73//! # fn clear_overflow_interrupt() {}
74//! # fn clear_compare_interrupt() {}
75//! # fn timer_get_value() -> u16 { 0 }
76//! use core::sync::atomic::{AtomicU32, Ordering};
77//!
78//! static HALF_PERIOD_COUNTER: AtomicU32 = AtomicU32::new(0);
79//!
80//! struct MyMonotonic;
81//!
82//! impl MyMonotonic {
83//! fn init() {
84//! timer_stop();
85//! timer_reset();
86//! HALF_PERIOD_COUNTER.store(0, Ordering::SeqCst);
87//! timer_enable_overflow_interrupt();
88//! timer_enable_compare_interrupt(0x8000);
89//! // Both the period counter and the timer are reset
90//! // to zero and the interrupts are enabled.
91//! // This means the period counter and the timer value
92//! // are in sync, so we can now enable the timer.
93//! timer_start();
94//! }
95//!
96//! fn on_interrupt() {
97//! if overflow_interrupt_happened() {
98//! clear_overflow_interrupt();
99//! let prev = HALF_PERIOD_COUNTER.fetch_add(1, Ordering::Relaxed);
100//! assert!(prev % 2 == 1, "Monotonic must have skipped an interrupt!");
101//! }
102//! if compare_interrupt_happened() {
103//! clear_compare_interrupt();
104//! let prev = HALF_PERIOD_COUNTER.fetch_add(1, Ordering::Relaxed);
105//! assert!(prev % 2 == 0, "Monotonic must have skipped an interrupt!");
106//! }
107//! }
108//!
109//! fn now() -> u64 {
110//! rtic_time::half_period_counter::calculate_now(
111//! || HALF_PERIOD_COUNTER.load(Ordering::Relaxed),
112//! || timer_get_value(),
113//! )
114//! }
115//! }
116//! ```
117//!
118
119use core::sync::atomic::{compiler_fence, Ordering};
120
121/// The value of the timer's count register.
122pub trait TimerValue {
123 /// Bit size of the timer. Does not need to be a multiple of `8`.
124 const BITS: u32;
125}
126macro_rules! impl_timer_value {
127 ($t:ty) => {
128 impl TimerValue for $t {
129 const BITS: u32 = Self::BITS;
130 }
131 };
132}
133impl_timer_value!(u8);
134impl_timer_value!(u16);
135impl_timer_value!(u32);
136impl_timer_value!(u64);
137
138/// Operations a type has to support
139/// in order to be used as the return value
140/// of [`calculate_now`].
141pub trait TimerOps: Copy {
142 /// All bits set to `1`.
143 const MAX: Self;
144 /// The lowest bit set to `1`.
145 const ONE: Self;
146 /// The `^` operation.
147 fn xor(self, other: Self) -> Self;
148 /// The `&` operation.
149 fn and(self, other: Self) -> Self;
150 /// The `+` operation.
151 fn add(self, other: Self) -> Self;
152 /// The `<<` operation.
153 fn left_shift(self, amount: u32) -> Self;
154}
155
156macro_rules! impl_timer_ops {
157 ($t:ty) => {
158 impl TimerOps for $t {
159 const MAX: Self = Self::MAX;
160 const ONE: Self = 1;
161
162 #[inline]
163 fn xor(self, other: Self) -> Self {
164 self ^ other
165 }
166
167 #[inline]
168 fn and(self, other: Self) -> Self {
169 self & other
170 }
171
172 #[inline]
173 fn add(self, other: Self) -> Self {
174 self + other
175 }
176
177 #[inline]
178 fn left_shift(self, amount: u32) -> Self {
179 self << amount
180 }
181 }
182 };
183}
184
185impl_timer_ops!(u16);
186impl_timer_ops!(u32);
187impl_timer_ops!(u64);
188impl_timer_ops!(u128);
189
190/// Calculates the current time from the half period counter and the timer value.
191///
192/// # Arguments
193///
194/// * `half_periods` - A closure/function that when called produces the period counter value. If read from an atomic, can use `Ordering::Relaxed`.
195/// * `timer_value` - A closure/function that when called produces the current timer value.
196pub fn calculate_now<P, T, F1, F2, O>(half_periods: F1, timer_value: F2) -> O
197where
198 T: TimerValue,
199 O: From<P> + From<T> + TimerOps,
200 F1: FnOnce() -> P,
201 F2: FnOnce() -> T,
202{
203 // This is timing critical; for fast-overflowing timers (like the 1MHz 16-bit timers on STM32),
204 // it could lead to erroneous behavior if preempted in between the two reads.
205 // Hence the critical section.
206 let (half_periods, timer_value) = critical_section::with(|_| {
207 // Important: half_periods **must** be read first.
208 // Otherwise the mathematical principle that prevents
209 // the race condition does not work.
210 let half_periods = O::from(half_periods());
211 compiler_fence(Ordering::Acquire);
212 let timer_value = O::from(timer_value());
213 (half_periods, timer_value)
214 });
215
216 // Credits to the `time-driver` of `embassy-stm32`.
217 //
218 // Given that our clock counter value is 32 bits.
219 //
220 // Clock timekeeping works with something we call "periods", which are time intervals
221 // of 2^31 ticks. The Clock counter value is 32 bits, so one "overflow cycle" is 2 periods.
222 //
223 // A `period` count is maintained in parallel to the Timer hardware `counter`, like this:
224 // - `period` and `counter` start at 0
225 // - `period` is incremented on overflow (at counter value 0)
226 // - `period` is incremented "midway" between overflows (at counter value 0x8000_0000)
227 //
228 // Therefore, when `period` is even, counter is in 0..0x7FFF_FFFF. When odd, counter is in 0x8000_0000..0xFFFF_FFFF
229 // This allows for now() to return the correct value even if it races an overflow.
230 //
231 // To get `now()`, `period` is read first, then `counter` is read. If the counter value matches
232 // the expected range for the `period` parity, we're done. If it doesn't, this means that
233 // a new period start has raced us between reading `period` and `counter`, so we assume the `counter` value
234 // corresponds to the next period.
235
236 let upper_half = half_periods.left_shift(T::BITS - 1);
237 let lower_half = O::ONE.left_shift(T::BITS - 1).and(upper_half);
238 upper_half.add(lower_half.xor(timer_value))
239}