network_manager_handlers.c 50 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192
  1. #ifdef NETWORK_HANDLERS_LOG_LEVEL
  2. #define LOG_LOCAL_LEVEL NETWORK_HANDLERS_LOG_LEVEL
  3. #endif
  4. #include "network_manager.h"
  5. #include <stdbool.h>
  6. #include <stdio.h>
  7. #include <stdlib.h>
  8. #include <string.h>
  9. #include "network_ethernet.h"
  10. #include "network_status.h"
  11. #include "network_wifi.h"
  12. #include "dns_server.h"
  13. #include "esp_log.h"
  14. #include "esp_system.h"
  15. #include "freertos/FreeRTOS.h"
  16. #include "platform_esp32.h"
  17. #include "esp_netif.h"
  18. #include "freertos/task.h"
  19. #include "cJSON.h"
  20. #include "cmd_system.h"
  21. #include "esp_app_format.h"
  22. #include "esp_event.h"
  23. #include "esp_ota_ops.h"
  24. #include "esp_wifi.h"
  25. #include "esp_wifi_types.h"
  26. #include "lwip/api.h"
  27. #include "lwip/err.h"
  28. #include "lwip/ip4_addr.h"
  29. #include "lwip/netdb.h"
  30. #include "mdns.h"
  31. #include "messaging.h"
  32. #include "platform_config.h"
  33. #include "trace.h"
  34. #include "accessors.h"
  35. #include "esp_err.h"
  36. #include "tools.h"
  37. #include "http_server_handlers.h"
  38. #include "network_manager.h"
  39. static const char TAG[]="network_handlers";
  40. EXT_RAM_ATTR static state_t network_states[TOTAL_NM_STATE];
  41. EXT_RAM_ATTR static state_t Wifi_Active_State[TOTAL_WIFI_ACTIVE_STATE];
  42. EXT_RAM_ATTR static state_t Eth_Active_State[TOTAL_ETH_ACTIVE_STATE];
  43. EXT_RAM_ATTR static state_t Wifi_Configuring_State[TOTAL_WIFI_CONFIGURING_STATE];
  44. static void network_interface_coexistence(state_machine_t* state_machine);
  45. static state_machine_result_t local_traverse_state(state_machine_t* const state_machine,
  46. const state_t* const target_state, const char * caller) ;
  47. static state_machine_result_t local_switch_state(state_machine_t* state_machine,
  48. const state_t* const target_state, const char * caller);
  49. void network_execute_cb(state_machine_t* const state_machine, const char * caller);
  50. int get_root_id(const state_t * state);
  51. const state_t* get_root( const state_t* const state);
  52. #define ADD_ROOT(name,...) ADD_ROOT_FORWARD_DECLARATION(name,...)
  53. #define ADD_ROOT_LEAF(name,...) ADD_ROOT_LEAF_FORWARD_DECLARATION(name,...)
  54. #define ADD_LEAF(name,...) ADD_LEAF_FORWARD_DECLARATION(name,...)
  55. ALL_NM_STATE
  56. ALL_ETH_STATE(, )
  57. ALL_WIFI_STATE(, )
  58. ALL_WIFI_CONFIGURING_STATE(, )
  59. #undef ADD_ROOT
  60. #undef ADD_ROOT_LEAF
  61. #undef ADD_LEAF
  62. /*
  63. * --------------------- Global variables ---------------------
  64. */
  65. #define ADD_ROOT(NAME, CHILD) ADD_ROOT_(NAME, CHILD)
  66. #define ADD_LEAF(NAME, PARENT, LEVEL) ADD_LEAF_(NAME, PARENT, LEVEL)
  67. #define ADD_ROOT_LEAF(NAME) ADD_ROOT_LEAF_(NAME)
  68. #define ADD_ROOT_(NAME, CHILD) \
  69. [NAME] = { \
  70. .Handler = NAME##_handler, \
  71. .Entry = NAME##_entry_handler, \
  72. .Exit = NAME##_exit_handler, \
  73. .Level = 0, \
  74. .Parent = NULL, \
  75. .Id = NAME, \
  76. .Node = CHILD, \
  77. },
  78. #define ADD_ROOT_LEAF_(NAME) \
  79. [NAME] = { \
  80. .Handler = NAME##_handler, \
  81. .Entry = NAME##_entry_handler, \
  82. .Exit = NAME##_exit_handler, \
  83. .Id = NAME, \
  84. .Parent = NULL, \
  85. },
  86. #define ADD_LEAF_(NAME, PARENT, LEVEL) \
  87. [NAME] = { \
  88. .Handler = NAME##_handler, \
  89. .Entry = NAME##_entry_handler, \
  90. .Exit = NAME##_exit_handler, \
  91. .Id = NAME, \
  92. .Level = LEVEL, \
  93. .Parent = PARENT, \
  94. },
  95. static void network_initialize_state_machine_globals(){
  96. static state_t loc_network_states[] =
  97. {
  98. ALL_NM_STATE};
  99. static state_t loc_Wifi_Active_State[] = {
  100. ALL_WIFI_STATE(&network_states[NETWORK_WIFI_ACTIVE_STATE], 1)
  101. };
  102. static state_t loc_Eth_Active_State[]={
  103. ALL_ETH_STATE(&network_states[NETWORK_ETH_ACTIVE_STATE], 1)
  104. };
  105. static state_t loc_Wifi_Configuring_State[]={
  106. ALL_WIFI_CONFIGURING_STATE(&network_states[NETWORK_WIFI_CONFIGURING_ACTIVE_STATE], 1)
  107. };
  108. memcpy(&network_states,&loc_network_states,sizeof(network_states));
  109. memcpy(&Eth_Active_State,&loc_Eth_Active_State,sizeof(Eth_Active_State));
  110. memcpy(&Wifi_Active_State,&loc_Wifi_Active_State,sizeof(Wifi_Active_State));
  111. memcpy(&Wifi_Configuring_State,&loc_Wifi_Configuring_State,sizeof(Wifi_Configuring_State));
  112. }
  113. #undef ADD_ROOT
  114. #undef ADD_ROOT_LEAF
  115. #undef ADD_LEAF
  116. #define HANDLE_GLOBAL_EVENT(m) \
  117. if (handle_global_event(m) == EVENT_HANDLED) \
  118. return EVENT_HANDLED;
  119. static void network_connect_active_ssid(state_machine_t* const State_Machine) {
  120. network_t* const nm = (network_t*)State_Machine;
  121. if (network_wifi_connect_active_ssid() != ESP_OK) {
  122. ESP_LOGE(TAG, "Oups. Something went wrong!");
  123. nm->wifi_connected = false;
  124. ESP_LOGD(TAG, "Checking if ethernet interface is connected");
  125. if (network_is_interface_connected(nm->eth_netif)) {
  126. ESP_LOGD(TAG, "Ethernet connection is found. Try to fallback there");
  127. network_async(EN_ETHERNET_FALLBACK);
  128. } else {
  129. // returning to AP mode
  130. nm->STA_duration = nm->sta_polling_min_ms;
  131. ESP_LOGD(TAG, "No ethernet and no wifi configured. Go to configuration mode");
  132. network_async_configure();
  133. }
  134. }
  135. }
  136. void initialize_network_handlers(state_machine_t* state_machine){
  137. MEMTRACE_PRINT_DELTA();
  138. network_initialize_state_machine_globals();
  139. MEMTRACE_PRINT_DELTA();
  140. NETWORK_INSTANTIATED_STATE_handler(state_machine);
  141. MEMTRACE_PRINT_DELTA();
  142. }
  143. static state_machine_result_t handle_global_event(state_machine_t* state_machine) {
  144. network_t * net_sm= ((network_t *)state_machine);
  145. switch (net_sm->Machine.Event) {
  146. case EN_UPDATE_STATUS:
  147. // handle the event, but don't swicth
  148. MEMTRACE_PRINT_DELTA_MESSAGE("handle EN_UPDATE_STATUS - start");
  149. network_status_update_basic_info();
  150. MEMTRACE_PRINT_DELTA_MESSAGE("handle EN_UPDATE_STATUS - end");
  151. return EVENT_HANDLED;
  152. /* code */
  153. break;
  154. case EN_REBOOT:
  155. ESP_LOGD(TAG,"Called for reboot type %d",net_sm->event_parameters->rtype);
  156. switch (net_sm->event_parameters->rtype) {
  157. case OTA:
  158. ESP_LOGD(TAG, " Calling guided_restart_ota.");
  159. guided_restart_ota();
  160. break;
  161. case RECOVERY:
  162. ESP_LOGD(TAG, " Calling guided_factory.");
  163. guided_factory();
  164. break;
  165. case RESTART:
  166. ESP_LOGD(TAG, " Calling simple_restart.");
  167. simple_restart();
  168. break;
  169. default:
  170. break;
  171. }
  172. return EVENT_UN_HANDLED;
  173. break;
  174. case EN_REBOOT_URL:
  175. if (net_sm->event_parameters->strval) {
  176. start_ota(net_sm->event_parameters->strval, NULL, 0);
  177. FREE_AND_NULL(net_sm->event_parameters->strval);
  178. }
  179. return EVENT_UN_HANDLED;
  180. break;
  181. default:
  182. break;
  183. }
  184. return EVENT_UN_HANDLED;
  185. }
  186. /*********************************************************************************************
  187. * INSTANTIATED_STATE
  188. */
  189. static state_machine_result_t NETWORK_INSTANTIATED_STATE_entry_handler(state_machine_t* const State_Machine) {
  190. network_handler_entry_print(State_Machine,true);
  191. NETWORK_EXECUTE_CB(State_Machine);
  192. network_handler_entry_print(State_Machine,false);
  193. return EVENT_HANDLED;
  194. }
  195. static state_machine_result_t NETWORK_INSTANTIATED_STATE_handler(state_machine_t* const State_Machine) {
  196. network_handler_print(State_Machine,true);
  197. state_machine_result_t result = EVENT_UN_HANDLED;
  198. network_t* const nm = (network_t *)State_Machine;
  199. State_Machine->State = &network_states[NETWORK_INSTANTIATED_STATE];
  200. State_Machine->Event = EN_START;
  201. char * valuestr=NULL;
  202. valuestr=config_alloc_get_default(NVS_TYPE_STR,"pollmx","600",0);
  203. if (valuestr) {
  204. nm->sta_polling_max_ms = atoi(valuestr)*1000;
  205. ESP_LOGD(TAG, "sta_polling_max_ms set to %d", nm->sta_polling_max_ms);
  206. FREE_AND_NULL(valuestr);
  207. }
  208. valuestr=config_alloc_get_default(NVS_TYPE_STR,"pollmin","15",0);
  209. if (valuestr) {
  210. nm->sta_polling_min_ms = atoi(valuestr)*1000;
  211. ESP_LOGD(TAG, "sta_polling_min_ms set to %d", nm->sta_polling_min_ms);
  212. FREE_AND_NULL(valuestr);
  213. }
  214. valuestr=config_alloc_get_default(NVS_TYPE_STR,"ethtmout","8",0);
  215. if (valuestr) {
  216. nm->eth_link_down_reboot_ms = atoi(valuestr)*1000;
  217. ESP_LOGD(TAG, "ethtmout set to %d", nm->eth_link_down_reboot_ms);
  218. FREE_AND_NULL(valuestr);
  219. }
  220. valuestr=config_alloc_get_default(NVS_TYPE_STR,"dhcp_tmout","8",0);
  221. if(valuestr){
  222. nm->dhcp_timeout = atoi(valuestr)*1000;
  223. ESP_LOGD(TAG, "dhcp_timeout set to %d", nm->dhcp_timeout);
  224. FREE_AND_NULL(valuestr);
  225. }
  226. HANDLE_GLOBAL_EVENT(State_Machine);
  227. if (State_Machine->Event == EN_START) {
  228. result= local_traverse_state(State_Machine, &network_states[NETWORK_INITIALIZING_STATE],__FUNCTION__);
  229. }
  230. network_handler_print(State_Machine,false);
  231. return result;
  232. }
  233. static state_machine_result_t NETWORK_INSTANTIATED_STATE_exit_handler(state_machine_t* const State_Machine) {
  234. network_exit_handler_print(State_Machine,true);
  235. network_exit_handler_print(State_Machine,false);
  236. return EVENT_HANDLED;
  237. }
  238. /*********************************************************************************************
  239. * INITIALIZING_STATE
  240. */
  241. static state_machine_result_t NETWORK_INITIALIZING_STATE_entry_handler(state_machine_t* const State_Machine) {
  242. network_handler_entry_print(State_Machine,true);
  243. MEMTRACE_PRINT_DELTA_MESSAGE(" Initializing tcp_ip adapter");
  244. esp_netif_init();
  245. MEMTRACE_PRINT_DELTA_MESSAGE(" Creating the default event loop");
  246. ESP_ERROR_CHECK(esp_event_loop_create_default());
  247. MEMTRACE_PRINT_DELTA_MESSAGE("Initializing network status");
  248. init_network_status();
  249. MEMTRACE_PRINT_DELTA_MESSAGE("Loading wifi global configuration");
  250. network_wifi_global_init();
  251. MEMTRACE_PRINT_DELTA_MESSAGE(" Registering event handler");
  252. esp_event_handler_register(IP_EVENT, ESP_EVENT_ANY_ID, &network_ip_event_handler, NULL);
  253. MEMTRACE_PRINT_DELTA_MESSAGE(" Initializing network done. Starting http server");
  254. // send a message to start the connections
  255. http_server_start();
  256. MEMTRACE_PRINT_DELTA_MESSAGE("Executing Callback");
  257. NETWORK_EXECUTE_CB(State_Machine);
  258. network_handler_entry_print(State_Machine,false);
  259. return EVENT_HANDLED;
  260. }
  261. static state_machine_result_t NETWORK_INITIALIZING_STATE_handler(state_machine_t* const State_Machine) {
  262. network_handler_print(State_Machine,true);
  263. state_machine_result_t result = EVENT_UN_HANDLED;
  264. HANDLE_GLOBAL_EVENT(State_Machine);
  265. switch (State_Machine->Event) {
  266. case EN_START:
  267. if (network_is_wifi_prioritized()) {
  268. ESP_LOGI(TAG, "WiFi connection is prioritized. Starting WiFi");
  269. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_INITIALIZING_STATE],__FUNCTION__);
  270. }
  271. else {
  272. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_STARTING_STATE],__FUNCTION__);
  273. }
  274. break;
  275. default:
  276. result= EVENT_UN_HANDLED;
  277. }
  278. network_handler_print(State_Machine,false);
  279. return result;
  280. }
  281. static state_machine_result_t NETWORK_INITIALIZING_STATE_exit_handler(state_machine_t* const State_Machine) {
  282. network_exit_handler_print(State_Machine,true);
  283. network_exit_handler_print(State_Machine,false);
  284. return EVENT_HANDLED;
  285. }
  286. /*********************************************************************************************
  287. * ETH_STARTING_STATE
  288. */
  289. static state_machine_result_t ETH_STARTING_STATE_entry_handler(state_machine_t* const State_Machine) {
  290. network_handler_entry_print(State_Machine,true);
  291. ESP_LOGD(TAG, "Looking for ethernet Interface");
  292. network_t* const nm = (network_t *)State_Machine;
  293. init_network_ethernet();
  294. if (!network_ethernet_enabled()) {
  295. network_async_fail();
  296. } else {
  297. nm->eth_netif = network_ethernet_get_interface();
  298. }
  299. NETWORK_EXECUTE_CB(State_Machine);
  300. network_handler_entry_print(State_Machine,false);
  301. return EVENT_HANDLED;
  302. }
  303. static state_machine_result_t ETH_STARTING_STATE_handler(state_machine_t* const State_Machine) {
  304. state_machine_result_t result = EVENT_HANDLED;
  305. network_handler_print(State_Machine,true);
  306. HANDLE_GLOBAL_EVENT(State_Machine);
  307. switch (State_Machine->Event) {
  308. case EN_FAIL:
  309. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_INITIALIZING_STATE],__FUNCTION__);
  310. break;
  311. case EN_SUCCESS:
  312. result= local_traverse_state(State_Machine, &network_states[NETWORK_ETH_ACTIVE_STATE],__FUNCTION__);
  313. break;
  314. default:
  315. ESP_LOGE(TAG, "No handler");
  316. result= EVENT_UN_HANDLED;
  317. }
  318. network_handler_print(State_Machine,false);
  319. return result;
  320. }
  321. static state_machine_result_t ETH_STARTING_STATE_exit_handler(state_machine_t* const State_Machine) {
  322. network_exit_handler_print(State_Machine,true);
  323. network_exit_handler_print(State_Machine,false);
  324. return EVENT_HANDLED;
  325. }
  326. /*********************************************************************************************
  327. * NETWORK_ETH_ACTIVE_STATE
  328. */
  329. static state_machine_result_t NETWORK_ETH_ACTIVE_STATE_entry_handler(state_machine_t* const State_Machine) {
  330. network_handler_entry_print(State_Machine,true);
  331. network_t* const nm = (network_t *)State_Machine;
  332. network_set_timer(nm->eth_link_down_reboot_ms,"Ethernet link not detected" );
  333. NETWORK_EXECUTE_CB(State_Machine);
  334. network_handler_entry_print(State_Machine,false);
  335. return EVENT_HANDLED;
  336. }
  337. static state_machine_result_t NETWORK_ETH_ACTIVE_STATE_handler(state_machine_t* const State_Machine) {
  338. network_handler_print(State_Machine,true);
  339. state_machine_result_t result = EVENT_UN_HANDLED;
  340. network_t* const nm = (network_t *)State_Machine;
  341. switch (State_Machine->Event) {
  342. case EN_CONNECT_NEW:
  343. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_CONNECTING_NEW_STATE],__FUNCTION__);
  344. break;
  345. case EN_LINK_UP:
  346. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_ACTIVE_LINKUP_STATE],__FUNCTION__);
  347. break;
  348. case EN_LINK_DOWN:
  349. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_ACTIVE_LINKDOWN_STATE],__FUNCTION__);
  350. break;
  351. case EN_ETH_GOT_IP:
  352. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_ACTIVE_CONNECTED_STATE],__FUNCTION__);
  353. break;
  354. case EN_ETHERNET_FALLBACK:
  355. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_ACTIVE_CONNECTED_STATE],__FUNCTION__);
  356. break;
  357. case EN_TIMER:
  358. ESP_LOGW(TAG, "Timeout %s. Rebooting to wifi",STR_OR_ALT(nm->timer_tag,"Ethernet link not detected"));
  359. network_prioritize_wifi(true);
  360. simple_restart();
  361. //result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_INITIALIZING_STATE],__FUNCTION__);
  362. break;
  363. case EN_SCAN:
  364. ESP_LOGW(TAG,"Wifi scan cannot be executed in this state");
  365. network_wifi_built_known_ap_list();
  366. break;
  367. case EN_DELETE: {
  368. ESP_LOGD(TAG, "WiFi disconnected by user");
  369. network_wifi_clear_config();
  370. network_status_update_ip_info(UPDATE_USER_DISCONNECT);
  371. result= EVENT_HANDLED;
  372. } break;
  373. default:
  374. HANDLE_GLOBAL_EVENT(State_Machine);
  375. result=EVENT_UN_HANDLED;
  376. }
  377. network_handler_print(State_Machine,false);
  378. return result;
  379. }
  380. static state_machine_result_t NETWORK_ETH_ACTIVE_STATE_exit_handler(state_machine_t* const State_Machine) {
  381. network_exit_handler_print(State_Machine,true);
  382. network_set_timer(0,NULL);
  383. network_exit_handler_print(State_Machine,false);
  384. return EVENT_HANDLED;
  385. }
  386. /*********************************************************************************************
  387. * ETH_CONNECTING_NEW_STATE
  388. */
  389. static state_machine_result_t ETH_CONNECTING_NEW_STATE_entry_handler(state_machine_t* const State_Machine) {
  390. network_t* const nm = (network_t *)State_Machine;
  391. network_handler_entry_print(State_Machine,true);
  392. network_start_stop_dhcp(nm->wifi_netif, true);
  393. network_wifi_connect(nm->event_parameters->ssid,nm->event_parameters->password);
  394. FREE_AND_NULL(nm->event_parameters->ssid);
  395. FREE_AND_NULL(nm->event_parameters->password);
  396. NETWORK_EXECUTE_CB(State_Machine);
  397. network_handler_entry_print(State_Machine,false);
  398. return EVENT_HANDLED;
  399. }
  400. static state_machine_result_t ETH_CONNECTING_NEW_STATE_handler(state_machine_t* const State_Machine) {
  401. HANDLE_GLOBAL_EVENT(State_Machine);
  402. network_handler_print(State_Machine,true);
  403. state_machine_result_t result = EVENT_HANDLED;
  404. switch (State_Machine->Event) {
  405. case EN_GOT_IP:
  406. result= local_traverse_state(State_Machine, &network_states[WIFI_CONNECTED_STATE],__FUNCTION__);
  407. break;
  408. case EN_LOST_CONNECTION:
  409. network_status_update_ip_info(UPDATE_FAILED_ATTEMPT);
  410. messaging_post_message(MESSAGING_ERROR, MESSAGING_CLASS_SYSTEM, "Unable to connect to new WiFi access point.");
  411. // no existing configuration, or wifi wasn't the active connection when connection
  412. // attempt was made
  413. network_async(EN_ETHERNET_FALLBACK);
  414. result = EVENT_HANDLED;
  415. break;
  416. default:
  417. result= EVENT_UN_HANDLED;
  418. }
  419. network_handler_print(State_Machine,false);
  420. return result;
  421. }
  422. static state_machine_result_t ETH_CONNECTING_NEW_STATE_exit_handler(state_machine_t* const State_Machine) {
  423. network_exit_handler_print(State_Machine,true);
  424. network_set_timer(0,NULL);
  425. network_exit_handler_print(State_Machine,false);
  426. return EVENT_HANDLED;
  427. }
  428. /*********************************************************************************************
  429. * ETH_ACTIVE_LINKDOWN
  430. */
  431. static state_machine_result_t ETH_ACTIVE_LINKDOWN_STATE_entry_handler(state_machine_t* const State_Machine) {
  432. network_handler_entry_print(State_Machine,true);
  433. network_t* const nm = (network_t *)State_Machine;
  434. network_set_timer(nm->eth_link_down_reboot_ms, "Ethernet link down" );
  435. NETWORK_EXECUTE_CB(State_Machine);
  436. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "Ethernet link down.");
  437. network_handler_entry_print(State_Machine,false);
  438. return EVENT_HANDLED;
  439. }
  440. static state_machine_result_t ETH_ACTIVE_LINKDOWN_STATE_handler(state_machine_t* const State_Machine) {
  441. network_handler_print(State_Machine,true);
  442. HANDLE_GLOBAL_EVENT(State_Machine);
  443. network_handler_print(State_Machine,false);
  444. return EVENT_UN_HANDLED;
  445. }
  446. static state_machine_result_t ETH_ACTIVE_LINKDOWN_STATE_exit_handler(state_machine_t* const State_Machine) {
  447. network_exit_handler_print(State_Machine,true);
  448. network_set_timer(0,NULL);
  449. network_exit_handler_print(State_Machine,false);
  450. return EVENT_HANDLED;
  451. }
  452. /*********************************************************************************************
  453. * ETH_ACTIVE_LINKUP_STATE
  454. */
  455. static state_machine_result_t ETH_ACTIVE_LINKUP_STATE_entry_handler(state_machine_t* const State_Machine) {
  456. network_handler_entry_print(State_Machine,true);
  457. network_t* const nm = (network_t *)State_Machine;
  458. network_set_timer(nm->dhcp_timeout, "DHCP timeout" );
  459. NETWORK_EXECUTE_CB(State_Machine);
  460. messaging_post_message(MESSAGING_INFO, MESSAGING_CLASS_SYSTEM, "Ethernet link up.");
  461. network_handler_entry_print(State_Machine,false);
  462. return EVENT_HANDLED;
  463. }
  464. static state_machine_result_t ETH_ACTIVE_LINKUP_STATE_handler(state_machine_t* const State_Machine) {
  465. state_machine_result_t result = EVENT_UN_HANDLED;
  466. network_handler_print(State_Machine,true);
  467. HANDLE_GLOBAL_EVENT(State_Machine);
  468. network_handler_print(State_Machine,false);
  469. return result;
  470. }
  471. static state_machine_result_t ETH_ACTIVE_LINKUP_STATE_exit_handler(state_machine_t* const State_Machine) {
  472. network_exit_handler_print(State_Machine,true);
  473. network_exit_handler_print(State_Machine,false);
  474. return EVENT_HANDLED;
  475. }
  476. /*********************************************************************************************
  477. * WIFI_UP_STATE
  478. */
  479. static state_machine_result_t NETWORK_WIFI_ACTIVE_STATE_entry_handler(state_machine_t* const State_Machine) {
  480. network_handler_entry_print(State_Machine,true);
  481. NETWORK_EXECUTE_CB(State_Machine);
  482. network_handler_entry_print(State_Machine,false);
  483. return EVENT_HANDLED;
  484. }
  485. static state_machine_result_t NETWORK_WIFI_ACTIVE_STATE_handler(state_machine_t* const State_Machine) {
  486. HANDLE_GLOBAL_EVENT(State_Machine);
  487. network_handler_print(State_Machine,true);
  488. state_machine_result_t result = EVENT_UN_HANDLED;
  489. switch (State_Machine->Event)
  490. {
  491. case EN_LINK_UP:
  492. ESP_LOGW(TAG, "Ethernet link up in wifi mode");
  493. break;
  494. case EN_ETH_GOT_IP:
  495. network_interface_coexistence(State_Machine);
  496. break;
  497. case EN_GOT_IP:
  498. network_status_update_ip_info(UPDATE_CONNECTION_OK);
  499. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTED_STATE],__FUNCTION__);
  500. break;
  501. case EN_SCAN:
  502. if (network_wifi_start_scan() == ESP_OK) {
  503. result= EVENT_HANDLED;
  504. }
  505. break;
  506. case EN_SCAN_DONE:
  507. if(wifi_scan_done() == ESP_OK) {
  508. result= EVENT_HANDLED;
  509. }
  510. break;
  511. case EN_CONNECT_NEW:
  512. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTING_NEW_STATE],__FUNCTION__);
  513. break;
  514. case EN_DELETE:
  515. result= local_traverse_state(State_Machine,&Wifi_Active_State[WIFI_USER_DISCONNECTED_STATE],__FUNCTION__);
  516. break;
  517. case EN_ETHERNET_FALLBACK:
  518. result= local_traverse_state(State_Machine, &Eth_Active_State[ETH_ACTIVE_CONNECTED_STATE],__FUNCTION__);
  519. break;
  520. default:
  521. break;
  522. }
  523. network_handler_print(State_Machine,false);
  524. return result;
  525. }
  526. static state_machine_result_t NETWORK_WIFI_ACTIVE_STATE_exit_handler(state_machine_t* const State_Machine) {
  527. network_exit_handler_print(State_Machine,true);
  528. network_exit_handler_print(State_Machine,false);
  529. return EVENT_HANDLED;
  530. }
  531. /*********************************************************************************************
  532. * WIFI_INITIALIZING_STATE
  533. */
  534. static state_machine_result_t WIFI_INITIALIZING_STATE_entry_handler(state_machine_t* const State_Machine) {
  535. network_t* const nm = (network_t *)State_Machine;
  536. network_handler_entry_print(State_Machine,true);
  537. if(!nm->wifi_netif){
  538. nm->wifi_netif = network_wifi_start();
  539. }
  540. if (!is_wifi_up()) {
  541. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "Wifi not started. Load Configuration");
  542. return EVENT_UN_HANDLED;
  543. }
  544. if (network_wifi_get_known_count()>0) {
  545. ESP_LOGI(TAG, "Existing wifi config found. Attempting to connect.");
  546. network_async_success();
  547. } else {
  548. /* no wifi saved: start soft AP! This is what should happen during a first run */
  549. ESP_LOGW(TAG, "No saved wifi. Starting AP configuration mode.");
  550. network_async_configure();
  551. }
  552. NETWORK_EXECUTE_CB(State_Machine);
  553. network_handler_entry_print(State_Machine,false);
  554. return EVENT_HANDLED;
  555. }
  556. static state_machine_result_t WIFI_INITIALIZING_STATE_handler(state_machine_t* const State_Machine) {
  557. HANDLE_GLOBAL_EVENT(State_Machine);
  558. network_handler_print(State_Machine,true);
  559. state_machine_result_t result = EVENT_HANDLED;
  560. switch (State_Machine->Event) {
  561. case EN_CONFIGURE:
  562. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_STATE],__FUNCTION__);
  563. break;
  564. case EN_SUCCESS:
  565. result= local_switch_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTING_STATE],__FUNCTION__);
  566. break;
  567. default:
  568. result= EVENT_UN_HANDLED;
  569. }
  570. network_handler_print(State_Machine,false);
  571. return result;
  572. }
  573. static state_machine_result_t WIFI_INITIALIZING_STATE_exit_handler(state_machine_t* const State_Machine) {
  574. network_exit_handler_print(State_Machine,true);
  575. network_exit_handler_print(State_Machine,false);
  576. return EVENT_HANDLED;
  577. }
  578. /*********************************************************************************************
  579. * WIFI_CONFIGURING_ACTIVE_STATE
  580. */
  581. static state_machine_result_t NETWORK_WIFI_CONFIGURING_ACTIVE_STATE_entry_handler(state_machine_t* const State_Machine) {
  582. network_handler_entry_print(State_Machine,true);
  583. network_t* const nm = (network_t *)State_Machine;
  584. nm->wifi_ap_netif = network_wifi_config_ap();
  585. dns_server_start(nm->wifi_ap_netif);
  586. network_wifi_start_scan();
  587. network_handler_entry_print(State_Machine,false);
  588. return EVENT_HANDLED;
  589. }
  590. static state_machine_result_t NETWORK_WIFI_CONFIGURING_ACTIVE_STATE_handler(state_machine_t* const State_Machine) {
  591. HANDLE_GLOBAL_EVENT(State_Machine);
  592. network_handler_print(State_Machine,true);
  593. state_machine_result_t result = EVENT_HANDLED;
  594. switch (State_Machine->Event) {
  595. case EN_SCAN:
  596. if (network_wifi_start_scan() == ESP_OK) {
  597. result= EVENT_HANDLED;
  598. }
  599. break;
  600. case EN_SCAN_DONE:
  601. ESP_LOGD(TAG,"Network configuration active, wifi scan completed");
  602. if(wifi_scan_done() == ESP_OK) {
  603. result= EVENT_HANDLED;
  604. }
  605. break;
  606. case EN_CONNECT_NEW:
  607. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_CONNECT_STATE],__FUNCTION__);
  608. break;
  609. case EN_LINK_UP:
  610. ESP_LOGW(TAG, "Ethernet link up in wifi mode");
  611. break;
  612. case EN_ETH_GOT_IP:
  613. network_interface_coexistence(State_Machine);
  614. break;
  615. default:
  616. result =EVENT_UN_HANDLED;
  617. }
  618. network_handler_print(State_Machine,false);
  619. return result;
  620. }
  621. static state_machine_result_t NETWORK_WIFI_CONFIGURING_ACTIVE_STATE_exit_handler(state_machine_t* const State_Machine) {
  622. network_exit_handler_print(State_Machine, true);
  623. /* bring down DNS hijack */
  624. ESP_LOGD(TAG, " Stopping DNS.");
  625. dns_server_stop();
  626. ESP_LOGD(TAG, "Changing wifi mode to STA.");
  627. network_wifi_set_sta_mode();
  628. network_exit_handler_print(State_Machine, false);
  629. return EVENT_HANDLED;
  630. }
  631. /*********************************************************************************************
  632. * WIFI_CONFIGURING_STATE
  633. */
  634. static state_machine_result_t WIFI_CONFIGURING_STATE_entry_handler(state_machine_t* const State_Machine) {
  635. network_handler_entry_print(State_Machine,true);
  636. NETWORK_EXECUTE_CB(State_Machine);
  637. network_handler_entry_print(State_Machine,false);
  638. return EVENT_HANDLED;
  639. }
  640. static state_machine_result_t WIFI_CONFIGURING_STATE_handler(state_machine_t* const State_Machine) {
  641. network_handler_print(State_Machine,true);
  642. HANDLE_GLOBAL_EVENT(State_Machine);
  643. network_handler_print(State_Machine,false);
  644. return EVENT_UN_HANDLED;
  645. }
  646. static state_machine_result_t WIFI_CONFIGURING_STATE_exit_handler(state_machine_t* const State_Machine) {
  647. network_exit_handler_print(State_Machine,true);
  648. network_exit_handler_print(State_Machine,false);
  649. return EVENT_HANDLED;
  650. }
  651. /*********************************************************************************************
  652. * WIFI_CONFIGURING_CONNECT_STATE
  653. */
  654. static state_machine_result_t WIFI_CONFIGURING_CONNECT_STATE_entry_handler(state_machine_t* const State_Machine) {
  655. network_t* const nm = (network_t *)State_Machine;
  656. network_handler_entry_print(State_Machine,true);
  657. network_start_stop_dhcp(nm->wifi_netif, true);
  658. network_wifi_connect(nm->event_parameters->ssid,nm->event_parameters->password);
  659. FREE_AND_NULL(nm->event_parameters->ssid);
  660. FREE_AND_NULL(nm->event_parameters->password);
  661. NETWORK_EXECUTE_CB(State_Machine);
  662. network_handler_entry_print(State_Machine,false);
  663. return EVENT_HANDLED;
  664. }
  665. static state_machine_result_t WIFI_CONFIGURING_CONNECT_STATE_handler(state_machine_t* const State_Machine) {
  666. HANDLE_GLOBAL_EVENT(State_Machine);
  667. network_handler_print(State_Machine,true);
  668. network_t* const nm = (network_t *)State_Machine;
  669. state_machine_result_t result = EVENT_HANDLED;
  670. switch (State_Machine->Event) {
  671. case EN_CONNECTED:
  672. result=EVENT_HANDLED;
  673. ESP_LOGD(TAG,"Wifi was connected. Waiting for IP address");
  674. network_set_timer(nm->dhcp_timeout,"DHCP Timeout");
  675. break;
  676. case EN_GOT_IP:
  677. network_status_update_ip_info(UPDATE_CONNECTION_OK);
  678. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_CONNECT_SUCCESS_STATE],__FUNCTION__);
  679. break;
  680. case EN_LOST_CONNECTION:
  681. network_status_update_ip_info(UPDATE_FAILED_ATTEMPT);
  682. result = local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_STATE],__FUNCTION__);
  683. break;
  684. default:
  685. result= EVENT_UN_HANDLED;
  686. }
  687. network_handler_print(State_Machine,false);
  688. return result;
  689. }
  690. static state_machine_result_t WIFI_CONFIGURING_CONNECT_STATE_exit_handler(state_machine_t* const State_Machine) {
  691. network_exit_handler_print(State_Machine,true);
  692. FREE_AND_NULL(((network_t *)State_Machine)->event_parameters->disconnected_event);
  693. network_set_timer(0,NULL);
  694. network_exit_handler_print(State_Machine,false);
  695. return EVENT_HANDLED;
  696. }
  697. /*********************************************************************************************
  698. * WIFI_CONFIGURING_CONNECT_SUCCESS_STATE
  699. */
  700. static state_machine_result_t WIFI_CONFIGURING_CONNECT_SUCCESS_STATE_entry_handler(state_machine_t* const State_Machine) {
  701. network_handler_entry_print(State_Machine,true);
  702. network_status_update_ip_info(UPDATE_CONNECTION_OK);
  703. ESP_LOGD(TAG, "Saving wifi configuration.");
  704. network_wifi_save_sta_config();
  705. NETWORK_EXECUTE_CB(State_Machine);
  706. network_handler_entry_print(State_Machine,false);
  707. return EVENT_HANDLED;
  708. }
  709. static state_machine_result_t WIFI_CONFIGURING_CONNECT_SUCCESS_STATE_handler(state_machine_t* const State_Machine) {
  710. network_handler_print(State_Machine,true);
  711. state_machine_result_t result = EVENT_HANDLED;
  712. switch (State_Machine->Event) {
  713. case EN_UPDATE_STATUS:
  714. network_status_update_basic_info();
  715. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_CONNECT_SUCCESS_GOTOSTA_STATE],__FUNCTION__);
  716. break;
  717. default:
  718. result= EVENT_UN_HANDLED;
  719. }
  720. // Process global handler at the end, since we want to overwrite
  721. // UPDATE_STATUS with our own logic above
  722. HANDLE_GLOBAL_EVENT(State_Machine);
  723. network_handler_print(State_Machine,false);
  724. return result;
  725. }
  726. static state_machine_result_t WIFI_CONFIGURING_CONNECT_SUCCESS_STATE_exit_handler(state_machine_t* const State_Machine) {
  727. network_exit_handler_print(State_Machine,true);
  728. network_exit_handler_print(State_Machine,false);
  729. return EVENT_HANDLED;
  730. }
  731. /*********************************************************************************************
  732. * WIFI_CONFIGURING_CONNECT_SUCCESS_GOTOSTA_STATE
  733. */
  734. static state_machine_result_t WIFI_CONFIGURING_CONNECT_SUCCESS_GOTOSTA_STATE_entry_handler(state_machine_t* const State_Machine) {
  735. network_handler_entry_print(State_Machine,true);
  736. ESP_LOGD(TAG, "Waiting for next status update event to turn off AP.");
  737. NETWORK_EXECUTE_CB(State_Machine);
  738. network_handler_entry_print(State_Machine,false);
  739. return EVENT_HANDLED;
  740. }
  741. static state_machine_result_t WIFI_CONFIGURING_CONNECT_SUCCESS_GOTOSTA_STATE_handler(state_machine_t* const State_Machine) {
  742. network_handler_print(State_Machine,true);
  743. state_machine_result_t result = EVENT_HANDLED;
  744. switch (State_Machine->Event) {
  745. case EN_UPDATE_STATUS:
  746. network_status_update_basic_info();
  747. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTED_STATE],__FUNCTION__);
  748. break;
  749. default:
  750. result= EVENT_UN_HANDLED;
  751. }
  752. // Process global handler at the end, since we want to overwrite
  753. // UPDATE_STATUS with our own logic above
  754. HANDLE_GLOBAL_EVENT(State_Machine);
  755. network_handler_print(State_Machine,false);
  756. return result;
  757. }
  758. static state_machine_result_t WIFI_CONFIGURING_CONNECT_SUCCESS_GOTOSTA_STATE_exit_handler(state_machine_t* const State_Machine) {
  759. network_exit_handler_print(State_Machine,true);
  760. network_exit_handler_print(State_Machine,false);
  761. return EVENT_HANDLED;
  762. }
  763. /*********************************************************************************************
  764. * WIFI_CONNECTING_STATE
  765. */
  766. static state_machine_result_t WIFI_CONNECTING_STATE_entry_handler(state_machine_t* const State_Machine) {
  767. network_t* const nm = (network_t *)State_Machine;
  768. network_handler_entry_print(State_Machine,true);
  769. network_start_stop_dhcp(nm->wifi_netif, true);
  770. network_connect_active_ssid(State_Machine);
  771. nm->STA_duration = nm->sta_polling_min_ms;
  772. /* create timer for background STA connection */
  773. network_set_timer(nm->STA_duration,"Wifi Polling timeout");
  774. NETWORK_EXECUTE_CB(State_Machine);
  775. network_handler_entry_print(State_Machine,false);
  776. return EVENT_HANDLED;
  777. }
  778. static state_machine_result_t WIFI_CONNECTING_STATE_handler(state_machine_t* const State_Machine) {
  779. HANDLE_GLOBAL_EVENT(State_Machine);
  780. state_machine_result_t result = EVENT_HANDLED;
  781. network_t* const nm = (network_t *)State_Machine;
  782. network_handler_print(State_Machine,true);
  783. switch (State_Machine->Event) {
  784. case EN_CONNECTED:
  785. // nothing to do here. Let's wait for IP address
  786. break;
  787. case EN_TIMER:
  788. // try connecting again.
  789. // todo: implement multi-ap logic
  790. ESP_LOGI(TAG, "Timer: %s ",STR_OR_ALT(nm->timer_tag,"Ethernet link not detected"));
  791. network_connect_active_ssid(State_Machine);
  792. break;
  793. default:
  794. result = EVENT_UN_HANDLED;
  795. }
  796. network_handler_print(State_Machine,false);
  797. return result;
  798. }
  799. static state_machine_result_t WIFI_CONNECTING_STATE_exit_handler(state_machine_t* const State_Machine) {
  800. network_exit_handler_print(State_Machine,true);
  801. network_exit_handler_print(State_Machine,false);
  802. return EVENT_HANDLED;
  803. }
  804. /*********************************************************************************************
  805. * WIFI_CONNECTING_NEW_STATE
  806. */
  807. static state_machine_result_t WIFI_CONNECTING_NEW_STATE_entry_handler(state_machine_t* const State_Machine) {
  808. network_t* const nm = (network_t *)State_Machine;
  809. network_handler_entry_print(State_Machine,true);
  810. network_start_stop_dhcp(nm->wifi_netif, true);
  811. network_wifi_connect(nm->event_parameters->ssid,nm->event_parameters->password);
  812. FREE_AND_NULL(nm->event_parameters->ssid);
  813. FREE_AND_NULL(nm->event_parameters->password);
  814. NETWORK_EXECUTE_CB(State_Machine);
  815. network_handler_entry_print(State_Machine,false);
  816. return EVENT_HANDLED;
  817. }
  818. static state_machine_result_t WIFI_CONNECTING_NEW_STATE_handler(state_machine_t* const State_Machine) {
  819. HANDLE_GLOBAL_EVENT(State_Machine);
  820. network_handler_print(State_Machine,true);
  821. state_machine_result_t result = EVENT_HANDLED;
  822. switch (State_Machine->Event) {
  823. case EN_GOT_IP:
  824. network_status_update_ip_info(UPDATE_CONNECTION_OK);
  825. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTED_STATE],__FUNCTION__);
  826. break;
  827. case EN_CONNECTED:
  828. ESP_LOGD(TAG,"Successfully connected to the new access point. Waiting for IP Address");
  829. result = EVENT_HANDLED;
  830. break;
  831. case EN_LOST_CONNECTION:
  832. if(((network_t *)State_Machine)->event_parameters->disconnected_event->reason == WIFI_REASON_ASSOC_LEAVE){
  833. ESP_LOGD(TAG,"Successfully disconnected from the existing access point. ");
  834. return EVENT_HANDLED;
  835. }
  836. ESP_LOGW(TAG,"Trying to connect failed");
  837. result = local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTING_NEW_FAILED_STATE],__FUNCTION__);
  838. break;
  839. default:
  840. result= EVENT_UN_HANDLED;
  841. }
  842. network_handler_print(State_Machine,false);
  843. return result;
  844. }
  845. static state_machine_result_t WIFI_CONNECTING_NEW_STATE_exit_handler(state_machine_t* const State_Machine) {
  846. network_exit_handler_print(State_Machine,true);
  847. network_set_timer(0,NULL);
  848. FREE_AND_NULL(((network_t *)State_Machine)->event_parameters->disconnected_event);
  849. network_exit_handler_print(State_Machine,false);
  850. return EVENT_HANDLED;
  851. }
  852. /*********************************************************************************************
  853. * WIFI_CONNECTING_NEW_FAILED_STATE
  854. */
  855. static state_machine_result_t WIFI_CONNECTING_NEW_FAILED_STATE_entry_handler(state_machine_t* const State_Machine) {
  856. network_t* const nm = (network_t *)State_Machine;
  857. network_handler_entry_print(State_Machine,true);
  858. if (nm->wifi_connected ) {
  859. // Wifi was already connected to an existing access point. Restore connection
  860. network_connect_active_ssid(State_Machine);
  861. }
  862. NETWORK_EXECUTE_CB(State_Machine);
  863. network_handler_entry_print(State_Machine,false);
  864. return EVENT_HANDLED;
  865. }
  866. static state_machine_result_t WIFI_CONNECTING_NEW_FAILED_STATE_handler(state_machine_t* const State_Machine) {
  867. HANDLE_GLOBAL_EVENT(State_Machine);
  868. network_handler_print(State_Machine,true);
  869. state_machine_result_t result = EVENT_HANDLED;
  870. switch (State_Machine->Event) {
  871. case EN_GOT_IP:
  872. network_status_update_ip_info(UPDATE_FAILED_ATTEMPT_AND_RESTORE);
  873. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_CONNECTED_STATE],__FUNCTION__);
  874. break;
  875. case EN_CONNECTED:
  876. ESP_LOGD(TAG,"Successfully connected to the previous access point. Waiting for IP Address");
  877. result = EVENT_HANDLED;
  878. break;
  879. case EN_LOST_CONNECTION:
  880. network_status_update_ip_info(UPDATE_FAILED_ATTEMPT);
  881. messaging_post_message(MESSAGING_ERROR, MESSAGING_CLASS_SYSTEM, "Unable to fall back to previous access point.");
  882. result = EVENT_HANDLED;
  883. break;
  884. default:
  885. result= EVENT_UN_HANDLED;
  886. }
  887. network_handler_print(State_Machine,false);
  888. return result;
  889. }
  890. static state_machine_result_t WIFI_CONNECTING_NEW_FAILED_STATE_exit_handler(state_machine_t* const State_Machine) {
  891. network_exit_handler_print(State_Machine,true);
  892. network_set_timer(0,NULL);
  893. FREE_AND_NULL(((network_t *)State_Machine)->event_parameters->disconnected_event);
  894. network_exit_handler_print(State_Machine,false);
  895. return EVENT_HANDLED;
  896. }
  897. /*********************************************************************************************
  898. * WIFI_CONNECTED_STATE
  899. */
  900. static state_machine_result_t WIFI_CONNECTED_STATE_entry_handler(state_machine_t* const State_Machine) {
  901. network_t* const nm = (network_t *)State_Machine;
  902. network_handler_entry_print(State_Machine,true);
  903. nm->last_connected = esp_timer_get_time();
  904. // cancel timeout pulse
  905. network_set_timer(0,NULL);
  906. ESP_LOGD(TAG, "Checking if wifi config changed.");
  907. if (network_wifi_sta_config_changed()) {
  908. ESP_LOGD(TAG, "Wifi Config changed. Saving it.");
  909. network_wifi_save_sta_config();
  910. }
  911. ESP_LOGD(TAG, "Updating the ip info json.");
  912. network_interface_coexistence(State_Machine);
  913. nm->wifi_connected = true;
  914. NETWORK_EXECUTE_CB(State_Machine);
  915. network_handler_entry_print(State_Machine,false);
  916. return EVENT_HANDLED;
  917. }
  918. static state_machine_result_t WIFI_CONNECTED_STATE_handler(state_machine_t* const State_Machine) {
  919. HANDLE_GLOBAL_EVENT(State_Machine);
  920. network_handler_print(State_Machine,true);
  921. state_machine_result_t result = EVENT_HANDLED;
  922. switch (State_Machine->Event) {
  923. case EN_LOST_CONNECTION:
  924. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_LOST_CONNECTION_STATE],__FUNCTION__);
  925. break;
  926. default:
  927. result = EVENT_UN_HANDLED;
  928. break;
  929. }
  930. network_handler_print(State_Machine,false);
  931. return result;
  932. }
  933. static state_machine_result_t WIFI_CONNECTED_STATE_exit_handler(state_machine_t* const State_Machine) {
  934. network_exit_handler_print(State_Machine,true);
  935. FREE_AND_NULL(((network_t *)State_Machine)->event_parameters->disconnected_event);
  936. network_exit_handler_print(State_Machine,false);
  937. return EVENT_HANDLED;
  938. }
  939. /*********************************************************************************************
  940. * WIFI_USER_DISCONNECTED_STATE
  941. */
  942. static state_machine_result_t WIFI_USER_DISCONNECTED_STATE_entry_handler(state_machine_t* const State_Machine) {
  943. network_handler_entry_print(State_Machine,true);
  944. ESP_LOGD(TAG, " WiFi disconnected by user");
  945. network_wifi_clear_config();
  946. network_status_update_ip_info(UPDATE_USER_DISCONNECT);
  947. NETWORK_EXECUTE_CB(State_Machine);
  948. network_handler_entry_print(State_Machine,false);
  949. return EVENT_HANDLED;
  950. }
  951. static state_machine_result_t WIFI_USER_DISCONNECTED_STATE_handler(state_machine_t* const State_Machine) {
  952. HANDLE_GLOBAL_EVENT(State_Machine);
  953. network_handler_print(State_Machine,true);
  954. state_machine_result_t result = EVENT_HANDLED;
  955. switch (State_Machine->Event) {
  956. case EN_LOST_CONNECTION:
  957. // this is a success! we're actually asking to disconnect
  958. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_STATE],__FUNCTION__);
  959. break;
  960. default:
  961. result= EVENT_UN_HANDLED;
  962. }
  963. network_handler_print(State_Machine,false);
  964. return result;
  965. }
  966. static state_machine_result_t WIFI_USER_DISCONNECTED_STATE_exit_handler(state_machine_t* const State_Machine) {
  967. network_exit_handler_print(State_Machine,true);
  968. network_exit_handler_print(State_Machine,false);
  969. return EVENT_HANDLED;
  970. }
  971. /*********************************************************************************************
  972. * WIFI_LOST_CONNECTION_STATE
  973. */
  974. static state_machine_result_t WIFI_LOST_CONNECTION_STATE_entry_handler(state_machine_t* const State_Machine) {
  975. network_t* const nm = (network_t *)State_Machine;
  976. network_handler_entry_print(State_Machine,true);
  977. ESP_LOGE(TAG, " WiFi Connection lost.");
  978. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "WiFi Connection lost");
  979. network_status_update_ip_info(UPDATE_LOST_CONNECTION);
  980. network_status_safe_reset_sta_ip_string();
  981. if (nm->last_connected > 0)
  982. nm->total_connected_time += ((esp_timer_get_time() - nm->last_connected) / (1000 * 1000));
  983. nm->last_connected = 0;
  984. nm->num_disconnect++;
  985. ESP_LOGW(TAG, " Wifi disconnected. Number of disconnects: %d, Average time connected: %d", nm->num_disconnect, nm->num_disconnect > 0 ? (nm->total_connected_time / nm->num_disconnect) : 0);
  986. if (nm->retries < WIFI_MANAGER_MAX_RETRY) {
  987. nm->retries++;
  988. ESP_LOGD(TAG, " Retrying connection connection, %d/%d.", nm->retries, WIFI_MANAGER_MAX_RETRY);
  989. network_connect_active_ssid( State_Machine);
  990. } else {
  991. /* In this scenario the connection was lost beyond repair */
  992. nm->retries = 0;
  993. ESP_LOGD(TAG,"Checking if ethernet interface is connected");
  994. if (network_is_interface_connected(nm->eth_netif)) {
  995. ESP_LOGW(TAG, "Cannot connect to Wifi. Falling back to Ethernet ");
  996. network_async(EN_ETHERNET_FALLBACK);
  997. } else {
  998. network_status_update_ip_info(UPDATE_LOST_CONNECTION);
  999. wifi_mode_t mode;
  1000. ESP_LOGW(TAG, " All connect retry attempts failed.");
  1001. /* put us in softAP mode first */
  1002. esp_wifi_get_mode(&mode);
  1003. if (WIFI_MODE_APSTA != mode) {
  1004. nm->STA_duration = nm->sta_polling_min_ms;
  1005. network_async_configure();
  1006. } else if (nm->STA_duration < nm->sta_polling_max_ms) {
  1007. nm->STA_duration *= 1.25;
  1008. }
  1009. /* keep polling for existing connection */
  1010. network_set_timer(nm->STA_duration, "Wifi Polling timeout");
  1011. ESP_LOGD(TAG, " STA search slow polling of %d", nm->STA_duration);
  1012. }
  1013. }
  1014. NETWORK_EXECUTE_CB(State_Machine);
  1015. network_handler_entry_print(State_Machine,false);
  1016. return EVENT_HANDLED;
  1017. }
  1018. static state_machine_result_t WIFI_LOST_CONNECTION_STATE_handler(state_machine_t* const State_Machine) {
  1019. HANDLE_GLOBAL_EVENT(State_Machine);
  1020. network_t* const nm = (network_t *)State_Machine;
  1021. state_machine_result_t result = EVENT_HANDLED;
  1022. network_handler_print(State_Machine,true);
  1023. switch (State_Machine->Event) {
  1024. case EN_CONFIGURE:
  1025. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONFIGURING_STATE],__FUNCTION__);
  1026. break;
  1027. case EN_TIMER:
  1028. ESP_LOGI(TAG, "Timer: %s ",STR_OR_ALT(nm->timer_tag,"Lost connection"));
  1029. result= local_traverse_state(State_Machine, &Wifi_Active_State[WIFI_LOST_CONNECTION_STATE],__FUNCTION__);
  1030. break;
  1031. case EN_CONNECT:
  1032. result= local_traverse_state(State_Machine, &Wifi_Configuring_State[WIFI_CONNECTING_STATE],__FUNCTION__);
  1033. break;
  1034. default:
  1035. result= EVENT_UN_HANDLED;
  1036. }
  1037. network_handler_print(State_Machine,false);
  1038. return result;
  1039. }
  1040. static state_machine_result_t WIFI_LOST_CONNECTION_STATE_exit_handler(state_machine_t* const State_Machine) {
  1041. network_exit_handler_print(State_Machine,true);
  1042. network_exit_handler_print(State_Machine,false);
  1043. return EVENT_HANDLED;
  1044. }
  1045. /*********************************************************************************************
  1046. * ETH_ACTIVE_CONNECTED_STATE
  1047. */
  1048. static state_machine_result_t ETH_ACTIVE_CONNECTED_STATE_entry_handler(state_machine_t* const State_Machine) {
  1049. network_t* const nm = (network_t *)State_Machine;
  1050. network_handler_entry_print(State_Machine,true);
  1051. network_status_update_ip_info(UPDATE_ETHERNET_CONNECTED);
  1052. nm->ethernet_connected = true;
  1053. // start a wifi Scan so web ui is populated with available entries
  1054. NETWORK_EXECUTE_CB(State_Machine);
  1055. network_handler_entry_print(State_Machine,false);
  1056. return EVENT_HANDLED;
  1057. }
  1058. static state_machine_result_t ETH_ACTIVE_CONNECTED_STATE_handler(state_machine_t* const State_Machine) {
  1059. HANDLE_GLOBAL_EVENT(State_Machine);
  1060. state_machine_result_t result = EVENT_HANDLED;
  1061. network_handler_print(State_Machine,true);
  1062. switch (State_Machine->Event) {
  1063. case EN_TIMER:
  1064. ESP_LOGD(TAG, "Ignoring ethernet link up timer check");
  1065. result= EVENT_HANDLED;
  1066. break;
  1067. default:
  1068. result= EVENT_UN_HANDLED;
  1069. }
  1070. network_handler_print(State_Machine,false);
  1071. return result;
  1072. }
  1073. static state_machine_result_t ETH_ACTIVE_CONNECTED_STATE_exit_handler(state_machine_t* const State_Machine) {
  1074. network_exit_handler_print(State_Machine,true);
  1075. network_exit_handler_print(State_Machine,false);
  1076. return EVENT_HANDLED;
  1077. }
  1078. static state_machine_result_t local_switch_state(state_machine_t* state_machine,
  1079. const state_t* const target_state, const char * caller) {
  1080. const state_t* source = state_machine->State;
  1081. NETWORK_PRINT_TRANSITION(true, "BEGIN SWITCH", ((network_t *)state_machine)->source_state, target_state, state_machine->Event, true,caller);
  1082. state_machine_result_t result = switch_state(state_machine, target_state);
  1083. NETWORK_PRINT_TRANSITION( false,"BEGIN SWITCH", ((network_t *)state_machine)->source_state, target_state, state_machine->Event, true,caller);
  1084. ((network_t *)state_machine)->source_state = source;
  1085. return result;
  1086. }
  1087. static state_machine_result_t local_traverse_state(state_machine_t* const state_machine,
  1088. const state_t* const target_state, const char * caller) {
  1089. const state_t * source = state_machine->State;
  1090. NETWORK_PRINT_TRANSITION( true,"BEGIN TRAVERSE", ((network_t *)state_machine)->source_state, target_state, state_machine->Event, true, caller);
  1091. state_machine_result_t result = traverse_state(state_machine, target_state);
  1092. NETWORK_PRINT_TRANSITION( false,"END TRAVERSE", ((network_t *)state_machine)->source_state, target_state, state_machine->Event, true,caller);
  1093. ((network_t *)state_machine)->source_state = source;
  1094. return result;
  1095. }
  1096. static void network_interface_coexistence(state_machine_t* state_machine) {
  1097. // this function is called whenever both wifi and ethernet are
  1098. // found to be active at the same time
  1099. network_t* nm = (network_t *)state_machine;
  1100. if (nm->wifi_connected && state_machine->Event == EN_ETH_GOT_IP) {
  1101. char* eth_reboot = config_alloc_get_default(NVS_TYPE_STR, "eth_boot", "N", 0);
  1102. network_prioritize_wifi(false);
  1103. if (strcasecmp(eth_reboot, "N")) {
  1104. ESP_LOGW(TAG, "Option eth_reboot set to reboot when ethernet is connected. Rebooting");
  1105. simple_restart();
  1106. } else {
  1107. ESP_LOGW(TAG, "Option eth_reboot set to not reboot when ethernet is connected. Using Wifi interface until next reboot");
  1108. }
  1109. FREE_AND_NULL(eth_reboot);
  1110. } else if (get_root(state_machine->State)->Id == NETWORK_ETH_ACTIVE_STATE){
  1111. messaging_post_message(MESSAGING_WARNING, MESSAGING_CLASS_SYSTEM, "Wifi Connected with Ethernet active. System reload needed");
  1112. simple_restart();
  1113. }
  1114. }