1mod integration;
59
60use derivative::Derivative;
61use net_types::ip::{Ip, Ipv4, Ipv6};
62use netstack3_base::{CoreTxMetadataContext, HandleableTimer, TimerHandler};
63use netstack3_datagram as datagram;
64use netstack3_device::WeakDeviceId;
65use netstack3_icmp_echo::{IcmpSocketTxMetadata, IcmpSockets};
66use netstack3_tcp::{
67 self as tcp, TcpCountersWithSocket, TcpCountersWithoutSocket, TcpState, TcpTimerId,
68};
69use netstack3_udp::{
70 UdpCountersWithSocket, UdpCountersWithoutSocket, UdpSocketTxMetadata, UdpState,
71};
72
73use crate::{BindingsContext, BindingsTypes, CoreCtx, IpExt};
74
75#[derive(Default, Clone)]
77pub struct TransportStateBuilder;
78
79impl TransportStateBuilder {
80 pub(crate) fn build_with_ctx<BC: BindingsContext>(
81 self,
82 bindings_ctx: &mut BC,
83 ) -> TransportLayerState<BC> {
84 let now = bindings_ctx.now();
85 let mut rng = bindings_ctx.rng();
86 TransportLayerState {
87 udpv4: Default::default(),
88 udpv6: Default::default(),
89 tcpv4: TcpState::new(now, &mut rng),
90 tcpv6: TcpState::new(now, &mut rng),
91 icmp_echo_v4: Default::default(),
92 icmp_echo_v6: Default::default(),
93 }
94 }
95}
96
97pub struct TransportLayerState<BT: BindingsTypes> {
99 udpv4: UdpState<Ipv4, WeakDeviceId<BT>, BT>,
100 udpv6: UdpState<Ipv6, WeakDeviceId<BT>, BT>,
101 tcpv4: TcpState<Ipv4, WeakDeviceId<BT>, BT>,
102 tcpv6: TcpState<Ipv6, WeakDeviceId<BT>, BT>,
103 icmp_echo_v4: IcmpSockets<Ipv4, WeakDeviceId<BT>, BT>,
104 icmp_echo_v6: IcmpSockets<Ipv6, WeakDeviceId<BT>, BT>,
105}
106
107impl<BT: BindingsTypes> TransportLayerState<BT> {
108 fn tcp_state<I: tcp::DualStackIpExt>(&self) -> &TcpState<I, WeakDeviceId<BT>, BT> {
109 I::map_ip((), |()| &self.tcpv4, |()| &self.tcpv6)
110 }
111
112 fn udp_state<I: datagram::IpExt>(&self) -> &UdpState<I, WeakDeviceId<BT>, BT> {
113 I::map_ip((), |()| &self.udpv4, |()| &self.udpv6)
114 }
115
116 pub(crate) fn icmp_echo_state<I: datagram::IpExt>(
117 &self,
118 ) -> &IcmpSockets<I, WeakDeviceId<BT>, BT> {
119 I::map_ip((), |()| &self.icmp_echo_v4, |()| &self.icmp_echo_v6)
120 }
121
122 pub(crate) fn udp_counters_with_socket<I: Ip>(&self) -> &UdpCountersWithSocket<I> {
123 I::map_ip((), |()| &self.udpv4.counters_with_socket, |()| &self.udpv6.counters_with_socket)
124 }
125
126 pub(crate) fn udp_counters_without_socket<I: Ip>(&self) -> &UdpCountersWithoutSocket<I> {
127 I::map_ip(
128 (),
129 |()| &self.udpv4.counters_without_socket,
130 |()| &self.udpv6.counters_without_socket,
131 )
132 }
133
134 pub(crate) fn tcp_counters_with_socket<I: Ip>(&self) -> &TcpCountersWithSocket<I> {
135 I::map_ip((), |()| &self.tcpv4.counters_with_socket, |()| &self.tcpv6.counters_with_socket)
136 }
137
138 pub(crate) fn tcp_counters_without_socket<I: Ip>(&self) -> &TcpCountersWithoutSocket<I> {
139 I::map_ip(
140 (),
141 |()| &self.tcpv4.counters_without_socket,
142 |()| &self.tcpv6.counters_without_socket,
143 )
144 }
145}
146
147#[derive(Derivative)]
149#[derivative(
150 Clone(bound = ""),
151 Eq(bound = ""),
152 PartialEq(bound = ""),
153 Hash(bound = ""),
154 Debug(bound = "")
155)]
156pub(crate) enum TransportLayerTimerId<BT: BindingsTypes> {
157 Tcp(TcpTimerId<WeakDeviceId<BT>, BT>),
158}
159
160impl<CC, BT> HandleableTimer<CC, BT> for TransportLayerTimerId<BT>
161where
162 BT: BindingsTypes,
163 CC: TimerHandler<BT, TcpTimerId<WeakDeviceId<BT>, BT>>,
164{
165 fn handle(self, core_ctx: &mut CC, bindings_ctx: &mut BT, timer: BT::UniqueTimerId) {
166 match self {
167 TransportLayerTimerId::Tcp(id) => core_ctx.handle_timer(bindings_ctx, id, timer),
168 }
169 }
170}
171
172impl<BT: BindingsTypes> From<TcpTimerId<WeakDeviceId<BT>, BT>> for TransportLayerTimerId<BT> {
173 fn from(id: TcpTimerId<WeakDeviceId<BT>, BT>) -> Self {
174 TransportLayerTimerId::Tcp(id)
175 }
176}
177
178#[derive(Derivative)]
180#[derivative(Debug = "transparent", Debug(bound = ""), Default(bound = ""))]
181#[cfg_attr(any(test, feature = "testutils"), derivative(PartialEq(bound = "")))]
182pub struct TxMetadata<BT: BindingsTypes>(TxMetadataInner<BT>);
183
184#[derive(Derivative)]
188#[derivative(Debug(bound = ""), Default(bound = ""))]
189#[cfg_attr(any(test, feature = "testutils"), derivative(PartialEq(bound = "")))]
190enum TxMetadataInner<BT: BindingsTypes> {
191 #[derivative(Default)]
192 None,
193 #[derivative(Debug = "transparent")]
194 Udpv4(UdpSocketTxMetadata<Ipv4, WeakDeviceId<BT>, BT>),
195 #[derivative(Debug = "transparent")]
196 Udpv6(UdpSocketTxMetadata<Ipv6, WeakDeviceId<BT>, BT>),
197 #[derivative(Debug = "transparent")]
198 Icmpv4(IcmpSocketTxMetadata<Ipv4, WeakDeviceId<BT>, BT>),
199 #[derivative(Debug = "transparent")]
200 Icmpv6(IcmpSocketTxMetadata<Ipv6, WeakDeviceId<BT>, BT>),
201}
202
203impl<I: IpExt, L, BT: BindingsTypes>
204 CoreTxMetadataContext<UdpSocketTxMetadata<I, WeakDeviceId<BT>, BT>, BT> for CoreCtx<'_, BT, L>
205{
206 fn convert_tx_meta(
207 &self,
208 tx_meta: UdpSocketTxMetadata<I, WeakDeviceId<BT>, BT>,
209 ) -> TxMetadata<BT> {
210 TxMetadata(I::map_ip_in(tx_meta, TxMetadataInner::Udpv4, TxMetadataInner::Udpv6))
211 }
212}
213
214impl<I: IpExt, L, BT: BindingsTypes>
215 CoreTxMetadataContext<IcmpSocketTxMetadata<I, WeakDeviceId<BT>, BT>, BT>
216 for CoreCtx<'_, BT, L>
217{
218 fn convert_tx_meta(
219 &self,
220 tx_meta: IcmpSocketTxMetadata<I, WeakDeviceId<BT>, BT>,
221 ) -> TxMetadata<BT> {
222 TxMetadata(I::map_ip_in(tx_meta, TxMetadataInner::Icmpv4, TxMetadataInner::Icmpv6))
223 }
224}