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
// Copyright 2019 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//! Execution contexts.
//!
//! This module defines "context" traits, which allow code in this crate to be
//! written agnostic to their execution context.
//!
//! All of the code in this crate operates in terms of "events". When an event
//! occurs (for example, a packet is received, an application makes a request,
//! or a timer fires), a function is called to handle that event. In response to
//! that event, the code may wish to emit new events (for example, to send a
//! packet, to respond to an application request, or to install a new timer).
//! The traits in this module provide the ability to emit new events. For
//! example, if, in order to handle some event, we need the ability to install
//! new timers, then the function to handle that event would take a
//! [`TimerContext`] parameter, which it could use to install new timers.
//!
//! Structuring code this way allows us to write code which is agnostic to
//! execution context - a test fake or any number of possible "real-world"
//! implementations of these traits all appear as indistinguishable, opaque
//! trait implementations to our code.
//!
//! The benefits are deeper than this, though. Large units of code can be
//! subdivided into smaller units that view each other as "contexts". For
//! example, the ARP implementation in the [`crate::device::arp`] module defines
//! the [`ArpContext`] trait, which is an execution context for ARP operations.
//! It is implemented both by the test fakes in that module, and also by the
//! Ethernet device implementation in the [`crate::device::ethernet`] module.
//!
//! This subdivision of code into small units in turn enables modularity. If,
//! for example, the IP code sees transport layer protocols as execution
//! contexts, then customizing which transport layer protocols are supported is
//! just a matter of providing a different implementation of the transport layer
//! context traits (this isn't what we do today, but we may in the future).
use lock_order::Unlocked;
use netstack3_base::ContextProvider;
use {netstack3_device as device, netstack3_ip as ip, netstack3_udp as udp};
use crate::marker::{BindingsContext, BindingsTypes};
use crate::state::StackState;
// Enable all blanket implementations on CoreCtx.
//
// Some blanket implementations are enabled individually to sidestep coherence
// issues with the fake context implementations in tests. We treat each of them
// individually so it's easier to split things into separate crates and avoids
// playing whack-a-mole with single markers that work for some traits/crates but
// not others.
impl<BC: BindingsContext, L> ip::marker::UseTransportIpContextBlanket for CoreCtx<'_, BC, L> {}
impl<BC: BindingsContext, L> ip::marker::UseIpSocketContextBlanket for CoreCtx<'_, BC, L> {}
impl<BC: BindingsContext, L> ip::marker::UseIpSocketHandlerBlanket for CoreCtx<'_, BC, L> {}
impl<BC: BindingsContext, L> ip::marker::UseDeviceIpSocketHandlerBlanket for CoreCtx<'_, BC, L> {}
impl<BC: BindingsContext, L> udp::UseUdpIpTransportContextBlanket for CoreCtx<'_, BC, L> {}
impl<BC: BindingsContext, L> device::marker::UseArpFrameMetadataBlanket for CoreCtx<'_, BC, L> {}
/// Provides access to core context implementations.
///
/// `L` is the current lock level of `CoreCtx`. The alias [`UnlockedCoreCtx`] is
/// provided at the [`Unlocked`] level.
pub type CoreCtx<'a, BT, L> = Locked<&'a StackState<BT>, L>;
pub(crate) type CoreCtxAndResource<'a, BT, R, L> =
Locked<lock_order::OwnedTupleWrapper<&'a StackState<BT>, &'a R>, L>;
/// An alias for an unlocked [`CoreCtx`].
pub type UnlockedCoreCtx<'a, BT> = CoreCtx<'a, BT, Unlocked>;
impl<'a, BT, L> ContextProvider for CoreCtx<'a, BT, L>
where
BT: BindingsTypes,
{
type Context = Self;
fn context(&mut self) -> &mut Self::Context {
self
}
}
pub(crate) use locked::{Locked, WrapLockLevel};
/// Prelude import to enable the lock wrapper traits.
pub(crate) mod prelude {
#[cfg(no_lock_order)]
pub(crate) use lock_order::wrap::disable::prelude::*;
#[cfg(not(no_lock_order))]
pub(crate) use lock_order::wrap::prelude::*;
}
/// Provides a crate-local wrapper for `[lock_order::Locked]`.
///
/// This module is intentionally private so usage is limited to the type alias
/// in [`CoreCtx`].
mod locked {
use super::{BindingsTypes, CoreCtx, StackState};
use core::ops::Deref;
use lock_order::wrap::LockedWrapper;
use lock_order::{Locked as ExternalLocked, TupleWrapper, Unlocked};
/// A crate-local wrapper on [`lock_order::Locked`].
pub struct Locked<T, L>(ExternalLocked<T, L>);
// SAFETY: This is only compiled when lock ordering is disabled.
unsafe impl<T, L> lock_order::wrap::disable::DisabledLockWrapper for Locked<T, L> {}
impl<T, L> LockedWrapper<T, L> for Locked<T, L>
where
T: Deref,
T::Target: Sized,
{
type AtLockLevel<'l, M> = Locked<&'l T::Target, M>
where
M: 'l,
T: 'l;
type CastWrapper<X> = Locked<X, L>
where
X: Deref,
X::Target: Sized;
fn wrap<'l, M>(locked: ExternalLocked<&'l T::Target, M>) -> Self::AtLockLevel<'l, M>
where
M: 'l,
T: 'l,
{
Locked(locked)
}
fn wrap_cast<R: Deref>(locked: ExternalLocked<R, L>) -> Self::CastWrapper<R>
where
R::Target: Sized,
{
Locked(locked)
}
fn get_mut(&mut self) -> &mut ExternalLocked<T, L> {
let Self(locked) = self;
locked
}
fn get(&self) -> &ExternalLocked<T, L> {
let Self(locked) = self;
locked
}
}
impl<'a, BT: BindingsTypes> CoreCtx<'a, BT, Unlocked> {
/// Creates a new `CoreCtx` from a borrowed [`StackState`].
pub fn new(stack_state: &'a StackState<BT>) -> Self {
Self(ExternalLocked::new(stack_state))
}
}
impl<'a, BT, R, L, T> Locked<T, L>
where
R: 'a,
T: Deref<Target = TupleWrapper<&'a StackState<BT>, &'a R>>,
BT: BindingsTypes,
{
pub(crate) fn cast_core_ctx(&mut self) -> CoreCtx<'_, BT, L> {
let Self(locked) = self;
crate::CoreCtx::<BT, L>::wrap(locked.cast_with(|c| c.left()))
}
}
/// Enables the [`WrapLockLevel`] type alias.
pub trait WrappedLockLevel {
type LockLevel;
}
impl<L> WrappedLockLevel for L {
/// All lock levels are actually [`Unlocked`].
#[cfg(no_lock_order)]
type LockLevel = Unlocked;
/// All lock levels are themselves.
#[cfg(not(no_lock_order))]
type LockLevel = L;
}
/// Wraps lock level `L` in [`WrappedLockLevel::LockLevel`], which allows
/// lock ordering to be disabled by build configuration.
///
/// Whenever using a concrete instantiation of a lock level (i.e. not in a
/// `LockBefore` trait bound) it must be wrapped in `WrapLockLevel` for
/// compilation with `cfg(no_lock_order)` to succeed.
pub(crate) type WrapLockLevel<L> = <L as WrappedLockLevel>::LockLevel;
}