1
0
mirror of https://github.com/esphome/esphome.git synced 2025-10-24 12:43:51 +01:00

[wifi] Optimize WiFi scanning to reduce copies and heap allocations (#11323)

This commit is contained in:
J. Nick Koston
2025-10-19 09:26:18 -10:00
committed by GitHub
parent 25f3b6a959
commit 33fea90c19
2 changed files with 23 additions and 19 deletions

View File

@@ -607,10 +607,12 @@ void WiFiComponent::check_scanning_finished() {
for (auto &ap : this->sta_) { for (auto &ap : this->sta_) {
if (res.matches(ap)) { if (res.matches(ap)) {
res.set_matches(true); res.set_matches(true);
if (!this->has_sta_priority(res.get_bssid())) { // Cache priority lookup - do single search instead of 2 separate searches
this->set_sta_priority(res.get_bssid(), ap.get_priority()); const bssid_t &bssid = res.get_bssid();
if (!this->has_sta_priority(bssid)) {
this->set_sta_priority(bssid, ap.get_priority());
} }
res.set_priority(this->get_sta_priority(res.get_bssid())); res.set_priority(this->get_sta_priority(bssid));
break; break;
} }
} }
@@ -629,8 +631,9 @@ void WiFiComponent::check_scanning_finished() {
return; return;
} }
WiFiAP connect_params; // Build connection params directly into selected_ap_ to avoid extra copy
WiFiScanResult scan_res = this->scan_result_[0]; const WiFiScanResult &scan_res = this->scan_result_[0];
WiFiAP &selected = this->selected_ap_;
for (auto &config : this->sta_) { for (auto &config : this->sta_) {
// search for matching STA config, at least one will match (from checks before) // search for matching STA config, at least one will match (from checks before)
if (!scan_res.matches(config)) { if (!scan_res.matches(config)) {
@@ -639,37 +642,38 @@ void WiFiComponent::check_scanning_finished() {
if (config.get_hidden()) { if (config.get_hidden()) {
// selected network is hidden, we use the data from the config // selected network is hidden, we use the data from the config
connect_params.set_hidden(true); selected.set_hidden(true);
connect_params.set_ssid(config.get_ssid()); selected.set_ssid(config.get_ssid());
// don't set BSSID and channel, there might be multiple hidden networks // Clear channel and BSSID for hidden networks - there might be multiple hidden networks
// but we can't know which one is the correct one. Rely on probe-req with just SSID. // but we can't know which one is the correct one. Rely on probe-req with just SSID.
selected.set_channel(0);
selected.set_bssid(optional<bssid_t>{});
} else { } else {
// selected network is visible, we use the data from the scan // selected network is visible, we use the data from the scan
// limit the connect params to only connect to exactly this network // limit the connect params to only connect to exactly this network
// (network selection is done during scan phase). // (network selection is done during scan phase).
connect_params.set_hidden(false); selected.set_hidden(false);
connect_params.set_ssid(scan_res.get_ssid()); selected.set_ssid(scan_res.get_ssid());
connect_params.set_channel(scan_res.get_channel()); selected.set_channel(scan_res.get_channel());
connect_params.set_bssid(scan_res.get_bssid()); selected.set_bssid(scan_res.get_bssid());
} }
// copy manual IP (if set) // copy manual IP (if set)
connect_params.set_manual_ip(config.get_manual_ip()); selected.set_manual_ip(config.get_manual_ip());
#ifdef USE_WIFI_WPA2_EAP #ifdef USE_WIFI_WPA2_EAP
// copy EAP parameters (if set) // copy EAP parameters (if set)
connect_params.set_eap(config.get_eap()); selected.set_eap(config.get_eap());
#endif #endif
// copy password (if set) // copy password (if set)
connect_params.set_password(config.get_password()); selected.set_password(config.get_password());
break; break;
} }
yield(); yield();
this->selected_ap_ = connect_params; this->start_connecting(this->selected_ap_, false);
this->start_connecting(connect_params, false);
} }
void WiFiComponent::dump_config() { void WiFiComponent::dump_config() {
@@ -902,7 +906,7 @@ WiFiScanResult::WiFiScanResult(const bssid_t &bssid, std::string ssid, uint8_t c
rssi_(rssi), rssi_(rssi),
with_auth_(with_auth), with_auth_(with_auth),
is_hidden_(is_hidden) {} is_hidden_(is_hidden) {}
bool WiFiScanResult::matches(const WiFiAP &config) { bool WiFiScanResult::matches(const WiFiAP &config) const {
if (config.get_hidden()) { if (config.get_hidden()) {
// User configured a hidden network, only match actually hidden networks // User configured a hidden network, only match actually hidden networks
// don't match SSID // don't match SSID

View File

@@ -170,7 +170,7 @@ class WiFiScanResult {
public: public:
WiFiScanResult(const bssid_t &bssid, std::string ssid, uint8_t channel, int8_t rssi, bool with_auth, bool is_hidden); WiFiScanResult(const bssid_t &bssid, std::string ssid, uint8_t channel, int8_t rssi, bool with_auth, bool is_hidden);
bool matches(const WiFiAP &config); bool matches(const WiFiAP &config) const;
bool get_matches() const; bool get_matches() const;
void set_matches(bool matches); void set_matches(bool matches);