network_manager.h 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. /*
  2. *
  3. * Sebastien L. 2023, sle118@hotmail.com
  4. * Philippe G. 2023, philippe_44@outlook.com
  5. *
  6. * This software is released under the MIT License.
  7. * https://opensource.org/licenses/MIT
  8. *
  9. * License Overview:
  10. * ----------------
  11. * The MIT License is a permissive open source license. As a user of this software, you are free to:
  12. * - Use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of this software.
  13. * - Use the software for private, commercial, or any other purposes.
  14. *
  15. * Conditions:
  16. * - You must include the above copyright notice and this permission notice in all
  17. * copies or substantial portions of the Software.
  18. *
  19. * The MIT License offers a high degree of freedom and is well-suited for both open source and
  20. * commercial applications. It places minimal restrictions on how the software can be used,
  21. * modified, and redistributed. For more details on the MIT License, please refer to the link above.
  22. */
  23. #pragma once
  24. #include "esp_system.h"
  25. #include "esp_wifi.h"
  26. #include "esp_wifi_types.h"
  27. #include "squeezelite-ota.h"
  28. #include "esp_eth.h"
  29. #include "freertos/event_groups.h"
  30. #include "hsm.h"
  31. #include "esp_log.h"
  32. #include "network_services.h"
  33. #ifdef __cplusplus
  34. extern "C" {
  35. #endif
  36. //! List of oven events
  37. #define ALL_NM_EVENTS \
  38. ADD_FIRST_EVENT(EN_LINK_UP) \
  39. ADD_EVENT(EN_LINK_DOWN)\
  40. ADD_EVENT(EN_CONFIGURE)\
  41. ADD_EVENT(EN_GOT_IP)\
  42. ADD_EVENT(EN_ETH_GOT_IP)\
  43. ADD_EVENT(EN_DELETE)\
  44. ADD_EVENT(EN_REMOVE)\
  45. ADD_EVENT(EN_TIMER)\
  46. ADD_EVENT(EN_START)\
  47. ADD_EVENT(EN_SCAN)\
  48. ADD_EVENT(EN_FAIL)\
  49. ADD_EVENT(EN_SUCCESS)\
  50. ADD_EVENT(EN_SCAN_DONE)\
  51. ADD_EVENT(EN_CONNECT)\
  52. ADD_EVENT(EN_CONNECT_NEW)\
  53. ADD_EVENT(EN_ADD)\
  54. ADD_EVENT(EN_REBOOT)\
  55. ADD_EVENT(EN_REBOOT_URL)\
  56. ADD_EVENT(EN_LOST_CONNECTION)\
  57. ADD_EVENT(EN_ETHERNET_FALLBACK)\
  58. ADD_EVENT(EN_UPDATE_STATUS)\
  59. ADD_EVENT(EN_CONNECTED)\
  60. ADD_EVENT(EN_COMMIT_CHANGES)\
  61. ADD_EVENT(EN_EXECUTE_CALLBACK)
  62. #define ADD_EVENT(name) name,
  63. #define ADD_FIRST_EVENT(name) name=1,
  64. typedef enum {
  65. ALL_NM_EVENTS
  66. } network_event_t;
  67. #undef ADD_EVENT
  68. #undef ADD_FIRST_EVENT
  69. typedef enum {
  70. OTA,
  71. RECOVERY,
  72. RESTART,
  73. } reboot_type_t;
  74. typedef void (*network_manager_cb_t)(void * ctx);
  75. typedef esp_err_t (*network_manager_ret_cb_t)(void * ctx);
  76. typedef struct {
  77. void * ctx;
  78. network_manager_cb_t cb;
  79. network_manager_ret_cb_t ret_cb;
  80. } callback_ctx_t;
  81. typedef struct {
  82. char * ssid;
  83. char * password;
  84. } network_credentials_t;
  85. typedef struct {
  86. network_event_t trigger;
  87. union
  88. {
  89. reboot_type_t rtype;
  90. char* strval;
  91. wifi_event_sta_disconnected_t* disconnected_event;
  92. callback_ctx_t cb_ctx;
  93. network_credentials_t credentials;
  94. ip_event_got_ip_t* got_ip_event_data;
  95. } ctx;
  96. esp_netif_t *netif;
  97. } queue_message;
  98. typedef struct
  99. {
  100. state_machine_t Machine; //!< Abstract state machine
  101. const state_t* source_state;
  102. bool ethernet_connected;
  103. TimerHandle_t state_timer;
  104. uint32_t STA_duration;
  105. int32_t total_connected_time;
  106. int64_t last_connected;
  107. uint16_t num_disconnect;
  108. uint16_t retries;
  109. uint16_t initial_retries;
  110. bool wifi_connected;
  111. esp_netif_t *wifi_netif;
  112. esp_netif_t *eth_netif;
  113. esp_netif_t *wifi_ap_netif;
  114. uint16_t sta_polling_min_ms;
  115. uint16_t sta_polling_max_ms;
  116. uint16_t ap_duration_ms;
  117. uint16_t eth_link_down_reboot_ms;
  118. uint16_t dhcp_timeout;
  119. uint16_t wifi_dhcp_fail_ms;
  120. queue_message * event_parameters;
  121. char * timer_tag;
  122. } network_t;
  123. /*
  124. * --------------------- External function prototype ---------------------
  125. */
  126. void network_initialize_task();
  127. network_t * network_get_state_machine();
  128. void network_event_simple(network_event_t trigger);
  129. void network_event(network_event_t trigger, void* param);
  130. void network_async_event(network_event_t trigger, void* param);
  131. void network_async(network_event_t trigger);
  132. void network_async_front(network_event_t trigger);
  133. void network_async_fail();
  134. void network_async_success();
  135. void network_async_link_up();
  136. void network_async_link_down();
  137. void network_async_configure();
  138. void network_async_got_ip(network_event_t trigger,ip_event_got_ip_t*event_data);
  139. void network_async_timer();
  140. void network_async_add(const char* ssid, const char* password);
  141. void network_async_commit_protowrapper(void* protoWrapperBase);
  142. void network_async_scan();
  143. void network_async_scan_done();
  144. void network_async_connect(const char * ssid, const char * password);
  145. void network_async_delete_connection(const char * ssid);
  146. void network_async_lost_connection(wifi_event_sta_disconnected_t * disconnected_event);
  147. void network_async_reboot(reboot_type_t rtype);
  148. void network_reboot_ota(char* url);
  149. void network_async_delete();
  150. void network_async_update_status();
  151. void network_async_eth_got_ip();
  152. void network_ip_event_handler(void* arg, esp_event_base_t event_base, int32_t event_id, void* event_data);
  153. bool network_is_interface_connected(esp_netif_t * interface);
  154. void network_check_recovery_running();
  155. void network_async_callback(void * ctx, network_manager_cb_t cb);
  156. void network_async_callback_withret(void * ctx, network_manager_ret_cb_t cb);
  157. const char* network_event_to_string(network_event_t state);
  158. void network_initialize_state_machine_globals();
  159. /** @brief Defines the auth mode as an access point
  160. * Value must be of type wifi_auth_mode_t
  161. * @see esp_wifi_types.h
  162. * @warning if set to WIFI_AUTH_OPEN, passowrd me be empty. See DEFAULT_AP_PASSWORD.
  163. */
  164. #define AP_AUTHMODE WIFI_AUTH_WPA2_PSK
  165. /** @brief Defines visibility of the access point. 0: visible AP. 1: hidden */
  166. #define DEFAULT_AP_SSID_HIDDEN 0
  167. /** @brief Defines access point's name. Default value: esp32. Run 'make menuconfig' to setup your own value or replace here by a string */
  168. #define DEFAULT_AP_SSID CONFIG_DEFAULT_AP_SSID
  169. /** @brief Defines access point's bandwidth.
  170. * Value: WIFI_BW_HT20 for 20 MHz or WIFI_BW_HT40 for 40 MHz
  171. * 20 MHz minimize channel interference but is not suitable for
  172. * applications with high data speeds
  173. */
  174. #define DEFAULT_AP_BANDWIDTH WIFI_BW_HT20
  175. #define DEFAULT_AP_CHANNEL CONFIG_DEFAULT_AP_CHANNEL
  176. #define DEFAULT_AP_GATEWAY CONFIG_DEFAULT_AP_GATEWAY
  177. #define DEFAULT_AP_NETMASK CONFIG_DEFAULT_AP_NETMASK
  178. void network_reboot_ota(char * url);
  179. /**
  180. * @brief simplified reason codes for a lost connection.
  181. *
  182. * esp-idf maintains a big list of reason codes which in practice are useless for most typical application.
  183. * UPDATE_CONNECTION_OK - Web UI expects this when attempting to connect to a new access point succeeds
  184. * UPDATE_FAILED_ATTEMPT - Web UI expects this when attempting to connect to a new access point fails
  185. * UPDATE_USER_DISCONNECT = 2,
  186. * UPDATE_LOST_CONNECTION = 3,
  187. * UPDATE_FAILED_ATTEMPT_AND_RESTORE - Web UI expects this when attempting to connect to a new access point fails and previous connection is restored
  188. * UPDATE_ETHERNET_CONNECTED = 5
  189. */
  190. /**
  191. * Frees up all memory allocated by the wifi_manager and kill the task.
  192. */
  193. void network_destroy();
  194. /**
  195. * Filters the AP scan list to unique SSIDs
  196. */
  197. void filter_unique( wifi_ap_record_t * aplist, uint16_t * ap_num);
  198. /**
  199. * @brief A standard wifi event handler as recommended by Espressif
  200. */
  201. esp_err_t network_manager_event_handler(void *ctx, system_event_t *event);
  202. /**
  203. * @brief Start the mDNS service
  204. */
  205. void network_manager_initialise_mdns();
  206. /**
  207. * @brief Register a callback to a custom function when specific network manager states are reached.
  208. */
  209. bool network_is_wifi_prioritized();
  210. void network_set_timer(uint16_t duration, const char * tag);
  211. void network_set_hostname(esp_netif_t * netif);
  212. esp_err_t network_get_ip_info_for_netif(esp_netif_t* netif, tcpip_adapter_ip_info_t* ipInfo);
  213. void network_start_stop_dhcp_client(esp_netif_t* netif, bool start);
  214. void network_start_stop_dhcps(esp_netif_t* netif, bool start);
  215. void network_prioritize_wifi(bool activate);
  216. #define ADD_ROOT_FORWARD_DECLARATION(name, ...) ADD_STATE_FORWARD_DECLARATION_(name)
  217. #define ADD_ROOT_LEAF_FORWARD_DECLARATION(name, ...) ADD_STATE_FORWARD_DECLARATION_(name)
  218. #define ADD_LEAF_FORWARD_DECLARATION(name, ...) ADD_STATE_FORWARD_DECLARATION_(name)
  219. #define ADD_STATE_FORWARD_DECLARATION_(name) \
  220. static state_machine_result_t name##_handler(state_machine_t* const State_Machine); \
  221. static state_machine_result_t name##_entry_handler(state_machine_t* const State_Machine); \
  222. static state_machine_result_t name##_exit_handler(state_machine_t* const State_Machine);
  223. void initialize_network_handlers(state_machine_t* state_machine);
  224. void network_manager_format_from_to_states(esp_log_level_t level, const char* prefix, state_t const * from_state, state_t const* current_state, network_event_t event,bool show_source, const char * caller );
  225. void network_manager_format_state_machine(esp_log_level_t level, const char* prefix, state_machine_t * state_machine, bool show_source, const char * caller) ;
  226. #if defined(LOG_LOCAL_LEVEL)
  227. #if LOG_LOCAL_LEVEL >=5
  228. #define NETWORK_PRINT_TRANSITION(begin, prefix, source,target, event, print_source,caller ) network_manager_format_from_to_states(ESP_LOG_VERBOSE, prefix, source,target, event, print_source,caller )
  229. #define NETWORK_DEBUG_STATE_MACHINE(begin, cb_prefix,state_machine,print_from,caller) network_manager_format_state_machine(ESP_LOG_DEBUG,cb_prefix,state_machine,print_from,caller)
  230. #define NETWORK_EXECUTE_CB(mch) network_execute_cb(mch,__FUNCTION__);
  231. #define network_handler_entry_print(State_Machine, begin) network_manager_format_state_machine(ESP_LOG_DEBUG,begin?"ENTRY START":"ENTRY END",State_Machine,false,__FUNCTION__)
  232. #define network_exit_handler_print(State_Machine, begin) network_manager_format_state_machine(ESP_LOG_DEBUG,begin?"EXIT START":"END END",State_Machine,false,__FUNCTION__)
  233. #define network_handler_print(State_Machine, begin) network_manager_format_state_machine(ESP_LOG_DEBUG,begin?"HANDLER START":"HANDLER END",State_Machine,false,__FUNCTION__)
  234. #elif LOG_LOCAL_LEVEL >= 4
  235. #define network_handler_entry_print(State_Machine, begin) if(begin) network_manager_format_state_machine(ESP_LOG_DEBUG,begin?"BEGIN ENTRY":"END ENTRY",State_Machine,false,"")
  236. #define network_exit_handler_print(State_Machine, begin) if(begin) network_manager_format_state_machine(ESP_LOG_DEBUG,begin?"BEGIN EXIT":"END EXIT",State_Machine,false,"")
  237. #define network_handler_print(State_Machine, begin) if(begin) network_manager_format_state_machine(ESP_LOG_DEBUG,begin?"HANDLER START":"HANDLER END",State_Machine,false,"")
  238. #define NETWORK_PRINT_TRANSITION(begin, prefix, source,target, event, print_source,caller ) if(begin) network_manager_format_from_to_states(ESP_LOG_DEBUG, prefix, source,target, event, print_source,caller )
  239. #define NETWORK_EXECUTE_CB(mch) network_execute_cb(mch,__FUNCTION__);
  240. #define NETWORK_DEBUG_STATE_MACHINE(begin, cb_prefix,state_machine,print_from,caller) if(begin) network_manager_format_state_machine(ESP_LOG_DEBUG,cb_prefix,state_machine,print_from,caller)
  241. #endif
  242. #endif
  243. #ifndef NETWORK_PRINT_TRANSITION
  244. #define network_exit_handler_print(nm, begin)
  245. #define network_handler_entry_print(State_Machine, begin)
  246. #define network_handler_print(State_Machine, begin)
  247. #define NETWORK_EXECUTE_CB(mch) network_execute_cb(mch,NULL)
  248. #define NETWORK_PRINT_TRANSITION(begin, prefix, source,target, event, print_source,caller )
  249. #define NETWORK_DEBUG_STATE_MACHINE(begin, cb_prefix,state_machine,print_from,caller)
  250. #endif
  251. #ifdef __cplusplus
  252. }
  253. #endif