mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
2db6b1578e
1. Use flag ESP32_WIFI_ENABLE_WPA3_SAE to control WPA3 code, disabling it code footprint reduces by 7.7kB in libwpa_supplicant.a 2. Fix handling of multiple AP credentials in WPS, apps need update to handle the new event for the fix to work
176 lines
6.5 KiB
C
176 lines
6.5 KiB
C
/* WiFi Connection Example using WPS
|
|
|
|
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.
|
|
*/
|
|
|
|
/*
|
|
This example demonstrates how to use WPS.
|
|
It supports two modes, which can be selected in menuconfig.
|
|
|
|
WPS_TYPE_PBC:
|
|
Start ESP32 and it will enter WPS PBC mode. Then push WPS button on the router.
|
|
ESP32 will receive SSID and password, and connect to the router.
|
|
|
|
WPS_TYPE_PIN:
|
|
Start ESP32, you'll see an eight-digit PIN number in log output.
|
|
Enter the PIN code on the router and then the ESP32 will get connected to router.
|
|
*/
|
|
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "freertos/event_groups.h"
|
|
#include "esp_wifi.h"
|
|
#include "esp_log.h"
|
|
#include "esp_wps.h"
|
|
#include "esp_event.h"
|
|
#include "nvs_flash.h"
|
|
#include <string.h>
|
|
|
|
|
|
/*set wps mode via project configuration */
|
|
#if CONFIG_EXAMPLE_WPS_TYPE_PBC
|
|
#define WPS_MODE WPS_TYPE_PBC
|
|
#elif CONFIG_EXAMPLE_WPS_TYPE_PIN
|
|
#define WPS_MODE WPS_TYPE_PIN
|
|
#else
|
|
#define WPS_MODE WPS_TYPE_DISABLE
|
|
#endif /*CONFIG_EXAMPLE_WPS_TYPE_PBC*/
|
|
|
|
#define MAX_RETRY_ATTEMPTS 2
|
|
|
|
#ifndef PIN2STR
|
|
#define PIN2STR(a) (a)[0], (a)[1], (a)[2], (a)[3], (a)[4], (a)[5], (a)[6], (a)[7]
|
|
#define PINSTR "%c%c%c%c%c%c%c%c"
|
|
#endif
|
|
|
|
static const char *TAG = "example_wps";
|
|
static esp_wps_config_t config = WPS_CONFIG_INIT_DEFAULT(WPS_MODE);
|
|
static wifi_config_t wps_ap_creds[MAX_WPS_AP_CRED];
|
|
static int s_ap_creds_num = 0;
|
|
static int s_retry_num = 0;
|
|
|
|
static void wifi_event_handler(void* arg, esp_event_base_t event_base,
|
|
int32_t event_id, void* event_data)
|
|
{
|
|
static int ap_idx = 1;
|
|
|
|
switch (event_id) {
|
|
case WIFI_EVENT_STA_START:
|
|
ESP_LOGI(TAG, "WIFI_EVENT_STA_START");
|
|
break;
|
|
case WIFI_EVENT_STA_DISCONNECTED:
|
|
ESP_LOGI(TAG, "WIFI_EVENT_STA_DISCONNECTED");
|
|
if (s_retry_num < MAX_RETRY_ATTEMPTS) {
|
|
ESP_ERROR_CHECK(esp_wifi_connect());
|
|
s_retry_num++;
|
|
} else if (ap_idx < s_ap_creds_num) {
|
|
/* Try the next AP credential if first one fails */
|
|
|
|
if (ap_idx < s_ap_creds_num) {
|
|
ESP_LOGI(TAG, "Connecting to SSID: %s, Passphrase: %s",
|
|
wps_ap_creds[ap_idx].sta.ssid, wps_ap_creds[ap_idx].sta.password);
|
|
ESP_ERROR_CHECK(esp_wifi_set_config(ESP_IF_WIFI_STA, &wps_ap_creds[ap_idx++]) );
|
|
ESP_ERROR_CHECK(esp_wifi_connect());
|
|
}
|
|
s_retry_num = 0;
|
|
} else {
|
|
ESP_LOGI(TAG, "Failed to connect!");
|
|
}
|
|
|
|
break;
|
|
case WIFI_EVENT_STA_WPS_ER_SUCCESS:
|
|
ESP_LOGI(TAG, "WIFI_EVENT_STA_WPS_ER_SUCCESS");
|
|
{
|
|
wifi_event_sta_wps_er_success_t *evt =
|
|
(wifi_event_sta_wps_er_success_t *)event_data;
|
|
int i;
|
|
|
|
if (evt) {
|
|
s_ap_creds_num = evt->ap_cred_cnt;
|
|
for (i = 0; i < s_ap_creds_num; i++) {
|
|
memcpy(wps_ap_creds[i].sta.ssid, evt->ap_cred[i].ssid,
|
|
sizeof(evt->ap_cred[i].ssid));
|
|
memcpy(wps_ap_creds[i].sta.password, evt->ap_cred[i].passphrase,
|
|
sizeof(evt->ap_cred[i].passphrase));
|
|
}
|
|
/* If multiple AP credentials are received from WPS, connect with first one */
|
|
ESP_LOGI(TAG, "Connecting to SSID: %s, Passphrase: %s",
|
|
wps_ap_creds[0].sta.ssid, wps_ap_creds[0].sta.password);
|
|
ESP_ERROR_CHECK(esp_wifi_set_config(ESP_IF_WIFI_STA, &wps_ap_creds[0]) );
|
|
}
|
|
/*
|
|
* If only one AP credential is received from WPS, there will be no event data and
|
|
* esp_wifi_set_config() is already called by WPS modules for backward compatibility
|
|
* with legacy apps. So directly attempt connection here.
|
|
*/
|
|
ESP_ERROR_CHECK(esp_wifi_wps_disable());
|
|
ESP_ERROR_CHECK(esp_wifi_connect());
|
|
}
|
|
break;
|
|
case WIFI_EVENT_STA_WPS_ER_FAILED:
|
|
ESP_LOGI(TAG, "WIFI_EVENT_STA_WPS_ER_FAILED");
|
|
ESP_ERROR_CHECK(esp_wifi_wps_disable());
|
|
ESP_ERROR_CHECK(esp_wifi_wps_enable(&config));
|
|
ESP_ERROR_CHECK(esp_wifi_wps_start(0));
|
|
break;
|
|
case WIFI_EVENT_STA_WPS_ER_TIMEOUT:
|
|
ESP_LOGI(TAG, "WIFI_EVENT_STA_WPS_ER_TIMEOUT");
|
|
ESP_ERROR_CHECK(esp_wifi_wps_disable());
|
|
ESP_ERROR_CHECK(esp_wifi_wps_enable(&config));
|
|
ESP_ERROR_CHECK(esp_wifi_wps_start(0));
|
|
break;
|
|
case WIFI_EVENT_STA_WPS_ER_PIN:
|
|
ESP_LOGI(TAG, "WIFI_EVENT_STA_WPS_ER_PIN");
|
|
/* display the PIN code */
|
|
wifi_event_sta_wps_er_pin_t* event = (wifi_event_sta_wps_er_pin_t*) event_data;
|
|
ESP_LOGI(TAG, "WPS_PIN = " PINSTR, PIN2STR(event->pin_code));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void got_ip_event_handler(void* arg, esp_event_base_t event_base,
|
|
int32_t event_id, void* event_data)
|
|
{
|
|
ip_event_got_ip_t* event = (ip_event_got_ip_t*) event_data;
|
|
ESP_LOGI(TAG, "got ip: %s", ip4addr_ntoa(&event->ip_info.ip));
|
|
}
|
|
|
|
/*init wifi as sta and start wps*/
|
|
static void start_wps(void)
|
|
{
|
|
tcpip_adapter_init();
|
|
ESP_ERROR_CHECK(esp_event_loop_create_default());
|
|
|
|
wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
|
|
ESP_ERROR_CHECK(esp_wifi_init(&cfg));
|
|
|
|
ESP_ERROR_CHECK(esp_event_handler_register(WIFI_EVENT, ESP_EVENT_ANY_ID, &wifi_event_handler, NULL));
|
|
ESP_ERROR_CHECK(esp_event_handler_register(IP_EVENT, IP_EVENT_STA_GOT_IP, &got_ip_event_handler, NULL));
|
|
|
|
ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA));
|
|
ESP_ERROR_CHECK(esp_wifi_start());
|
|
|
|
ESP_LOGI(TAG, "start wps...");
|
|
|
|
ESP_ERROR_CHECK(esp_wifi_wps_enable(&config));
|
|
ESP_ERROR_CHECK(esp_wifi_wps_start(0));
|
|
}
|
|
|
|
void app_main()
|
|
{
|
|
/* Initialize NVS — it is used to store PHY calibration data */
|
|
esp_err_t ret = nvs_flash_init();
|
|
if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) {
|
|
ESP_ERROR_CHECK(nvs_flash_erase());
|
|
ret = nvs_flash_init();
|
|
}
|
|
ESP_ERROR_CHECK( ret );
|
|
|
|
start_wps();
|
|
}
|