netstack3_base/
context.rs

1// Copyright 2024 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5//! Common context abstractions.
6
7/// A pair of core and bindings contexts.
8///
9/// This trait exists so implementers can be agnostic on the storage of the
10/// contexts, since all we need from a context pair is mutable references from
11/// both contexts.
12pub trait ContextPair {
13    /// The core context type held by this pair.
14    type CoreContext;
15    /// The bindings context type held by this pair.
16    type BindingsContext;
17
18    /// Gets a mutable reference to both contexts.
19    fn contexts(&mut self) -> (&mut Self::CoreContext, &mut Self::BindingsContext);
20
21    /// Gets a mutable reference to the core context.
22    fn core_ctx(&mut self) -> &mut Self::CoreContext {
23        let (core_ctx, _) = self.contexts();
24        core_ctx
25    }
26
27    /// Gets a mutable reference to the bindings context.
28    fn bindings_ctx(&mut self) -> &mut Self::BindingsContext {
29        let (_, bindings_ctx) = self.contexts();
30        bindings_ctx
31    }
32}
33
34impl<'a, C> ContextPair for &'a mut C
35where
36    C: ContextPair,
37{
38    type CoreContext = C::CoreContext;
39    type BindingsContext = C::BindingsContext;
40    fn contexts(&mut self) -> (&mut Self::CoreContext, &mut Self::BindingsContext) {
41        C::contexts(self)
42    }
43}
44
45/// A type that provides a context implementation.
46///
47/// This trait allows for [`CtxPair`] to hold context implementations
48/// agnostically of the storage method and how they're implemented. For example,
49/// tests usually create API structs with a mutable borrow to contexts, while
50/// the core context exposed to bindings is implemented on an owned [`CoreCtx`]
51/// type.
52///
53/// The shape of this trait is equivalent to [`core::ops::DerefMut`] but we
54/// decide against using that because of the automatic dereferencing semantics
55/// the compiler provides around implementers of `DerefMut`.
56pub trait ContextProvider {
57    /// The context provided by this `ContextProvider`.
58    type Context: Sized;
59
60    /// Gets a mutable borrow to this context.
61    fn context(&mut self) -> &mut Self::Context;
62}
63
64impl<'a, T: Sized> ContextProvider for &'a mut T {
65    type Context = T;
66
67    fn context(&mut self) -> &mut Self::Context {
68        &mut *self
69    }
70}
71
72/// A concrete implementation of [`ContextPair`].
73///
74///
75/// `CtxPair` provides a [`ContextPair`] implementation when `CC` and `BC` are
76/// [`ContextProvider`] and using their respective targets as the `CoreContext`
77/// and `BindingsContext` associated types.
78#[derive(Clone)]
79pub struct CtxPair<CC, BC> {
80    /// The core context.
81    pub core_ctx: CC,
82    /// The bindings context.
83    // We put `bindings_ctx` after `core_ctx` to make sure that `core_ctx` is
84    // dropped before `bindings_ctx` so that the existence of
85    // strongly-referenced device IDs in `bindings_ctx` causes test failures,
86    // forcing proper cleanup of device IDs in our unit tests.
87    //
88    // Note that if strongly-referenced (device) IDs exist when dropping the
89    // primary reference, the primary reference's drop impl will panic. See
90    // `crate::sync::PrimaryRc::drop` for details.
91    // TODO(https://fxbug.dev/320021524): disallow destructuring to actually
92    // uphold the intent above.
93    pub bindings_ctx: BC,
94}
95
96impl<CC, BC> CtxPair<CC, BC> {
97    /// Creates a new `CtxPair` with `core_ctx` and a default bindings context.
98    pub fn with_core_ctx(core_ctx: CC) -> Self
99    where
100        BC: Default,
101    {
102        Self { core_ctx, bindings_ctx: BC::default() }
103    }
104
105    /// Creates a new `CtxPair` with a default bindings context and a core
106    /// context resulting from `builder`.
107    pub fn with_default_bindings_ctx<F: FnOnce(&mut BC) -> CC>(builder: F) -> Self
108    where
109        BC: Default,
110    {
111        let mut bindings_ctx = BC::default();
112        let core_ctx = builder(&mut bindings_ctx);
113        Self { core_ctx, bindings_ctx }
114    }
115
116    /// Creates a new `CtxPair` with a mutable borrow of the contexts.
117    pub fn as_mut(&mut self) -> CtxPair<&mut CC, &mut BC> {
118        let Self { core_ctx, bindings_ctx } = self;
119        CtxPair { core_ctx, bindings_ctx }
120    }
121
122    /// Creates a new `CtxPair` with a default bindings context and the provided
123    /// core context builder.
124    pub fn new_with_builder(builder: CC::Builder) -> Self
125    where
126        CC: BuildableCoreContext<BC>,
127        BC: Default,
128    {
129        Self::with_default_bindings_ctx(|bindings_ctx| CC::build(bindings_ctx, builder))
130    }
131}
132
133impl<CC, BC> Default for CtxPair<CC, BC>
134where
135    CC: BuildableCoreContext<BC>,
136    CC::Builder: Default,
137    BC: Default,
138{
139    fn default() -> Self {
140        Self::new_with_builder(Default::default())
141    }
142}
143
144impl<CC, BC> ContextPair for CtxPair<CC, BC>
145where
146    CC: ContextProvider,
147    BC: ContextProvider,
148{
149    type CoreContext = CC::Context;
150    type BindingsContext = BC::Context;
151
152    fn contexts(&mut self) -> (&mut Self::CoreContext, &mut Self::BindingsContext) {
153        let Self { core_ctx, bindings_ctx } = self;
154        (core_ctx.context(), bindings_ctx.context())
155    }
156}
157
158/// A core context that can be created from some builder type.
159///
160/// This allows `CtxPair` to define its construction methods away from where
161/// core contexts are actually defined.
162pub trait BuildableCoreContext<BC> {
163    /// The builder type that can build this core context.
164    type Builder;
165
166    /// Consumes this builder and returns the context.
167    fn build(bindings_ctx: &mut BC, builder: Self::Builder) -> Self;
168}
169
170impl<O, BC> BuildableCoreContext<BC> for O
171where
172    O: Default,
173{
174    type Builder = ();
175    fn build(_bindings_ctx: &mut BC, (): ()) -> Self {
176        O::default()
177    }
178}