kernel/deferred_call.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265
// Licensed under the Apache License, Version 2.0 or the MIT License.
// SPDX-License-Identifier: Apache-2.0 OR MIT
// Copyright Tock Contributors 2022.
//! Hardware-independent kernel interface for deferred calls.
//!
//! This allows any struct in the kernel which implements [`DeferredCallClient`]
//! to set and receive deferred calls, Tock's version of software interrupts.
//!
//! These can be used to implement long-running in-kernel algorithms or software
//! devices that are supposed to work like hardware devices. Essentially, this
//! allows the chip to handle more important interrupts, and lets a kernel
//! component return the function call stack up to the scheduler, automatically
//! being called again.
//!
//! Usage
//! -----
//!
//! The `DEFCALLS` array size determines how many [`DeferredCall`]s may be
//! registered. By default this is set to 32. To support more deferred calls,
//! this file would need to be modified to use a larger variable for `BITMASK`
//! (e.g. `BITMASK` could be a u64 and the array size increased to 64). If more
//! than 32 deferred calls are created, the kernel will panic at the beginning
//! of the kernel loop.
//!
//! ```rust
//! use kernel::deferred_call::{DeferredCall, DeferredCallClient};
//! use kernel::static_init;
//!
//! struct SomeCapsule {
//! deferred_call: DeferredCall
//! }
//! impl SomeCapsule {
//! pub fn new() -> Self {
//! Self {
//! deferred_call: DeferredCall::new(),
//! }
//! }
//! }
//! impl DeferredCallClient for SomeCapsule {
//! fn handle_deferred_call(&self) {
//! // Your action here
//! }
//!
//! fn register(&'static self) {
//! self.deferred_call.register(self);
//! }
//! }
//!
//! // main.rs or your component must register the capsule with its deferred
//! // call. This should look like:
//! let some_capsule = unsafe { static_init!(SomeCapsule, SomeCapsule::new()) };
//! some_capsule.register();
//! ```
use crate::utilities::cells::OptionalCell;
use core::cell::Cell;
use core::marker::Copy;
use core::marker::PhantomData;
use core::ptr::addr_of;
/// This trait should be implemented by clients which need to receive
/// [`DeferredCall`]s.
// This trait is not intended to be used as a trait object; e.g. you should not
// create a `&dyn DeferredCallClient`. The `Sized` supertrait prevents this.
pub trait DeferredCallClient: Sized {
/// Software interrupt function that is called when the deferred call is
/// triggered.
fn handle_deferred_call(&self);
// This function should be implemented as
// `self.deferred_call.register(&self);`.
fn register(&'static self);
}
/// This struct serves as a lightweight alternative to the use of trait objects
/// (e.g. `&dyn DeferredCall`). Using a trait object will include a 20 byte
/// vtable per instance, but this alternative stores only the data and function
/// pointers, 8 bytes per instance.
#[derive(Copy, Clone)]
struct DynDefCallRef<'a> {
data: *const (),
callback: fn(*const ()),
_lifetime: PhantomData<&'a ()>,
}
impl<'a> DynDefCallRef<'a> {
// SAFETY: We define the callback function as being a closure which casts
// the passed pointer to be the appropriate type (a pointer to `T`) and then
// calls `T::handle_deferred_call()`. In practice, the closure is optimized
// away by LLVM when the ABI of the closure and the underlying function are
// identical, making this zero-cost, but saving us from having to trust that
// `fn(*const ())` and `fn handle_deferred_call(&self)` will always have the
// same calling convention for any type.
fn new<T: DeferredCallClient>(x: &'a T) -> Self {
Self {
data: core::ptr::from_ref(x) as *const (),
callback: |p| unsafe { T::handle_deferred_call(&*p.cast()) },
_lifetime: PhantomData,
}
}
}
impl DynDefCallRef<'_> {
// More efficient to pass by `self` if we don't have to implement
// `DeferredCallClient` directly.
fn handle_deferred_call(self) {
(self.callback)(self.data)
}
}
// The below constant lets us get around Rust not allowing short array
// initialization for non-default types.
const EMPTY: OptionalCell<DynDefCallRef<'static>> = OptionalCell::empty();
/// Counter for the number of deferred calls that have been created, this is
/// used to track that no more than 32 deferred calls have been created.
// All 3 of the below global statics are accessed only in this file, and all
// accesses are via immutable references. Tock is single threaded, so each will
// only ever be accessed via an immutable reference from the single kernel
// thread. TODO: Once Tock decides on an approach to replace `static mut` with
// some sort of `SyncCell`, migrate all three of these to that approach
// (https://github.com/tock/tock/issues/1545).
static mut CTR: Cell<usize> = Cell::new(0);
/// This bitmask tracks which of the up to 32 existing deferred calls have been
/// scheduled. Any bit that is set in that mask indicates the deferred call with
/// its [`DeferredCall::idx`] field set to the index of that bit has been
/// scheduled and not yet serviced.
static mut BITMASK: Cell<u32> = Cell::new(0);
/// An array that stores references to up to 32 `DeferredCall`s via the low-cost
/// [`DynDefCallRef`].
// This is a 256 byte array, but at least resides in `.bss`.
static mut DEFCALLS: [OptionalCell<DynDefCallRef<'static>>; 32] = [EMPTY; 32];
pub struct DeferredCall {
idx: usize,
}
impl DeferredCall {
/// Create a new deferred call with a unique ID.
pub fn new() -> Self {
// SAFETY: No accesses to CTR are via an &mut, and the Tock kernel is
// single-threaded so all accesses will occur from this thread.
let ctr = unsafe { &*addr_of!(CTR) };
let idx = ctr.get();
ctr.set(idx + 1);
DeferredCall { idx }
}
// To reduce monomorphization bloat, the non-generic portion of register is
// moved into this function without generic parameters.
#[inline(never)]
fn register_internal_non_generic(&self, handler: DynDefCallRef<'static>) {
// SAFETY: No accesses to DEFCALLS are via an &mut, and the Tock kernel
// is single-threaded so all accesses will occur from this thread.
let defcalls = unsafe { &*addr_of!(DEFCALLS) };
if self.idx >= defcalls.len() {
// This error will be caught by the scheduler at the beginning of
// the kernel loop, which is much better than panicking here, before
// the debug writer is setup. Also allows a single panic for
// creating too many deferred calls instead of NUM_DCS panics (this
// function is monomorphized).
return;
}
defcalls[self.idx].set(handler);
}
/// This function registers the passed client with this deferred call, such
/// that calls to [`DeferredCall::set()`] will schedule a callback on the
/// [`handle_deferred_call()`](DeferredCallClient::handle_deferred_call)
/// method of the passed client.
pub fn register<DC: DeferredCallClient>(&self, client: &'static DC) {
let handler = DynDefCallRef::new(client);
self.register_internal_non_generic(handler);
}
/// Schedule a deferred callback on the client associated with this deferred
/// call.
pub fn set(&self) {
// SAFETY: No accesses to BITMASK are via an &mut, and the Tock kernel
// is single-threaded so all accesses will occur from this thread.
let bitmask = unsafe { &*addr_of!(BITMASK) };
bitmask.set(bitmask.get() | (1 << self.idx));
}
/// Check if a deferred callback has been set and not yet serviced on this
/// deferred call.
pub fn is_pending(&self) -> bool {
// SAFETY: No accesses to BITMASK are via an &mut, and the Tock kernel
// is single-threaded so all accesses will occur from this thread.
let bitmask = unsafe { &*addr_of!(BITMASK) };
bitmask.get() & (1 << self.idx) == 1
}
/// Services and clears the next pending [`DeferredCall`], returns which
/// index was serviced.
pub fn service_next_pending() -> Option<usize> {
// SAFETY: No accesses to BITMASK/DEFCALLS are via an &mut, and the Tock
// kernel is single-threaded so all accesses will occur from this
// thread.
let bitmask = unsafe { &*addr_of!(BITMASK) };
let defcalls = unsafe { &*addr_of!(DEFCALLS) };
let val = bitmask.get();
if val == 0 {
None
} else {
let bit = val.trailing_zeros() as usize;
let new_val = val & !(1 << bit);
bitmask.set(new_val);
defcalls[bit].map(|dc| {
dc.handle_deferred_call();
bit
})
}
}
/// Returns true if any deferred calls are waiting to be serviced, false
/// otherwise.
pub fn has_tasks() -> bool {
// SAFETY: No accesses to BITMASK are via an &mut, and the Tock kernel
// is single-threaded so all accesses will occur from this thread.
let bitmask = unsafe { &*addr_of!(BITMASK) };
bitmask.get() != 0
}
/// This function should be called at the beginning of the kernel loop to
/// verify that deferred calls have been correctly initialized. This
/// function verifies two things:
///
/// 1. That <= [`DEFCALLS.len()`] deferred calls have been created, which is
/// the maximum this interface supports.
///
/// 2. That exactly as many deferred calls were registered as were created,
/// which helps to catch bugs if board maintainers forget to call
/// [`register()`](DeferredCall::register) on a created [`DeferredCall`].
///
/// Neither of these checks are necessary for soundness, but they are
/// necessary for confirming that [`DeferredCall`]s will actually be
/// delivered as expected. This function costs about 300 bytes, so you can
/// remove it if you are confident your setup will not exceed 32 deferred
/// calls, and that all of your components register their deferred calls.
// Ignore the clippy warning for using `.filter(|opt| opt.is_some())` since
// we don't actually have an Option (we have an OptionalCell) and
// IntoIterator is not implemented for OptionalCell.
#[allow(clippy::iter_filter_is_some)]
pub fn verify_setup() {
// SAFETY: No accesses to CTR/DEFCALLS are via an &mut, and the Tock
// kernel is single-threaded so all accesses will occur from this
// thread.
let ctr = unsafe { &*addr_of!(CTR) };
let defcalls = unsafe { &*addr_of!(DEFCALLS) };
let num_deferred_calls = ctr.get();
let num_registered_calls = defcalls.iter().filter(|opt| opt.is_some()).count();
if num_deferred_calls > defcalls.len() {
panic!("ERROR: too many deferred calls: {}", num_deferred_calls);
} else if num_deferred_calls != num_registered_calls {
panic!(
"ERROR: {} deferred calls, {} registered. A component may have forgotten to register a deferred call.",
num_deferred_calls, num_registered_calls
);
}
}
}