1pub(crate) mod api;
17pub(crate) mod counters;
18pub(crate) mod packet_queue;
19pub(crate) mod route;
20pub(crate) mod state;
21
22use core::sync::atomic::Ordering;
23
24use net_types::ip::{GenericOverIp, Ip, IpVersionMarker};
25use netstack3_base::{
26 AnyDevice, AtomicInstant, CounterContext, DeviceIdContext, EventContext, FrameDestination,
27 HandleableTimer, InstantBindingsTypes, InstantContext, TimerBindingsTypes, TimerContext,
28 WeakDeviceIdentifier,
29};
30use packet_formats::ip::IpPacket;
31use zerocopy::SplitByteSlice;
32
33use crate::internal::multicast_forwarding::counters::MulticastForwardingCounters;
34use crate::internal::multicast_forwarding::packet_queue::QueuePacketOutcome;
35use crate::internal::multicast_forwarding::route::{
36 Action, MulticastRouteEntry, MulticastRouteTargets,
37};
38use crate::multicast_forwarding::{
39 MulticastForwardingPendingPacketsContext, MulticastForwardingState,
40 MulticastForwardingStateContext, MulticastRoute, MulticastRouteKey,
41 MulticastRouteTableContext as _,
42};
43use crate::{IpLayerEvent, IpLayerIpExt};
44
45pub trait MulticastForwardingBindingsTypes: InstantBindingsTypes + TimerBindingsTypes {}
47impl<BT: InstantBindingsTypes + TimerBindingsTypes> MulticastForwardingBindingsTypes for BT {}
48
49pub trait MulticastForwardingBindingsContext<I: IpLayerIpExt, D>:
51 MulticastForwardingBindingsTypes + InstantContext + TimerContext + EventContext<IpLayerEvent<D, I>>
52{
53}
54impl<
55 I: IpLayerIpExt,
56 D,
57 BC: MulticastForwardingBindingsTypes
58 + InstantContext
59 + TimerContext
60 + EventContext<IpLayerEvent<D, I>>,
61> MulticastForwardingBindingsContext<I, D> for BC
62{
63}
64
65pub trait MulticastForwardingDeviceContext<I: IpLayerIpExt>: DeviceIdContext<AnyDevice> {
67 fn is_device_multicast_forwarding_enabled(&mut self, dev: &Self::DeviceId) -> bool;
69}
70
71#[derive(Clone, Debug, Eq, GenericOverIp, Hash, PartialEq)]
73#[generic_over_ip(I, Ip)]
74pub enum MulticastForwardingTimerId<I: Ip> {
75 PendingPacketsGc(IpVersionMarker<I>),
77}
78
79impl<
80 I: IpLayerIpExt,
81 BC: MulticastForwardingBindingsContext<I, CC::DeviceId>,
82 CC: MulticastForwardingStateContext<I, BC> + CounterContext<MulticastForwardingCounters<I>>,
83> HandleableTimer<CC, BC> for MulticastForwardingTimerId<I>
84{
85 fn handle(self, core_ctx: &mut CC, bindings_ctx: &mut BC, _: BC::UniqueTimerId) {
86 match self {
87 MulticastForwardingTimerId::PendingPacketsGc(_) => {
88 core_ctx.with_state(|state, ctx| match state {
89 MulticastForwardingState::Disabled => {}
92 MulticastForwardingState::Enabled(state) => {
93 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
94 .pending_table_gc
95 .increment();
96 let removed_count = ctx.with_pending_table_mut(state, |pending_table| {
97 pending_table.run_garbage_collection(bindings_ctx)
98 });
99 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
100 .pending_packet_drops_gc
101 .add(removed_count);
102 }
103 })
104 }
105 }
106 }
107}
108
109#[derive(Debug, Eq, Hash, PartialEq, GenericOverIp)]
111#[generic_over_ip(I, Ip)]
112pub enum MulticastForwardingEvent<I: IpLayerIpExt, D> {
113 MissingRoute {
115 key: MulticastRouteKey<I>,
117 input_interface: D,
119 },
120 WrongInputInterface {
122 key: MulticastRouteKey<I>,
124 actual_input_interface: D,
126 expected_input_interface: D,
129 },
130}
131
132impl<I: IpLayerIpExt, D> MulticastForwardingEvent<I, D> {
133 pub(crate) fn map_device<O, F: Fn(D) -> O>(self, map: F) -> MulticastForwardingEvent<I, O> {
134 match self {
135 MulticastForwardingEvent::MissingRoute { key, input_interface } => {
136 MulticastForwardingEvent::MissingRoute {
137 key,
138 input_interface: map(input_interface),
139 }
140 }
141 MulticastForwardingEvent::WrongInputInterface {
142 key,
143 actual_input_interface,
144 expected_input_interface,
145 } => MulticastForwardingEvent::WrongInputInterface {
146 key,
147 actual_input_interface: map(actual_input_interface),
148 expected_input_interface: map(expected_input_interface),
149 },
150 }
151 }
152}
153
154impl<I: IpLayerIpExt, D: WeakDeviceIdentifier> MulticastForwardingEvent<I, D> {
155 pub fn upgrade_device_id(self) -> Option<MulticastForwardingEvent<I, D::Strong>> {
157 match self {
158 MulticastForwardingEvent::MissingRoute { key, input_interface } => {
159 Some(MulticastForwardingEvent::MissingRoute {
160 key,
161 input_interface: input_interface.upgrade()?,
162 })
163 }
164 MulticastForwardingEvent::WrongInputInterface {
165 key,
166 actual_input_interface,
167 expected_input_interface,
168 } => Some(MulticastForwardingEvent::WrongInputInterface {
169 key,
170 actual_input_interface: actual_input_interface.upgrade()?,
171 expected_input_interface: expected_input_interface.upgrade()?,
172 }),
173 }
174 }
175}
176
177pub(crate) fn lookup_multicast_route_or_stash_packet<I, B, CC, BC>(
193 core_ctx: &mut CC,
194 bindings_ctx: &mut BC,
195 packet: &I::Packet<B>,
196 dev: &CC::DeviceId,
197 frame_dst: Option<FrameDestination>,
198) -> Option<MulticastRouteTargets<CC::DeviceId>>
199where
200 I: IpLayerIpExt,
201 B: SplitByteSlice,
202 CC: MulticastForwardingStateContext<I, BC>
203 + MulticastForwardingDeviceContext<I>
204 + CounterContext<MulticastForwardingCounters<I>>,
205 BC: MulticastForwardingBindingsContext<I, CC::DeviceId>,
206{
207 CounterContext::<MulticastForwardingCounters<I>>::counters(core_ctx).rx.increment();
208 let Some(key) = MulticastRouteKey::new(packet.src_ip(), packet.dst_ip()) else {
211 CounterContext::<MulticastForwardingCounters<I>>::counters(core_ctx)
212 .no_tx_invalid_key
213 .increment();
214 return None;
215 };
216
217 if !core_ctx.is_device_multicast_forwarding_enabled(dev) {
219 CounterContext::<MulticastForwardingCounters<I>>::counters(core_ctx)
220 .no_tx_disabled_dev
221 .increment();
222 return None;
223 }
224
225 core_ctx.with_state(|state, ctx| {
226 let Some(state) = state.enabled() else {
228 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
229 .no_tx_disabled_stack_wide
230 .increment();
231 return None;
232 };
233 ctx.with_route_table(state, |route_table, ctx| {
234 if let Some(MulticastRouteEntry {
235 route: MulticastRoute { input_interface, action },
236 stats,
237 }) = route_table.get(&key)
238 {
239 if dev != input_interface {
240 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
241 .no_tx_wrong_dev
242 .increment();
243 bindings_ctx.on_event(
244 MulticastForwardingEvent::WrongInputInterface {
245 key,
246 actual_input_interface: dev.clone(),
247 expected_input_interface: input_interface.clone(),
248 }
249 .into(),
250 );
251 return None;
252 }
253
254 stats.last_used.store_max(bindings_ctx.now(), Ordering::Relaxed);
255
256 match action {
257 Action::Forward(targets) => {
258 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
259 .tx
260 .increment();
261 return Some(targets.clone());
262 }
263 }
264 }
265 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
266 .pending_packets
267 .increment();
268 match ctx.with_pending_table_mut(state, |pending_table| {
269 pending_table.try_queue_packet(bindings_ctx, key.clone(), packet, dev, frame_dst)
270 }) {
271 QueuePacketOutcome::QueuedInNewQueue => {
272 bindings_ctx.on_event(
273 MulticastForwardingEvent::MissingRoute {
274 key,
275 input_interface: dev.clone(),
276 }
277 .into(),
278 );
279 }
280 QueuePacketOutcome::QueuedInExistingQueue => {}
281 QueuePacketOutcome::ExistingQueueFull => {
282 CounterContext::<MulticastForwardingCounters<I>>::counters(ctx)
283 .pending_packet_drops_queue_full
284 .increment();
285 }
286 }
287 return None;
288 })
289 })
290}
291
292#[cfg(test)]
293mod testutil {
294 use super::*;
295
296 use alloc::rc::Rc;
297 use alloc::vec::Vec;
298 use core::cell::RefCell;
299 use derivative::Derivative;
300 use net_declare::{net_ip_v4, net_ip_v6};
301 use net_types::MulticastAddr;
302 use net_types::ip::{Ipv4, Ipv4Addr, Ipv6, Ipv6Addr, Mtu};
303 use netstack3_base::socket::SocketIpAddr;
304 use netstack3_base::testutil::{FakeStrongDeviceId, MultipleDevicesId};
305 use netstack3_base::{
306 CoreTimerContext, CounterContext, CtxPair, FrameDestination, Marks,
307 NetworkSerializationContext, NetworkSerializer, ResourceCounterContext,
308 };
309 use netstack3_filter::ProofOfEgressCheck;
310 use netstack3_hashmap::HashSet;
311 use packet::{BufferMut, InnerPacketBuilder, NestablePacketBuilder as _, Serializer};
312 use packet_formats::ip::{IpPacketBuilder, IpProto};
313
314 use crate::device::IpDeviceSendContext;
315 use crate::internal::base::DeviceIpLayerMetadata;
316 use crate::internal::icmp::IcmpErrorHandler;
317 use crate::multicast_forwarding::{
318 MulticastForwardingApi, MulticastForwardingEnabledState, MulticastForwardingPendingPackets,
319 MulticastForwardingPendingPacketsContext, MulticastForwardingState, MulticastRouteTable,
320 MulticastRouteTableContext,
321 };
322 use crate::{IpCounters, IpDeviceMtuContext, IpLayerEvent, IpPacketDestination};
323
324 pub(crate) trait TestIpExt: IpLayerIpExt {
326 const SRC1: Self::Addr;
327 const SRC2: Self::Addr;
328 const DST1: Self::Addr;
329 const DST2: Self::Addr;
330 }
331
332 impl TestIpExt for Ipv4 {
333 const SRC1: Ipv4Addr = net_ip_v4!("192.0.2.1");
334 const SRC2: Ipv4Addr = net_ip_v4!("192.0.2.2");
335 const DST1: Ipv4Addr = net_ip_v4!("224.0.1.1");
336 const DST2: Ipv4Addr = net_ip_v4!("224.0.1.2");
337 }
338
339 impl TestIpExt for Ipv6 {
340 const SRC1: Ipv6Addr = net_ip_v6!("2001:0DB8::1");
341 const SRC2: Ipv6Addr = net_ip_v6!("2001:0DB8::2");
342 const DST1: Ipv6Addr = net_ip_v6!("ff0e::1");
343 const DST2: Ipv6Addr = net_ip_v6!("ff0e::2");
344 }
345
346 pub(crate) fn new_ip_packet_buf<I: IpLayerIpExt>(
348 src_addr: I::Addr,
349 dst_addr: I::Addr,
350 ) -> impl AsRef<[u8]> {
351 const TTL: u8 = 255;
352 const IP_BODY: [u8; 10] = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
354 I::PacketBuilder::new(src_addr, dst_addr, TTL, IpProto::Udp.into())
355 .wrap_body(IP_BODY.into_serializer())
356 .serialize_vec_outer(&mut NetworkSerializationContext::default())
357 .unwrap()
358 }
359
360 #[derive(Debug, PartialEq)]
361 pub(crate) struct SentPacket<I: IpLayerIpExt, D> {
362 pub(crate) dst: MulticastAddr<I::Addr>,
363 pub(crate) device: D,
364 }
365
366 #[derive(Derivative)]
367 #[derivative(Default(bound = ""))]
368 pub(crate) struct FakeCoreCtxState<I: IpLayerIpExt, D: FakeStrongDeviceId> {
369 pub(crate) multicast_forwarding:
373 Rc<RefCell<MulticastForwardingState<I, D, FakeBindingsCtx<I, D>>>>,
374 pub(crate) forwarding_enabled_devices: HashSet<D>,
376 pub(crate) sent_packets: Vec<SentPacket<I, D>>,
378 stack_wide_counters: IpCounters<I>,
379 per_device_counters: IpCounters<I>,
380 multicast_forwarding_counters: MulticastForwardingCounters<I>,
381 }
382
383 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> FakeCoreCtxState<I, D> {
384 pub(crate) fn set_multicast_forwarding_enabled_for_dev(&mut self, dev: D, enabled: bool) {
385 if enabled {
386 let _: bool = self.forwarding_enabled_devices.insert(dev);
387 } else {
388 let _: bool = self.forwarding_enabled_devices.remove(&dev);
389 }
390 }
391
392 pub(crate) fn take_sent_packets(&mut self) -> Vec<SentPacket<I, D>> {
393 core::mem::take(&mut self.sent_packets)
394 }
395 }
396
397 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> CounterContext<IpCounters<I>>
398 for FakeCoreCtxState<I, D>
399 {
400 fn counters(&self) -> &IpCounters<I> {
401 &self.stack_wide_counters
402 }
403 }
404
405 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> ResourceCounterContext<D, IpCounters<I>>
406 for FakeCoreCtxState<I, D>
407 {
408 fn per_resource_counters(&self, _resource: &D) -> &IpCounters<I> {
409 &self.per_device_counters
410 }
411 }
412
413 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> CounterContext<MulticastForwardingCounters<I>>
414 for FakeCoreCtxState<I, D>
415 {
416 fn counters(&self) -> &MulticastForwardingCounters<I> {
417 &self.multicast_forwarding_counters
418 }
419 }
420
421 pub(crate) type FakeBindingsCtx<I, D> = netstack3_base::testutil::FakeBindingsCtx<
422 MulticastForwardingTimerId<I>,
423 IpLayerEvent<D, I>,
424 (),
425 (),
426 >;
427 pub(crate) type FakeCoreCtx<I, D> =
428 netstack3_base::testutil::FakeCoreCtx<FakeCoreCtxState<I, D>, (), D>;
429
430 impl<I: IpLayerIpExt, D: FakeStrongDeviceId>
431 MulticastForwardingStateContext<I, FakeBindingsCtx<I, D>> for FakeCoreCtx<I, D>
432 {
433 type Ctx<'a> = FakeCoreCtx<I, D>;
434 fn with_state<
435 O,
436 F: FnOnce(
437 &MulticastForwardingState<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
438 &mut Self::Ctx<'_>,
439 ) -> O,
440 >(
441 &mut self,
442 cb: F,
443 ) -> O {
444 let state = self.state.multicast_forwarding.clone();
445 let borrow = state.borrow();
446 cb(&borrow, self)
447 }
448 fn with_state_mut<
449 O,
450 F: FnOnce(
451 &mut MulticastForwardingState<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
452 &mut Self::Ctx<'_>,
453 ) -> O,
454 >(
455 &mut self,
456 cb: F,
457 ) -> O {
458 let state = self.state.multicast_forwarding.clone();
459 let mut borrow = state.borrow_mut();
460 cb(&mut borrow, self)
461 }
462 }
463
464 impl<I: IpLayerIpExt, D: FakeStrongDeviceId>
465 MulticastRouteTableContext<I, FakeBindingsCtx<I, D>> for FakeCoreCtx<I, D>
466 {
467 type Ctx<'a> = FakeCoreCtx<I, D>;
468 fn with_route_table<
469 O,
470 F: FnOnce(
471 &MulticastRouteTable<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
472 &mut Self::Ctx<'_>,
473 ) -> O,
474 >(
475 &mut self,
476 state: &MulticastForwardingEnabledState<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
477 cb: F,
478 ) -> O {
479 let route_table = state.route_table().read();
480 cb(&route_table, self)
481 }
482 fn with_route_table_mut<
483 O,
484 F: FnOnce(
485 &mut MulticastRouteTable<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
486 &mut Self::Ctx<'_>,
487 ) -> O,
488 >(
489 &mut self,
490 state: &MulticastForwardingEnabledState<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
491 cb: F,
492 ) -> O {
493 let mut route_table = state.route_table().write();
494 cb(&mut route_table, self)
495 }
496 }
497
498 impl<I: IpLayerIpExt, D: FakeStrongDeviceId>
499 MulticastForwardingPendingPacketsContext<I, FakeBindingsCtx<I, D>> for FakeCoreCtx<I, D>
500 {
501 fn with_pending_table_mut<
502 O,
503 F: FnOnce(
504 &mut MulticastForwardingPendingPackets<I, Self::WeakDeviceId, FakeBindingsCtx<I, D>>,
505 ) -> O,
506 >(
507 &mut self,
508 state: &MulticastForwardingEnabledState<I, Self::DeviceId, FakeBindingsCtx<I, D>>,
509 cb: F,
510 ) -> O {
511 let mut pending_table = state.pending_table().lock();
512 cb(&mut pending_table)
513 }
514 }
515
516 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> MulticastForwardingDeviceContext<I>
517 for FakeCoreCtx<I, D>
518 {
519 fn is_device_multicast_forwarding_enabled(&mut self, device_id: &Self::DeviceId) -> bool {
520 self.state.forwarding_enabled_devices.contains(device_id)
521 }
522 }
523
524 impl<I: IpLayerIpExt, D: FakeStrongDeviceId>
525 CoreTimerContext<MulticastForwardingTimerId<I>, FakeBindingsCtx<I, D>>
526 for FakeCoreCtx<I, D>
527 {
528 fn convert_timer(
529 dispatch_id: MulticastForwardingTimerId<I>,
530 ) -> MulticastForwardingTimerId<I> {
531 dispatch_id
532 }
533 }
534
535 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> IpDeviceSendContext<I, FakeBindingsCtx<I, D>>
536 for FakeCoreCtx<I, D>
537 {
538 fn send_ip_frame<S>(
539 &mut self,
540 _bindings_ctx: &mut FakeBindingsCtx<I, D>,
541 device_id: &D,
542 destination: IpPacketDestination<I, &D>,
543 _ip_layer_metadata: DeviceIpLayerMetadata<FakeBindingsCtx<I, D>>,
544 _body: S,
545 _egress_proof: ProofOfEgressCheck,
546 ) -> Result<(), netstack3_base::SendFrameError<S>>
547 where
548 S: NetworkSerializer,
549 S::Buffer: BufferMut,
550 {
551 let dst = match destination {
552 IpPacketDestination::Multicast(dst) => dst,
553 dst => panic!("unexpected sent packet: destination={dst:?}"),
554 };
555 self.state.sent_packets.push(SentPacket { dst, device: device_id.clone() });
556 Ok(())
557 }
558 }
559
560 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> IpDeviceMtuContext<I> for FakeCoreCtx<I, D> {
561 fn get_mtu(&mut self, _device_id: &Self::DeviceId) -> Mtu {
562 Mtu::max()
563 }
564 }
565
566 impl<I: IpLayerIpExt, D: FakeStrongDeviceId> IcmpErrorHandler<I, FakeBindingsCtx<I, D>>
567 for FakeCoreCtx<I, D>
568 {
569 fn send_icmp_error_message<B: BufferMut>(
570 &mut self,
571 _bindings_ctx: &mut FakeBindingsCtx<I, D>,
572 _device: Option<&D>,
573 _frame_dst: Option<FrameDestination>,
574 _src_ip: SocketIpAddr<I::Addr>,
575 _dst_ip: SocketIpAddr<I::Addr>,
576 _original_packet: B,
577 _error: I::IcmpError,
578 _header_len: usize,
579 _proto: I::Proto,
580 _marks: &Marks,
581 ) {
582 unimplemented!()
583 }
584 }
585
586 pub(crate) fn new_api<I: IpLayerIpExt>() -> MulticastForwardingApi<
587 I,
588 CtxPair<FakeCoreCtx<I, MultipleDevicesId>, FakeBindingsCtx<I, MultipleDevicesId>>,
589 > {
590 MulticastForwardingApi::new(CtxPair::with_core_ctx(FakeCoreCtx::with_state(
591 Default::default(),
592 )))
593 }
594
595 pub(crate) fn with_pending_table<I, O, F, CC, BT>(core_ctx: &mut CC, cb: F) -> O
601 where
602 I: IpLayerIpExt,
603 CC: MulticastForwardingStateContext<I, BT>,
604 BT: MulticastForwardingBindingsTypes,
605 F: FnOnce(&mut MulticastForwardingPendingPackets<I, CC::WeakDeviceId, BT>) -> O,
606 {
607 core_ctx.with_state(|state, ctx| {
608 let state = state.enabled().unwrap();
609 ctx.with_route_table(state, |_routing_table, ctx| {
610 ctx.with_pending_table_mut(state, |pending_table| cb(pending_table))
611 })
612 })
613 }
614}
615
616#[cfg(test)]
617mod tests {
618 use super::*;
619
620 use alloc::vec;
621 use core::time::Duration;
622
623 use ip_test_macro::ip_test;
624 use netstack3_base::testutil::MultipleDevicesId;
625 use packet::ParseBuffer;
626 use test_case::test_case;
627 use testutil::TestIpExt;
628
629 use crate::internal::multicast_forwarding::route::MulticastRouteStats;
630 use crate::multicast_forwarding::MulticastRouteTarget;
631
632 struct LookupTestCase {
633 enabled: bool,
635 dev_enabled: bool,
637 right_key: bool,
639 right_dev: bool,
641 }
642 const LOOKUP_SUCCESS_CASE: LookupTestCase =
643 LookupTestCase { enabled: true, dev_enabled: true, right_key: true, right_dev: true };
644
645 #[ip_test(I)]
646 #[test_case(LOOKUP_SUCCESS_CASE => true; "success")]
647 #[test_case(LookupTestCase{enabled: false, ..LOOKUP_SUCCESS_CASE} => false; "disabled")]
648 #[test_case(LookupTestCase{dev_enabled: false, ..LOOKUP_SUCCESS_CASE} => false; "dev_disabled")]
649 #[test_case(LookupTestCase{right_key: false, ..LOOKUP_SUCCESS_CASE} => false; "wrong_key")]
650 #[test_case(LookupTestCase{right_dev: false, ..LOOKUP_SUCCESS_CASE} => false; "wrong_dev")]
651 fn lookup_route<I: TestIpExt>(test_case: LookupTestCase) -> bool {
652 let LookupTestCase { enabled, dev_enabled, right_key, right_dev } = test_case;
653 const FRAME_DST: Option<FrameDestination> = None;
654 let mut api = testutil::new_api::<I>();
655
656 let expected_key = MulticastRouteKey::new(I::SRC1, I::DST1).unwrap();
657 let actual_key = if right_key {
658 expected_key.clone()
659 } else {
660 MulticastRouteKey::new(I::SRC2, I::DST2).unwrap()
661 };
662
663 let expected_dev = MultipleDevicesId::A;
664 let actual_dev = if right_dev { expected_dev } else { MultipleDevicesId::B };
665
666 if enabled {
667 assert!(api.enable());
668 assert_eq!(
671 api.add_multicast_route(
672 expected_key.clone(),
673 MulticastRoute::new_forward(
674 expected_dev,
675 [MulticastRouteTarget {
676 output_interface: MultipleDevicesId::C,
677 min_ttl: 0
678 }]
679 .into()
680 )
681 .unwrap()
682 ),
683 Ok(None)
684 );
685 }
686
687 api.core_ctx().state.set_multicast_forwarding_enabled_for_dev(actual_dev, dev_enabled);
688
689 let (core_ctx, bindings_ctx) = api.contexts();
690 let creation_time = bindings_ctx.now();
691 bindings_ctx.timers.instant.sleep(Duration::from_secs(5));
692 let lookup_time = bindings_ctx.now();
693 assert!(lookup_time > creation_time);
694
695 let buf = testutil::new_ip_packet_buf::<I>(actual_key.src_addr(), actual_key.dst_addr());
696 let mut buf_ref = buf.as_ref();
697 let packet = buf_ref.parse::<I::Packet<_>>().expect("parse should succeed");
698
699 let route = lookup_multicast_route_or_stash_packet(
700 core_ctx,
701 bindings_ctx,
702 &packet,
703 &actual_dev,
704 FRAME_DST,
705 );
706
707 let mut expected_events = vec![];
709 if !right_key {
710 expected_events.push(IpLayerEvent::MulticastForwarding(
711 MulticastForwardingEvent::MissingRoute {
712 key: actual_key.clone(),
713 input_interface: actual_dev,
714 },
715 ));
716 }
717 if !right_dev {
718 expected_events.push(IpLayerEvent::MulticastForwarding(
719 MulticastForwardingEvent::WrongInputInterface {
720 key: actual_key,
721 actual_input_interface: actual_dev,
722 expected_input_interface: expected_dev,
723 },
724 ));
725 }
726 assert_eq!(bindings_ctx.take_events(), expected_events);
727
728 let lookup_succeeded = route.is_some();
729
730 if enabled {
731 let expected_stats = if lookup_succeeded {
733 MulticastRouteStats { last_used: lookup_time }
734 } else {
735 MulticastRouteStats { last_used: creation_time }
736 };
737 assert_eq!(api.get_route_stats(&expected_key), Ok(Some(expected_stats)));
738 }
739
740 let counters: &MulticastForwardingCounters<I> = api.core_ctx().counters();
742 assert_eq!(counters.rx.get(), 1);
743 assert_eq!(counters.tx.get(), if lookup_succeeded { 1 } else { 0 });
744 assert_eq!(counters.no_tx_disabled_dev.get(), if dev_enabled { 0 } else { 1 });
745 assert_eq!(counters.no_tx_disabled_stack_wide.get(), if enabled { 0 } else { 1 });
746 assert_eq!(counters.no_tx_wrong_dev.get(), if right_dev { 0 } else { 1 });
747 assert_eq!(counters.pending_packets.get(), if right_key { 0 } else { 1 });
748
749 lookup_succeeded
750 }
751}