Skip to main content

wlan_sme/serve/
client.rs

1// Copyright 2021 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
5use crate::client::{
6    self as client_sme, ConnectResult, ConnectTransactionEvent, ConnectTransactionStream,
7    RoamResult, ScheduledScanReceiver,
8};
9use crate::{MlmeEventStream, MlmeSink, MlmeStream};
10use fidl::endpoints::{ControlHandle, RequestStream, ServerEnd};
11use fidl_fuchsia_wlan_common as fidl_common;
12use fidl_fuchsia_wlan_ieee80211 as fidl_ieee80211;
13use fidl_fuchsia_wlan_mlme as fidl_mlme;
14use fidl_fuchsia_wlan_sme::{self as fidl_sme, ClientSmeRequest, TelemetryRequest};
15use fuchsia_sync::Mutex;
16use futures::channel::mpsc;
17use futures::prelude::*;
18use futures::select;
19use ieee80211::MacAddrBytes;
20use log::error;
21use std::pin::pin;
22use std::sync::Arc;
23use wlan_common::scan::write_vmo;
24
25pub type Endpoint = ServerEnd<fidl_sme::ClientSmeMarker>;
26type Sme = client_sme::ClientSme;
27
28#[allow(clippy::too_many_arguments, reason = "mass allow for https://fxbug.dev/381896734")]
29pub fn serve(
30    cfg: crate::Config,
31    device_info: fidl_mlme::DeviceInfo,
32    security_support: fidl_common::SecuritySupport,
33    spectrum_management_support: fidl_common::SpectrumManagementSupport,
34    event_stream: MlmeEventStream,
35    new_fidl_clients: mpsc::UnboundedReceiver<Endpoint>,
36    new_telemetry_fidl_clients: mpsc::UnboundedReceiver<
37        fidl::endpoints::ServerEnd<fidl_sme::TelemetryMarker>,
38    >,
39    inspector: fuchsia_inspect::Inspector,
40    inspect_node: fuchsia_inspect::Node,
41) -> (MlmeSink, MlmeStream, impl Future<Output = Result<(), anyhow::Error>>) {
42    let wpa3_supported =
43        security_support.mfp.as_ref().is_some_and(|mfp| mfp.supported.unwrap_or(false))
44            && security_support.sae.as_ref().is_some_and(|sae| {
45                sae.driver_handler_supported.unwrap_or(false)
46                    || sae.sme_handler_supported.unwrap_or(false)
47            });
48    let owe_supported =
49        security_support.mfp.as_ref().is_some_and(|mfp| mfp.supported.unwrap_or(false))
50            && security_support.owe.as_ref().is_some_and(|owe| owe.supported.unwrap_or(false));
51    let cfg = client_sme::ClientConfig::from_config(cfg, wpa3_supported, owe_supported);
52    let (sme, mlme_sink, mlme_stream, time_stream) = Sme::new(
53        cfg,
54        device_info,
55        inspector,
56        inspect_node,
57        security_support,
58        spectrum_management_support,
59    );
60    let fut = async move {
61        let sme = Arc::new(Mutex::new(sme));
62        let mlme_sme = super::serve_mlme_sme(event_stream, Arc::clone(&sme), time_stream);
63        let sme_fidl = super::serve_fidl(&*sme, new_fidl_clients, handle_fidl_request);
64        let telemetry_fidl =
65            super::serve_fidl(&*sme, new_telemetry_fidl_clients, handle_telemetry_fidl_request);
66        let mlme_sme = pin!(mlme_sme);
67        let sme_fidl = pin!(sme_fidl);
68        select! {
69            mlme_sme = mlme_sme.fuse() => mlme_sme?,
70            sme_fidl = sme_fidl.fuse() => match sme_fidl? {},
71            telemetry_fidl = telemetry_fidl.fuse() => match telemetry_fidl? {},
72        }
73        Ok(())
74    };
75    (mlme_sink, mlme_stream, fut)
76}
77
78async fn handle_fidl_request(
79    sme: &Mutex<Sme>,
80    request: fidl_sme::ClientSmeRequest,
81) -> Result<(), fidl::Error> {
82    #[allow(clippy::unit_arg, reason = "mass allow for https://fxbug.dev/381896734")]
83    match request {
84        ClientSmeRequest::Scan { req, responder } => Ok(scan(sme, req, |result| match result {
85            Ok(scan_results) => responder.send(Ok(write_vmo(scan_results)?)).map_err(|e| e.into()),
86            Err(e) => responder.send(Err(e)).map_err(|e| e.into()),
87        })
88        .await
89        .unwrap_or_else(|e| error!("Error handling a scan transaction: {:?}", e))),
90        ClientSmeRequest::Connect { req, txn, .. } => Ok(connect(sme, txn, req)
91            .await
92            .unwrap_or_else(|e| error!("Error handling a connect transaction: {:?}", e))),
93        ClientSmeRequest::Roam { req, .. } => Ok(roam(sme, req)),
94        ClientSmeRequest::Disconnect { responder, reason } => {
95            disconnect(sme, reason, responder);
96            Ok(())
97        }
98        ClientSmeRequest::Status { responder } => responder.send(&status(sme)),
99        ClientSmeRequest::WmmStatus { responder } => wmm_status(sme, responder).await,
100        ClientSmeRequest::StartScheduledScan { req, txn, responder } => {
101            start_scheduled_scan(sme, req, txn, responder).await
102        }
103
104        ClientSmeRequest::GetScheduledScanEnabled { responder } => {
105            let receiver = sme.lock().on_get_scheduled_scan_enabled_command();
106            let resp = match receiver.await {
107                Ok(result) => result,
108                Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
109            };
110            responder
111                .send(resp.as_ref().map(|r| !r.active_txn_ids.is_empty()).map_err(|e| *e))
112                .unwrap_or_else(|e| error!("Error sending response: {:?}", e));
113            Ok(())
114        }
115        ClientSmeRequest::ScanForController { req, responder } => {
116            Ok(scan(sme, req, |result| match result {
117                Ok(results) => responder.send(Ok(&results[..])).map_err(|e| e.into()),
118                Err(e) => responder.send(Err(e)).map_err(|e| e.into()),
119            })
120            .await
121            .unwrap_or_else(|e| error!("Error handling a test scan transaction: {:?}", e)))
122        }
123        ClientSmeRequest::SetMacAddress { mac_addr, responder } => {
124            Ok(set_mac_address(sme, mac_addr, responder).await?)
125        }
126        ClientSmeRequest::InstallApfPacketFilter { program, responder } => {
127            let receiver = sme.lock().install_apf_packet_filter(program);
128            let resp = match receiver.await {
129                Ok(result) => result,
130                Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
131            };
132            responder.send(resp).unwrap_or_else(|e| error!("Error sending response: {:?}", e));
133            Ok(())
134        }
135        ClientSmeRequest::ReadApfPacketFilterData { responder } => {
136            let receiver = sme.lock().read_apf_packet_filter_data();
137            let resp = match receiver.await {
138                Ok(result) => result.map(|r| r.memory),
139                Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
140            };
141            responder
142                .send(resp.as_ref().map(|m| &m[..]).map_err(|e| *e))
143                .unwrap_or_else(|e| error!("Error sending response: {:?}", e));
144            Ok(())
145        }
146        ClientSmeRequest::SetApfPacketFilterEnabled { enabled, responder } => {
147            let receiver = sme.lock().set_apf_packet_filter_enabled(enabled);
148            let resp = match receiver.await {
149                Ok(result) => result,
150                Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
151            };
152            responder.send(resp).unwrap_or_else(|e| error!("Error sending response: {:?}", e));
153            Ok(())
154        }
155        ClientSmeRequest::GetApfPacketFilterEnabled { responder } => {
156            let receiver = sme.lock().get_apf_packet_filter_enabled();
157            let resp = match receiver.await {
158                Ok(result) => result.map(|r| r.enabled),
159                Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
160            };
161            responder
162                .send(resp.as_ref().map(|e| *e).map_err(|e| *e))
163                .unwrap_or_else(|e| error!("Error sending response: {:?}", e));
164            Ok(())
165        }
166    }
167}
168
169async fn handle_telemetry_fidl_request(
170    sme: &Mutex<Sme>,
171    request: TelemetryRequest,
172) -> Result<(), fidl::Error> {
173    match request {
174        TelemetryRequest::QueryTelemetrySupport { responder, .. } => {
175            let support_fut = sme.lock().query_telemetry_support();
176            let support = support_fut
177                .await
178                .map_err(|_| zx::Status::CONNECTION_ABORTED.into_raw())
179                .and_then(|result| result);
180            responder.send(support.as_ref().map_err(|e| *e))
181        }
182        TelemetryRequest::GetIfaceStats { responder, .. } => {
183            let iface_stats_fut = sme.lock().iface_stats();
184            let iface_stats = iface_stats_fut
185                .await
186                .map_err(|_| zx::Status::CONNECTION_ABORTED.into_raw())
187                .and_then(|stats| match stats {
188                    fidl_mlme::GetIfaceStatsResponse::Stats(stats) => Ok(stats),
189                    fidl_mlme::GetIfaceStatsResponse::ErrorStatus(err) => Err(err),
190                });
191            responder.send(iface_stats.as_ref().map_err(|e| *e))
192        }
193        TelemetryRequest::GetHistogramStats { responder, .. } => {
194            let histogram_stats_fut = sme.lock().histogram_stats();
195            let histogram_stats = histogram_stats_fut
196                .await
197                .map_err(|_| zx::Status::CONNECTION_ABORTED.into_raw())
198                .and_then(|stats| match stats {
199                    fidl_mlme::GetIfaceHistogramStatsResponse::Stats(stats) => Ok(stats),
200                    fidl_mlme::GetIfaceHistogramStatsResponse::ErrorStatus(err) => Err(err),
201                });
202            responder.send(histogram_stats.as_ref().map_err(|e| *e))
203        }
204        TelemetryRequest::GetSignalReport { responder, .. } => {
205            let signal_report_fut = sme.lock().signal_report();
206            let signal_report = signal_report_fut
207                .await
208                .map_err(|_| zx::Status::CONNECTION_ABORTED.into_raw())
209                .and_then(|result| result);
210            responder.send(signal_report.as_ref().map_err(|e| *e))
211        }
212        TelemetryRequest::CloneInspectVmo { responder } => {
213            let inspect_vmo =
214                sme.lock().on_clone_inspect_vmo().ok_or_else(|| zx::Status::INTERNAL.into_raw());
215            responder.send(inspect_vmo)
216        }
217    }
218}
219
220async fn scan(
221    sme: &Mutex<Sme>,
222    request: fidl_sme::ScanRequest,
223    responder: impl FnOnce(
224        Result<Vec<fidl_sme::ScanResult>, fidl_sme::ScanErrorCode>,
225    ) -> Result<(), anyhow::Error>,
226) -> Result<(), anyhow::Error> {
227    let receiver = sme.lock().on_scan_command(request);
228    let receive_result = match receiver.await {
229        Ok(receive_result) => receive_result,
230        Err(e) => {
231            error!("Scan receiver error: {:?}", e);
232            responder(Err(fidl_sme::ScanErrorCode::InternalError))?;
233            return Ok(());
234        }
235    };
236
237    match receive_result {
238        Ok(scan_results) => {
239            let results = scan_results.into_iter().map(Into::into).collect::<Vec<_>>();
240            responder(Ok(results))
241        }
242        Err(mlme_scan_result_code) => {
243            let scan_error_code = match mlme_scan_result_code {
244                fidl_mlme::ScanResultCode::Success | fidl_mlme::ScanResultCode::InvalidArgs => {
245                    error!("Internal scan error: {:?}", mlme_scan_result_code);
246                    fidl_sme::ScanErrorCode::InternalError
247                }
248                fidl_mlme::ScanResultCode::NotSupported => fidl_sme::ScanErrorCode::NotSupported,
249                fidl_mlme::ScanResultCode::InternalError => {
250                    fidl_sme::ScanErrorCode::InternalMlmeError
251                }
252                fidl_mlme::ScanResultCode::ShouldWait => fidl_sme::ScanErrorCode::ShouldWait,
253                fidl_mlme::ScanResultCode::CanceledByDriverOrFirmware => {
254                    fidl_sme::ScanErrorCode::CanceledByDriverOrFirmware
255                }
256            };
257            responder(Err(scan_error_code))
258        }
259    }?;
260    Ok(())
261}
262
263async fn connect(
264    sme: &Mutex<Sme>,
265    txn: Option<ServerEnd<fidl_sme::ConnectTransactionMarker>>,
266    req: fidl_sme::ConnectRequest,
267) -> Result<(), anyhow::Error> {
268    #[allow(clippy::manual_map, reason = "mass allow for https://fxbug.dev/381896734")]
269    let handle = match txn {
270        None => None,
271        Some(txn) => Some(txn.into_stream().control_handle()),
272    };
273    let connect_txn_stream = sme.lock().on_connect_command(req);
274    serve_connect_txn_stream(handle, connect_txn_stream).await?;
275    Ok(())
276}
277
278/// Serves a scheduled scan session by processing a stream for events sent by MLME and processed in
279/// SME's scan scheduler, and a stream for client-initiated cancellations.
280async fn serve_sched_scan_session(
281    txn_handle: fidl_sme::ScheduledScanTransactionControlHandle,
282    txn_stream: fidl_sme::ScheduledScanTransactionRequestStream,
283    session: ScheduledScanReceiver,
284) -> Result<(), anyhow::Error> {
285    // Client will close the transaction channel to cancel session.
286    let mut cancellation_stream = txn_stream.fuse();
287    let mut session_stream = session.fuse();
288    loop {
289        futures::select! {
290            scan_results = session_stream.next() => {
291                if let Some(vmo) = scan_results {
292                    txn_handle.send_on_scheduled_scan_matches_available(vmo)?;
293                } else {
294                    // Stream closed naturally (SME dropped sender because firmware stopped it).
295                    txn_handle.shutdown_with_epitaph(zx::Status::OK);
296                    break;
297                }
298            },
299            _ = cancellation_stream.next() => {
300                // Client dropped transaction stream. Cleanup is handled by ScheduledScanReceiver's
301                // Drop implementation.
302                break;
303            }
304        }
305    }
306    Ok(())
307}
308
309/// Creates a new scheduled scan session and starts serving it.
310async fn start_scheduled_scan(
311    sme: &Mutex<Sme>,
312    req: fidl_common::ScheduledScanRequest,
313    txn_server: ServerEnd<fidl_sme::ScheduledScanTransactionMarker>,
314    responder: fidl_sme::ClientSmeStartScheduledScanResponder,
315) -> Result<(), fidl::Error> {
316    let txn_receiver_stream = txn_server.into_stream();
317    let txn_sender_handle = txn_receiver_stream.control_handle();
318    let (receiver, session) = sme.lock().on_start_scheduled_scan_command(req);
319    match receiver.await.unwrap_or_else(|_| Err(zx::Status::CANCELED.into_raw())) {
320        Ok(()) => {
321            responder.send(Ok(()))?;
322            serve_sched_scan_session(txn_sender_handle, txn_receiver_stream, session)
323                .await
324                .unwrap_or_else(|e| {
325                    error!("Error serving sched scan txn stream: {:?}", e);
326                });
327        }
328        Err(status) => {
329            responder.send(Err(status))?;
330            txn_sender_handle.shutdown_with_epitaph(zx::Status::from_raw(status));
331        }
332    }
333    Ok(())
334}
335
336async fn serve_connect_txn_stream(
337    handle: Option<fidl_sme::ConnectTransactionControlHandle>,
338    mut connect_txn_stream: ConnectTransactionStream,
339) -> Result<(), anyhow::Error> {
340    if let Some(handle) = handle {
341        loop {
342            match connect_txn_stream.next().await {
343                Some(event) => match event {
344                    ConnectTransactionEvent::OnConnectResult { result, is_reconnect } => {
345                        let connect_result = convert_connect_result(&result, is_reconnect);
346                        handle.send_on_connect_result(&connect_result)
347                    }
348                    ConnectTransactionEvent::OnRoamResult { result } => {
349                        let roam_result = convert_roam_result(&result);
350                        handle.send_on_roam_result(&roam_result)
351                    }
352                    ConnectTransactionEvent::OnDisconnect { info } => {
353                        handle.send_on_disconnect(&info)
354                    }
355                    ConnectTransactionEvent::OnSignalReport { ind } => {
356                        handle.send_on_signal_report(&ind)
357                    }
358                    ConnectTransactionEvent::OnChannelSwitched { info } => {
359                        handle.send_on_channel_switched(&info)
360                    }
361                }?,
362                // SME has dropped the ConnectTransaction endpoint, likely due to a disconnect.
363                None => return Ok(()),
364            }
365        }
366    }
367    Ok(())
368}
369
370fn roam(sme: &Mutex<Sme>, req: fidl_sme::RoamRequest) {
371    sme.lock().on_roam_command(req);
372}
373
374fn disconnect(
375    sme: &Mutex<Sme>,
376    policy_disconnect_reason: fidl_sme::UserDisconnectReason,
377    responder: fidl_sme::ClientSmeDisconnectResponder,
378) {
379    sme.lock().on_disconnect_command(policy_disconnect_reason, responder);
380}
381
382fn status(sme: &Mutex<Sme>) -> fidl_sme::ClientStatusResponse {
383    sme.lock().status().into()
384}
385
386async fn wmm_status(
387    sme: &Mutex<Sme>,
388    responder: fidl_sme::ClientSmeWmmStatusResponder,
389) -> Result<(), fidl::Error> {
390    let receiver = sme.lock().wmm_status();
391    let wmm_status = match receiver.await {
392        Ok(result) => result,
393        Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
394    };
395    responder.send(wmm_status.as_ref().map_err(|e| *e))
396}
397
398async fn set_mac_address(
399    sme: &Mutex<Sme>,
400    mac_addr: [u8; 6],
401    responder: fidl_sme::ClientSmeSetMacAddressResponder,
402) -> Result<(), fidl::Error> {
403    let receiver = sme.lock().set_mac_address(mac_addr);
404    let resp = match receiver.await {
405        Ok(result) => result,
406        Err(_) => Err(zx::sys::ZX_ERR_CANCELED),
407    };
408    responder.send(resp)
409}
410
411fn convert_connect_result(result: &ConnectResult, is_reconnect: bool) -> fidl_sme::ConnectResult {
412    let (code, is_credential_rejected) = match result {
413        ConnectResult::Success => (fidl_ieee80211::StatusCode::Success, false),
414        ConnectResult::Canceled => (fidl_ieee80211::StatusCode::Canceled, false),
415        ConnectResult::Failed(failure) => {
416            (failure.status_code(), failure.likely_due_to_credential_rejected())
417        }
418    };
419    fidl_sme::ConnectResult { code, is_credential_rejected, is_reconnect }
420}
421
422fn convert_roam_result(result: &RoamResult) -> fidl_sme::RoamResult {
423    match result {
424        RoamResult::Success(bss) => {
425            let bss_description =
426                Some(Box::new(fidl_ieee80211::BssDescription::from(*bss.clone())));
427            fidl_sme::RoamResult {
428                bssid: bss.bssid.to_array(),
429                status_code: fidl_ieee80211::StatusCode::Success,
430                // Must always be false on roam success.
431                original_association_maintained: false,
432                bss_description,
433                disconnect_info: None,
434                is_credential_rejected: false,
435            }
436        }
437        RoamResult::Failed(failure) => {
438            #[allow(clippy::manual_map, reason = "mass allow for https://fxbug.dev/381896734")]
439            fidl_sme::RoamResult {
440                bssid: failure.selected_bssid.to_array(),
441                status_code: failure.status_code,
442                // Current implementation assumes that all roam attempts incur disassociation from the
443                // original BSS. When this changes (e.g. due to Fast BSS Transition support), this
444                // hard-coded field should be set from the RoamResult enum.
445                original_association_maintained: false,
446                bss_description: match &failure.selected_bss {
447                    Some(bss) => Some(Box::new(bss.clone().into())),
448                    None => None,
449                },
450                disconnect_info: Some(Box::new(failure.disconnect_info)),
451                is_credential_rejected: failure.likely_due_to_credential_rejected(),
452            }
453        }
454    }
455}
456
457#[cfg(test)]
458mod tests {
459    use super::*;
460    use crate::client::{ConnectFailure, EstablishRsnaFailure, EstablishRsnaFailureReason};
461    use crate::test_utils;
462    use assert_matches::assert_matches;
463    use fidl::endpoints::create_proxy_and_stream;
464    use fidl_fuchsia_wlan_internal as fidl_internal;
465    use fidl_fuchsia_wlan_mlme::ScanResultCode;
466    use fidl_fuchsia_wlan_sme::{self as fidl_sme};
467    use fuchsia_async as fasync;
468    use futures::stream::StreamFuture;
469    use futures::task::Poll;
470    use rand::Rng;
471    use rand::prelude::ThreadRng;
472    use std::pin::pin;
473    use test_case::test_case;
474    use wlan_common::random_bss_description;
475    use wlan_common::scan::{self, Incompatible};
476    use wlan_rsn::auth;
477
478    #[test]
479    fn test_convert_connect_result() {
480        assert_eq!(
481            convert_connect_result(&ConnectResult::Success, false),
482            fidl_sme::ConnectResult {
483                code: fidl_ieee80211::StatusCode::Success,
484                is_credential_rejected: false,
485                is_reconnect: false,
486            }
487        );
488        assert_eq!(
489            convert_connect_result(&ConnectResult::Canceled, true),
490            fidl_sme::ConnectResult {
491                code: fidl_ieee80211::StatusCode::Canceled,
492                is_credential_rejected: false,
493                is_reconnect: true,
494            }
495        );
496        let connect_result =
497            ConnectResult::Failed(ConnectFailure::ScanFailure(ScanResultCode::ShouldWait));
498        assert_eq!(
499            convert_connect_result(&connect_result, false),
500            fidl_sme::ConnectResult {
501                code: fidl_ieee80211::StatusCode::Canceled,
502                is_credential_rejected: false,
503                is_reconnect: false,
504            }
505        );
506
507        let connect_result =
508            ConnectResult::Failed(ConnectFailure::EstablishRsnaFailure(EstablishRsnaFailure {
509                auth_method: Some(auth::MethodName::Psk),
510                reason: EstablishRsnaFailureReason::InternalError,
511            }));
512        assert_eq!(
513            convert_connect_result(&connect_result, false),
514            fidl_sme::ConnectResult {
515                code: fidl_ieee80211::StatusCode::EstablishRsnaFailure,
516                is_credential_rejected: false,
517                is_reconnect: false,
518            }
519        );
520
521        let connect_result =
522            ConnectResult::Failed(ConnectFailure::EstablishRsnaFailure(EstablishRsnaFailure {
523                auth_method: Some(auth::MethodName::Psk),
524                reason: EstablishRsnaFailureReason::RsnaResponseTimeout(
525                    wlan_rsn::Error::LikelyWrongCredential,
526                ),
527            }));
528        assert_eq!(
529            convert_connect_result(&connect_result, false),
530            fidl_sme::ConnectResult {
531                code: fidl_ieee80211::StatusCode::EstablishRsnaFailure,
532                is_credential_rejected: true,
533                is_reconnect: false,
534            }
535        );
536
537        let connect_result =
538            ConnectResult::Failed(ConnectFailure::EstablishRsnaFailure(EstablishRsnaFailure {
539                auth_method: Some(auth::MethodName::Psk),
540                reason: EstablishRsnaFailureReason::RsnaCompletionTimeout(
541                    wlan_rsn::Error::LikelyWrongCredential,
542                ),
543            }));
544        assert_eq!(
545            convert_connect_result(&connect_result, false),
546            fidl_sme::ConnectResult {
547                code: fidl_ieee80211::StatusCode::EstablishRsnaFailure,
548                is_credential_rejected: true,
549                is_reconnect: false,
550            }
551        );
552
553        let connect_result =
554            ConnectResult::Failed(ConnectFailure::EstablishRsnaFailure(EstablishRsnaFailure {
555                auth_method: Some(auth::MethodName::Psk),
556                reason: EstablishRsnaFailureReason::RsnaCompletionTimeout(
557                    wlan_rsn::Error::MissingGtkProvider,
558                ),
559            }));
560        assert_eq!(
561            convert_connect_result(&connect_result, false),
562            fidl_sme::ConnectResult {
563                code: fidl_ieee80211::StatusCode::EstablishRsnaFailure,
564                is_credential_rejected: false,
565                is_reconnect: false,
566            }
567        );
568
569        let connect_result =
570            ConnectResult::Failed(ConnectFailure::ScanFailure(ScanResultCode::InternalError));
571        assert_eq!(
572            convert_connect_result(&connect_result, false),
573            fidl_sme::ConnectResult {
574                code: fidl_ieee80211::StatusCode::RefusedReasonUnspecified,
575                is_credential_rejected: false,
576                is_reconnect: false,
577            }
578        );
579    }
580
581    // TODO(https://fxbug.dev/42164611): There is no test coverage for consistency between MLME scan results
582    // and SME scan results produced by wlanstack. In particular, the timestamp_nanos field
583    // of fidl_mlme::ScanResult is dropped in SME, and no tests reveal this problem.
584
585    #[test_case(1, true; "with 1 result")]
586    #[test_case(2, true; "with 2 results")]
587    #[test_case(30, true; "with 30 results")]
588    #[test_case(4000, true; "with 4000 results")]
589    #[test_case(50000, false; "with 50000 results")]
590    #[test_case(100000, false; "with 100000 results")]
591    fn scan_results_are_effectively_unbounded(number_of_scan_results: usize, randomize: bool) {
592        let mut exec = fasync::TestExecutor::new();
593        let (client_sme_proxy, mut client_sme_stream) =
594            create_proxy_and_stream::<fidl_sme::ClientSmeMarker>();
595
596        // Request scan
597        async fn request_and_collect_result(
598            client_sme_proxy: &fidl_sme::ClientSmeProxy,
599        ) -> fidl_sme::ClientSmeScanResult {
600            client_sme_proxy
601                .scan(&fidl_sme::ScanRequest::Passive(fidl_sme::PassiveScanRequest {
602                    channels: vec![],
603                }))
604                .await
605                .expect("FIDL request failed")
606        }
607
608        let result_fut = request_and_collect_result(&client_sme_proxy);
609        let mut result_fut = pin!(result_fut);
610
611        assert_matches!(exec.run_until_stalled(&mut result_fut), Poll::Pending);
612
613        // Generate and send scan results
614        let mut rng = rand::rng();
615        let scan_result_list = if randomize {
616            (0..number_of_scan_results).map(|_| random_scan_result(&mut rng).into()).collect()
617        } else {
618            vec![random_scan_result(&mut rng).into(); number_of_scan_results]
619        };
620        assert_matches!(exec.run_until_stalled(&mut client_sme_stream.next()),
621                        Poll::Ready(Some(Ok(fidl_sme::ClientSmeRequest::Scan {
622                            req: _, responder,
623                        }))) => {
624                            let vmo = write_vmo(scan_result_list.clone()).expect("failed to write VMO");
625                            responder.send(Ok(vmo)).expect("failed to send scan results");
626                        }
627        );
628
629        // Verify scan results
630        assert_matches!(exec.run_until_stalled(&mut result_fut), Poll::Ready(Ok(vmo)) => {
631            assert_eq!(scan_result_list, scan::read_vmo(vmo).expect("failed to read VMO"));
632        })
633    }
634
635    #[test]
636    fn test_serve_connect_txn_stream() {
637        let mut exec = fasync::TestExecutor::new();
638
639        let (sme_proxy, sme_connect_txn_stream) = mpsc::unbounded();
640        let (fidl_client_proxy, fidl_connect_txn_stream) =
641            create_proxy_and_stream::<fidl_sme::ConnectTransactionMarker>();
642        let fidl_client_fut = fidl_client_proxy.take_event_stream().into_future();
643        let mut fidl_client_fut = pin!(fidl_client_fut);
644        let fidl_connect_txn_handle = fidl_connect_txn_stream.control_handle();
645
646        let test_fut =
647            serve_connect_txn_stream(Some(fidl_connect_txn_handle), sme_connect_txn_stream);
648        let mut test_fut = pin!(test_fut);
649
650        // Test sending OnConnectResult
651        sme_proxy
652            .unbounded_send(ConnectTransactionEvent::OnConnectResult {
653                result: ConnectResult::Success,
654                is_reconnect: true,
655            })
656            .expect("expect sending ConnectTransactionEvent to succeed");
657        assert_matches!(exec.run_until_stalled(&mut test_fut), Poll::Pending);
658        let event = assert_matches!(poll_stream_fut(&mut exec, &mut fidl_client_fut), Poll::Ready(Some(Ok(event))) => event);
659        assert_matches!(
660            event,
661            fidl_sme::ConnectTransactionEvent::OnConnectResult {
662                result: fidl_sme::ConnectResult {
663                    code: fidl_ieee80211::StatusCode::Success,
664                    is_credential_rejected: false,
665                    is_reconnect: true,
666                }
667            }
668        );
669
670        // Test sending OnDisconnect
671        let input_info = fidl_sme::DisconnectInfo {
672            is_sme_reconnecting: true,
673            disconnect_source: fidl_sme::DisconnectSource::Mlme(fidl_sme::DisconnectCause {
674                reason_code: fidl_ieee80211::ReasonCode::UnspecifiedReason,
675                mlme_event_name: fidl_sme::DisconnectMlmeEventName::DeauthenticateIndication,
676            }),
677        };
678        sme_proxy
679            .unbounded_send(ConnectTransactionEvent::OnDisconnect { info: input_info })
680            .expect("expect sending ConnectTransactionEvent to succeed");
681        assert_matches!(exec.run_until_stalled(&mut test_fut), Poll::Pending);
682        let event = assert_matches!(poll_stream_fut(&mut exec, &mut fidl_client_fut), Poll::Ready(Some(Ok(event))) => event);
683        assert_matches!(event, fidl_sme::ConnectTransactionEvent::OnDisconnect { info: output_info } => {
684            assert_eq!(input_info, output_info);
685        });
686
687        // Test sending OnSignalReport
688        let input_ind = fidl_internal::SignalReportIndication { rssi_dbm: -40, snr_db: 30 };
689        sme_proxy
690            .unbounded_send(ConnectTransactionEvent::OnSignalReport { ind: input_ind })
691            .expect("expect sending ConnectTransactionEvent to succeed");
692        assert_matches!(exec.run_until_stalled(&mut test_fut), Poll::Pending);
693        let event = assert_matches!(poll_stream_fut(&mut exec, &mut fidl_client_fut), Poll::Ready(Some(Ok(event))) => event);
694        assert_matches!(event, fidl_sme::ConnectTransactionEvent::OnSignalReport { ind } => {
695            assert_eq!(input_ind, ind);
696        });
697
698        // Test sending OnChannelSwitched
699        let input_info = fidl_internal::ChannelSwitchInfo { new_channel: 8 };
700        sme_proxy
701            .unbounded_send(ConnectTransactionEvent::OnChannelSwitched { info: input_info })
702            .expect("expect sending ConnectTransactionEvent to succeed");
703        assert_matches!(exec.run_until_stalled(&mut test_fut), Poll::Pending);
704        let event = assert_matches!(poll_stream_fut(&mut exec, &mut fidl_client_fut), Poll::Ready(Some(Ok(event))) => event);
705        assert_matches!(event, fidl_sme::ConnectTransactionEvent::OnChannelSwitched { info } => {
706            assert_eq!(input_info, info);
707        });
708
709        // When SME proxy is dropped, the fut should terminate
710        std::mem::drop(sme_proxy);
711        assert_matches!(exec.run_until_stalled(&mut test_fut), Poll::Ready(Ok(())));
712    }
713
714    fn poll_stream_fut<S: Stream + std::marker::Unpin>(
715        exec: &mut fasync::TestExecutor,
716        stream_fut: &mut StreamFuture<S>,
717    ) -> Poll<Option<S::Item>> {
718        exec.run_until_stalled(stream_fut).map(|(item, stream)| {
719            *stream_fut = stream.into_future();
720            item
721        })
722    }
723
724    // Create roughly over 2k bytes ScanResult
725    fn random_scan_result(rng: &mut ThreadRng) -> wlan_common::scan::ScanResult {
726        use wlan_common::security::SecurityDescriptor;
727
728        // TODO(https://fxbug.dev/42164451): Merge this with a similar function in wlancfg.
729        wlan_common::scan::ScanResult {
730            compatibility: match rng.random_range(0..4) {
731                0 => wlan_common::scan::Compatible::expect_ok([SecurityDescriptor::OPEN]),
732                1 => wlan_common::scan::Compatible::expect_ok([SecurityDescriptor::WPA2_PERSONAL]),
733                2 => wlan_common::scan::Compatible::expect_ok([
734                    SecurityDescriptor::WPA2_PERSONAL,
735                    SecurityDescriptor::WPA3_PERSONAL,
736                ]),
737                _ => Incompatible::unknown(),
738            },
739            timestamp: zx::MonotonicInstant::from_nanos(rng.random()),
740            bss_description: random_bss_description!(),
741        }
742    }
743
744    #[test]
745    fn test_handle_fidl_request_apf() {
746        let mut exec = fasync::TestExecutor::new();
747        let inspector = fuchsia_inspect::Inspector::default();
748        let (sme, _mlme_sink, mut mlme_stream, _time_stream) = client_sme::ClientSme::new(
749            client_sme::ClientConfig::default(),
750            test_utils::fake_device_info([0; 6].into()),
751            inspector.clone(),
752            inspector.root().create_child("sme"),
753            wlan_common::test_utils::fake_features::fake_security_support(),
754            wlan_common::test_utils::fake_features::fake_spectrum_management_support_empty(),
755        );
756        let sme = Mutex::new(sme);
757
758        // Test InstallApfPacketFilter
759        let (proxy, stream) = create_proxy_and_stream::<fidl_sme::ClientSmeMarker>();
760        let program = vec![1, 2, 3];
761        let mut install_fut = proxy.install_apf_packet_filter(&program);
762        let mut stream = pin!(stream);
763
764        // Run handle_fidl_request for the install request
765        assert_matches!(exec.run_until_stalled(&mut stream.next()), Poll::Ready(Some(Ok(req))) => {
766            let mut handle_fut = pin!(handle_fidl_request(&sme, req));
767            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Pending);
768
769            // Forward from SME to MLME
770            assert_matches!(exec.run_until_stalled(&mut mlme_stream.next()), Poll::Ready(Some(crate::MlmeRequest::InstallApfPacketFilter(req, responder))) => {
771                assert_eq!(req.program, program);
772                responder.respond(Ok(()));
773            });
774
775            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Ready(Ok(())));
776        });
777        assert_matches!(exec.run_until_stalled(&mut install_fut), Poll::Ready(Ok(Ok(()))));
778
779        // Test ReadApfPacketFilterData
780        let mut read_fut = proxy.read_apf_packet_filter_data();
781        assert_matches!(exec.run_until_stalled(&mut stream.next()), Poll::Ready(Some(Ok(req))) => {
782            let mut handle_fut = pin!(handle_fidl_request(&sme, req));
783            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Pending);
784
785            // Forward from SME to MLME
786            assert_matches!(exec.run_until_stalled(&mut mlme_stream.next()), Poll::Ready(Some(crate::MlmeRequest::ReadApfPacketFilterData(responder))) => {
787                responder.respond(Ok(fidl_mlme::MlmeReadApfPacketFilterDataResponse {
788                    memory: vec![4, 5, 6],
789                }));
790            });
791
792            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Ready(Ok(())));
793        });
794        assert_matches!(exec.run_until_stalled(&mut read_fut), Poll::Ready(Ok(Ok(data))) => {
795            assert_eq!(data, vec![4, 5, 6]);
796        });
797
798        // Test SetApfPacketFilterEnabled
799        let mut set_enabled_fut = proxy.set_apf_packet_filter_enabled(true);
800        assert_matches!(exec.run_until_stalled(&mut stream.next()), Poll::Ready(Some(Ok(req))) => {
801            let mut handle_fut = pin!(handle_fidl_request(&sme, req));
802            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Pending);
803
804            // Forward from SME to MLME
805            assert_matches!(exec.run_until_stalled(&mut mlme_stream.next()), Poll::Ready(Some(crate::MlmeRequest::SetApfPacketFilterEnabled(req, responder))) => {
806                assert!(req.enabled);
807                responder.respond(Ok(()));
808            });
809
810            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Ready(Ok(())));
811        });
812        assert_matches!(exec.run_until_stalled(&mut set_enabled_fut), Poll::Ready(Ok(Ok(()))));
813
814        // Test GetApfPacketFilterEnabled
815        let mut get_enabled_fut = proxy.get_apf_packet_filter_enabled();
816        assert_matches!(exec.run_until_stalled(&mut stream.next()), Poll::Ready(Some(Ok(req))) => {
817            let mut handle_fut = pin!(handle_fidl_request(&sme, req));
818            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Pending);
819
820            // Forward from SME to MLME
821            assert_matches!(exec.run_until_stalled(&mut mlme_stream.next()), Poll::Ready(Some(crate::MlmeRequest::GetApfPacketFilterEnabled(responder))) => {
822                responder.respond(Ok(fidl_mlme::MlmeGetApfPacketFilterEnabledResponse {
823                    enabled: true,
824                }));
825            });
826
827            assert_matches!(exec.run_until_stalled(&mut handle_fut), Poll::Ready(Ok(())));
828        });
829        assert_matches!(exec.run_until_stalled(&mut get_enabled_fut), Poll::Ready(Ok(Ok(true))));
830    }
831}