netstack3_device/
state.rs1use alloc::sync::Arc;
8use core::fmt::Debug;
9
10use lock_order::lock::{OrderedLockAccess, OrderedLockRef};
11use net_types::ip::{Ipv4, Ipv6};
12use netstack3_base::sync::{RwLock, WeakRc};
13use netstack3_base::{
14 CoreTimerContext, Device, DeviceIdContext, Inspectable, TimerContext, WeakDeviceIdentifier,
15};
16use netstack3_ip::device::{DualStackIpDeviceState, IpAddressIdSpec, IpDeviceTimerId};
17use netstack3_ip::RawMetric;
18
19use crate::internal::base::{DeviceCounters, DeviceLayerTypes, OriginTracker};
20use crate::internal::socket::HeldDeviceSockets;
21
22pub trait DeviceStateSpec: Device + Sized + Send + Sync + 'static {
25 type State<BT: DeviceLayerTypes>: Send + Sync;
27 type External<BT: DeviceLayerTypes>: Send + Sync;
29 type CreationProperties: Debug;
31 type Counters: Inspectable;
33 type TimerId<D: WeakDeviceIdentifier>;
35
36 fn new_device_state<
38 CC: CoreTimerContext<Self::TimerId<CC::WeakDeviceId>, BC> + DeviceIdContext<Self>,
39 BC: DeviceLayerTypes + TimerContext,
40 >(
41 bindings_ctx: &mut BC,
42 self_id: CC::WeakDeviceId,
43 properties: Self::CreationProperties,
44 ) -> Self::State<BC>;
45
46 const IS_LOOPBACK: bool;
48 const DEBUG_TYPE: &'static str;
50}
51
52pub(crate) struct WeakCookie<T: DeviceStateSpec, BT: DeviceLayerTypes> {
59 pub(crate) bindings_id: BT::DeviceIdentifier,
60 pub(crate) weak_ref: WeakRc<BaseDeviceState<T, BT>>,
61}
62
63pub(crate) struct BaseDeviceState<T: DeviceStateSpec, BT: DeviceLayerTypes> {
64 pub(crate) ip: IpLinkDeviceState<T, BT>,
65 pub(crate) external_state: T::External<BT>,
66 pub(crate) weak_cookie: Arc<WeakCookie<T, BT>>,
67}
68
69pub type IpLinkDeviceState<T, BT> = IpLinkDeviceStateInner<<T as DeviceStateSpec>::State<BT>, BT>;
73
74pub struct IpLinkDeviceStateInner<T, BT: DeviceLayerTypes> {
78 pub ip: DualStackIpDeviceState<BT>,
80 pub link: T,
82 pub(crate) origin: OriginTracker,
83 pub(super) sockets: RwLock<HeldDeviceSockets<BT>>,
84 pub counters: DeviceCounters,
86}
87
88impl<T, BC: DeviceLayerTypes + TimerContext> IpLinkDeviceStateInner<T, BC> {
89 pub fn new<
91 D: WeakDeviceIdentifier,
92 A: IpAddressIdSpec,
93 CC: CoreTimerContext<IpDeviceTimerId<Ipv6, D, A>, BC>
94 + CoreTimerContext<IpDeviceTimerId<Ipv4, D, A>, BC>,
95 >(
96 bindings_ctx: &mut BC,
97 device_id: D,
98 link: T,
99 metric: RawMetric,
100 origin: OriginTracker,
101 ) -> Self {
102 Self {
103 ip: DualStackIpDeviceState::new::<D, A, CC>(bindings_ctx, device_id, metric),
104 link,
105 origin,
106 sockets: RwLock::new(HeldDeviceSockets::default()),
107 counters: DeviceCounters::default(),
108 }
109 }
110}
111
112impl<T, BT: DeviceLayerTypes> AsRef<DualStackIpDeviceState<BT>> for IpLinkDeviceStateInner<T, BT> {
113 fn as_ref(&self) -> &DualStackIpDeviceState<BT> {
114 &self.ip
115 }
116}
117
118impl<T, BT: DeviceLayerTypes> OrderedLockAccess<HeldDeviceSockets<BT>>
119 for IpLinkDeviceStateInner<T, BT>
120{
121 type Lock = RwLock<HeldDeviceSockets<BT>>;
122 fn ordered_lock_access(&self) -> OrderedLockRef<'_, Self::Lock> {
123 OrderedLockRef::new(&self.sockets)
124 }
125}