wlan_hw_sim/event/
filter.rs

1// Copyright 2023 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5//! Event handlers that filter their input events to more specific events.
6//!
7//! Filters are powered by the `AsEvent` trait, which extracts a reference to an event sub-type
8//! from a super-type (typically a sum type like `enum`s). The handlers composed by filters are
9//! only invoked when such a sub-type is received. Otherwise, filters return
10//! [`Handled::Unmatched`].
11//!
12//! [`Handled::Unmatched`]: crate::event::Handled
13
14use fidl_fuchsia_wlan_tap as fidl_tap;
15
16use crate::event::{Handled, Handler, StartMacArgs};
17
18// This event has no fields and is zero-size. This constant is used to return a static reference in
19// the corresponding `AsEvent` implementation.
20const START_MAC_ARGS: StartMacArgs = StartMacArgs;
21
22pub trait AsEvent<E> {
23    fn as_event(&self) -> Option<&E>;
24}
25
26impl<E> AsEvent<E> for E {
27    fn as_event(&self) -> Option<&E> {
28        Some(self)
29    }
30}
31
32impl AsEvent<fidl_tap::JoinBssArgs> for fidl_tap::WlantapPhyEvent {
33    fn as_event(&self) -> Option<&fidl_tap::JoinBssArgs> {
34        match self {
35            fidl_tap::WlantapPhyEvent::JoinBss { ref args } => Some(args),
36            _ => None,
37        }
38    }
39}
40
41impl AsEvent<fidl_tap::StartScanArgs> for fidl_tap::WlantapPhyEvent {
42    fn as_event(&self) -> Option<&fidl_tap::StartScanArgs> {
43        match self {
44            fidl_tap::WlantapPhyEvent::StartScan { ref args } => Some(args),
45            _ => None,
46        }
47    }
48}
49
50impl AsEvent<fidl_tap::SetChannelArgs> for fidl_tap::WlantapPhyEvent {
51    fn as_event(&self) -> Option<&fidl_tap::SetChannelArgs> {
52        match self {
53            fidl_tap::WlantapPhyEvent::SetChannel { ref args } => Some(args),
54            _ => None,
55        }
56    }
57}
58
59impl AsEvent<fidl_tap::SetCountryArgs> for fidl_tap::WlantapPhyEvent {
60    fn as_event(&self) -> Option<&fidl_tap::SetCountryArgs> {
61        match self {
62            fidl_tap::WlantapPhyEvent::SetCountry { ref args } => Some(args),
63            _ => None,
64        }
65    }
66}
67
68impl AsEvent<StartMacArgs> for fidl_tap::WlantapPhyEvent {
69    fn as_event(&self) -> Option<&StartMacArgs> {
70        match self {
71            fidl_tap::WlantapPhyEvent::WlanSoftmacStart { .. } => Some(&START_MAC_ARGS),
72            _ => None,
73        }
74    }
75}
76
77impl AsEvent<fidl_tap::TxArgs> for fidl_tap::WlantapPhyEvent {
78    fn as_event(&self) -> Option<&fidl_tap::TxArgs> {
79        match self {
80            fidl_tap::WlantapPhyEvent::Tx { ref args } => Some(args),
81            _ => None,
82        }
83    }
84}
85
86/// Filters [`WlantapPhyEvent`]s to [`JoinBssArgs`].
87///
88/// The composed event handler must accept [`JoinBssArgs`] and is only invoked when such an event
89/// is received by the filter. Otherwise, the filter returns [`Handled::Unmatched`].
90///
91/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
92/// [`JoinBssArgs`]: fidl_fuchsia_wlan_tap::JoinBssArgs
93/// [`WlantapPhyEvent`]: fidl_fuchsia_wlan_tap::WlantapPhyEvent
94pub fn on_join_bss<H, S, E>(handler: H) -> impl Handler<S, E, Output = H::Output>
95where
96    H: Handler<S, fidl_tap::JoinBssArgs>,
97    E: AsEvent<fidl_tap::JoinBssArgs>,
98{
99    filter(handler)
100}
101
102/// Filters [`WlantapPhyEvent`]s to [`StartScanArgs`].
103///
104/// The composed event handler must accept [`StartScanArgs`] and is only invoked when such an event
105/// is received by the filter. Otherwise, the filter returns [`Handled::Unmatched`].
106///
107/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
108/// [`StartScanArgs`]: fidl_fuchsia_wlan_tap::StartScanArgs
109/// [`WlantapPhyEvent`]: fidl_fuchsia_wlan_tap::WlantapPhyEvent
110pub fn on_scan<H, S, E>(handler: H) -> impl Handler<S, E, Output = H::Output>
111where
112    H: Handler<S, fidl_tap::StartScanArgs>,
113    E: AsEvent<fidl_tap::StartScanArgs>,
114{
115    filter(handler)
116}
117
118/// Filters [`WlantapPhyEvent`]s to [`SetChannelArgs`].
119///
120/// The composed event handler must accept [`SetChannelArgs`] and is only invoked when such an
121/// event is received by the filter. Otherwise, the filter returns [`Handled::Unmatched`].
122///
123/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
124/// [`SetChannelArgs`]: fidl_fuchsia_wlan_tap::SetChannelArgs
125/// [`WlantapPhyEvent`]: fidl_fuchsia_wlan_tap::WlantapPhyEvent
126pub fn on_set_channel<H, S, E>(handler: H) -> impl Handler<S, E, Output = H::Output>
127where
128    H: Handler<S, fidl_tap::SetChannelArgs>,
129    E: AsEvent<fidl_tap::SetChannelArgs>,
130{
131    filter(handler)
132}
133
134/// Filters [`WlantapPhyEvent`]s to [`SetCountryArgs`].
135///
136/// The composed event handler must accept [`SetCountryArgs`] and is only invoked when such an
137/// event is received by the filter. Otherwise, the filter returns [`Handled::Unmatched`].
138///
139/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
140/// [`SetCountryArgs`]: fidl_fuchsia_wlan_tap::SetCountryArgs
141/// [`WlantapPhyEvent`]: fidl_fuchsia_wlan_tap::WlantapPhyEvent
142pub fn on_set_country<H, S, E>(handler: H) -> impl Handler<S, E, Output = H::Output>
143where
144    H: Handler<S, fidl_tap::SetCountryArgs>,
145    E: AsEvent<fidl_tap::SetCountryArgs>,
146{
147    filter(handler)
148}
149
150/// Filters [`WlantapPhyEvent`]s to [`StartMacArgs`].
151///
152/// The composed event handler must accept [`StartMacArgs`] and is only invoked when such an
153/// event is received by the filter. Otherwise, the filter returns [`Handled::Unmatched`].
154///
155/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
156/// [`StartMacArgs`]: crate::event::StartMacArgs
157/// [`WlantapPhyEvent`]: fidl_fuchsia_wlan_tap::WlantapPhyEvent
158pub fn on_start_mac<H, S, E>(handler: H) -> impl Handler<S, E, Output = H::Output>
159where
160    H: Handler<S, StartMacArgs>,
161    E: AsEvent<StartMacArgs>,
162{
163    filter(handler)
164}
165
166/// Filters [`WlantapPhyEvent`]s to [`TxArgs`].
167///
168/// The composed event handler must accept [`TxArgs`] and is only invoked when such an event is
169/// received by the filter. Otherwise, the filter returns [`Handled::Unmatched`].
170///
171/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
172/// [`TxArgs`]: fidl_fuchsia_wlan_tap::TxArgs
173/// [`WlantapPhyEvent`]: fidl_fuchsia_wlan_tap::WlantapPhyEvent
174pub fn on_transmit<H, S, E>(handler: H) -> impl Handler<S, E, Output = H::Output>
175where
176    H: Handler<S, fidl_tap::TxArgs>,
177    E: AsEvent<fidl_tap::TxArgs>,
178{
179    filter(handler)
180}
181
182/// Filters `E1` events to more specific `E2` events.
183///
184/// It must be possible to extract a reference to an `E2` from an `E1` via `AsEvent` to use this
185/// function. The composed event handler must accept `E2` events and is only invoked when the
186/// filter receives an `E1` event from which an `E2` event can be retrieved. Otherwise, the filter
187/// returns [`Handled::Unmatched`].
188///
189/// [`Handled::Unmatched`]: crate::event::Handled::Unmatched
190fn filter<H, S, E1, E2>(mut handler: H) -> impl Handler<S, E1, Output = H::Output>
191where
192    H: Handler<S, E2>,
193    E1: AsEvent<E2>,
194{
195    move |state: &mut S, event: &E1| match event.as_event() {
196        Some(event) => handler.call(state, event),
197        _ => Handled::Unmatched,
198    }
199}