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}