1use net_types::ethernet::Mac;
8use net_types::ip::{Ip, IpVersionMarker};
9use net_types::{BroadcastAddr, MulticastAddr};
10
11use core::convert::Infallible as Never;
12use core::fmt::Debug;
13use packet::{BufferMut, SerializeError, Serializer};
14use thiserror::Error;
15
16use crate::error::ErrorAndSerializer;
17
18pub trait RecvFrameContext<Meta, BC> {
24 fn receive_frame<B: BufferMut + Debug>(
28 &mut self,
29 bindings_ctx: &mut BC,
30 metadata: Meta,
31 frame: B,
32 );
33}
34
35impl<CC, BC> ReceivableFrameMeta<CC, BC> for Never {
36 fn receive_meta<B: BufferMut + Debug>(
37 self,
38 _core_ctx: &mut CC,
39 _bindings_ctx: &mut BC,
40 _frame: B,
41 ) {
42 match self {}
43 }
44}
45
46pub trait ReceivableFrameMeta<CC, BC> {
54 fn receive_meta<B: BufferMut + Debug>(self, core_ctx: &mut CC, bindings_ctx: &mut BC, frame: B);
56}
57
58impl<CC, BC, Meta> RecvFrameContext<Meta, BC> for CC
59where
60 Meta: ReceivableFrameMeta<CC, BC>,
61{
62 fn receive_frame<B: BufferMut + Debug>(
63 &mut self,
64 bindings_ctx: &mut BC,
65 metadata: Meta,
66 frame: B,
67 ) {
68 metadata.receive_meta(self, bindings_ctx, frame)
69 }
70}
71
72#[derive(Error, Debug, PartialEq)]
74pub enum SendFrameErrorReason {
75 #[error("size constraints violated")]
77 SizeConstraintsViolation,
78 #[error("failed to allocate")]
80 Alloc,
81 #[error("transmit queue is full")]
83 QueueFull,
84}
85
86impl<A> From<SerializeError<A>> for SendFrameErrorReason {
87 fn from(e: SerializeError<A>) -> Self {
88 match e {
89 SerializeError::Alloc(_) => Self::Alloc,
90 SerializeError::SizeLimitExceeded => Self::SizeConstraintsViolation,
91 }
92 }
93}
94
95pub type SendFrameError<S> = ErrorAndSerializer<SendFrameErrorReason, S>;
97
98pub trait SendFrameContext<BC, Meta> {
100 fn send_frame<S>(
109 &mut self,
110 bindings_ctx: &mut BC,
111 metadata: Meta,
112 frame: S,
113 ) -> Result<(), SendFrameError<S>>
114 where
115 S: Serializer,
116 S::Buffer: BufferMut;
117}
118
119pub trait SendableFrameMeta<CC, BC> {
127 fn send_meta<S>(
129 self,
130 core_ctx: &mut CC,
131 bindings_ctx: &mut BC,
132 frame: S,
133 ) -> Result<(), SendFrameError<S>>
134 where
135 S: Serializer,
136 S::Buffer: BufferMut;
137}
138
139impl<CC, BC, Meta> SendFrameContext<BC, Meta> for CC
140where
141 Meta: SendableFrameMeta<CC, BC>,
142{
143 fn send_frame<S>(
144 &mut self,
145 bindings_ctx: &mut BC,
146 metadata: Meta,
147 frame: S,
148 ) -> Result<(), SendFrameError<S>>
149 where
150 S: Serializer,
151 S::Buffer: BufferMut,
152 {
153 metadata.send_meta(self, bindings_ctx, frame)
154 }
155}
156
157#[derive(Copy, Clone, Debug, Eq, PartialEq)]
163pub enum FrameDestination {
164 Individual {
166 local: bool,
168 },
169 Multicast,
173 Broadcast,
177}
178
179impl FrameDestination {
180 pub fn is_broadcast(self) -> bool {
182 self == FrameDestination::Broadcast
183 }
184
185 pub fn from_dest(destination: Mac, local_mac: Mac) -> Self {
187 BroadcastAddr::new(destination)
188 .map(Into::into)
189 .or_else(|| MulticastAddr::new(destination).map(Into::into))
190 .unwrap_or_else(|| FrameDestination::Individual { local: destination == local_mac })
191 }
192}
193
194impl From<BroadcastAddr<Mac>> for FrameDestination {
195 fn from(_value: BroadcastAddr<Mac>) -> Self {
196 Self::Broadcast
197 }
198}
199
200impl From<MulticastAddr<Mac>> for FrameDestination {
201 fn from(_value: MulticastAddr<Mac>) -> Self {
202 Self::Multicast
203 }
204}
205
206pub struct RecvIpFrameMeta<D, M, I: Ip> {
208 pub device: D,
210 pub frame_dst: Option<FrameDestination>,
217 pub ip_layer_metadata: M,
220 pub marker: IpVersionMarker<I>,
222}
223
224impl<D, M, I: Ip> RecvIpFrameMeta<D, M, I> {
225 pub fn new(
228 device: D,
229 frame_dst: Option<FrameDestination>,
230 ip_layer_metadata: M,
231 ) -> RecvIpFrameMeta<D, M, I> {
232 RecvIpFrameMeta { device, frame_dst, ip_layer_metadata, marker: IpVersionMarker::new() }
233 }
234}
235
236pub trait TxMetadataBindingsTypes {
248 type TxMetadata: Default + Debug + Send + Sync + 'static;
253}
254
255pub trait CoreTxMetadataContext<T, BT: TxMetadataBindingsTypes> {
260 fn convert_tx_meta(&self, tx_meta: T) -> BT::TxMetadata;
266}
267
268#[cfg(any(test, feature = "testutils"))]
269pub(crate) mod testutil {
270 use super::*;
271 use alloc::boxed::Box;
272 use alloc::vec::Vec;
273
274 use crate::testutil::FakeBindingsCtx;
275
276 pub struct FakeFrameCtx<Meta> {
278 frames: Vec<(Meta, Vec<u8>)>,
279 should_error_for_frame:
280 Option<Box<dyn FnMut(&Meta) -> Option<SendFrameErrorReason> + Send>>,
281 }
282
283 impl<Meta> FakeFrameCtx<Meta> {
284 pub fn set_should_error_for_frame<
287 F: Fn(&Meta) -> Option<SendFrameErrorReason> + Send + 'static,
288 >(
289 &mut self,
290 f: F,
291 ) {
292 self.should_error_for_frame = Some(Box::new(f));
293 }
294 }
295
296 impl<Meta> Default for FakeFrameCtx<Meta> {
297 fn default() -> FakeFrameCtx<Meta> {
298 FakeFrameCtx { frames: Vec::new(), should_error_for_frame: None }
299 }
300 }
301
302 impl<Meta> FakeFrameCtx<Meta> {
303 pub fn take_frames(&mut self) -> Vec<(Meta, Vec<u8>)> {
305 core::mem::take(&mut self.frames)
306 }
307
308 pub fn frames(&self) -> &[(Meta, Vec<u8>)] {
310 self.frames.as_slice()
311 }
312
313 pub fn push(&mut self, meta: Meta, frame: Vec<u8>) {
315 self.frames.push((meta, frame))
316 }
317 }
318
319 impl<Meta, BC> SendableFrameMeta<FakeFrameCtx<Meta>, BC> for Meta {
320 fn send_meta<S>(
321 self,
322 core_ctx: &mut FakeFrameCtx<Meta>,
323 _bindings_ctx: &mut BC,
324 frame: S,
325 ) -> Result<(), SendFrameError<S>>
326 where
327 S: Serializer,
328 S::Buffer: BufferMut,
329 {
330 if let Some(error) = core_ctx.should_error_for_frame.as_mut().and_then(|f| f(&self)) {
331 return Err(SendFrameError { serializer: frame, error });
332 }
333
334 let buffer = frame
335 .serialize_vec_outer()
336 .map_err(|(e, serializer)| SendFrameError { error: e.into(), serializer })?;
337 core_ctx.push(self, buffer.as_ref().to_vec());
338 Ok(())
339 }
340 }
341
342 pub trait WithFakeFrameContext<SendMeta> {
344 fn with_fake_frame_ctx_mut<O, F: FnOnce(&mut FakeFrameCtx<SendMeta>) -> O>(
346 &mut self,
347 f: F,
348 ) -> O;
349 }
350
351 impl<SendMeta> WithFakeFrameContext<SendMeta> for FakeFrameCtx<SendMeta> {
352 fn with_fake_frame_ctx_mut<O, F: FnOnce(&mut FakeFrameCtx<SendMeta>) -> O>(
353 &mut self,
354 f: F,
355 ) -> O {
356 f(self)
357 }
358 }
359
360 impl<TimerId, Event: Debug, State, FrameMeta> TxMetadataBindingsTypes
361 for FakeBindingsCtx<TimerId, Event, State, FrameMeta>
362 {
363 type TxMetadata = FakeTxMetadata;
364 }
365
366 #[derive(Default, Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
368 pub struct FakeTxMetadata;
369}
370
371#[cfg(test)]
372mod tests {
373 use super::*;
374
375 use net_declare::net_mac;
376 use net_types::{UnicastAddr, Witness as _};
377
378 #[test]
379 fn frame_destination_from_dest() {
380 const LOCAL_ADDR: Mac = net_mac!("88:88:88:88:88:88");
381
382 assert_eq!(
383 FrameDestination::from_dest(
384 UnicastAddr::new(net_mac!("00:11:22:33:44:55")).unwrap().get(),
385 LOCAL_ADDR
386 ),
387 FrameDestination::Individual { local: false }
388 );
389 assert_eq!(
390 FrameDestination::from_dest(LOCAL_ADDR, LOCAL_ADDR),
391 FrameDestination::Individual { local: true }
392 );
393 assert_eq!(
394 FrameDestination::from_dest(Mac::BROADCAST, LOCAL_ADDR),
395 FrameDestination::Broadcast,
396 );
397 assert_eq!(
398 FrameDestination::from_dest(
399 MulticastAddr::new(net_mac!("11:11:11:11:11:11")).unwrap().get(),
400 LOCAL_ADDR
401 ),
402 FrameDestination::Multicast
403 );
404 }
405}