1use 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
278async 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 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 txn_handle.shutdown_with_epitaph(zx::Status::OK);
296 break;
297 }
298 },
299 _ = cancellation_stream.next() => {
300 break;
303 }
304 }
305 }
306 Ok(())
307}
308
309async 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 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 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 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 #[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 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 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 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 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 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 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 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 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 fn random_scan_result(rng: &mut ThreadRng) -> wlan_common::scan::ScanResult {
726 use wlan_common::security::SecurityDescriptor;
727
728 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 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 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 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 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 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 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 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 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 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}