1use crate::wlan::types;
6use anyhow::{Context as _, Error};
7use fidl_fuchsia_wlan_common as fidl_common;
8use fidl_fuchsia_wlan_device_service::{DeviceMonitorMarker, DeviceMonitorProxy};
9use fuchsia_component::client::connect_to_protocol;
10use fuchsia_sync::RwLock;
11use ieee80211::{MacAddr, Ssid};
12use std::collections::HashMap;
13use wlan_common::scan::ScanResult;
14
15#[derive(Debug)]
23struct InnerWlanFacade {
24 #[allow(unused)]
26 scan_results: bool,
27}
28
29#[derive(Debug)]
30pub(crate) struct WlanFacade {
31 monitor_svc: DeviceMonitorProxy,
32 #[allow(unused)]
34 inner: RwLock<InnerWlanFacade>,
35}
36
37impl WlanFacade {
38 pub fn new() -> Result<WlanFacade, Error> {
39 let monitor_svc = connect_to_protocol::<DeviceMonitorMarker>()?;
40
41 Ok(WlanFacade { monitor_svc, inner: RwLock::new(InnerWlanFacade { scan_results: false }) })
42 }
43
44 pub async fn get_iface_id_list(&self) -> Result<Vec<u16>, Error> {
46 let wlan_iface_ids = wlan_service_util::get_iface_list(&self.monitor_svc)
47 .await
48 .context("Get Iface Id List: failed to get wlan iface list")?;
49 Ok(wlan_iface_ids)
50 }
51
52 pub async fn get_phy_id_list(&self) -> Result<Vec<u16>, Error> {
54 let wlan_phy_ids = wlan_service_util::get_phy_list(&self.monitor_svc)
55 .await
56 .context("Get Phy Id List: failed to get wlan phy list")?;
57 Ok(wlan_phy_ids)
58 }
59
60 pub async fn scan(&self) -> Result<Vec<String>, Error> {
61 let sme_proxy = wlan_service_util::client::get_first_sme(&self.monitor_svc)
63 .await
64 .context("Scan: failed to get client iface sme proxy")?;
65
66 let scan_result_list =
68 wlan_service_util::client::passive_scan(&sme_proxy).await.context("Scan failed")?;
69
70 let mut ssid_list = Vec::new();
72 for scan_result in &scan_result_list {
73 let scan_result: ScanResult = scan_result.clone().try_into()?;
74 let ssid = String::from_utf8_lossy(&scan_result.bss_description.ssid).into_owned();
75 ssid_list.push(ssid);
76 }
77 Ok(ssid_list)
78 }
79
80 async fn passive_scan(
81 &self,
82 ) -> Result<impl IntoIterator<Item = Result<ScanResult, Error>>, Error> {
83 let sme_proxy = wlan_service_util::client::get_first_sme(&self.monitor_svc)
85 .await
86 .context("Scan: failed to get client iface sme proxy")?;
87 Ok(wlan_service_util::client::passive_scan(&sme_proxy)
89 .await
90 .context("Scan failed")?
91 .into_iter()
92 .map(ScanResult::try_from))
93 }
94
95 pub async fn scan_for_bss_info(
96 &self,
97 ) -> Result<HashMap<String, Vec<Box<types::BssDescriptionDef>>>, Error> {
98 let mut scan_results_by_ssid_string = HashMap::new();
99 for scan_result in self.passive_scan().await? {
100 let scan_result = scan_result.context("Failed to convert scan result")?;
101 let entry = scan_results_by_ssid_string
102 .entry(String::from(scan_result.bss_description.ssid.to_string_not_redactable()))
103 .or_insert(vec![]);
104
105 let fidl_bss_desc: fidl_common::BssDescription = scan_result.bss_description.into();
106 entry.push(Box::new(fidl_bss_desc.into()));
107 }
108 Ok(scan_results_by_ssid_string)
109 }
110
111 pub async fn connect(
112 &self,
113 target_ssid: Ssid,
114 target_pwd: Vec<u8>,
115 target_bss_desc: fidl_common::BssDescription,
116 ) -> Result<bool, Error> {
117 let sme_proxy = wlan_service_util::client::get_first_sme(&self.monitor_svc)
118 .await
119 .context("Connect: failed to get client iface sme proxy")?;
120 wlan_service_util::client::connect(&sme_proxy, target_ssid, target_pwd, target_bss_desc)
121 .await
122 }
123
124 pub async fn create_iface(
125 &self,
126 phy_id: u16,
127 role: fidl_common::WlanMacRole,
128 sta_addr: MacAddr,
129 ) -> Result<u16, Error> {
130 let iface_id = wlan_service_util::create_iface(&self.monitor_svc, phy_id, role, sta_addr)
131 .await
132 .context("Create: Failed to create iface")?;
133
134 Ok(iface_id)
135 }
136
137 pub async fn destroy_iface(&self, iface_id: u16) -> Result<(), Error> {
142 wlan_service_util::destroy_iface(&self.monitor_svc, iface_id)
143 .await
144 .context("Destroy: Failed to destroy iface")
145 }
146
147 pub async fn disconnect(&self) -> Result<(), Error> {
148 wlan_service_util::client::disconnect_all(&self.monitor_svc)
149 .await
150 .context("Disconnect: Failed to disconnect ifaces")
151 }
152
153 pub async fn status(&self) -> Result<types::ClientStatusResponseDef, Error> {
154 let sme_proxy = wlan_service_util::client::get_first_sme(&self.monitor_svc)
156 .await
157 .context("Status: failed to get iface sme proxy")?;
158
159 let rsp = sme_proxy.status().await.context("failed to get status from sme_proxy")?;
160
161 Ok(rsp.into())
162 }
163
164 pub async fn query_iface(
165 &self,
166 iface_id: u16,
167 ) -> Result<types::QueryIfaceResponseWrapper, Error> {
168 let iface_info = self
169 .monitor_svc
170 .query_iface(iface_id)
171 .await
172 .context("Failed to query iface information")?
173 .map_err(|e| zx::Status::from_raw(e));
174
175 match iface_info {
176 Ok(info) => Ok(types::QueryIfaceResponseWrapper(info.into())),
177 Err(zx::Status::NOT_FOUND) => {
178 Err(format_err!("no iface information for ID: {}", iface_id))
179 }
180 Err(e) => Err(e.into()),
181 }
182 }
183}