mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
175 lines
5.4 KiB
C
175 lines
5.4 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 "lwip_default_hooks.h"
|
|
#include "openthread/error.h"
|
|
#include "openthread/ip6.h"
|
|
#include "openthread/link.h"
|
|
#include "openthread/message.h"
|
|
|
|
#define OPENTHREAD_IP6_MTU 1280
|
|
|
|
#if CONFIG_OPENTHREAD_BORDER_ROUTER
|
|
#if CONFIG_LWIP_IPV6_NUM_ADDRESSES != 12
|
|
#error CONFIG_LWIP_IPV6_NUM_ADDRESSES should be set to 12, please configure it using `idf.py menuconfig`
|
|
#endif
|
|
#endif
|
|
|
|
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;
|
|
#if CONFIG_LWIP_IPV4
|
|
netif->output = NULL;
|
|
#endif
|
|
netif->output_ip6 = openthread_output_ip6;
|
|
netif->mld_mac_filter = openthread_netif_multicast_handler;
|
|
netif_set_link_up(netif);
|
|
|
|
return ERR_OK;
|
|
}
|
|
|
|
const ip_addr_t *lwip_hook_ip6_select_source_address(struct netif *netif, const ip6_addr_t *dest)
|
|
{
|
|
const ip6_addr_t *cur_addr;
|
|
uint8_t idx = 0;
|
|
// Only process with ot netif.
|
|
if (!(netif->name[0] == 'o' && netif->name[1] == 't')) {
|
|
return NULL;
|
|
}
|
|
// Currently, prefer the address with the same prefix of the destination address.
|
|
// If no address found, return NULL for selection source address using the default algorithm.
|
|
for (idx = 0; idx < LWIP_IPV6_NUM_ADDRESSES; idx++) {
|
|
if (!ip6_addr_isvalid(netif_ip6_addr_state(netif, idx))) {
|
|
continue;
|
|
}
|
|
cur_addr = netif_ip6_addr(netif, idx);
|
|
if (ip6_addr_netcmp_zoneless(cur_addr, dest)) {
|
|
return netif_ip_addr6(netif, idx);
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|