1 // SPDX-License-Identifier: GPL-2.0 2 3 //! Intrusive high resolution timers. 4 //! 5 //! Allows running timer callbacks without doing allocations at the time of 6 //! starting the timer. For now, only one timer per type is allowed. 7 //! 8 //! # Vocabulary 9 //! 10 //! States: 11 //! 12 //! - Stopped: initialized but not started, or cancelled, or not restarted. 13 //! - Started: initialized and started or restarted. 14 //! - Running: executing the callback. 15 //! 16 //! Operations: 17 //! 18 //! * Start 19 //! * Cancel 20 //! * Restart 21 //! 22 //! Events: 23 //! 24 //! * Expire 25 //! 26 //! ## State Diagram 27 //! 28 //! ```text 29 //! Return NoRestart 30 //! +---------------------------------------------------------------------+ 31 //! | | 32 //! | | 33 //! | | 34 //! | Return Restart | 35 //! | +------------------------+ | 36 //! | | | | 37 //! | | | | 38 //! v v | | 39 //! +-----------------+ Start +------------------+ +--------+-----+--+ 40 //! | +---------------->| | | | 41 //! Init | | | | Expire | | 42 //! --------->| Stopped | | Started +---------->| Running | 43 //! | | Cancel | | | | 44 //! | |<----------------+ | | | 45 //! +-----------------+ +---------------+--+ +-----------------+ 46 //! ^ | 47 //! | | 48 //! +---------+ 49 //! Restart 50 //! ``` 51 //! 52 //! 53 //! A timer is initialized in the **stopped** state. A stopped timer can be 54 //! **started** by the `start` operation, with an **expiry** time. After the 55 //! `start` operation, the timer is in the **started** state. When the timer 56 //! **expires**, the timer enters the **running** state and the handler is 57 //! executed. After the handler has returned, the timer may enter the 58 //! **started* or **stopped** state, depending on the return value of the 59 //! handler. A timer in the **started** or **running** state may be **canceled** 60 //! by the `cancel` operation. A timer that is cancelled enters the **stopped** 61 //! state. 62 //! 63 //! A `cancel` or `restart` operation on a timer in the **running** state takes 64 //! effect after the handler has returned and the timer has transitioned 65 //! out of the **running** state. 66 //! 67 //! A `restart` operation on a timer in the **stopped** state is equivalent to a 68 //! `start` operation. 69 //! 70 //! When a type implements both `HrTimerPointer` and `Clone`, it is possible to 71 //! issue the `start` operation while the timer is in the **started** state. In 72 //! this case the `start` operation is equivalent to the `restart` operation. 73 //! 74 //! # Examples 75 //! 76 //! ## Using an intrusive timer living in a [`Box`] 77 //! 78 //! ``` 79 //! # use kernel::{ 80 //! # alloc::flags, 81 //! # impl_has_hr_timer, 82 //! # prelude::*, 83 //! # sync::{ 84 //! # atomic::{ordering, Atomic}, 85 //! # completion::Completion, 86 //! # Arc, 87 //! # }, 88 //! # time::{ 89 //! # hrtimer::{ 90 //! # RelativeMode, HrTimer, HrTimerCallback, HrTimerPointer, 91 //! # HrTimerRestart, HrTimerCallbackContext 92 //! # }, 93 //! # Delta, Monotonic, 94 //! # }, 95 //! # }; 96 //! 97 //! #[pin_data] 98 //! struct Shared { 99 //! #[pin] 100 //! flag: Atomic<u64>, 101 //! #[pin] 102 //! cond: Completion, 103 //! } 104 //! 105 //! impl Shared { 106 //! fn new() -> impl PinInit<Self> { 107 //! pin_init!(Self { 108 //! flag <- Atomic::new(0), 109 //! cond <- Completion::new(), 110 //! }) 111 //! } 112 //! } 113 //! 114 //! #[pin_data] 115 //! struct BoxIntrusiveHrTimer { 116 //! #[pin] 117 //! timer: HrTimer<Self>, 118 //! shared: Arc<Shared>, 119 //! } 120 //! 121 //! impl BoxIntrusiveHrTimer { 122 //! fn new() -> impl PinInit<Self, kernel::error::Error> { 123 //! try_pin_init!(Self { 124 //! timer <- HrTimer::new(), 125 //! shared: Arc::pin_init(Shared::new(), flags::GFP_KERNEL)?, 126 //! }) 127 //! } 128 //! } 129 //! 130 //! impl HrTimerCallback for BoxIntrusiveHrTimer { 131 //! type Pointer<'a> = Pin<KBox<Self>>; 132 //! 133 //! fn run(this: Pin<&mut Self>, _ctx: HrTimerCallbackContext<'_, Self>) -> HrTimerRestart { 134 //! pr_info!("Timer called\n"); 135 //! 136 //! let flag = this.shared.flag.fetch_add(1, ordering::Full); 137 //! this.shared.cond.complete_all(); 138 //! 139 //! if flag == 4 { 140 //! HrTimerRestart::NoRestart 141 //! } else { 142 //! HrTimerRestart::Restart 143 //! } 144 //! } 145 //! } 146 //! 147 //! impl_has_hr_timer! { 148 //! impl HasHrTimer<Self> for BoxIntrusiveHrTimer { 149 //! mode: RelativeMode<Monotonic>, field: self.timer 150 //! } 151 //! } 152 //! 153 //! let has_timer = Box::pin_init(BoxIntrusiveHrTimer::new(), GFP_KERNEL)?; 154 //! let shared = has_timer.shared.clone(); 155 //! let _handle = has_timer.start(Delta::from_micros(200)); 156 //! 157 //! while shared.flag.load(ordering::Relaxed) != 5 { 158 //! shared.cond.wait_for_completion(); 159 //! } 160 //! 161 //! pr_info!("Counted to 5\n"); 162 //! # Ok::<(), kernel::error::Error>(()) 163 //! ``` 164 //! 165 //! ## Using an intrusive timer in an [`Arc`] 166 //! 167 //! ``` 168 //! # use kernel::{ 169 //! # alloc::flags, 170 //! # impl_has_hr_timer, 171 //! # prelude::*, 172 //! # sync::{ 173 //! # atomic::{ordering, Atomic}, 174 //! # completion::Completion, 175 //! # Arc, ArcBorrow, 176 //! # }, 177 //! # time::{ 178 //! # hrtimer::{ 179 //! # RelativeMode, HrTimer, HrTimerCallback, HrTimerPointer, HrTimerRestart, 180 //! # HasHrTimer, HrTimerCallbackContext 181 //! # }, 182 //! # Delta, Monotonic, 183 //! # }, 184 //! # }; 185 //! 186 //! #[pin_data] 187 //! struct ArcIntrusiveHrTimer { 188 //! #[pin] 189 //! timer: HrTimer<Self>, 190 //! #[pin] 191 //! flag: Atomic<u64>, 192 //! #[pin] 193 //! cond: Completion, 194 //! } 195 //! 196 //! impl ArcIntrusiveHrTimer { 197 //! fn new() -> impl PinInit<Self> { 198 //! pin_init!(Self { 199 //! timer <- HrTimer::new(), 200 //! flag <- Atomic::new(0), 201 //! cond <- Completion::new(), 202 //! }) 203 //! } 204 //! } 205 //! 206 //! impl HrTimerCallback for ArcIntrusiveHrTimer { 207 //! type Pointer<'a> = Arc<Self>; 208 //! 209 //! fn run( 210 //! this: ArcBorrow<'_, Self>, 211 //! _ctx: HrTimerCallbackContext<'_, Self>, 212 //! ) -> HrTimerRestart { 213 //! pr_info!("Timer called\n"); 214 //! 215 //! let flag = this.flag.fetch_add(1, ordering::Full); 216 //! this.cond.complete_all(); 217 //! 218 //! if flag == 4 { 219 //! HrTimerRestart::NoRestart 220 //! } else { 221 //! HrTimerRestart::Restart 222 //! } 223 //! } 224 //! } 225 //! 226 //! impl_has_hr_timer! { 227 //! impl HasHrTimer<Self> for ArcIntrusiveHrTimer { 228 //! mode: RelativeMode<Monotonic>, field: self.timer 229 //! } 230 //! } 231 //! 232 //! let has_timer = Arc::pin_init(ArcIntrusiveHrTimer::new(), GFP_KERNEL)?; 233 //! let _handle = has_timer.clone().start(Delta::from_micros(200)); 234 //! 235 //! while has_timer.flag.load(ordering::Relaxed) != 5 { 236 //! has_timer.cond.wait_for_completion(); 237 //! } 238 //! 239 //! pr_info!("Counted to 5\n"); 240 //! # Ok::<(), kernel::error::Error>(()) 241 //! ``` 242 //! 243 //! ## Using a stack-based timer 244 //! 245 //! ``` 246 //! # use kernel::{ 247 //! # impl_has_hr_timer, 248 //! # prelude::*, 249 //! # sync::{ 250 //! # atomic::{ordering, Atomic}, 251 //! # completion::Completion, 252 //! # }, 253 //! # time::{ 254 //! # hrtimer::{ 255 //! # ScopedHrTimerPointer, HrTimer, HrTimerCallback, HrTimerPointer, HrTimerRestart, 256 //! # HasHrTimer, RelativeMode, HrTimerCallbackContext 257 //! # }, 258 //! # Delta, Monotonic, 259 //! # }, 260 //! # }; 261 //! # use pin_init::stack_pin_init; 262 //! 263 //! #[pin_data] 264 //! struct IntrusiveHrTimer { 265 //! #[pin] 266 //! timer: HrTimer<Self>, 267 //! #[pin] 268 //! flag: Atomic<u64>, 269 //! #[pin] 270 //! cond: Completion, 271 //! } 272 //! 273 //! impl IntrusiveHrTimer { 274 //! fn new() -> impl PinInit<Self> { 275 //! pin_init!(Self { 276 //! timer <- HrTimer::new(), 277 //! flag <- Atomic::new(0), 278 //! cond <- Completion::new(), 279 //! }) 280 //! } 281 //! } 282 //! 283 //! impl HrTimerCallback for IntrusiveHrTimer { 284 //! type Pointer<'a> = Pin<&'a Self>; 285 //! 286 //! fn run(this: Pin<&Self>, _ctx: HrTimerCallbackContext<'_, Self>) -> HrTimerRestart { 287 //! pr_info!("Timer called\n"); 288 //! 289 //! this.flag.store(1, ordering::Release); 290 //! this.cond.complete_all(); 291 //! 292 //! HrTimerRestart::NoRestart 293 //! } 294 //! } 295 //! 296 //! impl_has_hr_timer! { 297 //! impl HasHrTimer<Self> for IntrusiveHrTimer { 298 //! mode: RelativeMode<Monotonic>, field: self.timer 299 //! } 300 //! } 301 //! 302 //! stack_pin_init!( let has_timer = IntrusiveHrTimer::new() ); 303 //! has_timer.as_ref().start_scoped(Delta::from_micros(200), || { 304 //! while has_timer.flag.load(ordering::Relaxed) != 1 { 305 //! has_timer.cond.wait_for_completion(); 306 //! } 307 //! }); 308 //! 309 //! pr_info!("Flag raised\n"); 310 //! # Ok::<(), kernel::error::Error>(()) 311 //! ``` 312 //! 313 //! ## Using a mutable stack-based timer 314 //! 315 //! ``` 316 //! # use kernel::{ 317 //! # alloc::flags, 318 //! # impl_has_hr_timer, 319 //! # prelude::*, 320 //! # sync::{ 321 //! # atomic::{ordering, Atomic}, 322 //! # completion::Completion, 323 //! # Arc, 324 //! # }, 325 //! # time::{ 326 //! # hrtimer::{ 327 //! # ScopedHrTimerPointer, HrTimer, HrTimerCallback, HrTimerPointer, HrTimerRestart, 328 //! # HasHrTimer, RelativeMode, HrTimerCallbackContext 329 //! # }, 330 //! # Delta, Monotonic, 331 //! # }, 332 //! # }; 333 //! # use pin_init::stack_try_pin_init; 334 //! 335 //! #[pin_data] 336 //! struct Shared { 337 //! #[pin] 338 //! flag: Atomic<u64>, 339 //! #[pin] 340 //! cond: Completion, 341 //! } 342 //! 343 //! impl Shared { 344 //! fn new() -> impl PinInit<Self> { 345 //! pin_init!(Self { 346 //! flag <- Atomic::new(0), 347 //! cond <- Completion::new(), 348 //! }) 349 //! } 350 //! } 351 //! 352 //! #[pin_data] 353 //! struct IntrusiveHrTimer { 354 //! #[pin] 355 //! timer: HrTimer<Self>, 356 //! shared: Arc<Shared>, 357 //! } 358 //! 359 //! impl IntrusiveHrTimer { 360 //! fn new() -> impl PinInit<Self, kernel::error::Error> { 361 //! try_pin_init!(Self { 362 //! timer <- HrTimer::new(), 363 //! shared: Arc::pin_init(Shared::new(), flags::GFP_KERNEL)?, 364 //! }) 365 //! } 366 //! } 367 //! 368 //! impl HrTimerCallback for IntrusiveHrTimer { 369 //! type Pointer<'a> = Pin<&'a mut Self>; 370 //! 371 //! fn run(this: Pin<&mut Self>, _ctx: HrTimerCallbackContext<'_, Self>) -> HrTimerRestart { 372 //! pr_info!("Timer called\n"); 373 //! 374 //! let flag = this.shared.flag.fetch_add(1, ordering::Full); 375 //! this.shared.cond.complete_all(); 376 //! 377 //! if flag == 4 { 378 //! HrTimerRestart::NoRestart 379 //! } else { 380 //! HrTimerRestart::Restart 381 //! } 382 //! } 383 //! } 384 //! 385 //! impl_has_hr_timer! { 386 //! impl HasHrTimer<Self> for IntrusiveHrTimer { 387 //! mode: RelativeMode<Monotonic>, field: self.timer 388 //! } 389 //! } 390 //! 391 //! stack_try_pin_init!( let has_timer =? IntrusiveHrTimer::new() ); 392 //! let shared = has_timer.shared.clone(); 393 //! 394 //! has_timer.as_mut().start_scoped(Delta::from_micros(200), || { 395 //! while shared.flag.load(ordering::Relaxed) != 5 { 396 //! shared.cond.wait_for_completion(); 397 //! } 398 //! }); 399 //! 400 //! pr_info!("Counted to 5\n"); 401 //! # Ok::<(), kernel::error::Error>(()) 402 //! ``` 403 //! 404 //! [`Arc`]: kernel::sync::Arc 405 406 use super::{ClockSource, Delta, Instant}; 407 use crate::{prelude::*, types::Opaque}; 408 use core::{marker::PhantomData, ptr::NonNull}; 409 use pin_init::PinInit; 410 411 /// A type-alias to refer to the [`Instant<C>`] for a given `T` from [`HrTimer<T>`]. 412 /// 413 /// Where `C` is the [`ClockSource`] of the [`HrTimer`]. 414 pub type HrTimerInstant<T> = Instant<<<T as HasHrTimer<T>>::TimerMode as HrTimerMode>::Clock>; 415 416 /// A timer backed by a C `struct hrtimer`. 417 /// 418 /// # Invariants 419 /// 420 /// * `self.timer` is initialized by `bindings::hrtimer_setup`. 421 #[pin_data] 422 #[repr(C)] 423 pub struct HrTimer<T> { 424 #[pin] 425 timer: Opaque<bindings::hrtimer>, 426 _t: PhantomData<T>, 427 } 428 429 // SAFETY: Ownership of an `HrTimer` can be moved to other threads and 430 // used/dropped from there. 431 unsafe impl<T> Send for HrTimer<T> {} 432 433 // SAFETY: Timer operations are locked on the C side, so it is safe to operate 434 // on a timer from multiple threads. 435 unsafe impl<T> Sync for HrTimer<T> {} 436 437 impl<T> HrTimer<T> { 438 /// Return an initializer for a new timer instance. new() -> impl PinInit<Self> where T: HrTimerCallback, T: HasHrTimer<T>,439 pub fn new() -> impl PinInit<Self> 440 where 441 T: HrTimerCallback, 442 T: HasHrTimer<T>, 443 { 444 pin_init!(Self { 445 // INVARIANT: We initialize `timer` with `hrtimer_setup` below. 446 timer <- Opaque::ffi_init(move |place: *mut bindings::hrtimer| { 447 // SAFETY: By design of `pin_init!`, `place` is a pointer to a 448 // live allocation. hrtimer_setup will initialize `place` and 449 // does not require `place` to be initialized prior to the call. 450 unsafe { 451 bindings::hrtimer_setup( 452 place, 453 Some(T::Pointer::run), 454 <<T as HasHrTimer<T>>::TimerMode as HrTimerMode>::Clock::ID, 455 <T as HasHrTimer<T>>::TimerMode::C_MODE, 456 ); 457 } 458 }), 459 _t: PhantomData, 460 }) 461 } 462 463 /// Get a pointer to the contained `bindings::hrtimer`. 464 /// 465 /// This function is useful to get access to the value without creating 466 /// intermediate references. 467 /// 468 /// # Safety 469 /// 470 /// `this` must point to a live allocation of at least the size of `Self`. raw_get(this: *const Self) -> *mut bindings::hrtimer471 unsafe fn raw_get(this: *const Self) -> *mut bindings::hrtimer { 472 // SAFETY: The field projection to `timer` does not go out of bounds, 473 // because the caller of this function promises that `this` points to an 474 // allocation of at least the size of `Self`. 475 unsafe { Opaque::cast_into(core::ptr::addr_of!((*this).timer)) } 476 } 477 478 /// Cancel an initialized and potentially running timer. 479 /// 480 /// If the timer handler is running, this function will block until the 481 /// handler returns. 482 /// 483 /// Note that the timer might be started by a concurrent start operation. If 484 /// so, the timer might not be in the **stopped** state when this function 485 /// returns. 486 /// 487 /// Users of the `HrTimer` API would not usually call this method directly. 488 /// Instead they would use the safe [`HrTimerHandle::cancel`] on the handle 489 /// returned when the timer was started. 490 /// 491 /// This function is useful to get access to the value without creating 492 /// intermediate references. 493 /// 494 /// # Safety 495 /// 496 /// `this` must point to a valid `Self`. raw_cancel(this: *const Self) -> bool497 pub(crate) unsafe fn raw_cancel(this: *const Self) -> bool { 498 // SAFETY: `this` points to an allocation of at least `HrTimer` size. 499 let c_timer_ptr = unsafe { HrTimer::raw_get(this) }; 500 501 // If the handler is running, this will wait for the handler to return 502 // before returning. 503 // SAFETY: `c_timer_ptr` is initialized and valid. Synchronization is 504 // handled on the C side. 505 unsafe { bindings::hrtimer_cancel(c_timer_ptr) != 0 } 506 } 507 508 /// Forward the timer expiry for a given timer pointer. 509 /// 510 /// # Safety 511 /// 512 /// - `self_ptr` must point to a valid `Self`. 513 /// - The caller must either have exclusive access to the data pointed at by `self_ptr`, or be 514 /// within the context of the timer callback. 515 #[inline] raw_forward(self_ptr: *mut Self, now: HrTimerInstant<T>, interval: Delta) -> u64 where T: HasHrTimer<T>,516 unsafe fn raw_forward(self_ptr: *mut Self, now: HrTimerInstant<T>, interval: Delta) -> u64 517 where 518 T: HasHrTimer<T>, 519 { 520 // SAFETY: 521 // * The C API requirements for this function are fulfilled by our safety contract. 522 // * `self_ptr` is guaranteed to point to a valid `Self` via our safety contract 523 unsafe { 524 bindings::hrtimer_forward(Self::raw_get(self_ptr), now.as_nanos(), interval.as_nanos()) 525 } 526 } 527 528 /// Conditionally forward the timer. 529 /// 530 /// If the timer expires after `now`, this function does nothing and returns 0. If the timer 531 /// expired at or before `now`, this function forwards the timer by `interval` until the timer 532 /// expires after `now` and then returns the number of times the timer was forwarded by 533 /// `interval`. 534 /// 535 /// This function is mainly useful for timer types which can provide exclusive access to the 536 /// timer when the timer is not running. For forwarding the timer from within the timer callback 537 /// context, see [`HrTimerCallbackContext::forward()`]. 538 /// 539 /// Returns the number of overruns that occurred as a result of the timer expiry change. forward(self: Pin<&mut Self>, now: HrTimerInstant<T>, interval: Delta) -> u64 where T: HasHrTimer<T>,540 pub fn forward(self: Pin<&mut Self>, now: HrTimerInstant<T>, interval: Delta) -> u64 541 where 542 T: HasHrTimer<T>, 543 { 544 // SAFETY: `raw_forward` does not move `Self` 545 let this = unsafe { self.get_unchecked_mut() }; 546 547 // SAFETY: By existence of `Pin<&mut Self>`, the pointer passed to `raw_forward` points to a 548 // valid `Self` that we have exclusive access to. 549 unsafe { Self::raw_forward(this, now, interval) } 550 } 551 552 /// Conditionally forward the timer. 553 /// 554 /// This is a variant of [`forward()`](Self::forward) that uses an interval after the current 555 /// time of the base clock for the [`HrTimer`]. forward_now(self: Pin<&mut Self>, interval: Delta) -> u64 where T: HasHrTimer<T>,556 pub fn forward_now(self: Pin<&mut Self>, interval: Delta) -> u64 557 where 558 T: HasHrTimer<T>, 559 { 560 self.forward(HrTimerInstant::<T>::now(), interval) 561 } 562 563 /// Return the time expiry for this [`HrTimer`]. 564 /// 565 /// This value should only be used as a snapshot, as the actual expiry time could change after 566 /// this function is called. expires(&self) -> HrTimerInstant<T> where T: HasHrTimer<T>,567 pub fn expires(&self) -> HrTimerInstant<T> 568 where 569 T: HasHrTimer<T>, 570 { 571 // SAFETY: `self` is an immutable reference and thus always points to a valid `HrTimer`. 572 let c_timer_ptr = unsafe { HrTimer::raw_get(self) }; 573 574 // SAFETY: 575 // - Timers cannot have negative ktime_t values as their expiration time. 576 // - There's no actual locking here, a racy read is fine and expected 577 unsafe { 578 Instant::from_ktime( 579 // This `read_volatile` is intended to correspond to a READ_ONCE call. 580 // FIXME(read_once): Replace with `read_once` when available on the Rust side. 581 core::ptr::read_volatile(&raw const ((*c_timer_ptr).node.expires)), 582 ) 583 } 584 } 585 } 586 587 /// Implemented by pointer types that point to structs that contain a [`HrTimer`]. 588 /// 589 /// `Self` must be [`Sync`] because it is passed to timer callbacks in another 590 /// thread of execution (hard or soft interrupt context). 591 /// 592 /// Starting a timer returns a [`HrTimerHandle`] that can be used to manipulate 593 /// the timer. Note that it is OK to call the start function repeatedly, and 594 /// that more than one [`HrTimerHandle`] associated with a [`HrTimerPointer`] may 595 /// exist. A timer can be manipulated through any of the handles, and a handle 596 /// may represent a cancelled timer. 597 pub trait HrTimerPointer: Sync + Sized { 598 /// The operational mode associated with this timer. 599 /// 600 /// This defines how the expiration value is interpreted. 601 type TimerMode: HrTimerMode; 602 603 /// A handle representing a started or restarted timer. 604 /// 605 /// If the timer is running or if the timer callback is executing when the 606 /// handle is dropped, the drop method of [`HrTimerHandle`] should not return 607 /// until the timer is stopped and the callback has completed. 608 /// 609 /// Note: When implementing this trait, consider that it is not unsafe to 610 /// leak the handle. 611 type TimerHandle: HrTimerHandle; 612 613 /// Start the timer with expiry after `expires` time units. If the timer was 614 /// already running, it is restarted with the new expiry time. start(self, expires: <Self::TimerMode as HrTimerMode>::Expires) -> Self::TimerHandle615 fn start(self, expires: <Self::TimerMode as HrTimerMode>::Expires) -> Self::TimerHandle; 616 } 617 618 /// Unsafe version of [`HrTimerPointer`] for situations where leaking the 619 /// [`HrTimerHandle`] returned by `start` would be unsound. This is the case for 620 /// stack allocated timers. 621 /// 622 /// Typical implementers are pinned references such as [`Pin<&T>`]. 623 /// 624 /// # Safety 625 /// 626 /// Implementers of this trait must ensure that instances of types implementing 627 /// [`UnsafeHrTimerPointer`] outlives any associated [`HrTimerPointer::TimerHandle`] 628 /// instances. 629 pub unsafe trait UnsafeHrTimerPointer: Sync + Sized { 630 /// The operational mode associated with this timer. 631 /// 632 /// This defines how the expiration value is interpreted. 633 type TimerMode: HrTimerMode; 634 635 /// A handle representing a running timer. 636 /// 637 /// # Safety 638 /// 639 /// If the timer is running, or if the timer callback is executing when the 640 /// handle is dropped, the drop method of [`Self::TimerHandle`] must not return 641 /// until the timer is stopped and the callback has completed. 642 type TimerHandle: HrTimerHandle; 643 644 /// Start the timer after `expires` time units. If the timer was already 645 /// running, it is restarted at the new expiry time. 646 /// 647 /// # Safety 648 /// 649 /// Caller promises keep the timer structure alive until the timer is dead. 650 /// Caller can ensure this by not leaking the returned [`Self::TimerHandle`]. start(self, expires: <Self::TimerMode as HrTimerMode>::Expires) -> Self::TimerHandle651 unsafe fn start(self, expires: <Self::TimerMode as HrTimerMode>::Expires) -> Self::TimerHandle; 652 } 653 654 /// A trait for stack allocated timers. 655 /// 656 /// # Safety 657 /// 658 /// Implementers must ensure that `start_scoped` does not return until the 659 /// timer is dead and the timer handler is not running. 660 pub unsafe trait ScopedHrTimerPointer { 661 /// The operational mode associated with this timer. 662 /// 663 /// This defines how the expiration value is interpreted. 664 type TimerMode: HrTimerMode; 665 666 /// Start the timer to run after `expires` time units and immediately 667 /// after call `f`. When `f` returns, the timer is cancelled. start_scoped<T, F>(self, expires: <Self::TimerMode as HrTimerMode>::Expires, f: F) -> T where F: FnOnce() -> T668 fn start_scoped<T, F>(self, expires: <Self::TimerMode as HrTimerMode>::Expires, f: F) -> T 669 where 670 F: FnOnce() -> T; 671 } 672 673 // SAFETY: By the safety requirement of [`UnsafeHrTimerPointer`], dropping the 674 // handle returned by [`UnsafeHrTimerPointer::start`] ensures that the timer is 675 // killed. 676 unsafe impl<T> ScopedHrTimerPointer for T 677 where 678 T: UnsafeHrTimerPointer, 679 { 680 type TimerMode = T::TimerMode; 681 start_scoped<U, F>( self, expires: <<T as UnsafeHrTimerPointer>::TimerMode as HrTimerMode>::Expires, f: F, ) -> U where F: FnOnce() -> U,682 fn start_scoped<U, F>( 683 self, 684 expires: <<T as UnsafeHrTimerPointer>::TimerMode as HrTimerMode>::Expires, 685 f: F, 686 ) -> U 687 where 688 F: FnOnce() -> U, 689 { 690 // SAFETY: We drop the timer handle below before returning. 691 let handle = unsafe { UnsafeHrTimerPointer::start(self, expires) }; 692 let t = f(); 693 drop(handle); 694 t 695 } 696 } 697 698 /// Implemented by [`HrTimerPointer`] implementers to give the C timer callback a 699 /// function to call. 700 // This is split from `HrTimerPointer` to make it easier to specify trait bounds. 701 pub trait RawHrTimerCallback { 702 /// Type of the parameter passed to [`HrTimerCallback::run`]. It may be 703 /// [`Self`], or a pointer type derived from [`Self`]. 704 type CallbackTarget<'a>; 705 706 /// Callback to be called from C when timer fires. 707 /// 708 /// # Safety 709 /// 710 /// Only to be called by C code in the `hrtimer` subsystem. `this` must point 711 /// to the `bindings::hrtimer` structure that was used to start the timer. run(this: *mut bindings::hrtimer) -> bindings::hrtimer_restart712 unsafe extern "C" fn run(this: *mut bindings::hrtimer) -> bindings::hrtimer_restart; 713 } 714 715 /// Implemented by structs that can be the target of a timer callback. 716 pub trait HrTimerCallback { 717 /// The type whose [`RawHrTimerCallback::run`] method will be invoked when 718 /// the timer expires. 719 type Pointer<'a>: RawHrTimerCallback; 720 721 /// Called by the timer logic when the timer fires. run( this: <Self::Pointer<'_> as RawHrTimerCallback>::CallbackTarget<'_>, ctx: HrTimerCallbackContext<'_, Self>, ) -> HrTimerRestart where Self: Sized, Self: HasHrTimer<Self>722 fn run( 723 this: <Self::Pointer<'_> as RawHrTimerCallback>::CallbackTarget<'_>, 724 ctx: HrTimerCallbackContext<'_, Self>, 725 ) -> HrTimerRestart 726 where 727 Self: Sized, 728 Self: HasHrTimer<Self>; 729 } 730 731 /// A handle representing a potentially running timer. 732 /// 733 /// More than one handle representing the same timer might exist. 734 /// 735 /// # Safety 736 /// 737 /// When dropped, the timer represented by this handle must be cancelled, if it 738 /// is running. If the timer handler is running when the handle is dropped, the 739 /// drop method must wait for the handler to return before returning. 740 /// 741 /// Note: One way to satisfy the safety requirement is to call `Self::cancel` in 742 /// the drop implementation for `Self.` 743 pub unsafe trait HrTimerHandle { 744 /// Cancel the timer. If the timer is in the running state, block till the 745 /// handler has returned. 746 /// 747 /// Note that the timer might be started by a concurrent start operation. If 748 /// so, the timer might not be in the **stopped** state when this function 749 /// returns. 750 /// 751 /// Returns `true` if the timer was running. cancel(&mut self) -> bool752 fn cancel(&mut self) -> bool; 753 } 754 755 /// Implemented by structs that contain timer nodes. 756 /// 757 /// Clients of the timer API would usually safely implement this trait by using 758 /// the [`crate::impl_has_hr_timer`] macro. 759 /// 760 /// # Safety 761 /// 762 /// Implementers of this trait must ensure that the implementer has a 763 /// [`HrTimer`] field and that all trait methods are implemented according to 764 /// their documentation. All the methods of this trait must operate on the same 765 /// field. 766 pub unsafe trait HasHrTimer<T> { 767 /// The operational mode associated with this timer. 768 /// 769 /// This defines how the expiration value is interpreted. 770 type TimerMode: HrTimerMode; 771 772 /// Return a pointer to the [`HrTimer`] within `Self`. 773 /// 774 /// This function is useful to get access to the value without creating 775 /// intermediate references. 776 /// 777 /// # Safety 778 /// 779 /// `this` must be a valid pointer. raw_get_timer(this: *const Self) -> *const HrTimer<T>780 unsafe fn raw_get_timer(this: *const Self) -> *const HrTimer<T>; 781 782 /// Return a pointer to the struct that is containing the [`HrTimer`] pointed 783 /// to by `ptr`. 784 /// 785 /// This function is useful to get access to the value without creating 786 /// intermediate references. 787 /// 788 /// # Safety 789 /// 790 /// `ptr` must point to a [`HrTimer<T>`] field in a struct of type `Self`. timer_container_of(ptr: *mut HrTimer<T>) -> *mut Self where Self: Sized791 unsafe fn timer_container_of(ptr: *mut HrTimer<T>) -> *mut Self 792 where 793 Self: Sized; 794 795 /// Get pointer to the contained `bindings::hrtimer` struct. 796 /// 797 /// This function is useful to get access to the value without creating 798 /// intermediate references. 799 /// 800 /// # Safety 801 /// 802 /// `this` must be a valid pointer. c_timer_ptr(this: *const Self) -> *const bindings::hrtimer803 unsafe fn c_timer_ptr(this: *const Self) -> *const bindings::hrtimer { 804 // SAFETY: `this` is a valid pointer to a `Self`. 805 let timer_ptr = unsafe { Self::raw_get_timer(this) }; 806 807 // SAFETY: timer_ptr points to an allocation of at least `HrTimer` size. 808 unsafe { HrTimer::raw_get(timer_ptr) } 809 } 810 811 /// Start the timer contained in the `Self` pointed to by `self_ptr`. If 812 /// it is already running it is removed and inserted. 813 /// 814 /// # Safety 815 /// 816 /// - `this` must point to a valid `Self`. 817 /// - Caller must ensure that the pointee of `this` lives until the timer 818 /// fires or is canceled. start(this: *const Self, expires: <Self::TimerMode as HrTimerMode>::Expires)819 unsafe fn start(this: *const Self, expires: <Self::TimerMode as HrTimerMode>::Expires) { 820 // SAFETY: By function safety requirement, `this` is a valid `Self`. 821 unsafe { 822 bindings::hrtimer_start_range_ns( 823 Self::c_timer_ptr(this).cast_mut(), 824 expires.as_nanos(), 825 0, 826 <Self::TimerMode as HrTimerMode>::C_MODE, 827 ); 828 } 829 } 830 } 831 832 /// Restart policy for timers. 833 #[derive(Copy, Clone, PartialEq, Eq, Debug)] 834 #[repr(u32)] 835 pub enum HrTimerRestart { 836 /// Timer should not be restarted. 837 NoRestart = bindings::hrtimer_restart_HRTIMER_NORESTART, 838 /// Timer should be restarted. 839 Restart = bindings::hrtimer_restart_HRTIMER_RESTART, 840 } 841 842 impl HrTimerRestart { into_c(self) -> bindings::hrtimer_restart843 fn into_c(self) -> bindings::hrtimer_restart { 844 self as bindings::hrtimer_restart 845 } 846 } 847 848 /// Time representations that can be used as expiration values in [`HrTimer`]. 849 pub trait HrTimerExpires { 850 /// Converts the expiration time into a nanosecond representation. 851 /// 852 /// This value corresponds to a raw ktime_t value, suitable for passing to kernel 853 /// timer functions. The interpretation (absolute vs relative) depends on the 854 /// associated [HrTimerMode] in use. as_nanos(&self) -> i64855 fn as_nanos(&self) -> i64; 856 } 857 858 impl<C: ClockSource> HrTimerExpires for Instant<C> { 859 #[inline] as_nanos(&self) -> i64860 fn as_nanos(&self) -> i64 { 861 Instant::<C>::as_nanos(self) 862 } 863 } 864 865 impl HrTimerExpires for Delta { 866 #[inline] as_nanos(&self) -> i64867 fn as_nanos(&self) -> i64 { 868 Delta::as_nanos(*self) 869 } 870 } 871 872 mod private { 873 use crate::time::ClockSource; 874 875 pub trait Sealed {} 876 877 impl<C: ClockSource> Sealed for super::AbsoluteMode<C> {} 878 impl<C: ClockSource> Sealed for super::RelativeMode<C> {} 879 impl<C: ClockSource> Sealed for super::AbsolutePinnedMode<C> {} 880 impl<C: ClockSource> Sealed for super::RelativePinnedMode<C> {} 881 impl<C: ClockSource> Sealed for super::AbsoluteSoftMode<C> {} 882 impl<C: ClockSource> Sealed for super::RelativeSoftMode<C> {} 883 impl<C: ClockSource> Sealed for super::AbsolutePinnedSoftMode<C> {} 884 impl<C: ClockSource> Sealed for super::RelativePinnedSoftMode<C> {} 885 impl<C: ClockSource> Sealed for super::AbsoluteHardMode<C> {} 886 impl<C: ClockSource> Sealed for super::RelativeHardMode<C> {} 887 impl<C: ClockSource> Sealed for super::AbsolutePinnedHardMode<C> {} 888 impl<C: ClockSource> Sealed for super::RelativePinnedHardMode<C> {} 889 } 890 891 /// Operational mode of [`HrTimer`]. 892 pub trait HrTimerMode: private::Sealed { 893 /// The C representation of hrtimer mode. 894 const C_MODE: bindings::hrtimer_mode; 895 896 /// Type representing the clock source. 897 type Clock: ClockSource; 898 899 /// Type representing the expiration specification (absolute or relative time). 900 type Expires: HrTimerExpires; 901 } 902 903 /// Timer that expires at a fixed point in time. 904 pub struct AbsoluteMode<C: ClockSource>(PhantomData<C>); 905 906 impl<C: ClockSource> HrTimerMode for AbsoluteMode<C> { 907 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_ABS; 908 909 type Clock = C; 910 type Expires = Instant<C>; 911 } 912 913 /// Timer that expires after a delay from now. 914 pub struct RelativeMode<C: ClockSource>(PhantomData<C>); 915 916 impl<C: ClockSource> HrTimerMode for RelativeMode<C> { 917 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_REL; 918 919 type Clock = C; 920 type Expires = Delta; 921 } 922 923 /// Timer with absolute expiration time, pinned to its current CPU. 924 pub struct AbsolutePinnedMode<C: ClockSource>(PhantomData<C>); 925 impl<C: ClockSource> HrTimerMode for AbsolutePinnedMode<C> { 926 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_ABS_PINNED; 927 928 type Clock = C; 929 type Expires = Instant<C>; 930 } 931 932 /// Timer with relative expiration time, pinned to its current CPU. 933 pub struct RelativePinnedMode<C: ClockSource>(PhantomData<C>); 934 impl<C: ClockSource> HrTimerMode for RelativePinnedMode<C> { 935 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_REL_PINNED; 936 937 type Clock = C; 938 type Expires = Delta; 939 } 940 941 /// Timer with absolute expiration, handled in soft irq context. 942 pub struct AbsoluteSoftMode<C: ClockSource>(PhantomData<C>); 943 impl<C: ClockSource> HrTimerMode for AbsoluteSoftMode<C> { 944 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_ABS_SOFT; 945 946 type Clock = C; 947 type Expires = Instant<C>; 948 } 949 950 /// Timer with relative expiration, handled in soft irq context. 951 pub struct RelativeSoftMode<C: ClockSource>(PhantomData<C>); 952 impl<C: ClockSource> HrTimerMode for RelativeSoftMode<C> { 953 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_REL_SOFT; 954 955 type Clock = C; 956 type Expires = Delta; 957 } 958 959 /// Timer with absolute expiration, pinned to CPU and handled in soft irq context. 960 pub struct AbsolutePinnedSoftMode<C: ClockSource>(PhantomData<C>); 961 impl<C: ClockSource> HrTimerMode for AbsolutePinnedSoftMode<C> { 962 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_ABS_PINNED_SOFT; 963 964 type Clock = C; 965 type Expires = Instant<C>; 966 } 967 968 /// Timer with absolute expiration, pinned to CPU and handled in soft irq context. 969 pub struct RelativePinnedSoftMode<C: ClockSource>(PhantomData<C>); 970 impl<C: ClockSource> HrTimerMode for RelativePinnedSoftMode<C> { 971 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_REL_PINNED_SOFT; 972 973 type Clock = C; 974 type Expires = Delta; 975 } 976 977 /// Timer with absolute expiration, handled in hard irq context. 978 pub struct AbsoluteHardMode<C: ClockSource>(PhantomData<C>); 979 impl<C: ClockSource> HrTimerMode for AbsoluteHardMode<C> { 980 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_ABS_HARD; 981 982 type Clock = C; 983 type Expires = Instant<C>; 984 } 985 986 /// Timer with relative expiration, handled in hard irq context. 987 pub struct RelativeHardMode<C: ClockSource>(PhantomData<C>); 988 impl<C: ClockSource> HrTimerMode for RelativeHardMode<C> { 989 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_REL_HARD; 990 991 type Clock = C; 992 type Expires = Delta; 993 } 994 995 /// Timer with absolute expiration, pinned to CPU and handled in hard irq context. 996 pub struct AbsolutePinnedHardMode<C: ClockSource>(PhantomData<C>); 997 impl<C: ClockSource> HrTimerMode for AbsolutePinnedHardMode<C> { 998 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_ABS_PINNED_HARD; 999 1000 type Clock = C; 1001 type Expires = Instant<C>; 1002 } 1003 1004 /// Timer with relative expiration, pinned to CPU and handled in hard irq context. 1005 pub struct RelativePinnedHardMode<C: ClockSource>(PhantomData<C>); 1006 impl<C: ClockSource> HrTimerMode for RelativePinnedHardMode<C> { 1007 const C_MODE: bindings::hrtimer_mode = bindings::hrtimer_mode_HRTIMER_MODE_REL_PINNED_HARD; 1008 1009 type Clock = C; 1010 type Expires = Delta; 1011 } 1012 1013 /// Privileged smart-pointer for a [`HrTimer`] callback context. 1014 /// 1015 /// Many [`HrTimer`] methods can only be called in two situations: 1016 /// 1017 /// * When the caller has exclusive access to the `HrTimer` and the `HrTimer` is guaranteed not to 1018 /// be running. 1019 /// * From within the context of an `HrTimer`'s callback method. 1020 /// 1021 /// This type provides access to said methods from within a timer callback context. 1022 /// 1023 /// # Invariants 1024 /// 1025 /// * The existence of this type means the caller is currently within the callback for an 1026 /// [`HrTimer`]. 1027 /// * `self.0` always points to a live instance of [`HrTimer<T>`]. 1028 pub struct HrTimerCallbackContext<'a, T: HasHrTimer<T>>(NonNull<HrTimer<T>>, PhantomData<&'a ()>); 1029 1030 impl<'a, T: HasHrTimer<T>> HrTimerCallbackContext<'a, T> { 1031 /// Create a new [`HrTimerCallbackContext`]. 1032 /// 1033 /// # Safety 1034 /// 1035 /// This function relies on the caller being within the context of a timer callback, so it must 1036 /// not be used anywhere except for within implementations of [`RawHrTimerCallback::run`]. The 1037 /// caller promises that `timer` points to a valid initialized instance of 1038 /// [`bindings::hrtimer`]. 1039 /// 1040 /// The returned `Self` must not outlive the function context of [`RawHrTimerCallback::run`] 1041 /// where this function is called. from_raw(timer: *mut HrTimer<T>) -> Self1042 pub(crate) unsafe fn from_raw(timer: *mut HrTimer<T>) -> Self { 1043 // SAFETY: The caller guarantees `timer` is a valid pointer to an initialized 1044 // `bindings::hrtimer` 1045 // INVARIANT: Our safety contract ensures that we're within the context of a timer callback 1046 // and that `timer` points to a live instance of `HrTimer<T>`. 1047 Self(unsafe { NonNull::new_unchecked(timer) }, PhantomData) 1048 } 1049 1050 /// Conditionally forward the timer. 1051 /// 1052 /// This function is identical to [`HrTimer::forward()`] except that it may only be used from 1053 /// within the context of a [`HrTimer`] callback. forward(&mut self, now: HrTimerInstant<T>, interval: Delta) -> u641054 pub fn forward(&mut self, now: HrTimerInstant<T>, interval: Delta) -> u64 { 1055 // SAFETY: 1056 // - We are guaranteed to be within the context of a timer callback by our type invariants 1057 // - By our type invariants, `self.0` always points to a valid `HrTimer<T>` 1058 unsafe { HrTimer::<T>::raw_forward(self.0.as_ptr(), now, interval) } 1059 } 1060 1061 /// Conditionally forward the timer. 1062 /// 1063 /// This is a variant of [`HrTimerCallbackContext::forward()`] that uses an interval after the 1064 /// current time of the base clock for the [`HrTimer`]. forward_now(&mut self, duration: Delta) -> u641065 pub fn forward_now(&mut self, duration: Delta) -> u64 { 1066 self.forward(HrTimerInstant::<T>::now(), duration) 1067 } 1068 } 1069 1070 /// Use to implement the [`HasHrTimer<T>`] trait. 1071 /// 1072 /// See [`module`] documentation for an example. 1073 /// 1074 /// [`module`]: crate::time::hrtimer 1075 #[macro_export] 1076 macro_rules! impl_has_hr_timer { 1077 ( 1078 impl$({$($generics:tt)*})? 1079 HasHrTimer<$timer_type:ty> 1080 for $self:ty 1081 { 1082 mode : $mode:ty, 1083 field : self.$field:ident $(,)? 1084 } 1085 $($rest:tt)* 1086 ) => { 1087 // SAFETY: This implementation of `raw_get_timer` only compiles if the 1088 // field has the right type. 1089 unsafe impl$(<$($generics)*>)? $crate::time::hrtimer::HasHrTimer<$timer_type> for $self { 1090 type TimerMode = $mode; 1091 1092 #[inline] 1093 unsafe fn raw_get_timer( 1094 this: *const Self, 1095 ) -> *const $crate::time::hrtimer::HrTimer<$timer_type> { 1096 // SAFETY: The caller promises that the pointer is not dangling. 1097 unsafe { ::core::ptr::addr_of!((*this).$field) } 1098 } 1099 1100 #[inline] 1101 unsafe fn timer_container_of( 1102 ptr: *mut $crate::time::hrtimer::HrTimer<$timer_type>, 1103 ) -> *mut Self { 1104 // SAFETY: As per the safety requirement of this function, `ptr` 1105 // is pointing inside a `$timer_type`. 1106 unsafe { ::kernel::container_of!(ptr, $timer_type, $field) } 1107 } 1108 } 1109 } 1110 } 1111 1112 mod arc; 1113 pub use arc::ArcHrTimerHandle; 1114 mod pin; 1115 pub use pin::PinHrTimerHandle; 1116 mod pin_mut; 1117 pub use pin_mut::PinMutHrTimerHandle; 1118 // `box` is a reserved keyword, so prefix with `t` for timer 1119 mod tbox; 1120 pub use tbox::BoxHrTimerHandle; 1121