Skip to content
Snippets Groups Projects
Commit 6be81fbd authored by Timo Rothenpieler's avatar Timo Rothenpieler
Browse files

esp_now: use custom netif instead of raw one


Avoids parsing IPv6 packets to determine destination address.
Allows using 6Lo over ESP-NOW, which is required due to the low MTU of
ESP-NOW.

Co-authored-by: default avatarGunar Schorcht <gunar@schorcht.net>
parent 34c42cbc
No related branches found
No related tags found
No related merge requests found
/*
* Copyright (C) 2018 Timo Rothenpieler
*
* 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_esp_common_esp_now
* @{
*
* @file
* @brief Netif interface for the ESP-NOW WiFi P2P protocol
*
* @author Timo Rothenpieler <timo.rothenpieler@uni-bremen.de>
*/
#include <assert.h>
#include <stdlib.h>
#include <sys/uio.h>
#include "net/netdev.h"
#include "net/gnrc.h"
#include "esp_now_params.h"
#include "esp_now_netdev.h"
#include "esp_now_gnrc.h"
#include "net/gnrc/netif.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
static int _send(gnrc_netif_t *netif, gnrc_pktsnip_t *pkt)
{
uint8_t mac[ESP_NOW_ADDR_LEN];
esp_now_pkt_hdr_t esp_hdr;
netdev_t *dev = netif->dev;
assert(pkt != NULL);
if (pkt->type != GNRC_NETTYPE_NETIF) {
DEBUG("gnrc_esp_now: First header was not generic netif header\n");
gnrc_pktbuf_release(pkt);
return -EBADMSG;
}
gnrc_netif_hdr_t *netif_hdr = (gnrc_netif_hdr_t*)pkt->data;
gnrc_pktsnip_t *payload = pkt->next;
if (netif_hdr->flags & (GNRC_NETIF_HDR_FLAGS_BROADCAST | GNRC_NETIF_HDR_FLAGS_MULTICAST)) {
/* ESP-NOW does not support multicast, always broadcast */
memset(mac, 0xff, ESP_NOW_ADDR_LEN);
} else if (netif_hdr->dst_l2addr_len == ESP_NOW_ADDR_LEN) {
memcpy(mac, gnrc_netif_hdr_get_dst_addr(netif_hdr), ESP_NOW_ADDR_LEN);
} else {
DEBUG("gnrc_esp_now: destination address had unexpected format"
"(flags=%d, dst_l2addr_len=%d)\n", netif_hdr->flags, netif_hdr->dst_l2addr_len);
gnrc_pktbuf_release(pkt);
return -EBADMSG;
}
iolist_t esp_hdr_iolist = {
.iol_base = &esp_hdr,
.iol_len = sizeof(esp_hdr),
.iol_next = (iolist_t*)payload
};
iolist_t iolist = {
.iol_base = mac,
.iol_len = sizeof(mac),
.iol_next = &esp_hdr_iolist
};
switch (payload->type) {
#ifdef MODULE_GNRC_SIXLOWPAN
case GNRC_NETTYPE_SIXLOWPAN:
esp_hdr.flags = ESP_NOW_PKT_HDR_FLAG_SIXLO;
break;
#endif
default:
esp_hdr.flags = 0;
}
DEBUG("gnrc_esp_now: sending packet to %02x:%02x:%02x:%02x:%02x:%02x with size %u\n",
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5], (unsigned)payload->size);
int res = dev->driver->send(dev, &iolist);
gnrc_pktbuf_release(pkt);
return res;
}
static gnrc_pktsnip_t *_recv(gnrc_netif_t *netif)
{
netdev_t *dev = netif->dev;
esp_now_netdev_t *esp_now = (esp_now_netdev_t*)dev;
int bytes_expected = dev->driver->recv(dev, NULL, 0, NULL);
if (bytes_expected <= 0) {
DEBUG("gnrc_esp_now: failed receiving packet: %d\n", bytes_expected);
return NULL;
}
gnrc_pktsnip_t *pkt;
pkt = gnrc_pktbuf_add(NULL, NULL,
bytes_expected,
GNRC_NETTYPE_UNDEF);
if (!pkt) {
DEBUG("gnrc_esp_now: cannot allocate pktsnip.\n");
/* drop the packet */
dev->driver->recv(dev, NULL, bytes_expected, NULL);
return NULL;
}
int nread = dev->driver->recv(dev, pkt->data, bytes_expected, NULL);
if (nread <= 0) {
DEBUG("gnrc_esp_now: read error %d\n", nread);
goto err;
}
if (nread < bytes_expected) {
DEBUG("gnrc_esp_now: reallocating.\n");
gnrc_pktbuf_realloc_data(pkt, nread);
}
gnrc_pktsnip_t *mac_hdr;
mac_hdr = gnrc_pktbuf_mark(pkt, ESP_NOW_ADDR_LEN, GNRC_NETTYPE_UNDEF);
if (!mac_hdr) {
DEBUG("gnrc_esp_now: no space left in packet buffer\n");
goto err;
}
gnrc_pktsnip_t *esp_hdr;
esp_hdr = gnrc_pktbuf_mark(pkt, sizeof(esp_now_pkt_hdr_t), GNRC_NETTYPE_UNDEF);
if (!esp_hdr) {
DEBUG("gnrc_esp_now: no space left in packet buffer\n");
pkt = mac_hdr;
goto err;
}
esp_now_pkt_hdr_t *hdr = (esp_now_pkt_hdr_t*)esp_hdr->data;
#ifdef MODULE_L2FILTER
if (!l2filter_pass(dev->filter, mac_hdr->data, ESP_NOW_ADDR_LEN)) {
DEBUG("gnrc_esp_now: incoming packet filtered by l2filter\n");
pkt = mac_hdr;
goto err;
}
#endif
pkt->type = GNRC_NETTYPE_UNDEF;
#ifdef MODULE_GNRC_SIXLOWPAN
if (hdr->flags & ESP_NOW_PKT_HDR_FLAG_SIXLO) {
pkt->type = GNRC_NETTYPE_SIXLOWPAN;
}
#endif
gnrc_pktsnip_t *netif_hdr = gnrc_netif_hdr_build(mac_hdr->data, ESP_NOW_ADDR_LEN,
esp_now->addr, ESP_NOW_ADDR_LEN);
if (!netif_hdr) {
DEBUG("gnrc_esp_now: no space left in packet buffer\n");
pkt = mac_hdr;
goto err;
}
((gnrc_netif_hdr_t *)netif_hdr->data)->if_pid = netif->pid;
uint8_t *mac = mac_hdr->data;
DEBUG("gnrc_esp_now: received packet from %02x:%02x:%02x:%02x:%02x:%02x of length %u\n",
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5], pkt->size);
gnrc_pktbuf_remove_snip(pkt, mac_hdr);
gnrc_pktbuf_remove_snip(pkt, esp_hdr);
LL_APPEND(pkt, netif_hdr);
return pkt;
err:
gnrc_pktbuf_release(pkt);
return NULL;
}
static const gnrc_netif_ops_t _esp_now_ops = {
.send = _send,
.recv = _recv,
.get = gnrc_netif_get_from_netdev,
.set = gnrc_netif_set_from_netdev,
};
gnrc_netif_t *gnrc_netif_esp_now_create(char *stack, int stacksize, char priority,
char *name, netdev_t *dev)
{
return gnrc_netif_create(stack, stacksize, priority, name, dev, &_esp_now_ops);
}
/* device thread stack */
static char _esp_now_stack[ESP_NOW_STACKSIZE];
void auto_init_esp_now(void)
{
LOG_TAG_INFO("esp_now", "initializing ESP-NOW device\n");
esp_now_netdev_t *esp_now_dev = netdev_esp_now_setup();
if (!esp_now_dev) {
LOG_ERROR("[auto_init_netif] error initializing esp_now\n");
} else {
gnrc_netif_esp_now_create(_esp_now_stack, sizeof(_esp_now_stack),
ESP_NOW_PRIO,
"net-esp-now",
&esp_now_dev->netdev);
}
}
/** @} */
/*
* Copyright (C) 2018 Timo Rothenpieler
*
* 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_esp_common_esp_now
* @{
*
* @file
* @brief ESP-NOW adaption for @ref net_gnrc_netif
*
* @author Timo Rothenpieler <timo.rothenpieler@uni-bremen.de>
*/
#ifndef ESP_NOW_GNRC_H
#define ESP_NOW_GNRC_H
#include "net/gnrc/netif.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Creates the ESP-NOW network interface
* @see gnrc_netif_create
* @param [in] stack The stack for the network interface's thread.
* @param [in] stacksize Size of stack.
* @param [in] priority Priority for the network interface's thread.
* @param [in] name Name for the network interface. May be NULL.
* @param [in] dev Device for the interface.
*/
gnrc_netif_t *gnrc_netif_esp_now_create(char *stack, int stacksize, char priority,
char *name, netdev_t *dev);
#ifdef __cplusplus
}
#endif
#endif /* ESP_NOW_GNRC_H */
/** @} */
This diff is collapsed.
......@@ -14,32 +14,67 @@
* @brief Netdev interface for the ESP-NOW WiFi P2P protocol
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @author Timo Rothenpieler <timo.rothenpieler@uni-bremen.de>
*/
#ifndef ESP_NOW_NETDEV_H
#define ESP_NOW_NETDEV_H
#include "net/netdev.h"
#include "ringbuffer.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Maximum packet size that can be used with ESP-NOW
* @brief Maximum raw packet size that can be used with ESP-NOW (including headers)
*/
#define ESP_NOW_MAX_SIZE (250)
#define ESP_NOW_MAX_SIZE_RAW (250)
/**
* @brief Length of ESP-NOW addresses
*/
#define ESP_NOW_ADDR_LEN ETHERNET_ADDR_LEN
/**
* @brief Size of non-data header elements in ESP-NOW packet
*/
#define ESP_NOW_HEADER_LENGTH (sizeof(esp_now_pkt_hdr_t))
/**
* @brief Maximum packet size that can be used with ESP-NOW
*/
#define ESP_NOW_MAX_SIZE (ESP_NOW_MAX_SIZE_RAW - ESP_NOW_HEADER_LENGTH)
/**
* @brief buffer size used for RX buffering
*
* Reduce this value if your expected traffic does not include full IPv6 MTU
* sized packets.
*/
#ifndef ESP_NOW_BUFSIZE
#define ESP_NOW_BUFSIZE (1500)
#endif
/**
* @brief Reference to the netdev device driver struct
*/
extern const netdev_driver_t esp_now_driver;
/**
* @brief Header with neccesary flags for ESP-NOW packets
*/
typedef struct __attribute__((packed))
{
uint8_t flags; /**< Flags */
} esp_now_pkt_hdr_t;
/**
* @brief Packet is carrying 6Lo data
*/
#define ESP_NOW_PKT_HDR_FLAG_SIXLO (1)
/**
* @brief Device descriptor for ESP-NOW devices
*/
......@@ -47,28 +82,32 @@ typedef struct
{
netdev_t netdev; /**< netdev parent struct */
uint8_t addr[ESP_NOW_ADDR_LEN]; /**< device addr (MAC address) */
uint8_t rx_len; /**< number of bytes received */
uint8_t rx_buf[ESP_NOW_MAX_SIZE]; /**< receive buffer */
uint8_t rx_mac[ESP_NOW_ADDR_LEN]; /**< source address */
uint8_t addr[ESP_NOW_ADDR_LEN]; /**< device addr (MAC address) */
uint8_t tx_len; /**< number of bytes in transmit buffer */
uint8_t tx_buf[ESP_NOW_MAX_SIZE]; /**< transmit buffer */
uint8_t rx_mem[ESP_NOW_BUFSIZE]; /**< memory holding incoming packages */
ringbuffer_t rx_buf; /**< ringbuffer for incoming packages */
gnrc_netif_t* netif; /**< reference to the corresponding netif */
uint8_t tx_mem[ESP_NOW_MAX_SIZE_RAW]; /**< memory holding outgoing package */
#ifdef MODULE_GNRC
gnrc_nettype_t proto; /**< protocol for upper layer */
#endif
uint8_t peers_all; /**< number of peers reachable */
uint8_t peers_enc; /**< number of encrypted peers */
mutex_t dev_lock; /**< device is already in use */
mutex_t rx_lock; /**< rx_buf handling in progress */
bool recv_event; /**< ESP-NOW frame received */
bool scan_event; /**< ESP-NOW peers have to be scannged */
} esp_now_netdev_t;
/**
* @brief netdev <-> esp_npw glue code initialization function
*
* @return NULL on error, pointer to esp_now_netdev on success
*/
esp_now_netdev_t *netdev_esp_now_setup(void);
#ifdef __cplusplus
}
#endif
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment