/* * 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_esp_common_esp_now * @{ * * @file * @brief Netdev interface for the ESP-NOW WiFi P2P protocol * * @author Gunar Schorcht */ #define ENABLE_DEBUG (0) #include "debug.h" #include "log.h" #include "tools.h" #include #include #include #include "net/gnrc/netif/raw.h" #include "net/gnrc.h" #include "xtimer.h" #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 "irq_arch.h" #include "nvs_flash/include/nvs_flash.h" #include "esp_now_params.h" #include "esp_now_netdev.h" #include "net/ipv6/hdr.h" #include "net/gnrc/ipv6/nib.h" #define ESP_NOW_UNICAST 1 #define ESP_NOW_WIFI_STA 1 #define ESP_NOW_WIFI_SOFTAP 2 #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)) /** * 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; static const netdev_driver_t _esp_now_driver; /* device thread stack */ static char _esp_now_stack[ESP_NOW_STACKSIZE]; static inline int _get_mac_from_iid(uint8_t *iid, uint8_t *mac) { CHECK_PARAM_RET (iid != NULL, -EINVAL); CHECK_PARAM_RET (mac != NULL, -EINVAL); /* interface id according to */ /* https://tools.ietf.org/html/rfc4291#section-2.5.1 */ mac[0] = iid[0] ^ 0x02; /* invert bit1 */ mac[1] = iid[1]; mac[2] = iid[2]; mac[3] = iid[5]; mac[4] = iid[6]; mac[5] = iid[7]; return 0; } #if ESP_NOW_UNICAST static xtimer_t _esp_now_scan_peers_timer; static bool _esp_now_scan_peers_done = false; static bool _esp_now_add_peer (uint8_t* bssid, uint8_t channel, uint8_t* key) { if (esp_now_is_peer_exist(bssid)) { return false; } 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); 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); } #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 void IRAM_ATTR esp_now_scan_peers_done (void) { 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 which is not already a peer */ if (strncmp((char*)aps[i].ssid, ESP_NOW_AP_PREFIX, ESP_NOW_AP_PREFIX_LEN) == 0 && !esp_now_is_peer_exist(aps[i].bssid)) { /* add the AP as peer */ _esp_now_add_peer(aps[i].bssid, aps[i].primary, esp_now_params.key); } } critical_exit_var(state); } #if ENABLE_DEBUG esp_now_peer_num_t peer_num; esp_now_get_peer_num (&peer_num); DEBUG("associated peers total=%d, encrypted=%d\n", peer_num.total_num, peer_num.encrypt_num); #endif _esp_now_scan_peers_done = true; /* set the time for next scan */ xtimer_set (&_esp_now_scan_peers_timer, esp_now_params.scan_period); mutex_unlock(&_esp_now_dev.dev_lock); } static void esp_now_scan_peers_start (void) { DEBUG("%s\n", __func__); wifi_scan_config_t scan_cfg = { .ssid = NULL, .bssid = NULL, .channel = esp_now_params.channel, .show_hidden = true, .scan_type = WIFI_SCAN_TYPE_ACTIVE, .scan_time.active.min = 0, .scan_time.active.max = 120 /* TODO tune value */ }; esp_wifi_scan_start(&scan_cfg, false); } #define ESP_NOW_EVENT_SCAN_PEERS 1 static kernel_pid_t esp_now_event_handler_pid; char esp_now_event_handler_stack [THREAD_STACKSIZE_DEFAULT]; static void *esp_now_event_handler (void *arg) { msg_t event; while (1) { msg_receive(&event); switch (event.content.value) { case ESP_NOW_EVENT_SCAN_PEERS: esp_now_scan_peers_start (); break; } } return NULL; } static void IRAM_ATTR esp_now_scan_peers_timer_cb (void* arg) { DEBUG("%s\n", __func__); static msg_t event = { .content = { .value = ESP_NOW_EVENT_SCAN_PEERS } }; msg_send(&event, esp_now_event_handler_pid); } #else static const uint8_t _esp_now_mac[6] = { 0x82, 0x73, 0x79, 0x84, 0x79, 0x83 }; /* RIOTOS */ #endif /* ESP_NOW_UNICAST */ static IRAM_ATTR void esp_now_recv_cb (const uint8_t *mac, const uint8_t *data, int len) { #if ESP_NOW_UNICAST if (!_esp_now_scan_peers_done) { /* if peers are not scanned, we cannot receive anything */ return; } #endif if (_esp_now_dev.rx_len) { /* there is already a packet in receive buffer, we drop the new one */ return; } critical_enter(); #if 0 /* don't printf anything in ISR */ printf ("%s\n", __func__); printf ("%s: received %d byte from %02x:%02x:%02x:%02x:%02x:%02x\n", __func__, len, mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); esp_hexdump (data, len, 'b', 16); #endif _esp_now_dev.rx_len = len; memcpy(_esp_now_dev.rx_buf, data, len); memcpy(_esp_now_dev.rx_mac, mac, ESP_NOW_ADDR_LEN); if (_esp_now_dev.netdev.event_callback) { _esp_now_dev.netdev.event_callback((netdev_t*)&_esp_now_dev, NETDEV_EVENT_ISR); } critical_exit(); } static int _esp_now_sending = 0; static void IRAM_ATTR esp_now_send_cb (const uint8_t *mac, esp_now_send_status_t status) { 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--; } } /* * 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__); esp_now_scan_peers_done(); 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 extern esp_err_t esp_system_event_add_handler (system_event_cb_t handler, void *arg); static void esp_now_setup (esp_now_netdev_t* dev) { DEBUG("%s: %p\n", __func__, dev); /* * 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); esp_system_event_add_handler (_esp_system_event_handler, NULL); esp_err_t result; #if CONFIG_ESP32_WIFI_NVS_ENABLED result = nvs_flash_init(); if (result != ESP_OK) { LOG_TAG_ERROR("esp_now", "nfs_flash_init failed with return value %d\n", result); return; } #endif 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; } #ifdef CONFIG_WIFI_COUNTRY /* TODO */ #endif /* 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; } /* 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; } 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; } /* 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; } #if ESP_NOW_UNICAST==0 /* TODO */ /* all ESP-NOW nodes get the shared mac address on their station interface */ wifi_set_macaddr(STATION_IF, (uint8_t*)_esp_now_mac); #endif /* set the netdev driver */ dev->netdev.driver = &_esp_now_driver; /* initialize netdev data structure */ dev->peers_all = 0; dev->peers_enc = 0; 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; } esp_now_register_send_cb (esp_now_send_cb); esp_now_register_recv_cb (esp_now_recv_cb); #if ESP_NOW_UNICAST /* create the ESP_NOW event handler thread */ esp_now_event_handler_pid = thread_create(esp_now_event_handler_stack, sizeof(esp_now_event_handler_stack), ESP_NOW_PRIO + 1, THREAD_CREATE_WOUT_YIELD | THREAD_CREATE_STACKTEST, esp_now_event_handler, NULL, "net-esp-now-event"); /* 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_done(); #else /* ESP_NOW_UNICAST */ #if 0 int res = esp_now_add_peer((uint8_t*)_esp_now_mac, ESP_NOW_ROLE_COMBO, esp_now_params.channel, NULL, 0); DEBUG("%s: multicast node added %d\n", __func__, res); #endif #endif /* ESP_NOW_UNICAST */ } static int _init(netdev_t *netdev) { DEBUG("%s: %p\n", __func__, netdev); #ifdef MODULE_NETSTATS_L2 memset(&netdev->stats, 0x00, sizeof(netstats_t)); #endif return 0; } static int _send(netdev_t *netdev, const iolist_t *iolist) { #if ESP_NOW_UNICAST if (!_esp_now_scan_peers_done) { return -ENODEV; } #endif DEBUG("%s: %p %p\n", __func__, netdev, iolist); CHECK_PARAM_RET (netdev != NULL, -ENODEV); CHECK_PARAM_RET (iolist != NULL, -EINVAL); esp_now_netdev_t* dev = (esp_now_netdev_t*)netdev; mutex_lock(&dev->dev_lock); dev->tx_len = 0; /* load packet data into TX buffer */ for (const iolist_t *iol = iolist; iol; iol = iol->iol_next) { if (dev->tx_len + iol->iol_len > ESP_NOW_MAX_SIZE) { mutex_unlock(&dev->dev_lock); return -EOVERFLOW; } memcpy (dev->tx_buf + dev->tx_len, iol->iol_base, iol->iol_len); dev->tx_len += iol->iol_len; } #if ENABLE_DEBUG printf ("%s: send %d byte\n", __func__, dev->tx_len); /* esp_hexdump (dev->tx_buf, dev->tx_len, 'b', 16); */ #endif _esp_now_sending = 1; uint8_t* _esp_now_dst = 0; #if ESP_NOW_UNICAST ipv6_hdr_t* ipv6_hdr = (ipv6_hdr_t*)dev->tx_buf; uint8_t _esp_now_dst_from_iid[6]; if (ipv6_hdr->dst.u8[0] == 0xff) { /* packets to multicast prefix ff::/8 are sent to all peers */ DEBUG("multicast to all peers\n"); _esp_now_dst = 0; _esp_now_sending = dev->peers_all; #ifdef MODULE_NETSTATS_L2 netdev->stats.tx_mcast_count++; #endif } else if ((byteorder_ntohs(ipv6_hdr->dst.u16[0]) & 0xffc0) == 0xfe80) { /* for link local addresses fe80::/10, the MAC address is derived from dst address */ _get_mac_from_iid(&ipv6_hdr->dst.u8[8], _esp_now_dst_from_iid); DEBUG("link local to %02x:%02x:%02x:%02x:%02x:%02x\n", _esp_now_dst_from_iid[0], _esp_now_dst_from_iid[1], _esp_now_dst_from_iid[2], _esp_now_dst_from_iid[3], _esp_now_dst_from_iid[4], _esp_now_dst_from_iid[5]); _esp_now_dst = _esp_now_dst_from_iid; _esp_now_sending = 1; } else { #ifdef MODULE_GNRC_IPV6_NIB /* for other addresses, try to find an entry in NIB cache */ gnrc_ipv6_nib_nc_t nce; int ret = gnrc_ipv6_nib_get_next_hop_l2addr (&ipv6_hdr->dst, dev->netif, NULL, &nce); if (ret == 0) { /* entry was found in NIB, use MAC adress from the NIB cache entry */ DEBUG("global, next hop to neighbor %02x:%02x:%02x:%02x:%02x:%02x\n", nce.l2addr[0], nce.l2addr[1], nce.l2addr[2], nce.l2addr[3], nce.l2addr[4], nce.l2addr[5]); _esp_now_dst = nce.l2addr; _esp_now_sending = 1; } else { #endif /* entry was not found in NIB, send to all peers */ DEBUG("global, no neibhbor found, multicast to all peers\n"); _esp_now_dst = 0; _esp_now_sending = dev->peers_all; #ifdef MODULE_NETSTATS_L2 netdev->stats.tx_mcast_count++; #endif #ifdef MODULE_GNRC_IPV6_NIB } #endif } #else /* ESP_NOW_UNICAST */ ipv6_hdr_t* ipv6_hdr = (ipv6_hdr_t*)dev->tx_buf; uint8_t _esp_now_dst_from_iid[6]; _esp_now_dst = (uint8_t*)_esp_now_mac; _esp_now_sending = 1; if (ipv6_hdr->dst.u8[0] == 0xff) { /* packets to multicast prefix ff::/8 are sent to all peers */ DEBUG("multicast to all peers\n"); #ifdef MODULE_NETSTATS_L2 netdev->stats.tx_mcast_count++; #endif } else if ((byteorder_ntohs(ipv6_hdr->dst.u16[0]) & 0xffc0) == 0xfe80) { /* for link local addresses fe80::/10, the MAC address is derived from dst address */ _get_mac_from_iid(&ipv6_hdr->dst.u8[8], _esp_now_dst_from_iid); DEBUG("link local to %02x:%02x:%02x:%02x:%02x:%02x\n", _esp_now_dst_from_iid[0], _esp_now_dst_from_iid[1], _esp_now_dst_from_iid[2], _esp_now_dst_from_iid[3], _esp_now_dst_from_iid[4], _esp_now_dst_from_iid[5]); if (esp_now_is_peer_exist(_esp_now_dst_from_iid) > 0) { _esp_now_dst = _esp_now_dst_from_iid; } } else { /* for other addresses, try to find an entry in NIB cache */ gnrc_ipv6_nib_nc_t nce; int ret = gnrc_ipv6_nib_get_next_hop_l2addr (&ipv6_hdr->dst, dev->netif, NULL, &nce); if (ret == 0 && esp_now_is_peer_exist(nce.l2addr) > 0) { /* entry was found in NIB, use MAC adress from the NIB cache entry */ DEBUG("global, next hop to neighbor %02x:%02x:%02x:%02x:%02x:%02x\n", nce.l2addr[0], nce.l2addr[1], nce.l2addr[2], nce.l2addr[3], nce.l2addr[4], nce.l2addr[5]); _esp_now_dst = nce.l2addr; } else { /* entry was not found in NIB, send to all peers */ DEBUG("global, no neibhbor found, multicast to all peers\n"); #ifdef MODULE_NETSTATS_L2 netdev->stats.tx_mcast_count++; #endif } } #endif /* ESP_NOW_UNICAST */ 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]); } /* send the the packet to the peer(s) mac address */ if (esp_now_send (_esp_now_dst, dev->tx_buf, dev->tx_len) == 0) { while (_esp_now_sending > 0) { thread_yield_higher(); } #ifdef MODULE_NETSTATS_L2 netdev->stats.tx_bytes += dev->tx_len; netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE); #endif mutex_unlock(&dev->dev_lock); return dev->tx_len; } else { #ifdef MODULE_NETSTATS_L2 netdev->stats.tx_failed++; #endif } 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); CHECK_PARAM_RET (netdev != NULL, -ENODEV); esp_now_netdev_t* dev = (esp_now_netdev_t*)netdev; mutex_lock(&dev->dev_lock); uint8_t size = dev->rx_len; if (!buf && !len) { /* return the size without dropping received data */ mutex_unlock(&dev->dev_lock); return size; } if (!buf && len) { /* return the size and drop received data */ mutex_unlock(&dev->dev_lock); dev->rx_len = 0; return size; } if (buf && len && dev->rx_len) { if (dev->rx_len > len) { DEBUG("[esp_now] No space in receive buffers\n"); mutex_unlock(&dev->dev_lock); return -ENOBUFS; } #if ENABLE_DEBUG printf ("%s: received %d byte from %02x:%02x:%02x:%02x:%02x:%02x\n", __func__, dev->rx_len, dev->rx_mac[0], dev->rx_mac[1], dev->rx_mac[2], dev->rx_mac[3], dev->rx_mac[4], dev->rx_mac[5]); /* esp_hexdump (dev->rx_buf, dev->rx_len, 'b', 16); */ #endif if (esp_now_is_peer_exist(dev->rx_mac) <= 0) { _esp_now_add_peer(dev->rx_mac, esp_now_params.channel, esp_now_params.key); } memcpy(buf, dev->rx_buf, dev->rx_len); dev->rx_len = 0; #ifdef MODULE_NETSTATS_L2 netdev->stats.rx_count++; netdev->stats.rx_bytes += size; #endif mutex_unlock(&dev->dev_lock); return size; } mutex_unlock(&dev->dev_lock); return -EINVAL; } static inline int _get_iid(esp_now_netdev_t *dev, eui64_t *value, size_t max_len) { CHECK_PARAM_RET (max_len >= sizeof(eui64_t), -EOVERFLOW); /* interface id according to */ /* https://tools.ietf.org/html/rfc4291#section-2.5.1 */ value->uint8[0] = dev->addr[0] ^ 0x02; /* invert bit1 */ value->uint8[1] = dev->addr[1]; value->uint8[2] = dev->addr[2]; value->uint8[3] = 0xff; value->uint8[4] = 0xfe; value->uint8[5] = dev->addr[3]; value->uint8[6] = dev->addr[4]; value->uint8[7] = dev->addr[5]; return sizeof(eui64_t); } 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); CHECK_PARAM_RET (netdev != NULL, -ENODEV); CHECK_PARAM_RET (val != NULL, -EINVAL); esp_now_netdev_t *dev = (esp_now_netdev_t *)netdev; int res = -ENOTSUP; switch (opt) { case NETOPT_DEVICE_TYPE: CHECK_PARAM_RET (max_len >= sizeof(uint16_t), -EOVERFLOW); *((uint16_t *)val) = NETDEV_TYPE_ESP_NOW; res = sizeof(uint16_t); break; #ifdef MODULE_GNRC 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; #endif case NETOPT_MAX_PACKET_SIZE: CHECK_PARAM_RET (max_len >= sizeof(uint16_t), -EOVERFLOW); *((uint16_t *)val) = ESP_NOW_MAX_SIZE; res = sizeof(uint16_t); break; case NETOPT_ADDR_LEN: case NETOPT_SRC_LEN: CHECK_PARAM_RET (max_len >= sizeof(uint16_t), -EOVERFLOW); *((uint16_t *)val) = sizeof(dev->addr); res = sizeof(uint16_t); break; case NETOPT_ADDRESS: CHECK_PARAM_RET (max_len >= sizeof(dev->addr), -EOVERFLOW); memcpy(val, dev->addr, sizeof(dev->addr)); res = sizeof(dev->addr); break; case NETOPT_IPV6_IID: res = _get_iid(dev, val, max_len); break; #ifdef MODULE_NETSTATS_L2 case NETOPT_STATS: CHECK_PARAM_RET (max_len == sizeof(uintptr_t), -EOVERFLOW); *((netstats_t **)val) = &netdev->stats; res = sizeof(uintptr_t); break; #endif 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); CHECK_PARAM_RET (netdev != NULL, -ENODEV); CHECK_PARAM_RET (val != NULL, -EINVAL); esp_now_netdev_t *dev = (esp_now_netdev_t *) netdev; int res = -ENOTSUP; switch (opt) { #ifdef MODULE_GNRC 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; #endif 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); CHECK_PARAM (netdev != NULL); esp_now_netdev_t *dev = (esp_now_netdev_t *) netdev; dev->netdev.event_callback(netdev, NETDEV_EVENT_RX_COMPLETE); return; } static const netdev_driver_t _esp_now_driver = { .send = _send, .recv = _recv, .init = _init, .isr = _isr, .get = _get, .set = _set, }; void auto_init_esp_now (void) { LOG_TAG_INFO("esp_now", "initializing ESP-NOW device\n"); esp_now_setup(&_esp_now_dev); _esp_now_dev.netif = gnrc_netif_raw_create(_esp_now_stack, ESP_NOW_STACKSIZE, ESP_NOW_PRIO, "net-esp-now", (netdev_t *)&_esp_now_dev); } /** @} */