2018-07-30 21:40:10 +05:30
|
|
|
/* BLE based Provisioning Example
|
|
|
|
|
|
|
|
This example code is in the Public Domain (or CC0 licensed, at your option.)
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, this
|
|
|
|
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
|
|
|
CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <freertos/FreeRTOS.h>
|
|
|
|
#include <freertos/task.h>
|
|
|
|
#include <esp_system.h>
|
|
|
|
#include <esp_wifi.h>
|
2018-12-25 23:08:50 +05:30
|
|
|
#include <esp_event.h>
|
2018-07-30 21:40:10 +05:30
|
|
|
#include <esp_log.h>
|
|
|
|
#include <nvs_flash.h>
|
|
|
|
|
|
|
|
#include <lwip/err.h>
|
|
|
|
#include <lwip/sys.h>
|
|
|
|
|
|
|
|
#include "app_prov.h"
|
2021-02-18 18:45:21 +08:00
|
|
|
#include "qrcode.h"
|
2018-07-30 21:40:10 +05:30
|
|
|
|
2019-05-09 16:43:06 +02:00
|
|
|
#define EXAMPLE_AP_RECONN_ATTEMPTS CONFIG_EXAMPLE_AP_RECONN_ATTEMPTS
|
2021-02-18 18:45:21 +08:00
|
|
|
#define PROV_QR_VERSION "v1"
|
|
|
|
#define PROV_TRANSPORT_BLE "ble"
|
|
|
|
#define QRCODE_BASE_URL "https://espressif.github.io/esp-jumpstart/qrcode.html"
|
2018-12-25 23:08:50 +05:30
|
|
|
|
2018-07-30 21:40:10 +05:30
|
|
|
static const char *TAG = "app";
|
|
|
|
|
2019-07-16 16:33:30 +07:00
|
|
|
static void start_ble_provisioning(void);
|
2018-12-25 23:08:50 +05:30
|
|
|
|
|
|
|
static void event_handler(void* arg, esp_event_base_t event_base,
|
2021-01-11 13:24:00 +05:30
|
|
|
int32_t event_id, void* event_data)
|
2018-07-30 21:40:10 +05:30
|
|
|
{
|
2018-12-25 23:08:50 +05:30
|
|
|
static int s_retry_num_ap_not_found = 0;
|
|
|
|
static int s_retry_num_ap_auth_fail = 0;
|
|
|
|
|
|
|
|
if (event_base == WIFI_EVENT && event_id == WIFI_EVENT_STA_START) {
|
2018-07-30 21:40:10 +05:30
|
|
|
esp_wifi_connect();
|
2018-12-25 23:08:50 +05:30
|
|
|
} else if (event_base == WIFI_EVENT && event_id == WIFI_EVENT_STA_DISCONNECTED) {
|
|
|
|
wifi_event_sta_disconnected_t* disconnected = (wifi_event_sta_disconnected_t*) event_data;
|
|
|
|
switch (disconnected->reason) {
|
|
|
|
case WIFI_REASON_AUTH_EXPIRE:
|
|
|
|
case WIFI_REASON_4WAY_HANDSHAKE_TIMEOUT:
|
|
|
|
case WIFI_REASON_BEACON_TIMEOUT:
|
|
|
|
case WIFI_REASON_AUTH_FAIL:
|
|
|
|
case WIFI_REASON_ASSOC_FAIL:
|
|
|
|
case WIFI_REASON_HANDSHAKE_TIMEOUT:
|
|
|
|
ESP_LOGW(TAG, "connect to the AP fail : auth Error");
|
|
|
|
if (s_retry_num_ap_auth_fail < EXAMPLE_AP_RECONN_ATTEMPTS) {
|
|
|
|
s_retry_num_ap_auth_fail++;
|
|
|
|
esp_wifi_connect();
|
|
|
|
ESP_LOGI(TAG, "retry connecting to the AP...");
|
|
|
|
} else {
|
|
|
|
/* Restart provisioning if authentication fails */
|
|
|
|
start_ble_provisioning();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case WIFI_REASON_NO_AP_FOUND:
|
|
|
|
ESP_LOGW(TAG, "connect to the AP fail : not found");
|
|
|
|
if (s_retry_num_ap_not_found < EXAMPLE_AP_RECONN_ATTEMPTS) {
|
|
|
|
s_retry_num_ap_not_found++;
|
|
|
|
esp_wifi_connect();
|
|
|
|
ESP_LOGI(TAG, "retry to connecting to the AP...");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* None of the expected reasons */
|
|
|
|
esp_wifi_connect();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (event_base == IP_EVENT && event_id == IP_EVENT_STA_GOT_IP) {
|
|
|
|
ip_event_got_ip_t* event = (ip_event_got_ip_t*) event_data;
|
2019-08-20 14:40:34 +02:00
|
|
|
ESP_LOGI(TAG, "got ip:" IPSTR, IP2STR(&event->ip_info.ip));
|
2018-12-25 23:08:50 +05:30
|
|
|
s_retry_num_ap_not_found = 0;
|
|
|
|
s_retry_num_ap_auth_fail = 0;
|
2018-07-30 21:40:10 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-16 16:33:30 +07:00
|
|
|
static void wifi_init_sta(void)
|
2018-07-30 21:40:10 +05:30
|
|
|
{
|
2018-12-25 23:08:50 +05:30
|
|
|
/* Set our event handling */
|
|
|
|
ESP_ERROR_CHECK(esp_event_handler_register(WIFI_EVENT, ESP_EVENT_ANY_ID, event_handler, NULL));
|
|
|
|
ESP_ERROR_CHECK(esp_event_handler_register(IP_EVENT, IP_EVENT_STA_GOT_IP, event_handler, NULL));
|
|
|
|
|
2019-04-11 17:50:19 +05:30
|
|
|
/* Start Wi-Fi in station mode with credentials set during provisioning */
|
2018-12-25 23:08:50 +05:30
|
|
|
ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA));
|
|
|
|
ESP_ERROR_CHECK(esp_wifi_start());
|
2018-07-30 21:40:10 +05:30
|
|
|
}
|
|
|
|
|
2019-07-16 16:33:30 +07:00
|
|
|
static void start_ble_provisioning(void)
|
2018-07-30 21:40:10 +05:30
|
|
|
{
|
|
|
|
/* Security version */
|
|
|
|
int security = 0;
|
|
|
|
/* Proof of possession */
|
|
|
|
const protocomm_security_pop_t *pop = NULL;
|
|
|
|
|
2019-05-09 16:43:06 +02:00
|
|
|
#ifdef CONFIG_EXAMPLE_USE_SEC_1
|
2018-07-30 21:40:10 +05:30
|
|
|
security = 1;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Having proof of possession is optional */
|
2019-05-09 16:43:06 +02:00
|
|
|
#ifdef CONFIG_EXAMPLE_USE_POP
|
2018-07-30 21:40:10 +05:30
|
|
|
const static protocomm_security_pop_t app_pop = {
|
2019-05-09 16:43:06 +02:00
|
|
|
.data = (uint8_t *) CONFIG_EXAMPLE_POP,
|
|
|
|
.len = (sizeof(CONFIG_EXAMPLE_POP)-1)
|
2018-07-30 21:40:10 +05:30
|
|
|
};
|
|
|
|
pop = &app_pop;
|
|
|
|
#endif
|
|
|
|
|
2018-12-25 23:08:50 +05:30
|
|
|
ESP_ERROR_CHECK(app_prov_start_ble_provisioning(security, pop));
|
|
|
|
}
|
|
|
|
|
2021-02-18 18:45:21 +08:00
|
|
|
static void get_device_service_name(char *service_name, size_t max)
|
|
|
|
{
|
|
|
|
uint8_t eth_mac[6];
|
|
|
|
const char *ssid_prefix = "PROV_";
|
|
|
|
esp_wifi_get_mac(WIFI_IF_STA, eth_mac);
|
|
|
|
snprintf(service_name, max, "%s%02X%02X%02X",
|
|
|
|
ssid_prefix, eth_mac[3], eth_mac[4], eth_mac[5]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ble_prov_print_qr(void)
|
|
|
|
{
|
|
|
|
char payload[150] = {0};
|
|
|
|
char name[12] = {0};
|
|
|
|
char *pop = NULL;
|
|
|
|
#ifdef CONFIG_EXAMPLE_USE_POP
|
|
|
|
pop = CONFIG_EXAMPLE_POP;
|
|
|
|
#endif
|
|
|
|
get_device_service_name(name, sizeof(name));
|
|
|
|
if (pop) {
|
|
|
|
snprintf(payload, sizeof(payload), "{\"ver\":\"%s\",\"name\":\"%s\"" \
|
|
|
|
",\"pop\":\"%s\",\"transport\":\"%s\"}",
|
|
|
|
PROV_QR_VERSION, name, pop, PROV_TRANSPORT_BLE);
|
|
|
|
} else {
|
|
|
|
snprintf(payload, sizeof(payload), "{\"ver\":\"%s\",\"name\":\"%s\"" \
|
|
|
|
",\"transport\":\"%s\"}",
|
|
|
|
PROV_QR_VERSION, name, PROV_TRANSPORT_BLE);
|
|
|
|
}
|
|
|
|
#ifdef CONFIG_EXAMPLE_PROV_SHOW_QR
|
|
|
|
ESP_LOGI(TAG, "Scan this QR code from the provisioning application for Provisioning.");
|
|
|
|
esp_qrcode_config_t cfg = ESP_QRCODE_CONFIG_DEFAULT();
|
|
|
|
esp_qrcode_generate(&cfg, payload);
|
|
|
|
#endif /* CONFIG_APP_WIFI_PROV_SHOW_QR */
|
|
|
|
ESP_LOGI(TAG, "If QR code is not visible, copy paste the below URL in a browser.\n%s?data=%s", QRCODE_BASE_URL, payload);
|
|
|
|
}
|
|
|
|
|
2019-07-16 16:33:30 +07:00
|
|
|
void app_main(void)
|
2018-12-25 23:08:50 +05:30
|
|
|
{
|
2018-07-30 21:40:10 +05:30
|
|
|
/* Initialize networking stack */
|
2019-11-29 10:54:02 +01:00
|
|
|
ESP_ERROR_CHECK(esp_netif_init());
|
2018-07-30 21:40:10 +05:30
|
|
|
|
2018-12-25 23:08:50 +05:30
|
|
|
/* Create default event loop needed by the
|
|
|
|
* main app and the provisioning service */
|
|
|
|
ESP_ERROR_CHECK(esp_event_loop_create_default());
|
2018-07-30 21:40:10 +05:30
|
|
|
|
2019-04-11 17:50:19 +05:30
|
|
|
/* Initialize NVS needed by Wi-Fi */
|
|
|
|
ESP_ERROR_CHECK(nvs_flash_init());
|
|
|
|
|
2019-08-31 16:24:45 +02:00
|
|
|
/* Initialize Wi-Fi including netif with default config */
|
|
|
|
esp_netif_create_default_wifi_sta();
|
2019-04-11 17:50:19 +05:30
|
|
|
wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
|
|
|
|
ESP_ERROR_CHECK(esp_wifi_init(&cfg));
|
|
|
|
|
2018-07-30 21:40:10 +05:30
|
|
|
/* Check if device is provisioned */
|
|
|
|
bool provisioned;
|
|
|
|
if (app_prov_is_provisioned(&provisioned) != ESP_OK) {
|
|
|
|
ESP_LOGE(TAG, "Error getting device provisioning state");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (provisioned == false) {
|
|
|
|
/* If not provisioned, start provisioning via BLE */
|
|
|
|
ESP_LOGI(TAG, "Starting BLE provisioning");
|
2018-12-25 23:08:50 +05:30
|
|
|
start_ble_provisioning();
|
2021-02-18 18:45:21 +08:00
|
|
|
ble_prov_print_qr();
|
2018-07-30 21:40:10 +05:30
|
|
|
} else {
|
|
|
|
/* Else start as station with credentials set during provisioning */
|
|
|
|
ESP_LOGI(TAG, "Starting WiFi station");
|
2019-04-11 17:50:19 +05:30
|
|
|
wifi_init_sta();
|
2018-07-30 21:40:10 +05:30
|
|
|
}
|
|
|
|
}
|