9#include "lwip/ip_addr.h"
24static const char *
const TAG =
"wifi_lt";
39static constexpr size_t EVENT_QUEUE_SIZE = 16;
40static QueueHandle_t s_event_queue =
nullptr;
41static volatile uint32_t s_event_queue_overflow_count =
47 arduino_event_id_t event_id;
65 } sta_authmode_change;
90static uint8_t s_ignored_disconnect_count = 0;
95static constexpr uint8_t IGNORED_DISCONNECT_THRESHOLD = 3;
98 uint8_t current_mode = WiFi.getMode();
99 bool current_sta = current_mode & 0b01;
100 bool current_ap = current_mode & 0b10;
101 bool enable_sta = sta.
value_or(current_sta);
102 bool enable_ap = ap.
value_or(current_ap);
103 if (current_sta == enable_sta && current_ap == enable_ap)
106 if (enable_sta && !current_sta) {
107 ESP_LOGV(TAG,
"Enabling STA");
108 }
else if (!enable_sta && current_sta) {
109 ESP_LOGV(TAG,
"Disabling STA");
111 if (enable_ap && !current_ap) {
112 ESP_LOGV(TAG,
"Enabling AP");
113 }
else if (!enable_ap && current_ap) {
114 ESP_LOGV(TAG,
"Disabling AP");
122 bool ret = WiFi.mode(
static_cast<wifi_mode_t
>(
mode));
125 ESP_LOGW(TAG,
"Setting mode failed");
134 int8_t
val =
static_cast<int8_t
>(output_power * 4);
135 return WiFi.setTxPower(
val);
141 WiFi.setAutoReconnect(
false);
147#ifdef USE_WIFI_POWER_SAVE_LISTENERS
161 if (!manual_ip.has_value()) {
165 WiFi.config(manual_ip->static_ip, manual_ip->gateway, manual_ip->subnet, manual_ip->dns1, manual_ip->dns2);
174 addresses[0] = WiFi.localIP();
177 auto v6_addresses = WiFi.allLocalIPv6();
178 for (
auto address : v6_addresses) {
179 addresses[i++] = network::IPAddress(
address.toString().c_str());
195 String ssid = WiFi.SSID();
196 if (ssid && strcmp(ssid.c_str(), ap.get_ssid().c_str()) != 0) {
200#ifdef USE_WIFI_MANUAL_IP
214 s_ignored_disconnect_count = 0;
216 WiFiStatus
status = WiFi.begin(ap.get_ssid().c_str(), ap.get_password().empty() ? NULL : ap.get_password().c_str(),
218 ap.has_bssid() ? ap.get_bssid().data() : NULL);
219 if (status != WL_CONNECTED) {
220 ESP_LOGW(TAG,
"esp_wifi_connect failed: %d", status);
232 case WIFI_AUTH_WPA_PSK:
234 case WIFI_AUTH_WPA2_PSK:
236 case WIFI_AUTH_WPA_WPA2_PSK:
237 return "WPA/WPA2 PSK";
259 case WIFI_REASON_AUTH_EXPIRE:
260 return "Auth Expired";
261 case WIFI_REASON_AUTH_LEAVE:
263 case WIFI_REASON_ASSOC_EXPIRE:
264 return "Association Expired";
265 case WIFI_REASON_ASSOC_TOOMANY:
266 return "Too Many Associations";
267 case WIFI_REASON_NOT_AUTHED:
268 return "Not Authenticated";
269 case WIFI_REASON_NOT_ASSOCED:
270 return "Not Associated";
271 case WIFI_REASON_ASSOC_LEAVE:
272 return "Association Leave";
273 case WIFI_REASON_ASSOC_NOT_AUTHED:
274 return "Association not Authenticated";
275 case WIFI_REASON_DISASSOC_PWRCAP_BAD:
276 return "Disassociate Power Cap Bad";
277 case WIFI_REASON_DISASSOC_SUPCHAN_BAD:
278 return "Disassociate Supported Channel Bad";
279 case WIFI_REASON_IE_INVALID:
281 case WIFI_REASON_MIC_FAILURE:
282 return "Mic Failure";
283 case WIFI_REASON_4WAY_HANDSHAKE_TIMEOUT:
284 return "4-Way Handshake Timeout";
285 case WIFI_REASON_GROUP_KEY_UPDATE_TIMEOUT:
286 return "Group Key Update Timeout";
287 case WIFI_REASON_IE_IN_4WAY_DIFFERS:
288 return "IE In 4-Way Handshake Differs";
289 case WIFI_REASON_GROUP_CIPHER_INVALID:
290 return "Group Cipher Invalid";
291 case WIFI_REASON_PAIRWISE_CIPHER_INVALID:
292 return "Pairwise Cipher Invalid";
293 case WIFI_REASON_AKMP_INVALID:
294 return "AKMP Invalid";
295 case WIFI_REASON_UNSUPP_RSN_IE_VERSION:
296 return "Unsupported RSN IE version";
297 case WIFI_REASON_INVALID_RSN_IE_CAP:
298 return "Invalid RSN IE Cap";
299 case WIFI_REASON_802_1X_AUTH_FAILED:
300 return "802.1x Authentication Failed";
301 case WIFI_REASON_CIPHER_SUITE_REJECTED:
302 return "Cipher Suite Rejected";
303 case WIFI_REASON_BEACON_TIMEOUT:
304 return "Beacon Timeout";
305 case WIFI_REASON_NO_AP_FOUND:
306 return "AP Not Found";
307 case WIFI_REASON_AUTH_FAIL:
308 return "Authentication Failed";
309 case WIFI_REASON_ASSOC_FAIL:
310 return "Association Failed";
311 case WIFI_REASON_HANDSHAKE_TIMEOUT:
312 return "Handshake Failed";
313 case WIFI_REASON_CONNECTION_FAIL:
314 return "Connection Failed";
315 case WIFI_REASON_UNSPECIFIED:
317 return "Unspecified";
321#define ESPHOME_EVENT_ID_WIFI_READY ARDUINO_EVENT_WIFI_READY
322#define ESPHOME_EVENT_ID_WIFI_SCAN_DONE ARDUINO_EVENT_WIFI_SCAN_DONE
323#define ESPHOME_EVENT_ID_WIFI_STA_START ARDUINO_EVENT_WIFI_STA_START
324#define ESPHOME_EVENT_ID_WIFI_STA_STOP ARDUINO_EVENT_WIFI_STA_STOP
325#define ESPHOME_EVENT_ID_WIFI_STA_CONNECTED ARDUINO_EVENT_WIFI_STA_CONNECTED
326#define ESPHOME_EVENT_ID_WIFI_STA_DISCONNECTED ARDUINO_EVENT_WIFI_STA_DISCONNECTED
327#define ESPHOME_EVENT_ID_WIFI_STA_AUTHMODE_CHANGE ARDUINO_EVENT_WIFI_STA_AUTHMODE_CHANGE
328#define ESPHOME_EVENT_ID_WIFI_STA_GOT_IP ARDUINO_EVENT_WIFI_STA_GOT_IP
329#define ESPHOME_EVENT_ID_WIFI_STA_GOT_IP6 ARDUINO_EVENT_WIFI_STA_GOT_IP6
330#define ESPHOME_EVENT_ID_WIFI_STA_LOST_IP ARDUINO_EVENT_WIFI_STA_LOST_IP
331#define ESPHOME_EVENT_ID_WIFI_AP_START ARDUINO_EVENT_WIFI_AP_START
332#define ESPHOME_EVENT_ID_WIFI_AP_STOP ARDUINO_EVENT_WIFI_AP_STOP
333#define ESPHOME_EVENT_ID_WIFI_AP_STACONNECTED ARDUINO_EVENT_WIFI_AP_STACONNECTED
334#define ESPHOME_EVENT_ID_WIFI_AP_STADISCONNECTED ARDUINO_EVENT_WIFI_AP_STADISCONNECTED
335#define ESPHOME_EVENT_ID_WIFI_AP_STAIPASSIGNED ARDUINO_EVENT_WIFI_AP_STAIPASSIGNED
336#define ESPHOME_EVENT_ID_WIFI_AP_PROBEREQRECVED ARDUINO_EVENT_WIFI_AP_PROBEREQRECVED
337#define ESPHOME_EVENT_ID_WIFI_AP_GOT_IP6 ARDUINO_EVENT_WIFI_AP_GOT_IP6
344 if (s_event_queue ==
nullptr) {
349 auto *to_send =
new LTWiFiEvent{};
350 to_send->event_id = event;
354 case ESPHOME_EVENT_ID_WIFI_STA_CONNECTED: {
355 auto &it = info.wifi_sta_connected;
356 to_send->data.sta_connected.ssid_len = it.ssid_len;
357 memcpy(to_send->data.sta_connected.ssid, it.ssid,
358 std::min(
static_cast<size_t>(it.ssid_len),
sizeof(to_send->data.sta_connected.ssid) - 1));
359 memcpy(to_send->data.sta_connected.bssid, it.bssid, 6);
360 to_send->data.sta_connected.channel = it.channel;
361 to_send->data.sta_connected.authmode = it.authmode;
364 case ESPHOME_EVENT_ID_WIFI_STA_DISCONNECTED: {
365 auto &it = info.wifi_sta_disconnected;
366 to_send->data.sta_disconnected.ssid_len = it.ssid_len;
367 memcpy(to_send->data.sta_disconnected.ssid, it.ssid,
368 std::min(
static_cast<size_t>(it.ssid_len),
sizeof(to_send->data.sta_disconnected.ssid) - 1));
369 memcpy(to_send->data.sta_disconnected.bssid, it.bssid, 6);
370 to_send->data.sta_disconnected.reason = it.reason;
373 case ESPHOME_EVENT_ID_WIFI_STA_AUTHMODE_CHANGE: {
374 auto &it = info.wifi_sta_authmode_change;
375 to_send->data.sta_authmode_change.old_mode = it.old_mode;
376 to_send->data.sta_authmode_change.new_mode = it.new_mode;
379 case ESPHOME_EVENT_ID_WIFI_SCAN_DONE: {
380 auto &it = info.wifi_scan_done;
381 to_send->data.scan_done.status = it.status;
382 to_send->data.scan_done.number = it.number;
383 to_send->data.scan_done.scan_id = it.scan_id;
386 case ESPHOME_EVENT_ID_WIFI_AP_PROBEREQRECVED: {
387 auto &it = info.wifi_ap_probereqrecved;
388 memcpy(to_send->data.ap_probe_req.mac, it.mac, 6);
389 to_send->data.ap_probe_req.rssi = it.rssi;
392 case ESPHOME_EVENT_ID_WIFI_AP_STACONNECTED: {
393 auto &it = info.wifi_sta_connected;
394 memcpy(to_send->data.sta_connected.bssid, it.bssid, 6);
397 case ESPHOME_EVENT_ID_WIFI_AP_STADISCONNECTED: {
398 auto &it = info.wifi_sta_disconnected;
399 memcpy(to_send->data.sta_disconnected.bssid, it.bssid, 6);
402 case ESPHOME_EVENT_ID_WIFI_READY:
403 case ESPHOME_EVENT_ID_WIFI_STA_START:
404 case ESPHOME_EVENT_ID_WIFI_STA_STOP:
405 case ESPHOME_EVENT_ID_WIFI_STA_GOT_IP:
406 case ESPHOME_EVENT_ID_WIFI_STA_GOT_IP6:
407 case ESPHOME_EVENT_ID_WIFI_STA_LOST_IP:
408 case ESPHOME_EVENT_ID_WIFI_AP_START:
409 case ESPHOME_EVENT_ID_WIFI_AP_STOP:
410 case ESPHOME_EVENT_ID_WIFI_AP_STAIPASSIGNED:
420 if (xQueueSend(s_event_queue, &to_send, 0) != pdPASS) {
422 s_event_queue_overflow_count++;
431 switch (event->event_id) {
432 case ESPHOME_EVENT_ID_WIFI_READY: {
433 ESP_LOGV(TAG,
"Ready");
436 case ESPHOME_EVENT_ID_WIFI_SCAN_DONE: {
437 auto &it =
event->data.scan_done;
438 ESP_LOGV(TAG,
"Scan done: status=%" PRIu32
" number=%u scan_id=%u", it.status, it.number, it.scan_id);
442 case ESPHOME_EVENT_ID_WIFI_STA_START: {
443 ESP_LOGV(TAG,
"STA start");
447 case ESPHOME_EVENT_ID_WIFI_STA_STOP: {
448 ESP_LOGV(TAG,
"STA stop");
452 case ESPHOME_EVENT_ID_WIFI_STA_CONNECTED: {
453 auto &it =
event->data.sta_connected;
454 char bssid_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
456 ESP_LOGV(TAG,
"Connected ssid='%.*s' bssid=" LOG_SECRET(
"%s")
" channel=%u, authmode=%s", it.ssid_len,
461#ifdef USE_WIFI_CONNECT_STATE_LISTENERS
464 this->
pending_.connect_state =
true;
467#ifdef USE_WIFI_MANUAL_IP
470#ifdef USE_WIFI_IP_STATE_LISTENERS
479 case ESPHOME_EVENT_ID_WIFI_STA_DISCONNECTED: {
480 auto &it =
event->data.sta_disconnected;
494 s_ignored_disconnect_count++;
495 if (s_ignored_disconnect_count >= IGNORED_DISCONNECT_THRESHOLD) {
496 ESP_LOGW(TAG,
"Too many disconnect events (%u) while connecting, treating as failure (reason=%s)",
503 ESP_LOGV(TAG,
"Ignoring disconnect event with empty ssid while connecting (reason=%s, count=%u)",
509 if (it.reason == WIFI_REASON_NO_AP_FOUND) {
510 ESP_LOGW(TAG,
"Disconnected ssid='%.*s' reason='Probe Request Unsuccessful'", it.ssid_len,
511 (
const char *) it.ssid);
514 char bssid_s[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
516 ESP_LOGW(TAG,
"Disconnected ssid='%.*s' bssid=" LOG_SECRET(
"%s")
" reason='%s'", it.ssid_len,
521 uint8_t reason = it.reason;
522 if (reason == WIFI_REASON_AUTH_EXPIRE || reason == WIFI_REASON_BEACON_TIMEOUT ||
523 reason == WIFI_REASON_NO_AP_FOUND || reason == WIFI_REASON_ASSOC_FAIL ||
524 reason == WIFI_REASON_HANDSHAKE_TIMEOUT) {
529#ifdef USE_WIFI_CONNECT_STATE_LISTENERS
530 static constexpr uint8_t EMPTY_BSSID[6] = {};
532 listener->on_wifi_connect_state(
StringRef(), EMPTY_BSSID);
537 case ESPHOME_EVENT_ID_WIFI_STA_AUTHMODE_CHANGE: {
538 auto &it =
event->data.sta_authmode_change;
542 if (it.old_mode != WIFI_AUTH_OPEN && it.new_mode == WIFI_AUTH_OPEN) {
543 ESP_LOGW(TAG,
"Potential Authmode downgrade detected, disconnecting");
550 case ESPHOME_EVENT_ID_WIFI_STA_GOT_IP: {
551 char ip_buf[network::IP_ADDRESS_BUFFER_SIZE], gw_buf[network::IP_ADDRESS_BUFFER_SIZE];
555#ifdef USE_WIFI_IP_STATE_LISTENERS
562 case ESPHOME_EVENT_ID_WIFI_STA_GOT_IP6: {
563 ESP_LOGV(TAG,
"Got IPv6");
564#ifdef USE_WIFI_IP_STATE_LISTENERS
571 case ESPHOME_EVENT_ID_WIFI_STA_LOST_IP: {
572 ESP_LOGV(TAG,
"Lost IP");
576 case ESPHOME_EVENT_ID_WIFI_AP_START: {
577 ESP_LOGV(TAG,
"AP start");
580 case ESPHOME_EVENT_ID_WIFI_AP_STOP: {
581 ESP_LOGV(TAG,
"AP stop");
584 case ESPHOME_EVENT_ID_WIFI_AP_STACONNECTED: {
585#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
586 auto &it =
event->data.sta_connected;
587 char mac_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
589 ESP_LOGV(TAG,
"AP client connected MAC=%s", mac_buf);
593 case ESPHOME_EVENT_ID_WIFI_AP_STADISCONNECTED: {
594#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
595 auto &it =
event->data.sta_disconnected;
596 char mac_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
598 ESP_LOGV(TAG,
"AP client disconnected MAC=%s", mac_buf);
602 case ESPHOME_EVENT_ID_WIFI_AP_STAIPASSIGNED: {
603 ESP_LOGV(TAG,
"AP client assigned IP");
606 case ESPHOME_EVENT_ID_WIFI_AP_PROBEREQRECVED: {
607#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERY_VERBOSE
608 auto &it =
event->data.ap_probe_req;
609 char mac_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
611 ESP_LOGVV(TAG,
"AP receive Probe Request MAC=%s RSSI=%d", mac_buf, it.rssi);
622 s_event_queue = xQueueCreate(EVENT_QUEUE_SIZE,
sizeof(LTWiFiEvent *));
623 if (s_event_queue ==
nullptr) {
624 ESP_LOGE(TAG,
"Failed to create event queue");
636 switch (s_sta_state) {
660 int16_t err = WiFi.scanNetworks(
true,
true, passive, 200);
661 if (err != WIFI_SCAN_RUNNING) {
662 ESP_LOGV(TAG,
"WiFi.scanNetworks failed: %d", err);
672 int16_t num = WiFi.scanComplete();
676 this->
scan_result_.init(
static_cast<unsigned int>(num));
677 for (
int i = 0; i < num; i++) {
678 String ssid = WiFi.SSID(i);
679 wifi_auth_mode_t authmode = WiFi.encryptionType(i);
680 int32_t rssi = WiFi.RSSI(i);
681 uint8_t *bssid = WiFi.BSSID(i);
682 int32_t channel = WiFi.channel(i);
684 this->
scan_result_.emplace_back(
bssid_t{bssid[0], bssid[1], bssid[2], bssid[3], bssid[4], bssid[5]},
685 std::string(ssid.c_str()), channel, rssi, authmode != WIFI_AUTH_OPEN,
689#ifdef USE_WIFI_SCAN_RESULTS_LISTENERS
703 return WiFi.softAPConfig(manual_ip->static_ip, manual_ip->gateway, manual_ip->subnet);
705 return WiFi.softAPConfig(IPAddress(192, 168, 4, 1), IPAddress(192, 168, 4, 1), IPAddress(255, 255, 255, 0));
714#ifdef USE_WIFI_MANUAL_IP
716 ESP_LOGV(TAG,
"wifi_ap_ip_config_ failed");
721 ESP_LOGV(TAG,
"wifi_ap_ip_config_ failed");
728 return WiFi.softAP(ap.get_ssid().c_str(), ap.get_password().empty() ? NULL : ap.get_password().c_str(),
729 ap.has_channel() ? ap.get_channel() : 1, ap.get_hidden());
739 return WiFi.disconnect();
744 uint8_t *raw_bssid = WiFi.BSSID();
745 if (raw_bssid !=
nullptr) {
746 for (
size_t i = 0; i < bssid.size(); i++)
747 bssid[i] = raw_bssid[i];
754 String ssid = WiFi.SSID();
755 size_t len = std::min(
static_cast<size_t>(ssid.length()), SSID_BUFFER_SIZE - 1);
756 memcpy(buffer.data(), ssid.c_str(),
len);
758 return buffer.data();
767 if (s_event_queue ==
nullptr) {
772 if (s_event_queue_overflow_count > 0) {
773 ESP_LOGW(TAG,
"Event queue overflow, %" PRIu32
" events dropped", s_event_queue_overflow_count);
774 s_event_queue_overflow_count = 0;
779 if (xQueueReceive(s_event_queue, &event, 0) != pdTRUE) {
BedjetMode mode
BedJet operating mode.
const std::string & get_name() const
Get the name of this Application set by pre_setup().
StringRef is a reference to a string owned by something else.
value_type value_or(U const &v) const
const optional< ManualIP > & get_manual_ip() const
bool error_from_callback_
bool wifi_apply_power_save_()
WiFiPowerSaveMode power_save_
const WiFiAP * get_selected_sta_() const
struct esphome::wifi::WiFiComponent::@175 pending_
wifi_scan_vector_t< WiFiScanResult > scan_result_
StaticVector< WiFiScanResultsListener *, ESPHOME_WIFI_SCAN_RESULTS_LISTENERS > scan_results_listeners_
bool wifi_sta_ip_config_(const optional< ManualIP > &manual_ip)
network::IPAddress get_dns_address(int num)
void wifi_process_event_(IDFWiFiEvent *data)
bool wifi_start_ap_(const WiFiAP &ap)
StaticVector< WiFiConnectStateListener *, ESPHOME_WIFI_CONNECT_STATE_LISTENERS > connect_state_listeners_
int32_t get_wifi_channel()
const char * wifi_ssid_to(std::span< char, SSID_BUFFER_SIZE > buffer)
Write SSID to buffer without heap allocation.
network::IPAddress wifi_subnet_mask_()
network::IPAddress wifi_soft_ap_ip()
void wifi_event_callback_(arduino_event_id_t event, arduino_event_info_t info)
network::IPAddress wifi_gateway_ip_()
network::IPAddress wifi_dns_ip_(int num)
bool wifi_apply_hostname_()
bool wifi_sta_pre_setup_()
StaticVector< WiFiIPStateListener *, ESPHOME_WIFI_IP_STATE_LISTENERS > ip_state_listeners_
bool wifi_ap_ip_config_(const optional< ManualIP > &manual_ip)
StaticVector< WiFiPowerSaveListener *, ESPHOME_WIFI_POWER_SAVE_LISTENERS > power_save_listeners_
bool wifi_apply_output_power_(float output_power)
WiFiSTAConnectStatus wifi_sta_connect_status_()
bool wifi_sta_connect_(const WiFiAP &ap)
bool wifi_mode_(optional< bool > sta, optional< bool > ap)
network::IPAddresses wifi_sta_ip_addresses()
bool wifi_scan_start_(bool passive)
void wifi_scan_done_callback_()
std::array< IPAddress, 5 > IPAddresses
std::array< uint8_t, 6 > bssid_t
const LogString * get_auth_mode_str(uint8_t mode)
arduino_event_info_t esphome_wifi_event_info_t
const LogString * get_disconnect_reason_str(uint8_t reason)
@ ERROR_NETWORK_NOT_FOUND
arduino_event_id_t esphome_wifi_event_id_t
const LogString * get_op_mode_str(uint8_t mode)
void IRAM_ATTR HOT yield()
void IRAM_ATTR HOT delay(uint32_t ms)
Application App
Global storage of Application pointer - only one Application can exist.
char * format_mac_addr_upper(const uint8_t *mac, char *output)
Format MAC address as XX:XX:XX:XX:XX:XX (uppercase, colon separators)
char * str_to(char *buf) const
Write IP address to buffer. Buffer must be at least IP_ADDRESS_BUFFER_SIZE bytes.