1use alloc::collections::HashMap;
6use alloc::vec::Vec;
7use core::fmt::{Debug, Display};
8use core::num::NonZeroU64;
9
10use derivative::Derivative;
11use lock_order::lock::{OrderedLockAccess, OrderedLockRef};
12use net_types::ethernet::Mac;
13use net_types::ip::{Ip, IpVersion, Ipv4, Ipv6};
14use netstack3_base::sync::RwLock;
15use netstack3_base::{
16 Counter, Device, DeviceIdContext, HandleableTimer, Inspectable, Inspector, InspectorExt as _,
17 InstantContext, ReferenceNotifiers, TimerBindingsTypes, TimerHandler, TxMetadataBindingsTypes,
18};
19use netstack3_filter::FilterBindingsTypes;
20use netstack3_ip::device::Ipv6LinkLayerAddr;
21use netstack3_ip::nud::{LinkResolutionContext, NudCounters};
22use packet::Buf;
23
24use crate::blackhole::{BlackholeDeviceId, BlackholePrimaryDeviceId};
25use crate::internal::arp::ArpCounters;
26use crate::internal::ethernet::{EthernetLinkDevice, EthernetTimerId};
27use crate::internal::id::{
28 BaseDeviceId, BasePrimaryDeviceId, DeviceId, EthernetDeviceId, EthernetPrimaryDeviceId,
29 EthernetWeakDeviceId,
30};
31use crate::internal::loopback::{LoopbackDeviceId, LoopbackPrimaryDeviceId};
32use crate::internal::pure_ip::{PureIpDeviceId, PureIpPrimaryDeviceId};
33use crate::internal::queue::rx::ReceiveQueueBindingsContext;
34use crate::internal::queue::tx::TransmitQueueBindingsContext;
35use crate::internal::socket::{self, DeviceSocketCounters, HeldSockets};
36use crate::internal::state::DeviceStateSpec;
37
38pub struct DevicesIter<'s, BT: DeviceLayerTypes> {
44 pub(super) ethernet:
45 alloc::collections::hash_map::Values<'s, EthernetDeviceId<BT>, EthernetPrimaryDeviceId<BT>>,
46 pub(super) pure_ip:
47 alloc::collections::hash_map::Values<'s, PureIpDeviceId<BT>, PureIpPrimaryDeviceId<BT>>,
48 pub(super) blackhole: alloc::collections::hash_map::Values<
49 's,
50 BlackholeDeviceId<BT>,
51 BlackholePrimaryDeviceId<BT>,
52 >,
53 pub(super) loopback: core::option::Iter<'s, LoopbackPrimaryDeviceId<BT>>,
54}
55
56impl<'s, BT: DeviceLayerTypes> Iterator for DevicesIter<'s, BT> {
57 type Item = DeviceId<BT>;
58
59 fn next(&mut self) -> Option<Self::Item> {
60 let Self { ethernet, pure_ip, blackhole, loopback } = self;
61 ethernet
62 .map(|primary| primary.clone_strong().into())
63 .chain(pure_ip.map(|primary| primary.clone_strong().into()))
64 .chain(blackhole.map(|primary| primary.clone_strong().into()))
65 .chain(loopback.map(|primary| primary.clone_strong().into()))
66 .next()
67 }
68}
69
70#[allow(missing_docs)]
72pub enum Ipv6DeviceLinkLayerAddr {
73 Mac(Mac),
74 }
76
77impl Ipv6LinkLayerAddr for Ipv6DeviceLinkLayerAddr {
78 fn as_bytes(&self) -> &[u8] {
79 match self {
80 Ipv6DeviceLinkLayerAddr::Mac(a) => a.as_ref(),
81 }
82 }
83
84 fn eui64_iid(&self) -> [u8; 8] {
85 match self {
86 Ipv6DeviceLinkLayerAddr::Mac(a) => a.to_eui64(),
87 }
88 }
89}
90
91#[derive(Derivative)]
93#[derivative(
94 Clone(bound = ""),
95 Eq(bound = ""),
96 PartialEq(bound = ""),
97 Hash(bound = ""),
98 Debug(bound = "")
99)]
100pub struct DeviceLayerTimerId<BT: DeviceLayerTypes>(DeviceLayerTimerIdInner<BT>);
101
102#[derive(Derivative)]
103#[derivative(
104 Clone(bound = ""),
105 Eq(bound = ""),
106 PartialEq(bound = ""),
107 Hash(bound = ""),
108 Debug(bound = "")
109)]
110#[allow(missing_docs)]
111enum DeviceLayerTimerIdInner<BT: DeviceLayerTypes> {
112 Ethernet(EthernetTimerId<EthernetWeakDeviceId<BT>>),
113}
114
115impl<BT: DeviceLayerTypes> From<EthernetTimerId<EthernetWeakDeviceId<BT>>>
116 for DeviceLayerTimerId<BT>
117{
118 fn from(id: EthernetTimerId<EthernetWeakDeviceId<BT>>) -> DeviceLayerTimerId<BT> {
119 DeviceLayerTimerId(DeviceLayerTimerIdInner::Ethernet(id))
120 }
121}
122
123impl<CC, BT> HandleableTimer<CC, BT> for DeviceLayerTimerId<BT>
124where
125 BT: DeviceLayerTypes,
126 CC: TimerHandler<BT, EthernetTimerId<EthernetWeakDeviceId<BT>>>,
127{
128 fn handle(self, core_ctx: &mut CC, bindings_ctx: &mut BT, timer: BT::UniqueTimerId) {
129 let Self(id) = self;
130 match id {
131 DeviceLayerTimerIdInner::Ethernet(id) => core_ctx.handle_timer(bindings_ctx, id, timer),
132 }
133 }
134}
135
136#[derive(Derivative)]
138#[derivative(Default(bound = ""))]
139pub struct Devices<BT: DeviceLayerTypes> {
140 pub ethernet: HashMap<EthernetDeviceId<BT>, EthernetPrimaryDeviceId<BT>>,
142 pub pure_ip: HashMap<PureIpDeviceId<BT>, PureIpPrimaryDeviceId<BT>>,
144 pub blackhole: HashMap<BlackholeDeviceId<BT>, BlackholePrimaryDeviceId<BT>>,
146 pub loopback: Option<LoopbackPrimaryDeviceId<BT>>,
148}
149
150impl<BT: DeviceLayerTypes> Devices<BT> {
151 pub fn iter(&self) -> DevicesIter<'_, BT> {
153 let Self { ethernet, pure_ip, blackhole, loopback } = self;
154 DevicesIter {
155 ethernet: ethernet.values(),
156 pure_ip: pure_ip.values(),
157 blackhole: blackhole.values(),
158 loopback: loopback.iter(),
159 }
160 }
161}
162
163#[derive(Derivative)]
165#[derivative(Default(bound = ""))]
166pub struct DeviceLayerState<BT: DeviceLayerTypes> {
167 devices: RwLock<Devices<BT>>,
168 pub origin: OriginTracker,
170 pub shared_sockets: HeldSockets<BT>,
172 pub device_socket_counters: DeviceSocketCounters,
174 pub counters: DeviceCounters,
176 pub ethernet_counters: EthernetDeviceCounters,
178 pub pure_ip_counters: PureIpDeviceCounters,
180 pub nud_v4_counters: NudCounters<Ipv4>,
182 pub nud_v6_counters: NudCounters<Ipv6>,
184 pub arp_counters: ArpCounters,
186}
187
188impl<BT: DeviceLayerTypes> DeviceLayerState<BT> {
189 pub fn nud_counters<I: Ip>(&self) -> &NudCounters<I> {
191 I::map_ip((), |()| &self.nud_v4_counters, |()| &self.nud_v6_counters)
192 }
193}
194
195impl<BT: DeviceLayerTypes> OrderedLockAccess<Devices<BT>> for DeviceLayerState<BT> {
196 type Lock = RwLock<Devices<BT>>;
197 fn ordered_lock_access(&self) -> OrderedLockRef<'_, Self::Lock> {
198 OrderedLockRef::new(&self.devices)
199 }
200}
201
202#[derive(Default)]
204pub struct EthernetDeviceCounters {
205 pub recv_unsupported_ethertype: Counter,
207 pub recv_no_ethertype: Counter,
209}
210
211impl Inspectable for EthernetDeviceCounters {
212 fn record<I: Inspector>(&self, inspector: &mut I) {
213 inspector.record_child("Ethernet", |inspector| {
214 let Self { recv_no_ethertype, recv_unsupported_ethertype } = self;
215 inspector.record_child("Rx", |inspector| {
216 inspector.record_counter("NoEthertype", recv_no_ethertype);
217 inspector.record_counter("UnsupportedEthertype", recv_unsupported_ethertype);
218 });
219 })
220 }
221}
222
223#[derive(Default)]
225pub struct PureIpDeviceCounters {}
226
227impl Inspectable for PureIpDeviceCounters {
228 fn record<I: Inspector>(&self, _inspector: &mut I) {}
229}
230
231pub struct BlackholeDeviceCounters;
233
234impl Inspectable for BlackholeDeviceCounters {
235 fn record<I: Inspector>(&self, _inspector: &mut I) {}
236}
237
238#[derive(Default)]
240pub struct DeviceCounters {
241 pub send_total_frames: Counter,
244 pub send_frame: Counter,
246 pub send_queue_full: Counter,
248 pub send_serialize_error: Counter,
250 pub recv_frame: Counter,
252 pub recv_parse_error: Counter,
254 pub recv_ipv4_delivered: Counter,
256 pub recv_ipv6_delivered: Counter,
258 pub send_ipv4_frame: Counter,
260 pub send_ipv6_frame: Counter,
262 pub send_dropped_no_queue: Counter,
264 pub send_dropped_dequeue: Counter,
266}
267
268impl DeviceCounters {
269 pub fn send_frame<I: Ip>(&self) -> &Counter {
271 match I::VERSION {
272 IpVersion::V4 => &self.send_ipv4_frame,
273 IpVersion::V6 => &self.send_ipv6_frame,
274 }
275 }
276}
277
278impl Inspectable for DeviceCounters {
279 fn record<I: Inspector>(&self, inspector: &mut I) {
280 let Self {
281 recv_frame,
282 recv_ipv4_delivered,
283 recv_ipv6_delivered,
284 recv_parse_error,
285 send_dropped_no_queue,
286 send_frame,
287 send_ipv4_frame,
288 send_ipv6_frame,
289 send_queue_full,
290 send_serialize_error,
291 send_total_frames,
292 send_dropped_dequeue,
293 } = self;
294 inspector.record_child("Rx", |inspector| {
295 inspector.record_counter("TotalFrames", recv_frame);
296 inspector.record_counter("Malformed", recv_parse_error);
297 inspector.record_counter("Ipv4Delivered", recv_ipv4_delivered);
298 inspector.record_counter("Ipv6Delivered", recv_ipv6_delivered);
299 });
300 inspector.record_child("Tx", |inspector| {
301 inspector.record_counter("TotalFrames", send_total_frames);
302 inspector.record_counter("Sent", send_frame);
303 inspector.record_counter("SendIpv4Frame", send_ipv4_frame);
304 inspector.record_counter("SendIpv6Frame", send_ipv6_frame);
305 inspector.record_counter("NoQueue", send_dropped_no_queue);
306 inspector.record_counter("QueueFull", send_queue_full);
307 inspector.record_counter("SerializeError", send_serialize_error);
308 inspector.record_counter("DequeueDrop", send_dropped_dequeue);
309 });
310 }
311}
312#[derive(Clone, Debug, PartialEq)]
323pub struct OriginTracker(#[cfg(debug_assertions)] u64);
324
325impl Default for OriginTracker {
326 fn default() -> Self {
327 Self::new()
328 }
329}
330
331impl OriginTracker {
332 #[cfg_attr(not(debug_assertions), inline)]
339 fn new() -> Self {
340 Self(
341 #[cfg(debug_assertions)]
342 {
343 static COUNTER: core::sync::atomic::AtomicU64 =
344 core::sync::atomic::AtomicU64::new(0);
345 COUNTER.fetch_add(1, core::sync::atomic::Ordering::Relaxed)
346 },
347 )
348 }
349}
350
351pub trait OriginTrackerContext {
356 fn origin_tracker(&mut self) -> OriginTracker;
358}
359
360pub trait DeviceCollectionContext<D: Device + DeviceStateSpec, BT: DeviceLayerTypes>:
364 DeviceIdContext<D>
365{
366 fn insert(&mut self, device: BasePrimaryDeviceId<D, BT>);
368
369 fn remove(&mut self, device: &BaseDeviceId<D, BT>) -> Option<BasePrimaryDeviceId<D, BT>>;
371}
372
373pub trait DeviceReceiveFrameSpec {
379 type FrameMetadata<D>;
381}
382
383pub trait DeviceLayerStateTypes: InstantContext + FilterBindingsTypes {
385 type LoopbackDeviceState: Send + Sync + DeviceClassMatcher<Self::DeviceClass>;
387
388 type EthernetDeviceState: Send + Sync + DeviceClassMatcher<Self::DeviceClass>;
390
391 type PureIpDeviceState: Send + Sync + DeviceClassMatcher<Self::DeviceClass>;
393
394 type BlackholeDeviceState: Send + Sync + DeviceClassMatcher<Self::DeviceClass>;
396
397 type DeviceIdentifier: Send + Sync + Debug + Display + DeviceIdAndNameMatcher;
400}
401
402pub trait DeviceClassMatcher<DeviceClass> {
405 fn device_class_matches(&self, device_class: &DeviceClass) -> bool;
408}
409
410pub trait DeviceIdAndNameMatcher {
413 fn id_matches(&self, id: &NonZeroU64) -> bool;
415
416 fn name_matches(&self, name: &str) -> bool;
418}
419
420pub trait DeviceLayerTypes:
426 DeviceLayerStateTypes
427 + socket::DeviceSocketTypes
428 + LinkResolutionContext<EthernetLinkDevice>
429 + TimerBindingsTypes
430 + ReferenceNotifiers
431 + TxMetadataBindingsTypes
432 + 'static
433{
434}
435impl<
436 BC: DeviceLayerStateTypes
437 + socket::DeviceSocketTypes
438 + LinkResolutionContext<EthernetLinkDevice>
439 + TimerBindingsTypes
440 + ReferenceNotifiers
441 + TxMetadataBindingsTypes
442 + 'static,
443 > DeviceLayerTypes for BC
444{
445}
446
447pub trait DeviceLayerEventDispatcher:
449 DeviceLayerTypes
450 + ReceiveQueueBindingsContext<LoopbackDeviceId<Self>>
451 + TransmitQueueBindingsContext<EthernetDeviceId<Self>>
452 + TransmitQueueBindingsContext<LoopbackDeviceId<Self>>
453 + TransmitQueueBindingsContext<PureIpDeviceId<Self>>
454 + Sized
455{
456 type DequeueContext;
462
463 fn send_ethernet_frame(
473 &mut self,
474 device: &EthernetDeviceId<Self>,
475 frame: Buf<Vec<u8>>,
476 dequeue_context: Option<&mut Self::DequeueContext>,
477 ) -> Result<(), DeviceSendFrameError>;
478
479 fn send_ip_packet(
489 &mut self,
490 device: &PureIpDeviceId<Self>,
491 packet: Buf<Vec<u8>>,
492 ip_version: IpVersion,
493 dequeue_context: Option<&mut Self::DequeueContext>,
494 ) -> Result<(), DeviceSendFrameError>;
495}
496
497#[derive(Debug, PartialEq, Eq)]
499pub enum DeviceSendFrameError {
500 NoBuffers,
502}
503
504#[cfg(test)]
505mod tests {
506 use super::*;
507
508 #[test]
509 fn origin_tracker() {
510 let tracker = OriginTracker::new();
511 if cfg!(debug_assertions) {
512 assert_ne!(tracker, OriginTracker::new());
513 } else {
514 assert_eq!(tracker, OriginTracker::new());
515 }
516 assert_eq!(tracker.clone(), tracker);
517 }
518}