mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
Clean up wifi provisioning in the esp_event thread
Signed-off-by: Laukik Hase <laukik.hase@espressif.com>
This commit is contained in:
parent
885e501d99
commit
c9ce87a1a9
@ -36,6 +36,7 @@
|
||||
static const char *TAG = "wifi_prov_mgr";
|
||||
|
||||
ESP_EVENT_DEFINE_BASE(WIFI_PROV_EVENT);
|
||||
ESP_EVENT_DEFINE_BASE(WIFI_PROV_MGR_PVT_EVENT);
|
||||
|
||||
typedef enum {
|
||||
WIFI_PROV_STATE_IDLE,
|
||||
@ -47,6 +48,10 @@ typedef enum {
|
||||
WIFI_PROV_STATE_STOPPING
|
||||
} wifi_prov_mgr_state_t;
|
||||
|
||||
typedef enum {
|
||||
WIFI_PROV_MGR_STOP,
|
||||
} wifi_prov_mgr_pvt_event_t;
|
||||
|
||||
/**
|
||||
* @brief Structure for storing capabilities supported by
|
||||
* the provisioning service
|
||||
@ -100,6 +105,9 @@ struct wifi_prov_mgr_ctx {
|
||||
/* Handle for delayed Wi-Fi connection timer */
|
||||
esp_timer_handle_t wifi_connect_timer;
|
||||
|
||||
/* Handle for delayed cleanup timer */
|
||||
esp_timer_handle_t cleanup_delay_timer;
|
||||
|
||||
/* State of Wi-Fi Station */
|
||||
wifi_prov_sta_state_t wifi_state;
|
||||
|
||||
@ -411,6 +419,22 @@ static esp_err_t wifi_prov_mgr_start_service(const char *service_name, const cha
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = esp_event_handler_register(WIFI_PROV_MGR_PVT_EVENT, WIFI_PROV_MGR_STOP,
|
||||
wifi_prov_mgr_event_handler_internal, NULL);
|
||||
if (ret != ESP_OK) {
|
||||
ESP_LOGE(TAG, "Failed to register provisioning event handler");
|
||||
esp_event_handler_unregister(WIFI_EVENT, ESP_EVENT_ANY_ID,
|
||||
wifi_prov_mgr_event_handler_internal);
|
||||
esp_event_handler_unregister(IP_EVENT, IP_EVENT_STA_GOT_IP,
|
||||
wifi_prov_mgr_event_handler_internal);
|
||||
free(prov_ctx->wifi_scan_handlers);
|
||||
free(prov_ctx->wifi_prov_handlers);
|
||||
scheme->prov_stop(prov_ctx->pc);
|
||||
protocomm_delete(prov_ctx->pc);
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
||||
ESP_LOGI(TAG, "Provisioning started with service name : %s ",
|
||||
service_name ? service_name : "<NULL>");
|
||||
return ESP_OK;
|
||||
@ -480,9 +504,16 @@ void wifi_prov_mgr_endpoint_unregister(const char *ep_name)
|
||||
RELEASE_LOCK(prov_ctx_lock);
|
||||
}
|
||||
|
||||
static void prov_stop_task(void *arg)
|
||||
static void prov_stop_and_notify(bool is_async)
|
||||
{
|
||||
bool is_this_a_task = (bool) arg;
|
||||
esp_event_handler_unregister(WIFI_PROV_MGR_PVT_EVENT, WIFI_PROV_MGR_STOP,
|
||||
wifi_prov_mgr_event_handler_internal);
|
||||
|
||||
if (prov_ctx->cleanup_delay_timer) {
|
||||
esp_timer_stop(prov_ctx->cleanup_delay_timer);
|
||||
esp_timer_delete(prov_ctx->cleanup_delay_timer);
|
||||
prov_ctx->cleanup_delay_timer = NULL;
|
||||
}
|
||||
|
||||
wifi_prov_cb_func_t app_cb = prov_ctx->mgr_config.app_event_handler.event_cb;
|
||||
void *app_data = prov_ctx->mgr_config.app_event_handler.user_data;
|
||||
@ -492,8 +523,11 @@ static void prov_stop_task(void *arg)
|
||||
|
||||
/* This delay is so that the client side app is notified first
|
||||
* and then the provisioning is stopped. Generally 1000ms is enough. */
|
||||
uint32_t cleanup_delay = prov_ctx->cleanup_delay > 100 ? prov_ctx->cleanup_delay : 100;
|
||||
vTaskDelay(cleanup_delay / portTICK_PERIOD_MS);
|
||||
if (!is_async)
|
||||
{
|
||||
uint32_t cleanup_delay = prov_ctx->cleanup_delay > 100 ? prov_ctx->cleanup_delay : 100;
|
||||
vTaskDelay(cleanup_delay / portTICK_PERIOD_MS);
|
||||
}
|
||||
|
||||
/* All the extra application added endpoints are also
|
||||
* removed automatically when prov_stop is called */
|
||||
@ -517,10 +551,11 @@ static void prov_stop_task(void *arg)
|
||||
esp_wifi_set_mode(WIFI_MODE_STA);
|
||||
ESP_LOGI(TAG, "Provisioning stopped");
|
||||
|
||||
if (is_this_a_task) {
|
||||
ACQUIRE_LOCK(prov_ctx_lock);
|
||||
if (is_async) {
|
||||
/* NOTE: While calling this API in an async fashion,
|
||||
* the context lock prov_ctx_lock has already been taken
|
||||
*/
|
||||
prov_ctx->prov_state = WIFI_PROV_STATE_IDLE;
|
||||
RELEASE_LOCK(prov_ctx_lock);
|
||||
|
||||
ESP_LOGD(TAG, "execute_event_cb : %d", WIFI_PROV_END);
|
||||
if (scheme_cb) {
|
||||
@ -532,13 +567,11 @@ static void prov_stop_task(void *arg)
|
||||
if (esp_event_post(WIFI_PROV_EVENT, WIFI_PROV_END, NULL, 0, portMAX_DELAY) != ESP_OK) {
|
||||
ESP_LOGE(TAG, "Failed to post event WIFI_PROV_END");
|
||||
}
|
||||
|
||||
vTaskDelete(NULL);
|
||||
}
|
||||
}
|
||||
|
||||
/* This will do one of these:
|
||||
* 1) if blocking is false, start a task for stopping the provisioning service (returns true)
|
||||
* 1) if blocking is false, start a cleanup timer for stopping the provisioning service (returns true)
|
||||
* 2) if blocking is true, stop provisioning service immediately (returns true)
|
||||
* 3) if service was already in the process of termination, in blocking mode this will
|
||||
* wait till the service is stopped (returns false)
|
||||
@ -630,20 +663,18 @@ static bool wifi_prov_mgr_stop_service(bool blocking)
|
||||
/* Run the cleanup without launching a separate task. Also the
|
||||
* WIFI_PROV_END event is not emitted in this case */
|
||||
RELEASE_LOCK(prov_ctx_lock);
|
||||
prov_stop_task((void *)0);
|
||||
prov_stop_and_notify(false);
|
||||
ACQUIRE_LOCK(prov_ctx_lock);
|
||||
prov_ctx->prov_state = WIFI_PROV_STATE_IDLE;
|
||||
} else {
|
||||
/* Launch cleanup task to perform the cleanup asynchronously.
|
||||
/* Launch cleanup timer to perform the cleanup asynchronously.
|
||||
* It is important to do this asynchronously because, there are
|
||||
* situations in which the transport level resources have to be
|
||||
* released - some duration after - returning from a call to
|
||||
* wifi_prov_mgr_stop_provisioning(), like when it is called
|
||||
* inside a protocomm handler */
|
||||
if (xTaskCreate(prov_stop_task, "prov_stop_task", 4096, (void *)1, tskIDLE_PRIORITY, NULL) != pdPASS) {
|
||||
ESP_LOGE(TAG, "Failed to create prov_stop_task!");
|
||||
abort();
|
||||
}
|
||||
uint64_t cleanup_delay_ms = prov_ctx->cleanup_delay > 100 ? prov_ctx->cleanup_delay : 100;
|
||||
esp_timer_start_once(prov_ctx->cleanup_delay_timer, cleanup_delay_ms * 1000U);
|
||||
ESP_LOGD(TAG, "Provisioning scheduled for stopping");
|
||||
}
|
||||
return true;
|
||||
@ -655,6 +686,14 @@ static void stop_prov_timer_cb(void *arg)
|
||||
wifi_prov_mgr_stop_provisioning();
|
||||
}
|
||||
|
||||
static void cleanup_delay_timer_cb(void *arg)
|
||||
{
|
||||
esp_err_t ret = esp_event_post(WIFI_PROV_MGR_PVT_EVENT, WIFI_PROV_MGR_STOP, NULL, 0, pdMS_TO_TICKS(100));
|
||||
if (ret != ESP_OK) {
|
||||
ESP_LOGE(TAG, "Failed to post WIFI_PROV_MGR_STOP event! %d %s", ret, esp_err_to_name(ret));
|
||||
}
|
||||
}
|
||||
|
||||
esp_err_t wifi_prov_mgr_disable_auto_stop(uint32_t cleanup_delay)
|
||||
{
|
||||
if (!prov_ctx_lock) {
|
||||
@ -821,6 +860,7 @@ static void wifi_prov_mgr_event_handler_internal(
|
||||
ESP_LOGE(TAG, "Provisioning manager not initialized");
|
||||
return;
|
||||
}
|
||||
|
||||
ACQUIRE_LOCK(prov_ctx_lock);
|
||||
|
||||
/* If pointer to provisioning application data is NULL
|
||||
@ -905,6 +945,8 @@ static void wifi_prov_mgr_event_handler_internal(
|
||||
/* Execute user registered callback handler */
|
||||
execute_event_cb(WIFI_PROV_CRED_FAIL, (void *)&reason, sizeof(reason));
|
||||
}
|
||||
} else if (event_base == WIFI_PROV_MGR_PVT_EVENT && event_id == WIFI_PROV_MGR_STOP) {
|
||||
prov_stop_and_notify(true);
|
||||
}
|
||||
|
||||
RELEASE_LOCK(prov_ctx_lock);
|
||||
@ -1544,6 +1586,21 @@ esp_err_t wifi_prov_mgr_start_provisioning(wifi_prov_security_t security, const
|
||||
}
|
||||
}
|
||||
|
||||
esp_timer_create_args_t cleanup_delay_timer = {
|
||||
.callback = cleanup_delay_timer_cb,
|
||||
.arg = NULL,
|
||||
.dispatch_method = ESP_TIMER_TASK,
|
||||
.name = "cleanup_delay_tm"
|
||||
};
|
||||
ret = esp_timer_create(&cleanup_delay_timer, &prov_ctx->cleanup_delay_timer);
|
||||
if (ret != ESP_OK) {
|
||||
ESP_LOGE(TAG, "Failed to create cleanup delay timer");
|
||||
esp_timer_delete(prov_ctx->wifi_connect_timer);
|
||||
esp_timer_delete(prov_ctx->autostop_timer);
|
||||
goto err;
|
||||
}
|
||||
|
||||
|
||||
/* System APIs for BLE / Wi-Fi will be called inside wifi_prov_mgr_start_service(),
|
||||
* which may trigger system level events. Hence, releasing the context lock will
|
||||
* ensure that wifi_prov_mgr_event_handler() doesn't block the global event_loop
|
||||
@ -1555,6 +1612,7 @@ esp_err_t wifi_prov_mgr_start_provisioning(wifi_prov_security_t security, const
|
||||
if (ret != ESP_OK) {
|
||||
esp_timer_delete(prov_ctx->autostop_timer);
|
||||
esp_timer_delete(prov_ctx->wifi_connect_timer);
|
||||
esp_timer_delete(prov_ctx->cleanup_delay_timer);
|
||||
}
|
||||
ACQUIRE_LOCK(prov_ctx_lock);
|
||||
if (ret == ESP_OK) {
|
||||
|
Loading…
x
Reference in New Issue
Block a user