1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/esp_common/esp-now/esp_now_netdev.c

862 lines
25 KiB
C
Raw Normal View History

2018-10-08 12:20:49 +02: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.
*/
/**
2018-12-27 23:25:31 +01:00
* @ingroup cpu_esp_common_esp_now
2018-10-08 12:20:49 +02:00
* @{
*
* @file
* @brief Netdev interface for the ESP-NOW WiFi P2P protocol
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @author Timo Rothenpieler <timo.rothenpieler@uni-bremen.de>
2018-10-08 12:20:49 +02:00
*/
#include "log.h"
#include "tools.h"
#include <string.h>
#include <assert.h>
#include <errno.h>
#include "net/gnrc.h"
#include "xtimer.h"
#ifdef MCU_ESP32
2018-10-08 12:20:49 +02:00
#include "esp_common.h"
#include "esp_attr.h"
#include "esp_event_loop.h"
#include "esp_now.h"
#include "esp_system.h"
#include "esp_wifi.h"
#include "nvs_flash/include/nvs_flash.h"
#else
#include "common.h"
#include "espressif/c_types.h"
#include "espnow.h"
#include "esp/common_macros.h"
#include "sdk/sdk.h"
#endif /* MCU_ESP32 */
2018-10-08 12:20:49 +02:00
#include "irq_arch.h"
#include "od.h"
2018-10-08 12:20:49 +02:00
#include "esp_now_params.h"
#include "esp_now_netdev.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
2018-10-08 12:20:49 +02:00
#define ESP_NOW_UNICAST (1)
2018-10-08 12:20:49 +02:00
#define ESP_NOW_WIFI_STA (1)
#define ESP_NOW_WIFI_SOFTAP (2)
2018-10-08 12:20:49 +02:00
#define ESP_NOW_WIFI_STA_SOFTAP (ESP_NOW_WIFI_STA + ESP_NOW_WIFI_SOFTAP)
#define ESP_NOW_AP_PREFIX "RIOT_ESP_"
#define ESP_NOW_AP_PREFIX_LEN (strlen(ESP_NOW_AP_PREFIX))
#ifdef MCU_ESP32
#define esp_now_set_self_role(r)
#else
#define ESP_NOW_ROLE_IDLE (0)
#define ESP_NOW_ROLE_CONTROLLER (1)
#define ESP_NOW_ROLE_SLAVE (2)
#define ESP_NOW_ROLE_COMBO (3)
#define ESP_NOW_KEY_LEN (16)
#define ESP_OK (0)
#ifndef IRAM_ATTR
#define IRAM_ATTR IRAM
#endif
#endif /* MCU_ESP8266 */
2018-10-08 12:20:49 +02:00
/**
* There is only one ESP-NOW device. We define it as static device variable
* to have accesss to the device inside ESP-NOW interrupt routines which do
* not provide an argument that could be used as pointer to the ESP-NOW
* device which triggers the interrupt.
*/
static esp_now_netdev_t _esp_now_dev;
2018-10-08 12:20:49 +02:00
static const netdev_driver_t _esp_now_driver;
#ifdef MCU_ESP32
static bool _esp_now_add_peer(const uint8_t* bssid, uint8_t channel, uint8_t* key)
#else
static bool _esp_now_add_peer(uint8_t* bssid, uint8_t channel, uint8_t* key)
#endif
2018-10-08 12:20:49 +02:00
{
if (esp_now_is_peer_exist(bssid)) {
return false;
}
#ifdef MCU_ESP32
2018-10-08 12:20:49 +02:00
esp_now_peer_info_t peer = {};
memcpy(peer.peer_addr, bssid, ESP_NOW_ETH_ALEN);
peer.channel = channel;
peer.ifidx = ESP_IF_WIFI_AP;
if (esp_now_params.key) {
peer.encrypt = true;
memcpy(peer.lmk, esp_now_params.key, ESP_NOW_KEY_LEN);
}
esp_err_t ret = esp_now_add_peer(&peer);
#else
int ret = esp_now_add_peer(bssid, ESP_NOW_ROLE_COMBO, channel,
esp_now_params.key,
esp_now_params.key ? ESP_NOW_KEY_LEN : 0);
#endif
2018-10-08 12:20:49 +02:00
DEBUG("esp_now_add_peer node %02x:%02x:%02x:%02x:%02x:%02x "
"added with return value %d\n",
bssid[0], bssid[1], bssid[2], bssid[3], bssid[4], bssid[5], ret);
return (ret == ESP_OK);
}
#if ESP_NOW_UNICAST
static xtimer_t _esp_now_scan_peers_timer;
static bool _esp_now_scan_peers_done = false;
#ifdef MCU_ESP32
2018-10-08 12:20:49 +02:00
#define ESP_NOW_APS_BLOCK_SIZE 8 /* has to be power of two */
static wifi_ap_record_t* aps = NULL;
static uint32_t aps_size = 0;
static const wifi_scan_config_t scan_cfg = {
.ssid = NULL,
.bssid = NULL,
.channel = ESP_NOW_CHANNEL,
.show_hidden = true,
.scan_type = WIFI_SCAN_TYPE_ACTIVE,
.scan_time.active.min = 0,
.scan_time.active.max = 120 /* TODO tune value */
};
2018-12-10 14:03:54 +01:00
static void IRAM_ATTR esp_now_scan_peers_done(void)
2018-10-08 12:20:49 +02:00
{
mutex_lock(&_esp_now_dev.dev_lock);
esp_err_t ret;
uint16_t ap_num;
ret = esp_wifi_scan_get_ap_num(&ap_num);
DEBUG("wifi_scan_get_ap_num ret=%d num=%d\n", ret ,ap_num);
if (ret == ESP_OK && ap_num) {
uint32_t state;
/* reallocation of memory must not be disturbed */
critical_enter_var(state);
/* allocate memory for APs record list blockwise and fetch them the list */
if (ap_num > aps_size) {
if (aps) {
/* free allocated AP record list memory */
aps_size = 0;
free (aps);
}
/* allocate new memory */
aps_size = (ap_num & ~(ESP_NOW_APS_BLOCK_SIZE - 1)) + ESP_NOW_APS_BLOCK_SIZE;
aps = malloc(sizeof(wifi_ap_record_t) * aps_size);
ap_num = aps_size;
}
critical_exit_var(state);
ret = esp_wifi_scan_get_ap_records(&ap_num, aps);
DEBUG("wifi_scan_get_aps ret=%d num=%d\n", ret, ap_num);
critical_enter_var(state);
/* iterate over APs records */
for (uint16_t i = 0; i < ap_num; i++) {
/* check whether the AP is an ESP_NOW node */
if (strncmp((char*)aps[i].ssid, ESP_NOW_AP_PREFIX, ESP_NOW_AP_PREFIX_LEN) == 0) {
2018-10-08 12:20:49 +02:00
/* add the AP as peer */
_esp_now_add_peer(aps[i].bssid, aps[i].primary, esp_now_params.key);
}
}
critical_exit_var(state);
}
2018-12-10 14:03:54 +01:00
#if ENABLE_DEBUG
2018-10-08 12:20:49 +02:00
esp_now_peer_num_t peer_num;
2018-12-10 14:03:54 +01:00
esp_now_get_peer_num(&peer_num);
2018-10-08 12:20:49 +02:00
DEBUG("associated peers total=%d, encrypted=%d\n",
peer_num.total_num, peer_num.encrypt_num);
#endif /* ENABLE_DEBUG */
2018-10-08 12:20:49 +02:00
_esp_now_scan_peers_done = true;
mutex_unlock(&_esp_now_dev.dev_lock);
}
#else /* MCU_ESP32 */
static const struct scan_config scan_cfg = {
.ssid = 0,
.bssid = 0,
.channel = ESP_NOW_CHANNEL,
.show_hidden = true
};
static void IRAM esp_now_scan_peers_done(void *arg, STATUS status)
{
/* This function is executed in interrupt context */
DEBUG("%s: %p %d\n", __func__, arg, status);
CHECK_PARAM (status == OK);
CHECK_PARAM (arg != NULL);
critical_enter();
struct bss_info *bss_link = (struct bss_info*)arg;
/* iterate over APs records */
while (bss_link) {
/* check whether the AP is an ESP_NOW node */
if (strncmp((char*)bss_link->ssid, ESP_NOW_AP_PREFIX, ESP_NOW_AP_PREFIX_LEN) == 0) {
/* add the AP as peer */
_esp_now_add_peer(bss_link->bssid, bss_link->channel, esp_now_params.key);
}
bss_link = STAILQ_NEXT(bss_link, next);
}
#if ENABLE_DEBUG
uint8_t peers_all;
uint8_t peers_enc;
esp_now_get_cnt_info (&peers_all, &peers_enc);
DEBUG("associated peers total=%d, encrypted=%d\n", peers_all, peers_enc);
#endif /* ENABLE_DEBUG */
_esp_now_scan_peers_done = true;
critical_exit();
}
#endif /* MCU_ESP32 */
2018-12-10 14:03:54 +01:00
static void esp_now_scan_peers_start(void)
2018-10-08 12:20:49 +02:00
{
DEBUG("%s\n", __func__);
/* set the time for next scan */
xtimer_set(&_esp_now_scan_peers_timer, esp_now_params.scan_period);
/* start the scan */
#ifdef MCU_ESP32
esp_wifi_scan_start(&scan_cfg, false);
#else
wifi_station_scan((struct scan_config*)&scan_cfg, esp_now_scan_peers_done);
#endif
2018-10-08 12:20:49 +02:00
}
2018-12-10 14:03:54 +01:00
static void IRAM_ATTR esp_now_scan_peers_timer_cb(void* arg)
2018-10-08 12:20:49 +02:00
{
DEBUG("%s\n", __func__);
esp_now_netdev_t* dev = (esp_now_netdev_t*)arg;
if (dev->netdev.event_callback) {
dev->scan_event++;
dev->netdev.event_callback((netdev_t*)dev, NETDEV_EVENT_ISR);
}
2018-10-08 12:20:49 +02:00
}
#else /* ESP_NOW_UNICAST */
2018-10-08 12:20:49 +02:00
static const uint8_t _esp_now_mac[6] = { 0x82, 0x73, 0x79, 0x84, 0x79, 0x83 }; /* RIOTOS */
#endif /* ESP_NOW_UNICAST */
static bool _in_recv_cb = false;
#ifdef MCU_ESP32
2018-12-10 14:03:54 +01:00
static IRAM_ATTR void esp_now_recv_cb(const uint8_t *mac, const uint8_t *data, int len)
#else
static IRAM_ATTR void esp_now_recv_cb(uint8_t *mac, uint8_t *data, uint8_t len)
#endif
2018-10-08 12:20:49 +02:00
{
2018-12-10 14:03:54 +01:00
#if ESP_NOW_UNICAST
2018-10-08 12:20:49 +02:00
if (!_esp_now_scan_peers_done) {
/* if peers are not scanned, we cannot receive anything */
return;
}
#endif /* ESP_NOW_UNICAST */
2018-10-08 12:20:49 +02:00
/*
* The function `esp_now_recv_cb` is executed in the context of the `wifi`
* thread. The ISRs handling the hardware interrupts from the WiFi
* interface pass events to a message queue of the `wifi` thread which is
* sequentially processed by the `wifi` thread to asynchronously execute
* callback functions such as `esp_now_recv_cb`.
*
* It should be therefore not possible to reenter function
* `esp_now_recv_cb`. To avoid inconsistencies this is checked by an
* additional boolean variable . This can not be realized by a mutex
* because `esp_now_recv_cb` would be reentered from same thread context.
*/
if (_in_recv_cb) {
return;
}
_in_recv_cb = true;
/*
* Since it is not possible to reenter the function `esp_now_recv_cb`, and
* the functions netif::_ recv and esp_now_netdev::_ recv are called
* directly in the same thread context, neither a mutual exclusion has to
* be realized nor have the interrupts to be deactivated.
* Therefore we can read directly from the `buf` and don't need a receive
* buffer.
*/
if (_esp_now_dev.rx_len) {
DEBUG("%s: buffer full, dropping incoming packet of %d bytes\n", __func__, len);
_in_recv_cb = false;
2018-10-08 12:20:49 +02:00
return;
}
_esp_now_dev.rx_mac = (uint8_t*)mac;
_esp_now_dev.rx_data = (uint8_t*)data;
_esp_now_dev.rx_len = len;
2018-10-08 12:20:49 +02:00
/*
* Since we are not in the interrupt context, we do not have to pass
* `NETDEV_EVENT_ISR` first. We can call the receive function directly.
*/
if (_esp_now_dev.netdev.event_callback) {
_esp_now_dev.netdev.event_callback((netdev_t*)&_esp_now_dev,
NETDEV_EVENT_RX_COMPLETE);
}
_in_recv_cb = false;
2018-10-08 12:20:49 +02:00
}
static volatile int _esp_now_sending = 0;
2018-10-08 12:20:49 +02:00
#ifdef MCU_ESP32
2018-12-10 14:03:54 +01:00
static void IRAM_ATTR esp_now_send_cb(const uint8_t *mac, esp_now_send_status_t status)
#else
static void IRAM_ATTR esp_now_send_cb(uint8_t *mac, uint8_t status)
#endif
2018-10-08 12:20:49 +02:00
{
DEBUG("%s: sent to %02x:%02x:%02x:%02x:%02x:%02x with status %d\n",
__func__,
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5], status);
if (_esp_now_sending) {
_esp_now_sending--;
}
}
#ifdef MCU_ESP32
2018-10-08 12:20:49 +02:00
/*
* Event handler for esp system events.
*/
static esp_err_t IRAM_ATTR _esp_system_event_handler(void *ctx, system_event_t *event)
{
switch(event->event_id) {
case SYSTEM_EVENT_STA_START:
DEBUG("%s WiFi started\n", __func__);
break;
case SYSTEM_EVENT_SCAN_DONE:
DEBUG("%s WiFi scan done\n", __func__);
#if ESP_NOW_UNICAST
2018-10-08 12:20:49 +02:00
esp_now_scan_peers_done();
#endif /* ESP_NOW_UNICAST */
2018-10-08 12:20:49 +02:00
break;
default:
break;
}
return ESP_OK;
}
/**
* Default WiFi configuration, overwrite them with your configs
*/
#ifndef CONFIG_WIFI_STA_SSID
#define CONFIG_WIFI_STA_SSID "RIOT_AP"
#endif
#ifndef CONFIG_WIFI_STA_PASSWORD
#define CONFIG_WIFI_STA_PASSWORD "ThisistheRIOTporttoESP"
#endif
#ifndef CONFIG_WIFI_STA_CHANNEL
#define CONFIG_WIFI_STA_CHANNEL 0
#endif
#define CONFIG_WIFI_STA_SCAN_METHOD WIFI_ALL_CHANNEL_SCAN
#define CONFIG_WIFI_STA_SORT_METHOD WIFI_CONNECT_AP_BY_SIGNAL
#define CONFIG_WIFI_STA_RSSI -127
#define CONFIG_WIFI_STA_AUTHMODE WIFI_AUTH_WPA_WPA2_PSK
#define CONFIG_WIFI_AP_PASSWORD ESP_NOW_SOFT_AP_PASSPHRASE
#define CONFIG_WIFI_AP_CHANNEL ESP_NOW_CHANNEL
#define CONFIG_WIFI_AP_AUTH WIFI_AUTH_WPA_WPA2_PSK
#define CONFIG_WIFI_AP_HIDDEN false
#define CONFIG_WIFI_AP_BEACON 100
#define CONFIG_WIFI_AP_MAX_CONN 4
2018-12-10 14:03:54 +01:00
extern esp_err_t esp_system_event_add_handler(system_event_cb_t handler,
void *arg);
2018-10-08 12:20:49 +02:00
#else /* MCU_ESP32 */
#define ESP_IF_WIFI_STA (STATION_IF)
#endif /* MCU_ESP32 */
esp_now_netdev_t *netdev_esp_now_setup(void)
2018-10-08 12:20:49 +02:00
{
esp_now_netdev_t* dev = &_esp_now_dev;
2018-10-08 12:20:49 +02:00
DEBUG("%s: %p\n", __func__, dev);
if (dev->netdev.driver) {
DEBUG("%s: early returning previously initialized device\n", __func__);
return dev;
}
#ifdef MCU_ESP32
2018-10-08 12:20:49 +02:00
/*
* Init the WiFi driver. TODO It is not only required before ESP_NOW is
* initialized but also before other WiFi functions are used. Once other
* WiFi functions are realized it has to be moved to a more common place.
*/
extern portMUX_TYPE g_intr_lock_mux;
mutex_init(&g_intr_lock_mux);
/* initialize buffer */
dev->rx_len = 0;
2018-12-10 14:03:54 +01:00
esp_system_event_add_handler(_esp_system_event_handler, NULL);
2018-10-08 12:20:49 +02:00
esp_err_t result;
2018-12-10 14:03:54 +01:00
#if CONFIG_ESP32_WIFI_NVS_ENABLED
2018-10-08 12:20:49 +02:00
result = nvs_flash_init();
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now",
"nfs_flash_init failed with return value %d\n", result);
return NULL;
2018-10-08 12:20:49 +02:00
}
#endif /* CONFIG_ESP32_WIFI_NVS_ENABLED */
2018-10-08 12:20:49 +02:00
wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
result = esp_wifi_init(&cfg);
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now",
"esp_wifi_init failed with return value %d\n", result);
return NULL;
2018-10-08 12:20:49 +02:00
}
2018-12-10 14:03:54 +01:00
#ifdef CONFIG_WIFI_COUNTRY
2018-10-08 12:20:49 +02:00
/* TODO */
#endif /* CONFIG_WIFI_COUNTRY */
2018-10-08 12:20:49 +02:00
/* we use predefined station configuration */
wifi_config_t wifi_config_sta = {
.sta = {
.ssid = CONFIG_WIFI_STA_SSID,
.password = CONFIG_WIFI_STA_PASSWORD,
.channel = CONFIG_WIFI_STA_CHANNEL,
.scan_method = CONFIG_WIFI_STA_SCAN_METHOD,
.sort_method = CONFIG_WIFI_STA_SORT_METHOD,
.threshold.rssi = CONFIG_WIFI_STA_RSSI,
.threshold.authmode = CONFIG_WIFI_STA_AUTHMODE
}
};
/* get SoftAP interface mac address and store it as device addresss */
esp_read_mac(dev->addr, ESP_MAC_WIFI_SOFTAP);
/* prepare the ESP_NOW configuration for SoftAP */
wifi_config_t wifi_config_ap = {};
strcpy ((char*)wifi_config_ap.ap.password, esp_now_params.softap_pass);
sprintf((char*)wifi_config_ap.ap.ssid, "%s%02x%02x%02x%02x%02x%02x",
ESP_NOW_AP_PREFIX,
dev->addr[0], dev->addr[1], dev->addr[2],
dev->addr[3], dev->addr[4], dev->addr[5]);
wifi_config_ap.ap.ssid_len = strlen((char*)wifi_config_ap.ap.ssid);
wifi_config_ap.ap.channel = esp_now_params.channel;
wifi_config_ap.ap.authmode = CONFIG_WIFI_AP_AUTH;
wifi_config_ap.ap.ssid_hidden = CONFIG_WIFI_AP_HIDDEN;
wifi_config_ap.ap.max_connection = CONFIG_WIFI_AP_MAX_CONN;
wifi_config_ap.ap.beacon_interval = CONFIG_WIFI_AP_BEACON;
/* set the WiFi interface to Station + SoftAP mode without DHCP */
result = esp_wifi_set_mode(WIFI_MODE_STA | WIFI_MODE_AP);
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now",
"esp_wifi_set_mode failed with return value %d\n",
result);
return NULL;
2018-10-08 12:20:49 +02:00
}
/* set the Station and SoftAP configuration */
result = esp_wifi_set_config(ESP_IF_WIFI_STA, &wifi_config_sta);
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now", "esp_wifi_set_config station failed with "
"return value %d\n", result);
return NULL;
2018-10-08 12:20:49 +02:00
}
result = esp_wifi_set_config(ESP_IF_WIFI_AP, &wifi_config_ap);
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now",
"esp_wifi_set_mode softap failed with return value %d\n",
result);
return NULL;
2018-10-08 12:20:49 +02:00
}
/* start the WiFi driver */
result = esp_wifi_start();
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now",
"esp_wifi_start failed with return value %d\n", result);
return NULL;
2018-10-08 12:20:49 +02:00
}
#if !ESP_NOW_UNICAST
2018-10-08 12:20:49 +02:00
/* all ESP-NOW nodes get the shared mac address on their station interface */
esp_wifi_set_mac(ESP_IF_WIFI_STA, (uint8_t*)_esp_now_mac);
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
#else /* MCU_ESP32 */
int result;
/* set the WiFi interface to Station + SoftAP mode without DHCP */
wifi_set_opmode_current(ESP_NOW_WIFI_STA_SOFTAP);
wifi_softap_dhcps_stop();
/* get SoftAP mac address and store it in device address */
wifi_get_macaddr(SOFTAP_IF, dev->addr);
/* set the SoftAP configuration */
struct softap_config ap_conf;
strcpy((char*)ap_conf.password, esp_now_params.softap_pass);
sprintf((char*)ap_conf.ssid, "%s%02x%02x%02x%02x%02x%02x", ESP_NOW_AP_PREFIX,
dev->addr[0], dev->addr[1], dev->addr[2],
dev->addr[3], dev->addr[4], dev->addr[5]);
ap_conf.ssid_len = strlen((char*)ap_conf.ssid);
ap_conf.channel = esp_now_params.channel; /* support 1 ~ 13 */
ap_conf.authmode = AUTH_WPA2_PSK; /* don't support AUTH_WEP in softAP mode. */
ap_conf.ssid_hidden = 0; /* default 0 */
ap_conf.max_connection = 4; /* default 4, max 4 */
ap_conf.beacon_interval = 100; /* support 100 ~ 60000 ms, default 100 */
wifi_softap_set_config_current(&ap_conf);
#if !ESP_NOW_UNICAST
2018-10-08 12:20:49 +02:00
/* all ESP-NOW nodes get the shared mac address on their station interface */
wifi_set_macaddr(ESP_IF_WIFI_STA, (uint8_t*)_esp_now_mac);
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
#endif /* MCU_ESP32 */
2018-10-08 12:20:49 +02:00
/* set the netdev driver */
dev->netdev.driver = &_esp_now_driver;
/* initialize netdev data structure */
dev->scan_event = 0;
2018-10-08 12:20:49 +02:00
mutex_init(&dev->dev_lock);
/* initialize ESP-NOW and register callback functions */
result = esp_now_init();
if (result != ESP_OK) {
LOG_TAG_ERROR("esp_now", "esp_now_init failed with return value %d\n",
result);
return NULL;
2018-10-08 12:20:49 +02:00
}
2018-12-10 14:03:54 +01:00
esp_now_register_send_cb(esp_now_send_cb);
esp_now_register_recv_cb(esp_now_recv_cb);
esp_now_set_self_role(ESP_NOW_ROLE_COMBO);
2018-10-08 12:20:49 +02:00
2018-12-10 14:03:54 +01:00
#if ESP_NOW_UNICAST
2018-10-08 12:20:49 +02:00
/* timer for peer scan initialization */
_esp_now_scan_peers_done = false;
_esp_now_scan_peers_timer.callback = &esp_now_scan_peers_timer_cb;
_esp_now_scan_peers_timer.arg = dev;
/* execute the first scan */
esp_now_scan_peers_start();
2018-10-08 12:20:49 +02:00
2018-12-10 14:03:54 +01:00
#else /* ESP_NOW_UNICAST */
bool res = _esp_now_add_peer((uint8_t*)_esp_now_mac, esp_now_params.channel,
esp_now_params.key);
DEBUG("%s: multicast node add %s\n", __func__, res ? "success" : "error");
2018-12-10 14:03:54 +01:00
#endif /* ESP_NOW_UNICAST */
return dev;
2018-10-08 12:20:49 +02:00
}
static int _init(netdev_t *netdev)
{
DEBUG("%s: %p\n", __func__, netdev);
return 0;
}
static int _send(netdev_t *netdev, const iolist_t *iolist)
{
2018-12-10 14:03:54 +01:00
#if ESP_NOW_UNICAST
2018-10-08 12:20:49 +02:00
if (!_esp_now_scan_peers_done) {
return -ENODEV;
}
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
DEBUG("%s: %p %p\n", __func__, netdev, iolist);
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(netdev != NULL, -ENODEV);
CHECK_PARAM_RET(iolist != NULL && iolist->iol_len == ESP_NOW_ADDR_LEN, -EINVAL);
CHECK_PARAM_RET(iolist->iol_next != NULL, -EINVAL);
2018-10-08 12:20:49 +02:00
esp_now_netdev_t *dev = (esp_now_netdev_t*)netdev;
2018-10-08 12:20:49 +02:00
mutex_lock(&dev->dev_lock);
#if ESP_NOW_UNICAST
uint8_t* _esp_now_dst = NULL;
for (uint8_t i = 0; i < ESP_NOW_ADDR_LEN; i++) {
if (((uint8_t*)iolist->iol_base)[i] != 0xff) {
_esp_now_dst = iolist->iol_base;
break;
2018-10-08 12:20:49 +02:00
}
}
#else
const uint8_t* _esp_now_dst = _esp_now_mac;
2018-12-10 14:03:54 +01:00
#endif
iolist = iolist->iol_next;
2018-10-08 12:20:49 +02:00
uint8_t *data_pos = dev->tx_mem;
uint8_t data_len = 0;
2018-10-08 12:20:49 +02:00
while (iolist) {
if (((int)data_len + iolist->iol_len) > ESP_NOW_MAX_SIZE_RAW) {
DEBUG("%s: payload length exceeds maximum(%u>%u)\n", __func__,
data_len + iolist->iol_len, ESP_NOW_MAX_SIZE_RAW);
return -EBADMSG;
2018-10-08 12:20:49 +02:00
}
memcpy(data_pos, iolist->iol_base, iolist->iol_len);
data_pos += iolist->iol_len;
data_len += iolist->iol_len;
2018-10-08 12:20:49 +02:00
iolist = iolist->iol_next;
2018-10-08 12:20:49 +02:00
}
DEBUG("%s: send %u byte\n", __func__, (unsigned)data_len);
#if defined(MODULE_OD) && ENABLE_DEBUG
od_hex_dump(dev->tx_mem, data_len, OD_WIDTH_DEFAULT);
#endif
2018-10-08 12:20:49 +02:00
if (_esp_now_dst) {
DEBUG("%s: send to esp_now addr %02x:%02x:%02x:%02x:%02x:%02x\n", __func__,
_esp_now_dst[0], _esp_now_dst[1], _esp_now_dst[2],
_esp_now_dst[3], _esp_now_dst[4], _esp_now_dst[5]);
} else {
DEBUG("%s: send esp_now broadcast\n", __func__);
2018-10-08 12:20:49 +02:00
}
_esp_now_sending = 1;
/* send the packet to the peer(s) mac address */
if (esp_now_send((uint8_t*)_esp_now_dst, dev->tx_mem, data_len) == ESP_OK) {
2018-10-08 12:20:49 +02:00
while (_esp_now_sending > 0) {
thread_yield_higher();
}
2018-12-10 14:03:54 +01:00
#ifdef MODULE_NETSTATS_L2
2018-10-08 12:20:49 +02:00
netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE);
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
mutex_unlock(&dev->dev_lock);
return data_len;
} else {
_esp_now_sending = 0;
2018-10-08 12:20:49 +02:00
}
mutex_unlock(&dev->dev_lock);
return -EIO;
}
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
{
DEBUG("%s: %p %p %u %p\n", __func__, netdev, buf, len, info);
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(netdev != NULL, -ENODEV);
2018-10-08 12:20:49 +02:00
esp_now_netdev_t* dev = (esp_now_netdev_t*)netdev;
/* we store source mac address and received data in `buf` */
uint16_t size = dev->rx_len ? ESP_NOW_ADDR_LEN + dev->rx_len : 0;
2018-10-08 12:20:49 +02:00
if (!buf) {
/* get the size of the frame */
if (len > 0 && size) {
/* if len > 0, drop the frame */
dev->rx_len = 0;
}
2018-10-08 12:20:49 +02:00
return size;
}
if (len < size) {
/* buffer is smaller than the number of received bytes */
DEBUG("[esp_now] No space in receive buffers\n");
/* newest API requires to drop the frame in that case */
dev->rx_len = 0;
return -ENOBUFS;
}
/* copy the buffer */
memcpy(buf, dev->rx_mac, ESP_NOW_ADDR_LEN);
memcpy((char *)buf + ESP_NOW_ADDR_LEN, dev->rx_data, dev->rx_len);
dev->rx_len = 0;
uint8_t *mac = buf;
DEBUG("%s: received %d byte from %02x:%02x:%02x:%02x:%02x:%02x\n",
__func__, size - ESP_NOW_ADDR_LEN,
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
#if defined(MODULE_OD) && ENABLE_DEBUG
od_hex_dump(buf + ESP_NOW_ADDR_LEN, size - ESP_NOW_ADDR_LEN, OD_WIDTH_DEFAULT);
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
#if ESP_NOW_UNICAST
if (esp_now_is_peer_exist(mac) <= 0) {
_esp_now_add_peer(mac, esp_now_params.channel, esp_now_params.key);
}
#endif
return size;
2018-10-08 12:20:49 +02:00
}
static int _get(netdev_t *netdev, netopt_t opt, void *val, size_t max_len)
{
DEBUG("%s: %s %p %p %u\n", __func__, netopt2str(opt), netdev, val, max_len);
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(netdev != NULL, -ENODEV);
CHECK_PARAM_RET(val != NULL, -EINVAL);
2018-10-08 12:20:49 +02:00
2018-12-10 14:03:54 +01:00
esp_now_netdev_t *dev = (esp_now_netdev_t*)netdev;
2018-10-08 12:20:49 +02:00
int res = -ENOTSUP;
switch (opt) {
case NETOPT_DEVICE_TYPE:
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(max_len >= sizeof(uint16_t), -EOVERFLOW);
*((uint16_t *)val) = NETDEV_TYPE_ESP_NOW;
2018-10-08 12:20:49 +02:00
res = sizeof(uint16_t);
break;
2018-12-10 14:03:54 +01:00
#ifdef MODULE_GNRC
2018-10-08 12:20:49 +02:00
case NETOPT_PROTO:
CHECK_PARAM_RET(max_len == sizeof(gnrc_nettype_t), -EOVERFLOW);
*((gnrc_nettype_t *)val) = dev->proto;
res = sizeof(gnrc_nettype_t);
break;
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
case NETOPT_MAX_PDU_SIZE:
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(max_len >= sizeof(uint16_t), -EOVERFLOW);
2018-10-08 12:20:49 +02:00
*((uint16_t *)val) = ESP_NOW_MAX_SIZE;
res = sizeof(uint16_t);
break;
case NETOPT_ADDR_LEN:
case NETOPT_SRC_LEN:
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(max_len >= sizeof(uint16_t), -EOVERFLOW);
2018-10-08 12:20:49 +02:00
*((uint16_t *)val) = sizeof(dev->addr);
res = sizeof(uint16_t);
break;
case NETOPT_ADDRESS:
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(max_len >= sizeof(dev->addr), -EOVERFLOW);
2018-10-08 12:20:49 +02:00
memcpy(val, dev->addr, sizeof(dev->addr));
res = sizeof(dev->addr);
break;
default:
DEBUG("%s: %s not supported\n", __func__, netopt2str(opt));
break;
}
return res;
}
static int _set(netdev_t *netdev, netopt_t opt, const void *val, size_t max_len)
{
DEBUG("%s: %s %p %p %u\n", __func__, netopt2str(opt), netdev, val, max_len);
2018-12-10 14:03:54 +01:00
CHECK_PARAM_RET(netdev != NULL, -ENODEV);
CHECK_PARAM_RET(val != NULL, -EINVAL);
2018-10-08 12:20:49 +02:00
esp_now_netdev_t *dev = (esp_now_netdev_t *) netdev;
int res = -ENOTSUP;
switch (opt) {
2018-12-10 14:03:54 +01:00
#ifdef MODULE_GNRC
2018-10-08 12:20:49 +02:00
case NETOPT_PROTO:
CHECK_PARAM_RET(max_len == sizeof(gnrc_nettype_t), -EOVERFLOW);
dev->proto = *((gnrc_nettype_t *)val);
res = sizeof(gnrc_nettype_t);
break;
2018-12-10 14:03:54 +01:00
#endif
2018-10-08 12:20:49 +02:00
case NETOPT_ADDRESS:
CHECK_PARAM_RET(max_len >= sizeof(dev->addr), -EOVERFLOW);
memcpy(dev->addr, val, sizeof(dev->addr));
res = sizeof(dev->addr);
break;
default:
DEBUG("%s: %s not supported\n", __func__, netopt2str(opt));
break;
}
return res;
}
static void _isr(netdev_t *netdev)
{
DEBUG("%s: %p\n", __func__, netdev);
2018-12-10 14:03:54 +01:00
CHECK_PARAM(netdev != NULL);
2018-10-08 12:20:49 +02:00
2018-12-10 14:03:54 +01:00
esp_now_netdev_t *dev = (esp_now_netdev_t*)netdev;
critical_enter();
if (dev->scan_event) {
dev->scan_event--;
critical_exit();
#if ESP_NOW_UNICAST
esp_now_scan_peers_start();
#endif
}
2018-10-08 12:20:49 +02:00
return;
}
static const netdev_driver_t _esp_now_driver =
{
.send = _send,
.recv = _recv,
.init = _init,
.isr = _isr,
.get = _get,
.set = _set,
};
/** @} */