2019-01-16 18:04:45 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cpu_esp8266_esp_wifi
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Network device driver for the ESP8266 WiFi interface
|
|
|
|
*
|
|
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "log.h"
|
|
|
|
#include "tools.h"
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "net/ethernet.h"
|
|
|
|
#include "net/ipv4/addr.h"
|
|
|
|
#include "net/gnrc/netif/ethernet.h"
|
|
|
|
#include "net/netdev/eth.h"
|
|
|
|
#include "od.h"
|
|
|
|
#include "xtimer.h"
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "espressif/c_types.h"
|
|
|
|
#include "espnow.h"
|
|
|
|
#include "esp/common_macros.h"
|
|
|
|
#include "irq_arch.h"
|
|
|
|
#include "sdk/sdk.h"
|
|
|
|
|
|
|
|
#include "lwip/igmp.h"
|
|
|
|
#include "lwip/udp.h"
|
|
|
|
|
|
|
|
#include "esp_wifi_params.h"
|
|
|
|
#include "esp_wifi_netdev.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#define ESP_WIFI_DEBUG(f, ...) \
|
|
|
|
DEBUG("[esp_wifi] %s: " f "\n", __func__, ## __VA_ARGS__)
|
|
|
|
|
2019-01-20 11:09:59 +01:00
|
|
|
#define ESP_WIFI_LOG_INFO(f, ...) \
|
|
|
|
LOG_INFO("[esp_wifi] " f "\n", ## __VA_ARGS__)
|
|
|
|
|
|
|
|
#define ESP_WIFI_LOG_ERROR(f, ...) \
|
|
|
|
LOG_ERROR("[esp_wifi] " f "\n", ## __VA_ARGS__)
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
#define ESP_WIFI_STATION_MODE (STATION_MODE)
|
|
|
|
#define ESP_WIFI_AP_MODE (SOFTAP_MODE)
|
|
|
|
#define ESP_WIFI_STATION_AP_MODE (STATIONAP_MODE)
|
2019-01-24 02:17:58 +01:00
|
|
|
#define ESP_WIFI_MODE (STATIONAP_MODE)
|
2019-01-16 18:04:45 +01:00
|
|
|
|
|
|
|
#define ESP_WIFI_STATION_IF (STATION_IF)
|
|
|
|
#define ESP_WIFI_SOFTAP_IF (SOFTAP_IF)
|
|
|
|
|
2019-01-23 10:30:17 +01:00
|
|
|
#define ESP_WIFI_RECONNECT_TIME (20 * US_PER_SEC)
|
2019-01-29 18:16:21 +01:00
|
|
|
#define ESP_WIFI_HEAP_MARGIN (2 * ETHERNET_MAX_LEN)
|
2019-01-23 10:32:44 +01:00
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
#define MAC_STR "%02x:%02x:%02x:%02x:%02x:%02x"
|
|
|
|
#define MAC_STR_ARG(m) m[0], m[1], m[2], m[3], m[4], m[5]
|
2019-01-23 10:33:26 +01:00
|
|
|
|
|
|
|
#define PBUF_IEEE80211_HLEN (36)
|
|
|
|
|
2019-01-23 10:30:17 +01:00
|
|
|
/** Timer used to reconnect automatically after 20 seconds if not connected */
|
|
|
|
static xtimer_t _esp_wifi_reconnect_timer;
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
/**
|
|
|
|
* There is only one ESP WIFI device. We define it as static device variable
|
|
|
|
* to have accesss to the device inside ESP WIFI interrupt routines which do
|
|
|
|
* not provide an argument that could be used as pointer to the ESP WIFI
|
|
|
|
* device which triggers the interrupt.
|
|
|
|
*/
|
|
|
|
static esp_wifi_netdev_t _esp_wifi_dev;
|
|
|
|
|
|
|
|
/** forward declaration of the driver functions structure */
|
|
|
|
static const netdev_driver_t _esp_wifi_driver;
|
|
|
|
|
|
|
|
/** Stack for the netif thread */
|
|
|
|
static char _esp_wifi_stack[ESP_WIFI_STACKSIZE];
|
|
|
|
|
|
|
|
/** Static station configuration used for the WiFi interface */
|
|
|
|
static const struct station_config station_cfg = {
|
|
|
|
.bssid_set = 0, /* no check of MAC address of AP */
|
|
|
|
.ssid = ESP_WIFI_SSID,
|
|
|
|
.password = ESP_WIFI_PASS,
|
|
|
|
};
|
|
|
|
|
2019-01-24 02:17:58 +01:00
|
|
|
#ifndef MODULE_ESP_NOW
|
|
|
|
/**
|
2019-01-24 09:39:45 +01:00
|
|
|
* Static const configuration for the SoftAP which is used to configure the
|
2019-01-24 02:17:58 +01:00
|
|
|
* SoftAP interface if ESP-NOW is not enabled.
|
|
|
|
*
|
|
|
|
* Since we need to use the WiFi interface in SoftAP + Station mode for
|
|
|
|
* stability reasons, although in fact only the station interface is required,
|
|
|
|
* we make the SoftAP interface invisible and unusable. This configuration
|
|
|
|
*
|
|
|
|
* - uses the same hidden SSID that the station interface uses to
|
|
|
|
* connect to the AP,
|
|
|
|
* - uses the same channel that the station interface uses to connect to the AP,
|
|
|
|
* - defines a very long beacon interval
|
|
|
|
* - doesn't allow any connection.
|
|
|
|
*/
|
|
|
|
static const struct softap_config softap_cfg = {
|
|
|
|
.ssid = ESP_WIFI_SSID,
|
|
|
|
.ssid_len = sizeof(ESP_WIFI_SSID) / sizeof(ESP_WIFI_SSID[0]),
|
|
|
|
.ssid_hidden = 1, /* don't make the AP visible */
|
|
|
|
.password = ESP_WIFI_PASS,
|
|
|
|
.authmode = AUTH_WPA2_PSK,
|
|
|
|
.max_connection = 0, /* don't allow connections */
|
|
|
|
.beacon_interval = 60000, /* send beacon only every 60 s */
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
extern struct netif * eagle_lwip_getif(uint8 index);
|
|
|
|
|
2019-01-23 10:28:06 +01:00
|
|
|
/** guard variable to avoid reentrance to _send */
|
|
|
|
static bool _in_send = false;
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
/** guard variable to avoid reentrance to _esp_wifi_recv_cb */
|
|
|
|
static bool _in_esp_wifi_recv_cb = false;
|
|
|
|
|
2019-01-23 10:30:17 +01:00
|
|
|
/**
|
|
|
|
* @brief Reconnect function called back by the reconnect timer
|
|
|
|
*/
|
|
|
|
static void IRAM _esp_wifi_reconnect_timer_cb(void* arg)
|
|
|
|
{
|
|
|
|
DEBUG("%s\n", __func__);
|
|
|
|
|
|
|
|
esp_wifi_netdev_t* dev = (esp_wifi_netdev_t*)arg;
|
|
|
|
|
|
|
|
if (dev->state == ESP_WIFI_DISCONNECTED ||
|
|
|
|
dev->state == ESP_WIFI_CONNECTING) {
|
2019-01-24 02:49:39 +01:00
|
|
|
ESP_WIFI_LOG_INFO("trying to reconnect to ssid " ESP_WIFI_SSID);
|
|
|
|
|
2019-01-23 10:30:17 +01:00
|
|
|
wifi_station_disconnect();
|
|
|
|
wifi_station_connect();
|
|
|
|
dev->state = ESP_WIFI_CONNECTING;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the time for next connection check */
|
|
|
|
xtimer_set(&_esp_wifi_reconnect_timer, ESP_WIFI_RECONNECT_TIME);
|
|
|
|
}
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
/**
|
2019-01-21 16:24:11 +01:00
|
|
|
* @brief Callback when ethernet frame is received. Has to run in IRAM.
|
2019-01-16 18:04:45 +01:00
|
|
|
*/
|
2019-01-23 10:28:51 +01:00
|
|
|
void IRAM _esp_wifi_recv_cb(struct pbuf *pb, struct netif *netif)
|
2019-01-16 18:04:45 +01:00
|
|
|
{
|
|
|
|
assert(pb != NULL);
|
|
|
|
assert(netif != NULL);
|
|
|
|
|
|
|
|
/*
|
2019-01-24 17:26:43 +01:00
|
|
|
* Function `esp_wifi_recv_cb` is executed in the context of the `ets`
|
|
|
|
* thread. ISRs which handle hardware interrupts from the WiFi interface
|
|
|
|
* simply pass events to a message queue of the `ets` thread which are then
|
2019-01-21 16:24:11 +01:00
|
|
|
* sequentially processed by the `ets` thread to asynchronously execute
|
2019-01-16 18:04:45 +01:00
|
|
|
* callback functions such as `esp_wifi_recv_cb`.
|
|
|
|
*
|
|
|
|
* It should be therefore not possible to reenter function
|
2019-01-20 11:01:18 +01:00
|
|
|
* `esp_wifi_recv_cb`. If it does occur inspite of that, we use a
|
2019-01-20 11:09:59 +01:00
|
|
|
* guard variable to avoid inconsistencies. This can not be realized
|
2019-01-20 11:01:18 +01:00
|
|
|
* by a mutex because `esp_wifi_recv_cb` would be reentered from same
|
|
|
|
* thread context.
|
2019-01-16 18:04:45 +01:00
|
|
|
*/
|
2019-01-24 09:39:45 +01:00
|
|
|
if (_in_esp_wifi_recv_cb) {
|
2019-01-20 11:01:18 +01:00
|
|
|
pbuf_free(pb);
|
2019-01-16 18:04:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
_in_esp_wifi_recv_cb = true;
|
|
|
|
|
2019-01-20 11:01:18 +01:00
|
|
|
critical_enter();
|
2019-01-16 18:04:45 +01:00
|
|
|
|
2019-01-21 16:24:11 +01:00
|
|
|
/* first, check packet buffer for the minimum packet size */
|
2019-01-24 09:51:43 +01:00
|
|
|
if (pb->tot_len < sizeof(ethernet_hdr_t)) {
|
2019-01-16 18:04:45 +01:00
|
|
|
ESP_WIFI_DEBUG("frame length is less than the size of an Ethernet"
|
2019-01-24 09:51:43 +01:00
|
|
|
"header (%u < %u)", pb->tot_len, sizeof(ethernet_hdr_t));
|
2019-01-16 18:04:45 +01:00
|
|
|
pbuf_free(pb);
|
2019-01-20 11:01:18 +01:00
|
|
|
_in_esp_wifi_recv_cb = false;
|
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-01-20 11:01:18 +01:00
|
|
|
/* check whether the receive buffer is already holding a frame */
|
|
|
|
if (_esp_wifi_dev.rx_len) {
|
2019-01-16 18:04:45 +01:00
|
|
|
ESP_WIFI_DEBUG("buffer used, dropping incoming frame of %d bytes",
|
|
|
|
pb->tot_len);
|
|
|
|
pbuf_free(pb);
|
2019-01-20 11:01:18 +01:00
|
|
|
_in_esp_wifi_recv_cb = false;
|
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-01-21 16:22:35 +01:00
|
|
|
/* check whether packet buffer fits into receive buffer */
|
2019-01-24 09:51:43 +01:00
|
|
|
if (pb->tot_len > ETHERNET_MAX_LEN) {
|
2019-01-21 16:22:35 +01:00
|
|
|
ESP_WIFI_DEBUG("frame length is greater than the maximum size of an "
|
2019-01-24 09:51:43 +01:00
|
|
|
"Ethernet frame (%u > %u)", pb->tot_len, ETHERNET_MAX_LEN);
|
2019-01-21 16:22:35 +01:00
|
|
|
pbuf_free(pb);
|
|
|
|
_in_esp_wifi_recv_cb = false;
|
|
|
|
critical_exit();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-01-24 17:26:43 +01:00
|
|
|
/* we have to store the frame in the buffer and free lwIP pbuf immediatly */
|
2019-01-20 11:01:18 +01:00
|
|
|
_esp_wifi_dev.rx_len = pb->tot_len;
|
|
|
|
pbuf_copy_partial(pb, _esp_wifi_dev.rx_buf, _esp_wifi_dev.rx_len, 0);
|
|
|
|
pbuf_free(pb);
|
2019-01-16 18:04:45 +01:00
|
|
|
|
2019-01-20 11:01:18 +01:00
|
|
|
/*
|
2019-01-24 17:26:43 +01:00
|
|
|
* Since _esp_wifi_recv_cb is not executed in interrupt context but in
|
|
|
|
* the context of the `ets` thread, it is not necessary to pass the
|
|
|
|
* `NETDEV_EVENT_ISR` event first. Instead, the receive function can be
|
|
|
|
* called directly which result in much faster handling, a less frame lost
|
|
|
|
* rate and more robustness. There is no need for a mutex anymore to
|
|
|
|
* synchronize the access to the receive buffer between _esp_wifi_recv_cb
|
|
|
|
* and _recv function.
|
2019-01-20 11:01:18 +01:00
|
|
|
*/
|
2019-01-16 18:04:45 +01:00
|
|
|
if (_esp_wifi_dev.netdev.event_callback) {
|
|
|
|
_esp_wifi_dev.netdev.event_callback(&_esp_wifi_dev.netdev,
|
2019-01-24 17:26:43 +01:00
|
|
|
NETDEV_EVENT_RX_COMPLETE);
|
2019-01-16 18:04:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
_in_esp_wifi_recv_cb = false;
|
2019-01-20 11:01:18 +01:00
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
}
|
|
|
|
|
2019-01-26 17:33:28 +01:00
|
|
|
#define BEACON_TIMEOUT (200)
|
|
|
|
#define HANDSHAKE_TIMEOUT (204)
|
|
|
|
|
|
|
|
static const char *_esp_wifi_disc_reasons [] = {
|
|
|
|
"INVALID", /* 0 */
|
|
|
|
"UNSPECIFIED", /* 1 */
|
|
|
|
"AUTH_EXPIRE", /* 2 */
|
|
|
|
"AUTH_LEAVE", /* 3 */
|
|
|
|
"ASSOC_EXPIRE", /* 4 */
|
|
|
|
"ASSOC_TOOMANY", /* 5 */
|
|
|
|
"NOT_AUTHED", /* 6 */
|
|
|
|
"NOT_ASSOCED", /* 7 */
|
|
|
|
"ASSOC_LEAVE", /* 8 */
|
|
|
|
"ASSOC_NOT_AUTHED", /* 9 */
|
|
|
|
"DISASSOC_PWRCAP_BAD", /* 10 (11h) */
|
|
|
|
"DISASSOC_SUPCHAN_BAD", /* 11 (11h) */
|
|
|
|
"IE_INVALID", /* 13 (11i) */
|
|
|
|
"MIC_FAILURE", /* 14 (11i) */
|
|
|
|
"4WAY_HANDSHAKE_TIMEOUT", /* 15 (11i) */
|
|
|
|
"GROUP_KEY_UPDATE_TIMEOUT", /* 16 (11i) */
|
|
|
|
"IE_IN_4WAY_DIFFERS", /* 17 (11i) */
|
|
|
|
"GROUP_CIPHER_INVALID", /* 18 (11i) */
|
|
|
|
"PAIRWISE_CIPHER_INVALID", /* 19 (11i) */
|
|
|
|
"AKMP_INVALID", /* 20 (11i) */
|
|
|
|
"UNSUPP_RSN_IE_VERSION", /* 21 (11i) */
|
|
|
|
"INVALID_RSN_IE_CAP", /* 22 (11i) */
|
|
|
|
"802_1X_AUTH_FAILED", /* 23 (11i) */
|
|
|
|
"CIPHER_SUITE_REJECTED", /* 24 (11i) */
|
|
|
|
"BEACON_TIMEOUT", /* 200 */
|
|
|
|
"NO_AP_FOUND", /* 201 */
|
|
|
|
"AUTH_FAIL", /* 202 */
|
|
|
|
"ASSOC_FAIL", /* 203 */
|
|
|
|
"HANDSHAKE_TIMEOUT" /* 204 */
|
|
|
|
};
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
/**
|
|
|
|
* @brief Event handler for esp system events.
|
|
|
|
*/
|
|
|
|
static void _esp_wifi_handle_event_cb(System_Event_t *evt)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("event %d", evt->event);
|
|
|
|
|
2019-01-26 17:33:28 +01:00
|
|
|
uint8_t reason;
|
|
|
|
const char* reason_str = "UNKNOWN";
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
switch (evt->event) {
|
|
|
|
case EVENT_STAMODE_CONNECTED:
|
2019-01-20 11:09:59 +01:00
|
|
|
ESP_WIFI_LOG_INFO("connected to ssid %s, channel %d",
|
|
|
|
evt->event_info.connected.ssid,
|
|
|
|
evt->event_info.connected.channel);
|
2019-01-23 09:20:24 +01:00
|
|
|
_esp_wifi_dev.state = ESP_WIFI_CONNECTED;
|
2019-01-26 17:33:28 +01:00
|
|
|
_esp_wifi_dev.event = EVENT_STAMODE_CONNECTED;
|
|
|
|
_esp_wifi_dev.netdev.event_callback(&_esp_wifi_dev.netdev, NETDEV_EVENT_ISR);
|
2019-01-16 18:04:45 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case EVENT_STAMODE_DISCONNECTED:
|
2019-01-26 17:33:28 +01:00
|
|
|
reason = evt->event_info.disconnected.reason;
|
|
|
|
if (reason < REASON_BEACON_TIMEOUT) {
|
|
|
|
reason_str = _esp_wifi_disc_reasons[reason];
|
|
|
|
}
|
|
|
|
else if (reason <= REASON_HANDSHAKE_TIMEOUT) {
|
|
|
|
reason_str = _esp_wifi_disc_reasons[reason - REASON_BEACON_TIMEOUT];
|
|
|
|
}
|
|
|
|
ESP_WIFI_LOG_INFO("disconnected from ssid %s, reason %d (%s)",
|
2019-01-20 11:09:59 +01:00
|
|
|
evt->event_info.disconnected.ssid,
|
2019-01-26 17:33:28 +01:00
|
|
|
evt->event_info.disconnected.reason, reason_str);
|
2019-01-23 09:20:24 +01:00
|
|
|
_esp_wifi_dev.state = ESP_WIFI_DISCONNECTED;
|
2019-01-26 17:33:28 +01:00
|
|
|
_esp_wifi_dev.event = EVENT_STAMODE_DISCONNECTED;
|
|
|
|
_esp_wifi_dev.netdev.event_callback(&_esp_wifi_dev.netdev, NETDEV_EVENT_ISR);
|
2019-01-16 18:04:45 +01:00
|
|
|
break;
|
|
|
|
|
2019-01-24 02:17:58 +01:00
|
|
|
case EVENT_SOFTAPMODE_STACONNECTED:
|
|
|
|
ESP_WIFI_LOG_INFO("station " MACSTR " join, aid %d",
|
|
|
|
MAC2STR(evt->event_info.sta_connected.mac),
|
|
|
|
evt->event_info.sta_connected.aid);
|
|
|
|
break;
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _init(netdev_t *netdev)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%p", netdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLE_DEBUG
|
|
|
|
/** buffer for sent packet dump */
|
|
|
|
uint8_t _send_pkt_buf[ETHERNET_MAX_LEN];
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/** function used to send an ethernet frame over WiFi */
|
|
|
|
extern err_t ieee80211_output_pbuf(struct netif *netif, struct pbuf *p);
|
|
|
|
|
2019-01-29 18:16:21 +01:00
|
|
|
/** function to get free heap */
|
|
|
|
unsigned int IRAM get_free_heap_size (void);
|
|
|
|
|
2019-01-23 10:28:51 +01:00
|
|
|
static int IRAM _send(netdev_t *netdev, const iolist_t *iolist)
|
2019-01-16 18:04:45 +01:00
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%p %p", netdev, iolist);
|
|
|
|
|
|
|
|
assert(netdev != NULL);
|
|
|
|
assert(iolist != NULL);
|
|
|
|
|
|
|
|
if (_in_send) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
_in_send = true;
|
|
|
|
|
|
|
|
esp_wifi_netdev_t *dev = (esp_wifi_netdev_t*)netdev;
|
|
|
|
|
2019-01-20 11:16:33 +01:00
|
|
|
critical_enter();
|
2019-01-23 09:20:24 +01:00
|
|
|
if (dev->state != ESP_WIFI_CONNECTED) {
|
2019-01-16 18:04:45 +01:00
|
|
|
ESP_WIFI_DEBUG("WiFi is still not connected to AP, cannot send");
|
|
|
|
_in_send = false;
|
2019-01-20 11:16:33 +01:00
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2019-02-02 14:37:21 +01:00
|
|
|
#ifndef MODULE_ESP_NOW
|
2019-01-24 02:17:58 +01:00
|
|
|
if (wifi_get_opmode() != ESP_WIFI_MODE) {
|
|
|
|
ESP_WIFI_DEBUG("WiFi is not in correct mode, cannot send");
|
2019-01-16 18:04:45 +01:00
|
|
|
_in_send = false;
|
2019-01-20 11:16:33 +01:00
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
2019-02-02 14:37:21 +01:00
|
|
|
#endif
|
2019-01-16 18:04:45 +01:00
|
|
|
|
|
|
|
const iolist_t *iol = iolist;
|
|
|
|
size_t iol_len = 0;
|
|
|
|
|
|
|
|
/* determine the frame size */
|
|
|
|
while (iol) {
|
|
|
|
iol_len += iol->iol_len;
|
|
|
|
iol = iol->iol_next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* limit checks */
|
|
|
|
if (iol_len > ETHERNET_MAX_LEN) {
|
2019-01-21 16:22:35 +01:00
|
|
|
ESP_WIFI_DEBUG("frame length exceeds the maximum (%u > %u)",
|
2019-01-16 18:04:45 +01:00
|
|
|
iol_len, ETHERNET_MAX_LEN);
|
|
|
|
_in_send = false;
|
2019-01-20 11:16:33 +01:00
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iol_len < sizeof(ethernet_hdr_t)) {
|
|
|
|
ESP_WIFI_DEBUG("frame length is less than the size of an Ethernet"
|
|
|
|
"header (%u < %u)", iol_len, sizeof(ethernet_hdr_t));
|
|
|
|
_in_send = false;
|
2019-01-20 11:16:33 +01:00
|
|
|
critical_exit();
|
2019-01-16 18:04:45 +01:00
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct netif *sta_netif = (struct netif *)eagle_lwip_getif(ESP_WIFI_STATION_IF);
|
|
|
|
netif_set_default(sta_netif);
|
|
|
|
|
2019-01-29 18:16:21 +01:00
|
|
|
struct pbuf *pb;
|
|
|
|
|
|
|
|
if (get_free_heap_size() < ESP_WIFI_HEAP_MARGIN ||
|
|
|
|
(pb = pbuf_alloc(PBUF_LINK, iol_len, PBUF_RAM)) == NULL ||
|
|
|
|
(pb->tot_len < iol_len)) {
|
2019-01-27 12:14:18 +01:00
|
|
|
ESP_WIFI_LOG_ERROR("could not allocate buffer to send %d bytes ", iol_len);
|
2019-01-20 11:22:15 +01:00
|
|
|
/*
|
|
|
|
* The memory of EPS8266 is quite small. Therefore, it may happen on
|
2019-01-21 16:24:11 +01:00
|
|
|
* heavy network load that we run into out of memory and we have
|
2019-01-26 17:23:21 +01:00
|
|
|
* to wait until lwIP pbuf has been flushed. We slow down sending a bit.
|
2019-01-20 11:22:15 +01:00
|
|
|
*/
|
2019-01-20 11:16:33 +01:00
|
|
|
critical_exit();
|
2019-01-20 11:22:15 +01:00
|
|
|
/* wait 20 ms */
|
|
|
|
xtimer_usleep(20 * US_PER_MS);
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
_in_send = false;
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct pbuf *pbi = pb;
|
|
|
|
uint8_t *pbi_payload = pb->payload;
|
|
|
|
size_t pbi_pos = 0;
|
|
|
|
|
|
|
|
/* prepare lwIP packet buffer direct from iolist without any buffer */
|
|
|
|
for (const iolist_t *iol = iolist; iol && pbi; iol = iol->iol_next) {
|
|
|
|
uint8_t *iol_base = iol->iol_base;
|
|
|
|
for (unsigned i = 0; i < iol->iol_len && pbi; i++) {
|
|
|
|
pbi_payload[pbi_pos++] = iol_base[i];
|
|
|
|
if (pbi_pos >= pbi->len) {
|
|
|
|
pbi = pbi->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLE_DEBUG
|
|
|
|
pbi = pb;
|
|
|
|
pbi_pos = 0;
|
|
|
|
|
|
|
|
for (; pbi; pbi = pbi->next) {
|
|
|
|
memcpy(_send_pkt_buf + pbi_pos, pbi->payload, pbi->len);
|
|
|
|
pbi_pos += pbi->len;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ethernet_hdr_t* hdr = (const ethernet_hdr_t *)_send_pkt_buf;
|
|
|
|
|
|
|
|
ESP_WIFI_DEBUG("send %u byte to " MAC_STR,
|
|
|
|
(unsigned)iol_len, MAC_STR_ARG(hdr->dst));
|
|
|
|
#if MODULE_OD
|
|
|
|
od_hex_dump(_send_pkt_buf, iol_len, OD_WIDTH_DEFAULT);
|
|
|
|
#endif /* MODULE_OD */
|
|
|
|
#endif /* ENABLE_DEBUG */
|
|
|
|
|
2019-01-26 17:32:04 +01:00
|
|
|
critical_exit();
|
2019-01-29 18:16:21 +01:00
|
|
|
/* sta_netif->linkoutput = ieee80211_output_pbuf */
|
2019-01-26 17:32:04 +01:00
|
|
|
err_t res = sta_netif->linkoutput(sta_netif, pb);
|
2019-01-16 18:04:45 +01:00
|
|
|
pbuf_free(pb);
|
|
|
|
|
2019-01-26 17:32:04 +01:00
|
|
|
if (res == ERR_OK) {
|
2019-01-23 10:32:44 +01:00
|
|
|
/* There was no ieee80211_output_pbuf error and no send timeout. */
|
2019-01-16 18:04:45 +01:00
|
|
|
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE);
|
|
|
|
_in_send = false;
|
|
|
|
return iol_len;
|
|
|
|
}
|
|
|
|
else {
|
2019-01-23 10:32:44 +01:00
|
|
|
/* There was either a ieee80211_output_pbuf error or send timed out. */
|
2019-01-16 18:04:45 +01:00
|
|
|
_in_send = false;
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%p %p %u %p", netdev, buf, len, info);
|
|
|
|
|
|
|
|
assert(netdev != NULL);
|
|
|
|
|
|
|
|
esp_wifi_netdev_t* dev = (esp_wifi_netdev_t*)netdev;
|
|
|
|
|
2019-01-20 11:01:18 +01:00
|
|
|
uint16_t size = dev->rx_len ? dev->rx_len : 0;
|
2019-01-16 18:04:45 +01:00
|
|
|
|
|
|
|
if (!buf) {
|
|
|
|
/* get the size of the frame */
|
|
|
|
if (len > 0 && size) {
|
|
|
|
/* if len > 0, drop the frame */
|
2019-01-20 11:01:18 +01:00
|
|
|
dev->rx_len = 0;
|
2019-01-16 18:04:45 +01:00
|
|
|
}
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len < size) {
|
|
|
|
/* buffer is smaller than the number of received bytes */
|
|
|
|
ESP_WIFI_DEBUG("not enough space in receive buffer");
|
|
|
|
/* newest API requires to drop the frame in that case */
|
2019-01-20 11:01:18 +01:00
|
|
|
dev->rx_len = 0;
|
2019-01-16 18:04:45 +01:00
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy the buffer and free */
|
2019-01-20 11:01:18 +01:00
|
|
|
memcpy(buf, dev->rx_buf, dev->rx_len);
|
|
|
|
dev->rx_len = 0;
|
2019-01-16 18:04:45 +01:00
|
|
|
|
|
|
|
#if ENABLE_DEBUG
|
|
|
|
ethernet_hdr_t *hdr = (ethernet_hdr_t *)buf;
|
|
|
|
|
|
|
|
ESP_WIFI_DEBUG("received %u byte from addr " MAC_STR,
|
|
|
|
size, MAC_STR_ARG(hdr->src));
|
|
|
|
#if MODULE_OD
|
|
|
|
od_hex_dump(buf, size, OD_WIDTH_DEFAULT);
|
|
|
|
#endif /* MODULE_OD */
|
|
|
|
#endif /* ENABLE_DEBUG */
|
|
|
|
|
2019-01-20 11:16:33 +01:00
|
|
|
return size;
|
2019-01-16 18:04:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%s %p %p %u", netopt2str(opt), netdev, val, max_len);
|
|
|
|
|
|
|
|
assert(netdev != NULL);
|
|
|
|
assert(val != NULL);
|
|
|
|
|
|
|
|
esp_wifi_netdev_t *dev = (esp_wifi_netdev_t*)netdev;
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
|
|
|
|
case NETOPT_IS_WIRED:
|
|
|
|
return -ENOTSUP;
|
|
|
|
|
|
|
|
case NETOPT_LINK_CONNECTED:
|
|
|
|
assert(max_len == 1);
|
2019-01-23 09:20:24 +01:00
|
|
|
if (dev->state == ESP_WIFI_CONNECTED) {
|
2019-01-16 18:04:45 +01:00
|
|
|
*((netopt_enable_t *)val) = NETOPT_ENABLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*((netopt_enable_t *)val) = NETOPT_DISABLE;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
assert(max_len >= sizeof(dev->mac));
|
|
|
|
memcpy(val, dev->mac, sizeof(dev->mac));
|
|
|
|
return sizeof(dev->mac);
|
|
|
|
|
|
|
|
default:
|
|
|
|
return netdev_eth_get(netdev, opt, val, max_len);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const void *val, size_t max_len)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%s %p %p %u", netopt2str(opt), netdev, val, max_len);
|
|
|
|
|
|
|
|
assert(netdev != NULL);
|
|
|
|
assert(val != NULL);
|
|
|
|
|
|
|
|
esp_wifi_netdev_t *dev = (esp_wifi_netdev_t *) netdev;
|
|
|
|
|
|
|
|
switch (opt) {
|
|
|
|
|
|
|
|
case NETOPT_ADDRESS:
|
|
|
|
assert(max_len >= sizeof(dev->mac));
|
|
|
|
memcpy(dev->mac, val, sizeof(dev->mac));
|
|
|
|
return sizeof(dev->mac);
|
|
|
|
|
|
|
|
default:
|
|
|
|
return netdev_eth_set(netdev, opt, val, max_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _isr(netdev_t *netdev)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%p", netdev);
|
|
|
|
|
|
|
|
assert(netdev != NULL);
|
2019-01-26 17:34:13 +01:00
|
|
|
|
|
|
|
esp_wifi_netdev_t *dev = (esp_wifi_netdev_t *) netdev;
|
|
|
|
|
|
|
|
switch (dev->event) {
|
|
|
|
case EVENT_STAMODE_CONNECTED:
|
|
|
|
dev->netdev.event_callback(netdev, NETDEV_EVENT_LINK_UP);
|
|
|
|
break;
|
|
|
|
case EVENT_STAMODE_DISCONNECTED:
|
|
|
|
dev->netdev.event_callback(netdev, NETDEV_EVENT_LINK_DOWN);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
_esp_wifi_dev.event = EVENT_MAX; /* no event */
|
|
|
|
|
|
|
|
return;
|
2019-01-16 18:04:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/** override lwIP ethernet_intput to get ethernet frames */
|
|
|
|
extern err_t __real_ethernet_input(struct pbuf *pb, struct netif* netif);
|
|
|
|
|
|
|
|
err_t __wrap_ethernet_input(struct pbuf *pb, struct netif* netif)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("%p %p", pb, netif);
|
2019-01-23 10:19:28 +01:00
|
|
|
if (_esp_wifi_dev.state == ESP_WIFI_CONNECTED) {
|
|
|
|
_esp_wifi_recv_cb(pb, netif);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
__real_ethernet_input(pb, netif);
|
|
|
|
}
|
2019-01-16 18:04:45 +01:00
|
|
|
return ERR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const netdev_driver_t _esp_wifi_driver = {
|
|
|
|
.send = _send,
|
|
|
|
.recv = _recv,
|
|
|
|
.init = _init,
|
|
|
|
.isr = _isr,
|
|
|
|
.get = _get,
|
|
|
|
.set = _set,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void _esp_wifi_setup(void)
|
|
|
|
{
|
|
|
|
esp_wifi_netdev_t* dev = &_esp_wifi_dev;
|
|
|
|
|
|
|
|
ESP_WIFI_DEBUG("%p", dev);
|
|
|
|
|
|
|
|
if (dev->netdev.driver) {
|
|
|
|
ESP_WIFI_DEBUG("early returning previously initialized device");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* initialize netdev data structure */
|
2019-01-20 11:01:18 +01:00
|
|
|
dev->rx_len = 0;
|
2019-01-23 09:20:24 +01:00
|
|
|
dev->state = ESP_WIFI_DISCONNECTED;
|
2019-01-26 17:34:13 +01:00
|
|
|
dev->event = EVENT_MAX;
|
2019-01-16 18:04:45 +01:00
|
|
|
|
|
|
|
/* set the netdev driver */
|
|
|
|
dev->netdev.driver = &_esp_wifi_driver;
|
|
|
|
|
2019-01-24 02:17:58 +01:00
|
|
|
#ifndef MODULE_ESP_NOW
|
|
|
|
/* set the WiFi interface mode */
|
|
|
|
if (!wifi_set_opmode_current(ESP_WIFI_MODE)) {
|
2019-01-20 11:09:59 +01:00
|
|
|
ESP_WIFI_LOG_ERROR("could not set WiFi working mode");
|
2019-01-16 18:04:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-01-24 02:17:58 +01:00
|
|
|
/* set the WiFi SoftAP configuration */
|
|
|
|
if (!wifi_softap_set_config_current((struct softap_config *)&softap_cfg)) {
|
|
|
|
ESP_WIFI_LOG_ERROR("could not set WiFi configuration");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* set the WiFi station configuration */
|
2019-01-16 18:04:45 +01:00
|
|
|
if (!wifi_station_set_config_current((struct station_config *)&station_cfg)) {
|
2019-01-20 11:09:59 +01:00
|
|
|
ESP_WIFI_LOG_ERROR("could not set WiFi configuration");
|
2019-01-16 18:04:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get station mac address and store it in device address */
|
|
|
|
if (!wifi_get_macaddr(ESP_WIFI_STATION_IF, dev->mac)) {
|
2019-01-20 11:09:59 +01:00
|
|
|
ESP_WIFI_LOG_ERROR("could not get MAC address of WiFi interface");
|
2019-01-16 18:04:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
ESP_WIFI_DEBUG("own MAC addr is " MAC_STR, MAC_STR_ARG(dev->mac));
|
|
|
|
|
2019-01-25 01:14:00 +01:00
|
|
|
/* set auto reconnect policy */
|
|
|
|
wifi_station_set_reconnect_policy(true);
|
|
|
|
wifi_station_set_auto_connect(true);
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
/* register callbacks */
|
|
|
|
wifi_set_event_handler_cb(_esp_wifi_handle_event_cb);
|
|
|
|
|
2019-01-23 10:30:17 +01:00
|
|
|
/* reconnect timer initialization */
|
|
|
|
_esp_wifi_reconnect_timer.callback = &_esp_wifi_reconnect_timer_cb;
|
|
|
|
_esp_wifi_reconnect_timer.arg = dev;
|
|
|
|
|
|
|
|
/* set the the reconnect timer */
|
|
|
|
xtimer_set(&_esp_wifi_reconnect_timer, ESP_WIFI_RECONNECT_TIME);
|
|
|
|
|
2019-01-30 08:27:21 +01:00
|
|
|
/* avoid the WiFi modem going into sleep mode */
|
|
|
|
wifi_set_sleep_type(NONE_SLEEP_T);
|
|
|
|
|
2019-01-16 18:04:45 +01:00
|
|
|
/* connect */
|
2019-01-23 10:16:21 +01:00
|
|
|
wifi_station_connect();
|
|
|
|
_esp_wifi_dev.state = ESP_WIFI_CONNECTING;
|
2019-01-16 18:04:45 +01:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void auto_init_esp_wifi(void)
|
|
|
|
{
|
|
|
|
ESP_WIFI_DEBUG("auto initializing netdev\n");
|
|
|
|
|
|
|
|
/* setup netdev device */
|
|
|
|
_esp_wifi_setup();
|
|
|
|
|
|
|
|
/* create netif */
|
|
|
|
gnrc_netif_ethernet_create(_esp_wifi_stack, ESP_WIFI_STACKSIZE,
|
2019-02-02 14:37:21 +01:00
|
|
|
#ifdef MODULE_ESP_NOW
|
|
|
|
ESP_WIFI_PRIO - 1,
|
|
|
|
#else
|
|
|
|
ESP_WIFI_PRIO,
|
|
|
|
#endif
|
|
|
|
"esp-wifi",
|
2019-01-16 18:04:45 +01:00
|
|
|
(netdev_t *)&_esp_wifi_dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @} */
|