Skip to content

Commit

Permalink
Reimplement scan result reading to use less stack
Browse files Browse the repository at this point in the history
  • Loading branch information
bugadani committed Oct 18, 2023
1 parent d0a448b commit 790c068
Showing 1 changed file with 125 additions and 35 deletions.
160 changes: 125 additions & 35 deletions esp-wifi/src/wifi/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -899,55 +899,141 @@ impl<'d> WifiController<'d> {

Ok(mode == wifi_mode_t_WIFI_MODE_AP || mode == wifi_mode_t_WIFI_MODE_APSTA)
}
}

struct ScanResults {
count: usize,
}

fn scan_result_count(&mut self) -> Result<usize, WifiError> {
let mut bss_total: u16 = 0;
impl ScanResults {
pub fn new() -> Self {
let mut bss_total = 0;

unsafe {
esp_wifi_result!(crate::binary::include::esp_wifi_scan_get_ap_num(
&mut bss_total
))
.map_err(|e| {
crate::binary::include::esp_wifi_clear_ap_list();
e
})?;
// `esp_wifi_scan_get_ap_num` shouldn't return an error. It's documentation lists state
// errors that would trip previous checks, and ESP_ERR_INVALID_ARG.
let count = esp_wifi_result!(unsafe {
crate::binary::include::esp_wifi_scan_get_ap_num(&mut bss_total)
})
.map(|_| bss_total as usize)
.unwrap_or(0);

Self { count }
}

pub fn remaining(&self) -> usize {
self.count
}

#[cfg(not(esp32c2))]
fn read_into<const N: usize>(
&mut self,
scanned: &mut heapless::Vec<AccessPointInfo, N>,
) -> Result<(), WifiError> {
while !scanned.is_full() {
if let Some(record) = self.next() {
// It is okay to ignore the result here, because we checked previously whether the
// vector is full.
_ = scanned.push(record?);
} else {
break;
}
}

Ok(bss_total as usize)
Ok(())
}

fn scan_results<const N: usize>(
#[cfg(esp32c2)]
fn read_into<const N: usize>(
&mut self,
) -> Result<heapless::Vec<AccessPointInfo, N>, WifiError> {
let mut scanned = heapless::Vec::<AccessPointInfo, N>::new();
scanned: &mut heapless::Vec<AccessPointInfo, N>,
) -> Result<(), WifiError> {
let mut bss_total: u16 = N as u16;
let mut records: [MaybeUninit<crate::binary::include::wifi_ap_record_t>; N] =
[MaybeUninit::uninit(); N];

unsafe {
let mut records: [MaybeUninit<crate::binary::include::wifi_ap_record_t>; N] =
[MaybeUninit::uninit(); N];

esp_wifi_result!(crate::binary::include::esp_wifi_scan_get_ap_records(
&mut bss_total,
records[0].as_mut_ptr(),
))
.map_err(|e| {
// upon scan failure, list should be cleared to avoid memory leakage
crate::binary::include::esp_wifi_clear_ap_list();
e
})?;

for i in 0..bss_total {
let record = MaybeUninit::assume_init_ref(&records[i as usize]);
let ap_info = convert_ap_info(record);

scanned.push(ap_info).ok();
))?;

for i in 0..(bss_total as usize).min(N) {
if !scanned.is_full() {
let record = MaybeUninit::assume_init_ref(&records[i]);
let ap_info = convert_ap_info(record);
scanned.push(ap_info).ok();
} else {
break;
}
}
}

// `esp_wifi_scan_get_ap_records` frees the whole list, regardless of how many records were
// actually returned.
self.count = 0;
Ok(())
}

pub fn take_results<const N: usize>(
&mut self,
) -> Result<heapless::Vec<AccessPointInfo, N>, WifiError> {
let mut scanned = heapless::Vec::<AccessPointInfo, N>::new();

self.read_into(&mut scanned)?;

Ok(scanned)
}
}

#[cfg(not(esp32c2))]
impl Iterator for ScanResults {
type Item = Result<AccessPointInfo, WifiError>;

fn next(&mut self) -> Option<Self::Item> {
extern "C" {
fn esp_mesh_scan_get_ap_record(
ap_record: *mut crate::binary::include::wifi_ap_record_t,
buffer: *mut crate::binary::c_types::c_void,
) -> crate::binary::include::esp_err_t;
}

if self.count == 0 {
return None;
}

self.count -= 1;

let mut record: MaybeUninit<crate::binary::include::wifi_ap_record_t> =
MaybeUninit::uninit();

unsafe {
let result = esp_wifi_result!(esp_mesh_scan_get_ap_record(
record.as_mut_ptr(),
core::ptr::null_mut(),
));

match result {
Ok(_) => {
let record = MaybeUninit::assume_init_ref(&record);
let ap_info = convert_ap_info(record);

Some(Ok(ap_info))
}

Err(e) => Some(Err(e)),
}
}
}
}

impl Drop for ScanResults {
fn drop(&mut self) {
unsafe {
crate::binary::include::esp_wifi_clear_ap_list();
}
}
}

// see https://docs.rs/smoltcp/0.7.1/smoltcp/phy/index.html
impl<'d> Device for WifiDevice<'d> {
type RxToken<'a> = WifiRxToken where Self: 'a;
Expand Down Expand Up @@ -1090,10 +1176,12 @@ impl Wifi for WifiController<'_> {
) -> Result<(heapless::Vec<AccessPointInfo, N>, usize), Self::Error> {
esp_wifi_result!(crate::wifi::wifi_start_scan(true))?;

let count = self.scan_result_count()?;
let result = self.scan_results()?;
let mut scan_result = ScanResults::new();

let count = scan_result.remaining();
let ap_list = scan_result.take_results()?;

Ok((result, count))
Ok((ap_list, count))
}

/// Get the currently used configuration.
Expand Down Expand Up @@ -1418,10 +1506,12 @@ mod asynch {
esp_wifi_result!(wifi_start_scan(false))?;
WifiEventFuture::new(WifiEvent::ScanDone).await;

let count = self.scan_result_count()?;
let result = self.scan_results()?;
let mut scan_result = ScanResults::new();

let count = scan_result.remaining();
let ap_list = scan_result.take_results()?;

Ok((result, count))
Ok((ap_list, count))
}

/// Async version of [`embedded_svc::wifi::Wifi`]'s `start` method
Expand Down

0 comments on commit 790c068

Please sign in to comment.