2
0

network_wifi.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805
  1. #ifdef NETWORK_WIFI_LOG_LEVEL
  2. #define LOG_LOCAL_LEVEL NETWORK_WIFI_LOG_LEVEL
  3. #endif
  4. #include "network_wifi.h"
  5. #include "Config.h"
  6. #include "WifiList.h"
  7. #include "accessors.h"
  8. #include "bootstate.h"
  9. #include "cJSON.h"
  10. #include "dns_server.h"
  11. #include "esp_event.h"
  12. #include "esp_log.h"
  13. #include "esp_system.h"
  14. #include "esp_wifi.h"
  15. #include "esp_wifi_types.h"
  16. #include "lwip/sockets.h"
  17. #include "messaging.h"
  18. #include "network_status.h"
  19. #include "platform_esp32.h"
  20. #include "tools.h"
  21. #include "trace.h"
  22. #include <string.h>
  23. #include "esp_sntp.h"
  24. WifiList* wifi_config_aps;
  25. EXT_RAM_ATTR WifiList scan_results_aps("scan_results");
  26. static void network_wifi_event_handler(void* arg, esp_event_base_t event_base, int32_t event_id, void* event_data);
  27. static const char* get_disconnect_code_desc(uint8_t reason);
  28. esp_err_t network_wifi_get_blob(void* target, size_t size, const char* key);
  29. static const char TAG[] = "network_wifi";
  30. esp_netif_t* wifi_netif;
  31. esp_netif_t* wifi_ap_netif;
  32. extern sys_status_data status;
  33. static void time_sync_notification_cb(struct timeval *tv)
  34. {
  35. ESP_LOGI(TAG, "Notification of a time synchronization event");
  36. wifi_config_aps->UpdateFromClock();
  37. scan_results_aps.UpdateFromClock();
  38. }
  39. static bool is_initialized() { return (platform != NULL) && platform->has_net; }
  40. // void network_wifi_update_connected() {
  41. // wifi_config_t config;
  42. // esp_err_t err = esp_wifi_get_config(WIFI_IF_STA, &config);
  43. // if (err != ESP_OK) {
  44. // ESP_LOGE(TAG, "Unable to update connected STA: %s", esp_err_to_name(err));
  45. // return;
  46. // }
  47. // auto sta = wifi_config_aps->AddUpdate(&config.sta);
  48. // ESP_LOGD(TAG, "Added/Updated STA %s to credentials list, password: %s", sta.ssid, sta.password);
  49. // wifi_config_aps->PrintWifiSTAEntry(sta);
  50. // if (sys_status_obj.Lock()) {
  51. // sta.connected = true;
  52. // auto status = sys_status_obj.get();
  53. // status.has_net = true;
  54. // status.net.has_wifi = true;
  55. // memcpy(&status.net.wifi.connected_sta, &sta, sizeof(status.net.wifi.connected_sta));
  56. // status.net.wifi.has_connected_sta = true;
  57. // sys_status_obj.Unlock();
  58. // }
  59. // config_raise_changed(false);
  60. // }
  61. size_t network_wifi_get_known_count() {
  62. size_t result = 0;
  63. if (!is_initialized()) {
  64. ESP_LOGE(TAG, "Config not initialized");
  65. return 0;
  66. }
  67. result = wifi_config_aps->GetCount();
  68. ESP_LOGD(TAG, "SSID Count from config: %d", result);
  69. return result;
  70. }
  71. const wifi_sta_config_t* network_wifi_get_active_config() {
  72. static wifi_config_t config;
  73. esp_err_t err = ESP_OK;
  74. memset(&config, 0x00, sizeof(config));
  75. if ((err = esp_wifi_get_config(WIFI_IF_STA, &config)) == ESP_OK) {
  76. return &config.sta;
  77. } else {
  78. ESP_LOGD(TAG, "Could not get wifi STA config: %s", esp_err_to_name(err));
  79. }
  80. return nullptr;
  81. }
  82. bool network_wifi_is_known_ap(const char* ssid) { return wifi_config_aps->Exists(ssid); }
  83. esp_netif_t* network_wifi_get_interface() { return wifi_netif; }
  84. esp_netif_t* network_wifi_get_ap_interface() { return wifi_ap_netif; }
  85. esp_err_t network_wifi_set_sta_mode() {
  86. if (!wifi_netif) {
  87. ESP_LOGE(TAG, "Wifi not initialized. Cannot set sta mode");
  88. return ESP_ERR_INVALID_STATE;
  89. }
  90. ESP_LOGD(TAG, "Set Mode to STA");
  91. esp_err_t err = esp_wifi_set_mode(WIFI_MODE_STA);
  92. if (err != ESP_OK) {
  93. ESP_LOGE(TAG, "Error setting mode to STA: %s", esp_err_to_name(err));
  94. } else {
  95. ESP_LOGI(TAG, "Starting wifi");
  96. err = esp_wifi_start();
  97. if (err != ESP_OK) {
  98. ESP_LOGE(TAG, "Error starting wifi: %s", esp_err_to_name(err));
  99. }
  100. }
  101. ESP_LOGD(TAG,"Done setting wifi mode to STA");
  102. return err;
  103. }
  104. esp_netif_t* network_wifi_start() {
  105. esp_err_t err = ESP_OK;
  106. if (!wifi_netif) {
  107. wifi_netif = esp_netif_create_default_wifi_sta();
  108. wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
  109. err = esp_wifi_init(&cfg);
  110. if (err != ESP_OK) {
  111. ESP_LOGE(TAG, "Error initializing wifi: %s", esp_err_to_name(err));
  112. return wifi_netif;
  113. }
  114. // network_wifi_register_handlers();
  115. ESP_ERROR_CHECK_WITHOUT_ABORT(esp_event_handler_instance_register(WIFI_EVENT, ESP_EVENT_ANY_ID, &network_wifi_event_handler, NULL, NULL));
  116. ESP_ERROR_CHECK_WITHOUT_ABORT(esp_wifi_set_storage(WIFI_STORAGE_RAM));
  117. }
  118. network_wifi_set_sta_mode();
  119. network_set_hostname(wifi_netif);
  120. return wifi_netif;
  121. }
  122. void destroy_network_wifi() {}
  123. bool network_wifi_sta_config_changed() { return wifi_config_aps->Update(network_wifi_get_active_config()); }
  124. esp_err_t network_wifi_set_connected(bool connected) {
  125. wifi_config_t config;
  126. bool state_changed = false;
  127. bool config_changed = false;
  128. ESP_LOGD(TAG, "Resetting connected list");
  129. if (wifi_config_aps->Lock()) {
  130. wifi_config_aps->ResetConnected();
  131. wifi_config_aps->Unlock();
  132. }
  133. ESP_LOGD(TAG, "Erasing connected sta object");
  134. if (sys_status_obj.get()->net.wifi.has_connected_sta && sys_status_obj.Lock()) {
  135. pb_release(&sys_net_wifi_entry_msg, &sys_status_obj.get()->net.wifi.connected_sta);
  136. sys_status_obj.get()->net.has_wifi = true;
  137. sys_status_obj.get()->has_net = true;
  138. sys_status_obj.get()->net.wifi.has_connected_sta = false;
  139. sys_status_obj.Unlock();
  140. }
  141. if (stateWrapper.get()->has_connected_sta && stateWrapper.Lock()) {
  142. pb_release(&sys_net_wifi_entry_msg, stateWrapper.get());
  143. stateWrapper.get()->has_connected_sta = false;
  144. stateWrapper.Unlock();
  145. state_changed = true;
  146. }
  147. esp_err_t err = esp_wifi_get_config(WIFI_IF_STA, &config);
  148. if (err != ESP_OK) {
  149. ESP_LOGE(TAG, "Error retrieving wifi config");
  150. } else {
  151. if (!connected) {
  152. ESP_LOGI(TAG, "Connected list was reset");
  153. } else {
  154. ESP_LOGI(TAG, "Updating connected sta to %s", WifiList::GetSSID(&config.sta).c_str());
  155. ESP_LOGD(TAG, "Locking credentials manager");
  156. if (wifi_config_aps->Lock()) {
  157. ESP_LOGD(TAG, "Adding/updating credentials");
  158. wifi_config_aps->AddUpdate(&config.sta).connected = true;
  159. ESP_LOGD(TAG, "Unlocking credentials manager");
  160. config_changed = true;
  161. wifi_config_aps->Unlock();
  162. }
  163. }
  164. }
  165. if (config_changed) {
  166. ESP_LOGD(TAG, "Raising changed event on the config wrapper object");
  167. configWrapper.RaiseChangedAsync();
  168. }
  169. if (state_changed) {
  170. ESP_LOGD(TAG, "Raising changed event on the state wrapper object");
  171. stateWrapper.RaiseChangedAsync();
  172. }
  173. return err;
  174. }
  175. static void network_wifi_event_handler(void* arg, esp_event_base_t event_base, int32_t event_id, void* event_data) {
  176. if (event_base != WIFI_EVENT) return;
  177. switch (event_id) {
  178. case WIFI_EVENT_WIFI_READY:
  179. ESP_LOGD(TAG, "WIFI_EVENT_WIFI_READY");
  180. break;
  181. case WIFI_EVENT_SCAN_DONE:
  182. ESP_LOGD(TAG, "WIFI_EVENT_SCAN_DONE");
  183. network_async_scan_done();
  184. break;
  185. case WIFI_EVENT_STA_AUTHMODE_CHANGE:
  186. ESP_LOGD(TAG, "WIFI_EVENT_STA_AUTHMODE_CHANGE");
  187. break;
  188. case WIFI_EVENT_AP_START:
  189. ESP_LOGD(TAG, "WIFI_EVENT_AP_START");
  190. break;
  191. case WIFI_EVENT_AP_STOP:
  192. ESP_LOGD(TAG, "WIFI_EVENT_AP_STOP");
  193. break;
  194. case WIFI_EVENT_AP_PROBEREQRECVED: {
  195. wifi_event_ap_probe_req_rx_t* s = (wifi_event_ap_probe_req_rx_t*)event_data;
  196. char* mac = alloc_get_formatted_mac_string(s->mac);
  197. if (mac) {
  198. ESP_LOGD(TAG, "WIFI_EVENT_AP_PROBEREQRECVED. RSSI: %d, MAC: %s", s->rssi, STR_OR_BLANK(mac));
  199. }
  200. FREE_AND_NULL(mac);
  201. } break;
  202. case WIFI_EVENT_STA_WPS_ER_SUCCESS:
  203. ESP_LOGD(TAG, "WIFI_EVENT_STA_WPS_ER_SUCCESS");
  204. break;
  205. case WIFI_EVENT_STA_WPS_ER_FAILED:
  206. ESP_LOGD(TAG, "WIFI_EVENT_STA_WPS_ER_FAILED");
  207. break;
  208. case WIFI_EVENT_STA_WPS_ER_TIMEOUT:
  209. ESP_LOGD(TAG, "WIFI_EVENT_STA_WPS_ER_TIMEOUT");
  210. break;
  211. case WIFI_EVENT_STA_WPS_ER_PIN:
  212. ESP_LOGD(TAG, "WIFI_EVENT_STA_WPS_ER_PIN");
  213. break;
  214. case WIFI_EVENT_AP_STACONNECTED: {
  215. wifi_event_ap_staconnected_t* stac = (wifi_event_ap_staconnected_t*)event_data;
  216. char* mac = alloc_get_formatted_mac_string(stac->mac);
  217. if (mac) {
  218. ESP_LOGD(TAG, "WIFI_EVENT_AP_STACONNECTED. aid: %d, mac: %s", stac->aid, STR_OR_BLANK(mac));
  219. }
  220. FREE_AND_NULL(mac);
  221. } break;
  222. case WIFI_EVENT_AP_STADISCONNECTED:
  223. ESP_LOGD(TAG, "WIFI_EVENT_AP_STADISCONNECTED");
  224. break;
  225. case WIFI_EVENT_STA_START:
  226. ESP_LOGD(TAG, "WIFI_EVENT_STA_START");
  227. break;
  228. case WIFI_EVENT_STA_STOP:
  229. ESP_LOGD(TAG, "WIFI_EVENT_STA_STOP");
  230. break;
  231. case WIFI_EVENT_STA_CONNECTED: {
  232. ESP_LOGD(TAG, "WIFI_EVENT_STA_CONNECTED. ");
  233. network_async(EN_CONNECTED);
  234. } break;
  235. case WIFI_EVENT_STA_DISCONNECTED: {
  236. if (is_restarting()) {
  237. ESP_LOGD(TAG, "Disconnect event ignored while restarting");
  238. return;
  239. }
  240. wifi_event_sta_disconnected_t* s = (wifi_event_sta_disconnected_t*)event_data;
  241. char* bssid = alloc_get_formatted_mac_string(s->bssid);
  242. ESP_LOGW(TAG, "WIFI_EVENT_STA_DISCONNECTED. From BSSID: %s, reason code: %d (%s)", STR_OR_BLANK(bssid), s->reason,
  243. get_disconnect_code_desc(s->reason));
  244. FREE_AND_NULL(bssid);
  245. if (s->reason == WIFI_REASON_ROAMING) {
  246. ESP_LOGI(TAG, "WiFi Roaming to new access point");
  247. } else {
  248. network_async_lost_connection((wifi_event_sta_disconnected_t*)event_data);
  249. network_wifi_set_connected(false);
  250. }
  251. } break;
  252. default:
  253. break;
  254. }
  255. }
  256. void network_wifi_global_init() {
  257. ESP_LOGD(TAG, "running network_wifi_global_init");
  258. assert(platform != nullptr);
  259. wifi_config_aps = reinterpret_cast<WifiList*>(platform->net.credentials);
  260. assert(wifi_config_aps != nullptr);
  261. sys_status_obj.get()->has_net = true;
  262. sys_status_obj.get()->net.has_wifi = true;
  263. sys_status_obj.get()->net.wifi.scan_result = reinterpret_cast<sys_net_wifi_entry*>(&scan_results_aps);
  264. sntp_set_time_sync_notification_cb(time_sync_notification_cb);
  265. }
  266. esp_netif_t* network_wifi_config_ap() {
  267. esp_netif_ip_info_t info;
  268. esp_err_t err = ESP_OK;
  269. wifi_config_t ap_config;
  270. memset(&ap_config, 0x00, sizeof(ap_config));
  271. ESP_LOGI(TAG, "Configuring Access Point.");
  272. if (!wifi_ap_netif) {
  273. wifi_ap_netif = esp_netif_create_default_wifi_ap();
  274. }
  275. if (!is_initialized()) {
  276. ESP_LOGE(TAG, "Config not initialized");
  277. return wifi_ap_netif;
  278. }
  279. if (platform->has_net && platform->net.has_ap && platform->net.ap.has_ip) {
  280. inet_pton(AF_INET, platform->net.ap.ip.ip, (ip4_addr_t*)&info.ip); /* access point is on a static IP */
  281. inet_pton(AF_INET, platform->net.ap.ip.gw, (ip4_addr_t*)&info.gw); /* access point is on a static IP */
  282. inet_pton(AF_INET, platform->net.ap.ip.netmask, (ip4_addr_t*)&info.netmask); /* access point is on a static IP */
  283. } else {
  284. ESP_LOGE(TAG, "Access point not configured. Bailing out");
  285. return nullptr;
  286. }
  287. ap_config.ap.max_connection = platform->net.ap.max_connection > 0 ? platform->net.ap.max_connection : 4;
  288. ap_config.ap.beacon_interval = platform->net.ap.beacon_interval > 0 ? platform->net.ap.beacon_interval : 100;
  289. ap_config.ap.ssid_hidden = platform->net.ap.hidden;
  290. ap_config.ap.authmode = WifiList::GetESPAuthMode(platform->net.ap.auth_mode);
  291. ap_config.ap.channel = platform->net.ap.channel;
  292. /* In order to change the IP info structure, we have to first stop
  293. * the DHCP server on the new interface
  294. */
  295. network_start_stop_dhcps(wifi_ap_netif, false);
  296. ESP_LOGD(TAG, "Setting tcp_ip info for access point");
  297. if ((err = esp_netif_set_ip_info(wifi_ap_netif, &info)) != ESP_OK) {
  298. ESP_LOGE(TAG, "Setting tcp_ip info for interface TCPIP_ADAPTER_IF_AP. Error %s", esp_err_to_name(err));
  299. return wifi_ap_netif;
  300. }
  301. network_start_stop_dhcps(wifi_ap_netif, true);
  302. strlcpy((char*)ap_config.ap.ssid, platform->names.wifi_ap_name, sizeof(ap_config.ap.ssid));
  303. strlcpy((char*)ap_config.ap.password, STR_OR_BLANK(platform->net.ap.password), sizeof(ap_config.ap.password));
  304. ESP_LOGI(TAG, "AP SSID: %s, PASSWORD: %s Auth Mode: %d SSID: %s Max Connections: %d Beacon interval: %d", (char*)ap_config.ap.ssid,
  305. (char*)ap_config.ap.password, ap_config.ap.authmode, ap_config.ap.ssid_hidden ? "HIDDEN" : "VISIBLE", ap_config.ap.max_connection,
  306. ap_config.ap.beacon_interval);
  307. const char* msg = "Setting wifi mode as WIFI_MODE_APSTA";
  308. ESP_LOGD(TAG, "%s", msg);
  309. if ((err = esp_wifi_set_mode(WIFI_MODE_APSTA)) != ESP_OK) {
  310. ESP_LOGE(TAG, "%s. Error %s", msg, esp_err_to_name(err));
  311. return wifi_ap_netif;
  312. }
  313. msg = "Setting wifi AP configuration for WIFI_IF_AP";
  314. ESP_LOGD(TAG, "%s", msg);
  315. if ((err = esp_wifi_set_config(WIFI_IF_AP, &ap_config)) != ESP_OK) /* stop AP DHCP server */
  316. {
  317. ESP_LOGE(TAG, "%s . Error %s", msg, esp_err_to_name(err));
  318. return wifi_ap_netif;
  319. }
  320. msg = "Setting wifi bandwidth";
  321. ESP_LOGD(TAG, "%s (%d)", msg, DEFAULT_AP_BANDWIDTH);
  322. if ((err = esp_wifi_set_bandwidth(WIFI_IF_AP, DEFAULT_AP_BANDWIDTH)) != ESP_OK) /* stop AP
  323. DHCP server */
  324. {
  325. ESP_LOGE(TAG, "%s failed. Error %s", msg, esp_err_to_name(err));
  326. return wifi_ap_netif;
  327. }
  328. msg = "Setting wifi power save";
  329. ESP_LOGD(TAG, "%s (%s)", msg, sys_net_config_ps_types_name(platform->net.power_save_mode));
  330. if ((err = esp_wifi_set_ps((wifi_ps_type_t)platform->net.power_save_mode)) != ESP_OK) /* stop AP DHCP server */
  331. {
  332. ESP_LOGE(TAG, "%s failed. Error %s", msg, esp_err_to_name(err));
  333. return wifi_ap_netif;
  334. }
  335. ESP_LOGD(TAG, "Done configuring Soft Access Point");
  336. return wifi_ap_netif;
  337. }
  338. void network_wifi_filter_unique(wifi_ap_record_t* aplist, uint16_t* aps) {
  339. int total_unique;
  340. wifi_ap_record_t* first_free;
  341. total_unique = *aps;
  342. first_free = NULL;
  343. for (int i = 0; i < *aps - 1; i++) {
  344. wifi_ap_record_t* ap = &aplist[i];
  345. /* skip the previously removed APs */
  346. if (ap->ssid[0] == 0) continue;
  347. /* remove the identical SSID+authmodes */
  348. for (int j = i + 1; j < *aps; j++) {
  349. wifi_ap_record_t* ap1 = &aplist[j];
  350. if ((strcmp((const char*)ap->ssid, (const char*)ap1->ssid) == 0) &&
  351. (ap->authmode == ap1->authmode)) { /* same SSID, different auth mode is skipped */
  352. /* save the rssi for the display */
  353. if ((ap1->rssi) > (ap->rssi)) ap->rssi = ap1->rssi;
  354. /* clearing the record */
  355. memset(ap1, 0, sizeof(wifi_ap_record_t));
  356. }
  357. }
  358. }
  359. /* reorder the list so APs follow each other in the list */
  360. for (int i = 0; i < *aps; i++) {
  361. wifi_ap_record_t* ap = &aplist[i];
  362. /* skipping all that has no name */
  363. if (ap->ssid[0] == 0) {
  364. /* mark the first free slot */
  365. if (first_free == NULL) first_free = ap;
  366. total_unique--;
  367. continue;
  368. }
  369. if (first_free != NULL) {
  370. memcpy(first_free, ap, sizeof(wifi_ap_record_t));
  371. memset(ap, 0, sizeof(wifi_ap_record_t));
  372. /* find the next free slot */
  373. for (int j = 0; j < *aps; j++) {
  374. if (aplist[j].ssid[0] == 0) {
  375. first_free = &aplist[j];
  376. break;
  377. }
  378. }
  379. }
  380. }
  381. /* update the length of the list */
  382. *aps = total_unique;
  383. }
  384. bool network_wifi_update_scan_list(wifi_ap_record_t* accessp_records, uint16_t count) {
  385. bool changed = false;
  386. // reset RSSI values. This will help when choosing the
  387. // strongest signal to connect to
  388. wifi_config_aps->ResetRSSI();
  389. // clear current access point list
  390. scan_results_aps.Clear();
  391. if (count > 0) {
  392. WifiList::PrintWifiSTAEntryTitle();
  393. ESP_LOGD(TAG, "Found %d AP(s) during scan:",count);
  394. for (int i = 0; i < count; i++) {
  395. auto& entry = scan_results_aps.AddUpdate(&accessp_records[i]);
  396. scan_results_aps.UpdateLastSeen(&accessp_records[i]);
  397. if (wifi_config_aps->Exists(&accessp_records[i]) && configWrapper.Lock()) {
  398. ESP_LOGD(TAG,"AP with known password. updating last seen");
  399. changed = wifi_config_aps->Update(&accessp_records[i]) || changed;
  400. changed = wifi_config_aps->UpdateLastSeen(&accessp_records[i]) || changed;
  401. configWrapper.Unlock();
  402. }
  403. ESP_LOGV(TAG,"Address of entry is %p", static_cast<void*>(&entry));
  404. if (LOG_LOCAL_LEVEL >= ESP_LOG_DEBUG) {
  405. WifiList::PrintWifiSTAEntry(entry);
  406. }
  407. }
  408. } else {
  409. ESP_LOGI(TAG, "No nearby access point found");
  410. }
  411. if (changed) {
  412. config_raise_changed(false);
  413. }
  414. return true;
  415. }
  416. esp_err_t wifi_scan_done() {
  417. esp_err_t err = ESP_OK;
  418. wifi_ap_record_t* accessp_records = NULL;
  419. ESP_LOGD(TAG, "Getting AP list records");
  420. uint16_t ap_num = platform->net.max_ap_num;
  421. if ((err = esp_wifi_scan_get_ap_num(&ap_num)) != ESP_OK) {
  422. ESP_LOGE(TAG, "Failed to retrieve scan results count. Error %s", esp_err_to_name(err));
  423. return err;
  424. }
  425. if (ap_num <= 0) {
  426. ESP_LOGI(TAG, "No AP found during scan");
  427. return err;
  428. }
  429. accessp_records = (wifi_ap_record_t*)malloc_init_external(sizeof(wifi_ap_record_t) * ap_num);
  430. if (!accessp_records) {
  431. ESP_LOGE(TAG, "Alloc failed for scan results");
  432. return ESP_FAIL;
  433. }
  434. if ((err = esp_wifi_scan_get_ap_records(&ap_num, accessp_records)) != ESP_OK) {
  435. ESP_LOGE(TAG, "Failed to retrieve scan results list. Error %s", esp_err_to_name(err));
  436. } else {
  437. /* make sure the http server isn't trying to access the list while it gets refreshed */
  438. ESP_LOGD(TAG, "Retrieving scan list");
  439. /* Will remove the duplicate SSIDs from the list and update ap_num */
  440. network_wifi_filter_unique(accessp_records, &ap_num);
  441. if (!network_wifi_update_scan_list(accessp_records, ap_num)) {
  442. ESP_LOGE(TAG, "could not get access to json mutex in wifi_scan");
  443. err = ESP_FAIL;
  444. }
  445. ESP_LOGD(TAG, "Done retrieving scan list");
  446. }
  447. free(accessp_records);
  448. return err;
  449. }
  450. bool is_wifi_up() { return wifi_netif != NULL; }
  451. esp_err_t network_wifi_start_scan() {
  452. wifi_scan_config_t scan_config = {.ssid = 0, .bssid = 0, .channel = 0, .show_hidden = true, .scan_type = WIFI_SCAN_TYPE_ACTIVE};
  453. esp_err_t err = ESP_OK;
  454. ESP_LOGI(TAG, "Initiating wifi network scan");
  455. if (!is_wifi_up()) {
  456. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "Wifi not started. Cannot scan");
  457. return ESP_FAIL;
  458. }
  459. /* if a scan is already in progress this message is simply ignored thanks to the
  460. * WIFI_MANAGER_SCAN_BIT uxBit */
  461. if ((err = esp_wifi_scan_start(&scan_config, false)) != ESP_OK) {
  462. ESP_LOGW(TAG, "Unable to start scan; %s ", esp_err_to_name(err));
  463. // set_status_message(WARNING, "Wifi Connecting. Cannot start scan.");
  464. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "Scanning failed: %s", esp_err_to_name(err));
  465. }
  466. return err;
  467. }
  468. bool network_wifi_is_ap_mode() {
  469. wifi_mode_t mode;
  470. /* update config to latest and attempt connection */
  471. return esp_wifi_get_mode(&mode) == ESP_OK && mode == WIFI_MODE_AP;
  472. }
  473. bool network_wifi_is_sta_mode() {
  474. wifi_mode_t mode;
  475. /* update config to latest and attempt connection */
  476. return esp_wifi_get_mode(&mode) == ESP_OK && mode == WIFI_MODE_STA;
  477. }
  478. bool network_wifi_is_ap_sta_mode() {
  479. wifi_mode_t mode;
  480. /* update config to latest and attempt connection */
  481. return esp_wifi_get_mode(&mode) == ESP_OK && mode == WIFI_MODE_APSTA;
  482. }
  483. esp_err_t network_wifi_connect(const char* ssid, const char* password) {
  484. esp_err_t err = ESP_OK;
  485. wifi_config_t config;
  486. bool changed = false;
  487. memset(&config, 0x00, sizeof(config));
  488. ESP_LOGD(TAG, "network_wifi_connect");
  489. network_wifi_set_connected(false);
  490. if (!is_wifi_up()) {
  491. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "Wifi not started. Cannot connect");
  492. return ESP_FAIL;
  493. }
  494. if (!ssid || !password || strlen(ssid) == 0) {
  495. ESP_LOGE(TAG, "Cannot connect wifi. wifi config is null!");
  496. return ESP_ERR_INVALID_ARG;
  497. }
  498. wifi_mode_t wifi_mode;
  499. err = esp_wifi_get_mode(&wifi_mode);
  500. if (err == ESP_ERR_WIFI_NOT_INIT) {
  501. ESP_LOGW(TAG, "Wifi not initialized. Attempting to start sta mode");
  502. network_wifi_start();
  503. } else if (err != ESP_OK) {
  504. ESP_LOGE(TAG, "Could not retrieve wifi mode : %s", esp_err_to_name(err));
  505. } else if (wifi_mode != WIFI_MODE_STA && wifi_mode != WIFI_MODE_APSTA) {
  506. ESP_LOGD(TAG, "Changing wifi mode to STA");
  507. err = network_wifi_set_sta_mode();
  508. if (err != ESP_OK) {
  509. ESP_LOGE(TAG, "Could not set mode to STA. Cannot connect to SSID %s", ssid);
  510. return err;
  511. }
  512. }
  513. // copy configuration and connect
  514. strlcpy((char*)config.sta.ssid, ssid, sizeof(config.sta.ssid));
  515. if (password) {
  516. strlcpy((char*)config.sta.password, password, sizeof(config.sta.password));
  517. }
  518. config.sta.scan_method = platform->net.wifi_connect_fast_scan ? WIFI_FAST_SCAN : WIFI_ALL_CHANNEL_SCAN;
  519. auto existing = wifi_config_aps->Get(&config.sta);
  520. if (existing != nullptr) {
  521. if (existing->channel > 0 && existing->channel <= 13) {
  522. ESP_LOGD(TAG, "Trying to connect on channel %d", existing->channel);
  523. config.sta.channel = existing->channel;
  524. } else {
  525. ESP_LOGD(TAG, "Access point known, but no channel defined. Scanning");
  526. }
  527. }
  528. // First Disconnect
  529. esp_wifi_disconnect();
  530. config.sta.scan_method = WIFI_ALL_CHANNEL_SCAN;
  531. if ((err = esp_wifi_set_config(WIFI_IF_STA, &config)) != ESP_OK) {
  532. ESP_LOGE(TAG, "Failed to set STA configuration. Error %s", esp_err_to_name(err));
  533. }
  534. if (err == ESP_OK) {
  535. ESP_LOGI(TAG, "Wifi Connecting to %s...", ssid);
  536. if ((err = esp_wifi_connect()) != ESP_OK) {
  537. ESP_LOGE(TAG, "Failed to initiate wifi connection. Error %s", esp_err_to_name(err));
  538. } else {
  539. if (wifi_config_aps->Lock()) {
  540. ESP_LOGD(TAG, "Storing/updating credentials entry");
  541. if (wifi_config_aps->Exists(&config.sta)) {
  542. ESP_LOGD(TAG, "Existing entry. Updating it");
  543. changed = wifi_config_aps->Update(&config.sta);
  544. } else {
  545. ESP_LOGD(TAG, "New entry. Adding it");
  546. wifi_config_aps->AddUpdate(&config.sta);
  547. }
  548. auto entry = wifi_config_aps->Get(&config.sta);
  549. if (entry != nullptr) {
  550. wifi_config_aps->UpdateLastTry(&config.sta);
  551. WifiList::PrintWifiSTAEntryTitle();
  552. WifiList::PrintWifiSTAEntry(*entry);
  553. }
  554. wifi_config_aps->Unlock();
  555. }
  556. if (changed) {
  557. configWrapper.RaiseChangedAsync();
  558. }
  559. }
  560. }
  561. return err;
  562. }
  563. esp_err_t network_wifi_connect_ssid(const char* ssid) {
  564. sys_net_wifi_entry* item = wifi_config_aps->Get(std::string(ssid));
  565. if (item != nullptr) {
  566. return network_wifi_connect(item->ssid, item->password);
  567. } else {
  568. ESP_LOGE(TAG, "Attempting to connect to an unknown ssid: %s", ssid);
  569. }
  570. return ESP_FAIL;
  571. }
  572. esp_err_t network_wifi_connect_active_ssid() {
  573. sys_net_wifi_entry* connected = NULL;
  574. if (sys_state->has_connected_sta && strlen(sys_state->connected_sta.ssid) > 0) {
  575. connected = &sys_state->connected_sta;
  576. if (LOG_LOCAL_LEVEL >= ESP_LOG_INFO) {
  577. ESP_LOGI(TAG, "Connecting to active access point");
  578. WifiList::PrintWifiSTAEntryTitle();
  579. WifiList::PrintWifiSTAEntry(sys_state->connected_sta);
  580. }
  581. return network_wifi_connect(connected->ssid, connected->password);
  582. }
  583. return ESP_ERR_NOT_FOUND;
  584. }
  585. esp_err_t network_wifi_add_ssid(const char* ssid, const char* password) {
  586. bool changed = false;
  587. auto conn = wifi_config_aps->Get(ssid);
  588. if (conn != nullptr) {
  589. ESP_LOGW(TAG, "Existing ssid %s. Updating password", ssid);
  590. if (strcmp(conn->password, password) != 0) {
  591. strncpy(conn->password, password, sizeof(conn->password));
  592. changed = true;
  593. } else {
  594. ESP_LOGI(TAG, "No change to ssid %s", password);
  595. }
  596. } else {
  597. ESP_LOGI(TAG, "Adding new ssid %s with password %s to credentials", ssid, STR_OR_BLANK(password));
  598. auto seen = scan_results_aps.Get(ssid);
  599. if (seen) {
  600. ESP_LOGD(TAG,"Adding with extra info from scan");
  601. wifi_config_aps->AddUpdate(seen, STR_OR_BLANK(password));
  602. changed = true;
  603. } else {
  604. ESP_LOGD(TAG,"Adding without extra info from scan");
  605. wifi_config_aps->AddUpdate(ssid, STR_OR_BLANK(password));
  606. changed = true;
  607. }
  608. }
  609. if (changed) {
  610. config_raise_changed(false);
  611. }
  612. return ESP_OK;
  613. }
  614. esp_err_t network_wifi_remove_ssid(const char* ssid) {
  615. wifi_config_t config;
  616. esp_err_t err = esp_wifi_get_config(WIFI_IF_STA, &config);
  617. if (err != ESP_OK) {
  618. ESP_LOGE(TAG, "Error retrieving wifi config");
  619. }
  620. if ((network_wifi_is_sta_mode() || network_wifi_is_ap_sta_mode()) && WifiList::GetSSID(&config.sta) == ssid) {
  621. ESP_LOGI(TAG, "Disconnecting from deleted ssid: %s", ssid);
  622. esp_wifi_disconnect();
  623. }
  624. if (wifi_config_aps->RemoveCredential(std::string(ssid))) {
  625. ESP_LOGI(TAG, "SSID %s was removed", ssid);
  626. config_raise_changed(false);
  627. } else {
  628. ESP_LOGI(TAG, "SSID %s could not be removed", ssid);
  629. }
  630. return ESP_OK;
  631. }
  632. void network_wifi_activate_strongest_ssid() {
  633. sys_state->has_connected_sta = true;
  634. auto strongest = wifi_config_aps->GetStrongestSTA();
  635. if (LOG_LOCAL_LEVEL >= ESP_LOG_DEBUG) {
  636. ESP_LOGD(TAG, "Setting strongest AP as active");
  637. WifiList::PrintWifiSTAEntryTitle();
  638. WifiList::PrintWifiSTAEntry(strongest);
  639. }
  640. WifiList::Update(sys_state->connected_sta, strongest);
  641. }
  642. void network_wifi_clear_config() {
  643. /* erase configuration */
  644. wifi_config_aps->RemoveCredential(network_wifi_get_active_config());
  645. esp_err_t err = ESP_OK;
  646. if ((err = esp_wifi_disconnect()) != ESP_OK) {
  647. ESP_LOGW(TAG, "Could not disconnect from deleted network : %s", esp_err_to_name(err));
  648. }
  649. }
  650. void print_wifi_ap_status(const wifi_sta_config_t* ap_status) {
  651. if (ap_status == NULL) {
  652. printf("AP status is NULL\n");
  653. return;
  654. }
  655. printf("SSID: %s\n", ap_status->ssid);
  656. printf("Password: %s\n", ap_status->password);
  657. const char* scan_method = (ap_status->scan_method == WIFI_FAST_SCAN) ? "Fast Scan" : "All Channel Scan";
  658. printf("Scan Method: %s\n", scan_method);
  659. printf("BSSID Set: %s\n", ap_status->bssid_set ? "Yes" : "No");
  660. printf("BSSID: %02x:%02x:%02x:%02x:%02x:%02x\n", ap_status->bssid[0], ap_status->bssid[1], ap_status->bssid[2], ap_status->bssid[3],
  661. ap_status->bssid[4], ap_status->bssid[5]);
  662. printf("Channel: %d\n", ap_status->channel);
  663. printf("Listen Interval: %d\n", ap_status->listen_interval);
  664. const char* sort_method = (ap_status->sort_method == WIFI_CONNECT_AP_BY_SIGNAL) ? "By Signal" : "By Security";
  665. printf("Sort Method: %s\n", sort_method);
  666. printf("RSSI Threshold: %d\n", ap_status->threshold.rssi);
  667. printf("Auth Mode Threshold: %d\n", ap_status->threshold.authmode);
  668. printf("PMF Capable: %s\n", ap_status->pmf_cfg.capable ? "Yes" : "No");
  669. printf("PMF Required: %s\n", ap_status->pmf_cfg.required ? "Yes" : "No");
  670. // Add other fields as needed
  671. }
  672. void network_wifi_get_status() {
  673. wifi_config_t config;
  674. esp_err_t err = esp_wifi_get_config(WIFI_IF_STA, &config);
  675. if (err == ESP_OK) {
  676. print_wifi_ap_status(&config.sta);
  677. } else {
  678. ESP_LOGE(TAG, "Error getting wifi status: %s", esp_err_to_name(err));
  679. }
  680. if (wifi_config_aps->GetCount() > 0 || scan_results_aps.GetCount() > 0) {
  681. WifiList::PrintWifiSTAEntryTitle();
  682. }
  683. ESP_LOGI(TAG, "Known access points:");
  684. for (int i = 0; i < wifi_config_aps->GetCount(); i++) {
  685. const auto entry = wifi_config_aps->GetIndex(i);
  686. if (entry != nullptr) {
  687. WifiList::PrintWifiSTAEntry(*entry);
  688. }
  689. }
  690. if (scan_results_aps.GetCount() > 0) {
  691. ESP_LOGI(TAG, "List of nearby access points");
  692. for (int i = 0; i < scan_results_aps.GetCount(); i++) {
  693. const auto entry = scan_results_aps.GetIndex(i);
  694. if (entry != nullptr) {
  695. WifiList::PrintWifiSTAEntry(*entry);
  696. }
  697. }
  698. }
  699. }
  700. const char* get_disconnect_code_desc(uint8_t reason) {
  701. switch (reason) {
  702. ENUM_TO_STRING(WIFI_REASON_UNSPECIFIED);
  703. ENUM_TO_STRING(WIFI_REASON_AUTH_EXPIRE);
  704. ENUM_TO_STRING(WIFI_REASON_AUTH_LEAVE);
  705. ENUM_TO_STRING(WIFI_REASON_ASSOC_EXPIRE);
  706. ENUM_TO_STRING(WIFI_REASON_ASSOC_TOOMANY);
  707. ENUM_TO_STRING(WIFI_REASON_NOT_AUTHED);
  708. ENUM_TO_STRING(WIFI_REASON_NOT_ASSOCED);
  709. ENUM_TO_STRING(WIFI_REASON_ASSOC_LEAVE);
  710. ENUM_TO_STRING(WIFI_REASON_ASSOC_NOT_AUTHED);
  711. ENUM_TO_STRING(WIFI_REASON_DISASSOC_PWRCAP_BAD);
  712. ENUM_TO_STRING(WIFI_REASON_DISASSOC_SUPCHAN_BAD);
  713. ENUM_TO_STRING(WIFI_REASON_IE_INVALID);
  714. ENUM_TO_STRING(WIFI_REASON_MIC_FAILURE);
  715. ENUM_TO_STRING(WIFI_REASON_4WAY_HANDSHAKE_TIMEOUT);
  716. ENUM_TO_STRING(WIFI_REASON_GROUP_KEY_UPDATE_TIMEOUT);
  717. ENUM_TO_STRING(WIFI_REASON_IE_IN_4WAY_DIFFERS);
  718. ENUM_TO_STRING(WIFI_REASON_GROUP_CIPHER_INVALID);
  719. ENUM_TO_STRING(WIFI_REASON_PAIRWISE_CIPHER_INVALID);
  720. ENUM_TO_STRING(WIFI_REASON_AKMP_INVALID);
  721. ENUM_TO_STRING(WIFI_REASON_UNSUPP_RSN_IE_VERSION);
  722. ENUM_TO_STRING(WIFI_REASON_INVALID_RSN_IE_CAP);
  723. ENUM_TO_STRING(WIFI_REASON_802_1X_AUTH_FAILED);
  724. ENUM_TO_STRING(WIFI_REASON_CIPHER_SUITE_REJECTED);
  725. ENUM_TO_STRING(WIFI_REASON_INVALID_PMKID);
  726. ENUM_TO_STRING(WIFI_REASON_BEACON_TIMEOUT);
  727. ENUM_TO_STRING(WIFI_REASON_NO_AP_FOUND);
  728. ENUM_TO_STRING(WIFI_REASON_AUTH_FAIL);
  729. ENUM_TO_STRING(WIFI_REASON_ASSOC_FAIL);
  730. ENUM_TO_STRING(WIFI_REASON_HANDSHAKE_TIMEOUT);
  731. ENUM_TO_STRING(WIFI_REASON_CONNECTION_FAIL);
  732. ENUM_TO_STRING(WIFI_REASON_AP_TSF_RESET);
  733. ENUM_TO_STRING(WIFI_REASON_ROAMING);
  734. }
  735. return "";
  736. }