mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
144 lines
4.3 KiB
C
144 lines
4.3 KiB
C
/*
|
|
* SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include "esp_netif.h"
|
|
#include "esp_netif_net_stack.h"
|
|
#include "esp_openthread.h"
|
|
#include "esp_openthread_lock.h"
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "lwip/err.h"
|
|
#include "lwip/esp_netif_net_stack.h"
|
|
#include "lwip/netif.h"
|
|
#include "lwip/pbuf.h"
|
|
#include "openthread/error.h"
|
|
#include "openthread/ip6.h"
|
|
#include "openthread/link.h"
|
|
#include "openthread/message.h"
|
|
|
|
#define OPENTHREAD_IP6_MTU 1280
|
|
|
|
static err_t openthread_netif_init(struct netif *netif);
|
|
static void openthread_netif_input(void *h, void *buffer, size_t len, void *eb);
|
|
|
|
const struct esp_netif_netstack_config g_esp_netif_netstack_default_openthread = {
|
|
.lwip = {
|
|
.init_fn = openthread_netif_init,
|
|
.input_fn = openthread_netif_input,
|
|
}};
|
|
|
|
static err_t openthread_output_ip6(struct netif *netif, struct pbuf *p, const struct ip6_addr *peer_addr)
|
|
{
|
|
struct pbuf *q = p;
|
|
esp_netif_t *esp_netif = esp_netif_get_handle_from_netif_impl(netif);
|
|
esp_err_t ret = ESP_FAIL;
|
|
if (!esp_netif) {
|
|
LWIP_DEBUGF(NETIF_DEBUG, ("corresponding esp-netif is NULL: netif=%p pbuf=%p len=%d\n", netif, p, p->len));
|
|
return ERR_IF;
|
|
}
|
|
|
|
if (q->next == NULL) {
|
|
ret = esp_netif_transmit(esp_netif, q->payload, q->len);
|
|
} else {
|
|
LWIP_DEBUGF(PBUF_DEBUG, ("low_level_output: pbuf is a list, application may has bug"));
|
|
q = pbuf_alloc(PBUF_RAW_TX, p->tot_len, PBUF_RAM);
|
|
if (q != NULL) {
|
|
pbuf_copy(q, p);
|
|
} else {
|
|
return ERR_MEM;
|
|
}
|
|
ret = esp_netif_transmit(esp_netif, q->payload, q->len);
|
|
/* content in payload has been copied to OpenThread queue, it's safe to free pbuf now */
|
|
pbuf_free(q);
|
|
}
|
|
/* Check error */
|
|
switch (ret) {
|
|
case ESP_ERR_NO_MEM:
|
|
return ERR_MEM;
|
|
|
|
case ESP_OK:
|
|
return ERR_OK;
|
|
|
|
default:
|
|
return ERR_ABRT;
|
|
}
|
|
}
|
|
|
|
static void openthread_netif_input(void *h, void *buffer, size_t len, void *eb)
|
|
{
|
|
struct netif *netif = h;
|
|
struct pbuf *p;
|
|
otMessage *message = (otMessage *)buffer;
|
|
|
|
if (unlikely(buffer == NULL || !netif_is_up(netif))) {
|
|
return;
|
|
}
|
|
|
|
/* Allocate LINK buffer in case it's forwarded to WiFi/ETH */
|
|
p = pbuf_alloc(PBUF_LINK, len, PBUF_POOL);
|
|
if (p == NULL) {
|
|
LWIP_DEBUGF(NETIF_DEBUG, ("Failed to allocate input pbuf for OpenThread netif\n"));
|
|
return;
|
|
}
|
|
|
|
if (unlikely(otMessageRead(message, 0, p->payload, len) != otMessageGetLength(message))) {
|
|
LWIP_DEBUGF(NETIF_DEBUG, ("Failed to read OpenThread message\n"));
|
|
pbuf_free(p);
|
|
return;
|
|
}
|
|
|
|
/* full packet send to tcpip_thread to process */
|
|
if (unlikely(netif->input(p, netif) != ERR_OK)) {
|
|
LWIP_DEBUGF(NETIF_DEBUG, ("openthread_netif_input: IP input error\n"));
|
|
pbuf_free(p);
|
|
}
|
|
/* the pbuf will be free in upper layer, eg: tcpip_input */
|
|
}
|
|
|
|
static err_t openthread_netif_multicast_handler(struct netif *netif, const ip6_addr_t *group,
|
|
enum netif_mac_filter_action action)
|
|
{
|
|
otError error = OT_ERROR_NONE;
|
|
otIp6Address multicast_addr;
|
|
|
|
memcpy(multicast_addr.mFields.m8, group->addr, sizeof(group->addr));
|
|
esp_openthread_task_switching_lock_acquire(portMAX_DELAY);
|
|
if (action == NETIF_ADD_MAC_FILTER) {
|
|
error = otIp6SubscribeMulticastAddress(esp_openthread_get_instance(), &multicast_addr);
|
|
} else {
|
|
error = otIp6UnsubscribeMulticastAddress(esp_openthread_get_instance(), &multicast_addr);
|
|
}
|
|
esp_openthread_task_switching_lock_release();
|
|
switch (error) {
|
|
case OT_ERROR_NONE:
|
|
case OT_ERROR_ALREADY:
|
|
return ERR_OK;
|
|
case OT_ERROR_NO_BUFS:
|
|
return ERR_MEM;
|
|
case OT_ERROR_INVALID_ARGS:
|
|
return ERR_ARG;
|
|
default:
|
|
return ERR_IF;
|
|
}
|
|
}
|
|
|
|
static err_t openthread_netif_init(struct netif *netif)
|
|
{
|
|
netif->name[0] = 'o';
|
|
netif->name[1] = 't';
|
|
netif->hwaddr_len = sizeof(otExtAddress);
|
|
memset(netif->hwaddr, 0, sizeof(netif->hwaddr));
|
|
netif->mtu = OPENTHREAD_IP6_MTU;
|
|
netif->flags = NETIF_FLAG_BROADCAST;
|
|
netif->output = NULL;
|
|
netif->output_ip6 = openthread_output_ip6;
|
|
netif->mld_mac_filter = openthread_netif_multicast_handler;
|
|
netif_set_link_up(netif);
|
|
|
|
return ERR_OK;
|
|
}
|