2018-08-15 12:58:57 +02:00
|
|
|
/** @file
|
|
|
|
* @brief IPv6 Neighbor related functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 Intel Corporation
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* By default this prints too much data, set the value to 1 to see
|
|
|
|
* neighbor cache contents.
|
|
|
|
*/
|
|
|
|
#define NET_DEBUG_NBR 0
|
|
|
|
|
2018-11-30 11:54:56 +01:00
|
|
|
#include <logging/log.h>
|
|
|
|
LOG_MODULE_DECLARE(net_ipv6, CONFIG_NET_IPV6_LOG_LEVEL);
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <net/net_core.h>
|
|
|
|
#include <net/net_pkt.h>
|
|
|
|
#include <net/net_stats.h>
|
|
|
|
#include <net/net_context.h>
|
|
|
|
#include <net/net_mgmt.h>
|
|
|
|
#include "net_private.h"
|
|
|
|
#include "connection.h"
|
|
|
|
#include "icmpv6.h"
|
|
|
|
#include "udp_internal.h"
|
|
|
|
#include "tcp_internal.h"
|
|
|
|
#include "ipv6.h"
|
|
|
|
#include "nbr.h"
|
|
|
|
#include "6lo.h"
|
|
|
|
#include "route.h"
|
|
|
|
#include "net_stats.h"
|
|
|
|
|
|
|
|
/* Timeout value to be used when allocating net buffer during various
|
|
|
|
* neighbor discovery procedures.
|
|
|
|
*/
|
|
|
|
#define ND_NET_BUF_TIMEOUT K_MSEC(100)
|
|
|
|
|
|
|
|
/* Timeout for various buffer allocations in this file. */
|
|
|
|
#define NET_BUF_TIMEOUT K_MSEC(50)
|
|
|
|
|
|
|
|
/* Maximum reachable time value specified in RFC 4861 section
|
|
|
|
* 6.2.1. Router Configuration Variables, AdvReachableTime
|
|
|
|
*/
|
|
|
|
#define MAX_REACHABLE_TIME 3600000
|
|
|
|
|
|
|
|
/* IPv6 minimum link MTU specified in RFC 8200 section 5
|
|
|
|
* Packet Size Issues
|
|
|
|
*/
|
|
|
|
#define MIN_IPV6_MTU NET_IPV6_MTU
|
|
|
|
#define MAX_IPV6_MTU 0xffff
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
#if defined(CONFIG_NET_IPV6_NBR_CACHE) || defined(CONFIG_NET_IPV6_ND)
|
|
|
|
/* Global stale counter, whenever ipv6 neighbor enters into
|
|
|
|
* stale state, stale counter is incremented by one.
|
|
|
|
* When network stack tries to add new neighbor and if table
|
|
|
|
* is full, oldest (oldest stale counter) neighbor in stale
|
|
|
|
* state will be removed from the table and new entry will be
|
|
|
|
* added.
|
|
|
|
*/
|
|
|
|
static u32_t stale_counter;
|
|
|
|
|
|
|
|
static struct k_sem nbr_lock;
|
|
|
|
#endif
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
static struct k_delayed_work ipv6_nd_reachable_timer;
|
|
|
|
static void ipv6_nd_reachable_timeout(struct k_work *work);
|
2019-03-15 09:35:28 +01:00
|
|
|
static void ipv6_nd_restart_reachable_timer(struct net_nbr *nbr, s64_t time);
|
2018-08-15 12:58:57 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_NBR_CACHE)
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
/* Protocol constants from RFC 4861 Chapter 10 */
|
2018-08-15 12:58:57 +02:00
|
|
|
#define MAX_MULTICAST_SOLICIT 3
|
|
|
|
#define MAX_UNICAST_SOLICIT 3
|
2019-03-15 09:35:28 +01:00
|
|
|
#define DELAY_FIRST_PROBE_TIME K_SECONDS(5)
|
|
|
|
#define RETRANS_TIMER K_MSEC(1000)
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
extern void net_neighbor_data_remove(struct net_nbr *nbr);
|
|
|
|
extern void net_neighbor_table_clear(struct net_nbr_table *table);
|
|
|
|
|
|
|
|
/** Neighbor Solicitation reply timer */
|
|
|
|
static struct k_delayed_work ipv6_ns_reply_timer;
|
|
|
|
|
|
|
|
NET_NBR_POOL_INIT(net_neighbor_pool,
|
|
|
|
CONFIG_NET_IPV6_MAX_NEIGHBORS,
|
|
|
|
sizeof(struct net_ipv6_nbr_data),
|
|
|
|
net_neighbor_data_remove,
|
|
|
|
0);
|
|
|
|
|
|
|
|
NET_NBR_TABLE_INIT(NET_NBR_GLOBAL,
|
|
|
|
neighbor,
|
|
|
|
net_neighbor_pool,
|
|
|
|
net_neighbor_table_clear);
|
|
|
|
|
|
|
|
const char *net_ipv6_nbr_state2str(enum net_ipv6_nbr_state state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
|
|
|
case NET_IPV6_NBR_STATE_INCOMPLETE:
|
|
|
|
return "incomplete";
|
|
|
|
case NET_IPV6_NBR_STATE_REACHABLE:
|
|
|
|
return "reachable";
|
|
|
|
case NET_IPV6_NBR_STATE_STALE:
|
|
|
|
return "stale";
|
|
|
|
case NET_IPV6_NBR_STATE_DELAY:
|
|
|
|
return "delay";
|
|
|
|
case NET_IPV6_NBR_STATE_PROBE:
|
|
|
|
return "probe";
|
|
|
|
case NET_IPV6_NBR_STATE_STATIC:
|
|
|
|
return "static";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "<invalid state>";
|
|
|
|
}
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
static inline struct net_nbr *get_nbr(int idx)
|
|
|
|
{
|
|
|
|
return &net_neighbor_pool[idx].nbr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct net_nbr *get_nbr_from_data(struct net_ipv6_nbr_data *data)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
struct net_nbr *nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (nbr->data == (u8_t *)data) {
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
static void ipv6_nbr_set_state(struct net_nbr *nbr,
|
|
|
|
enum net_ipv6_nbr_state new_state)
|
|
|
|
{
|
|
|
|
if (new_state == net_ipv6_nbr_data(nbr)->state ||
|
|
|
|
net_ipv6_nbr_data(nbr)->state == NET_IPV6_NBR_STATE_STATIC) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
NET_DBG("nbr %p %s -> %s", nbr,
|
|
|
|
net_ipv6_nbr_state2str(net_ipv6_nbr_data(nbr)->state),
|
|
|
|
net_ipv6_nbr_state2str(new_state));
|
|
|
|
|
|
|
|
net_ipv6_nbr_data(nbr)->state = new_state;
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
if (net_ipv6_nbr_data(nbr)->state == NET_IPV6_NBR_STATE_STALE) {
|
|
|
|
if (stale_counter + 1 != UINT32_MAX) {
|
|
|
|
net_ipv6_nbr_data(nbr)->stale_counter = stale_counter++;
|
|
|
|
} else {
|
|
|
|
/* Global stale counter reached UINT32_MAX, reset it and
|
|
|
|
* respective neighbors stale counter too.
|
|
|
|
*/
|
|
|
|
struct net_nbr *n = NULL;
|
|
|
|
struct net_ipv6_nbr_data *data = NULL;
|
|
|
|
int i;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
k_sem_take(&nbr_lock, K_FOREVER);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
stale_counter = 0U;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
n = get_nbr(i);
|
|
|
|
if (!n || !n->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = net_ipv6_nbr_data(nbr);
|
|
|
|
if (!data) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->state != NET_IPV6_NBR_STATE_STALE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->stale_counter = stale_counter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_sem_give(&nbr_lock);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct iface_cb_data {
|
|
|
|
net_nbr_cb_t cb;
|
|
|
|
void *user_data;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void iface_cb(struct net_if *iface, void *user_data)
|
|
|
|
{
|
|
|
|
struct iface_cb_data *data = user_data;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
struct net_nbr *nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (!nbr->ref || nbr->iface != iface) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->cb(nbr, data->user_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void net_ipv6_nbr_foreach(net_nbr_cb_t cb, void *user_data)
|
|
|
|
{
|
|
|
|
struct iface_cb_data cb_data = {
|
|
|
|
.cb = cb,
|
|
|
|
.user_data = user_data,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Return the neighbors according to network interface. This makes it
|
|
|
|
* easier in the callback to use the neighbor information.
|
|
|
|
*/
|
|
|
|
net_if_foreach(iface_cb, &cb_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if NET_DEBUG_NBR
|
|
|
|
void nbr_print(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
struct net_nbr *nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (!nbr->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
NET_DBG("[%d] %p %d/%d/%d/%d/%d pending %p iface %p idx %d "
|
|
|
|
"ll %s addr %s",
|
|
|
|
i, nbr, nbr->ref, net_ipv6_nbr_data(nbr)->ns_count,
|
|
|
|
net_ipv6_nbr_data(nbr)->is_router,
|
|
|
|
net_ipv6_nbr_data(nbr)->state,
|
|
|
|
net_ipv6_nbr_data(nbr)->link_metric,
|
|
|
|
net_ipv6_nbr_data(nbr)->pending,
|
|
|
|
nbr->iface, nbr->idx,
|
|
|
|
nbr->idx == NET_NBR_LLADDR_UNKNOWN ? "?" :
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ll_addr(
|
2018-08-15 12:58:57 +02:00
|
|
|
net_nbr_get_lladdr(nbr->idx)->addr,
|
2018-10-02 13:57:55 +02:00
|
|
|
net_nbr_get_lladdr(nbr->idx)->len)),
|
|
|
|
log_strdup(net_sprint_ipv6_addr(
|
|
|
|
&net_ipv6_nbr_data(nbr)->addr)));
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define nbr_print(...)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static struct net_nbr *nbr_lookup(struct net_nbr_table *table,
|
|
|
|
struct net_if *iface,
|
|
|
|
struct in6_addr *addr)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
struct net_nbr *nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (!nbr->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iface && nbr->iface != iface) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (net_ipv6_addr_cmp(&net_ipv6_nbr_data(nbr)->addr, addr)) {
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void nbr_clear_ns_pending(struct net_ipv6_nbr_data *data)
|
|
|
|
{
|
|
|
|
data->send_ns = 0;
|
|
|
|
|
|
|
|
if (data->pending) {
|
|
|
|
net_pkt_unref(data->pending);
|
|
|
|
data->pending = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void nbr_free(struct net_nbr *nbr)
|
|
|
|
{
|
|
|
|
NET_DBG("nbr %p", nbr);
|
|
|
|
|
|
|
|
nbr_clear_ns_pending(net_ipv6_nbr_data(nbr));
|
|
|
|
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable = 0;
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable_timeout = 0;
|
|
|
|
|
|
|
|
net_nbr_unref(nbr);
|
|
|
|
net_nbr_unlink(nbr, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool net_ipv6_nbr_rm(struct net_if *iface, struct in6_addr *addr)
|
|
|
|
{
|
|
|
|
struct net_nbr *nbr;
|
|
|
|
#if defined(CONFIG_NET_MGMT_EVENT_INFO)
|
|
|
|
struct net_event_ipv6_nbr info;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
nbr = nbr_lookup(&net_neighbor.table, iface, addr);
|
|
|
|
if (!nbr) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove any routes with nbr as nexthop in first place */
|
|
|
|
net_route_del_by_nexthop(iface, addr);
|
|
|
|
|
|
|
|
nbr_free(nbr);
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_MGMT_EVENT_INFO)
|
|
|
|
info.idx = -1;
|
|
|
|
net_ipaddr_copy(&info.addr, addr);
|
|
|
|
net_mgmt_event_notify_with_info(NET_EVENT_IPV6_NBR_DEL,
|
|
|
|
iface, (void *) &info,
|
|
|
|
sizeof(struct net_event_ipv6_nbr));
|
|
|
|
#else
|
|
|
|
net_mgmt_event_notify(NET_EVENT_IPV6_NBR_DEL, iface);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define NS_REPLY_TIMEOUT K_SECONDS(1)
|
|
|
|
|
|
|
|
static void ipv6_ns_reply_timeout(struct k_work *work)
|
|
|
|
{
|
|
|
|
s64_t current = k_uptime_get();
|
|
|
|
struct net_nbr *nbr = NULL;
|
|
|
|
struct net_ipv6_nbr_data *data;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
ARG_UNUSED(work);
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
s64_t remaining;
|
|
|
|
nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (!nbr || !nbr->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = net_ipv6_nbr_data(nbr);
|
|
|
|
if (!data) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data->send_ns) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
remaining = data->send_ns + NS_REPLY_TIMEOUT - current;
|
|
|
|
|
|
|
|
if (remaining > 0) {
|
|
|
|
if (!k_delayed_work_remaining_get(
|
|
|
|
&ipv6_ns_reply_timer)) {
|
|
|
|
k_delayed_work_submit(&ipv6_ns_reply_timer,
|
|
|
|
remaining);
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->send_ns = 0;
|
|
|
|
|
|
|
|
/* We did not receive reply to a sent NS */
|
|
|
|
if (!data->pending) {
|
|
|
|
/* Silently return, this is not an error as the work
|
|
|
|
* cannot be cancelled in certain cases.
|
|
|
|
*/
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
NET_DBG("NS nbr %p pending %p timeout to %s", nbr,
|
|
|
|
data->pending,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(
|
|
|
|
&NET_IPV6_HDR(data->pending)->dst)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* To unref when pending variable was set */
|
|
|
|
net_pkt_unref(data->pending);
|
|
|
|
|
|
|
|
/* To unref the original pkt allocation */
|
|
|
|
net_pkt_unref(data->pending);
|
|
|
|
|
|
|
|
data->pending = NULL;
|
|
|
|
|
|
|
|
net_nbr_unref(nbr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void nbr_init(struct net_nbr *nbr, struct net_if *iface,
|
|
|
|
struct in6_addr *addr, bool is_router,
|
|
|
|
enum net_ipv6_nbr_state state)
|
|
|
|
{
|
|
|
|
nbr->idx = NET_NBR_LLADDR_UNKNOWN;
|
|
|
|
nbr->iface = iface;
|
|
|
|
|
|
|
|
net_ipaddr_copy(&net_ipv6_nbr_data(nbr)->addr, addr);
|
|
|
|
ipv6_nbr_set_state(nbr, state);
|
|
|
|
net_ipv6_nbr_data(nbr)->is_router = is_router;
|
|
|
|
net_ipv6_nbr_data(nbr)->pending = NULL;
|
|
|
|
net_ipv6_nbr_data(nbr)->send_ns = 0;
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable = 0;
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable_timeout = 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct net_nbr *nbr_new(struct net_if *iface,
|
|
|
|
struct in6_addr *addr, bool is_router,
|
|
|
|
enum net_ipv6_nbr_state state)
|
|
|
|
{
|
|
|
|
struct net_nbr *nbr = net_nbr_get(&net_neighbor.table);
|
|
|
|
|
|
|
|
if (!nbr) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
nbr_init(nbr, iface, addr, is_router, state);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
NET_DBG("nbr %p iface %p state %d IPv6 %s",
|
2018-10-02 13:57:55 +02:00
|
|
|
nbr, iface, state,
|
|
|
|
log_strdup(net_sprint_ipv6_addr(addr)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
|
2018-07-06 10:35:07 +02:00
|
|
|
static void dbg_update_neighbor_lladdr(struct net_linkaddr *new_lladdr,
|
|
|
|
struct net_linkaddr_storage *old_lladdr,
|
|
|
|
struct in6_addr *addr)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
|
|
|
char out[sizeof("xx:xx:xx:xx:xx:xx:xx:xx")];
|
|
|
|
|
|
|
|
snprintk(out, sizeof(out), "%s",
|
|
|
|
net_sprint_ll_addr(old_lladdr->addr, old_lladdr->len));
|
|
|
|
|
|
|
|
NET_DBG("Updating neighbor %s lladdr %s (was %s)",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(addr)),
|
|
|
|
log_strdup(net_sprint_ll_addr(new_lladdr->addr,
|
|
|
|
new_lladdr->len)),
|
|
|
|
log_strdup(out));
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-07-06 10:35:07 +02:00
|
|
|
static void dbg_update_neighbor_lladdr_raw(u8_t *new_lladdr,
|
|
|
|
struct net_linkaddr_storage *old_lladdr,
|
|
|
|
struct in6_addr *addr)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
|
|
|
struct net_linkaddr lladdr = {
|
|
|
|
.len = old_lladdr->len,
|
|
|
|
.addr = new_lladdr,
|
|
|
|
};
|
|
|
|
|
|
|
|
dbg_update_neighbor_lladdr(&lladdr, old_lladdr, addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define dbg_addr(action, pkt_str, src, dst) \
|
|
|
|
do { \
|
|
|
|
NET_DBG("%s %s from %s to %s", action, pkt_str, \
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(src)), \
|
|
|
|
log_strdup(net_sprint_ipv6_addr(dst))); \
|
2018-08-15 12:58:57 +02:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define dbg_addr_recv(pkt_str, src, dst) \
|
|
|
|
dbg_addr("Received", pkt_str, src, dst)
|
|
|
|
|
|
|
|
#define dbg_addr_sent(pkt_str, src, dst) \
|
|
|
|
dbg_addr("Sent", pkt_str, src, dst)
|
|
|
|
|
|
|
|
#define dbg_addr_with_tgt(action, pkt_str, src, dst, target) \
|
|
|
|
do { \
|
|
|
|
NET_DBG("%s %s from %s to %s, target %s", action, \
|
|
|
|
pkt_str, \
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(src)), \
|
|
|
|
log_strdup(net_sprint_ipv6_addr(dst)), \
|
|
|
|
log_strdup(net_sprint_ipv6_addr(target))); \
|
2018-08-15 12:58:57 +02:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define dbg_addr_recv_tgt(pkt_str, src, dst, tgt) \
|
|
|
|
dbg_addr_with_tgt("Received", pkt_str, src, dst, tgt)
|
|
|
|
|
|
|
|
#define dbg_addr_sent_tgt(pkt_str, src, dst, tgt) \
|
|
|
|
dbg_addr_with_tgt("Sent", pkt_str, src, dst, tgt)
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
static void ipv6_nd_remove_old_stale_nbr(void)
|
|
|
|
{
|
|
|
|
struct net_nbr *nbr = NULL;
|
|
|
|
struct net_ipv6_nbr_data *data = NULL;
|
|
|
|
int nbr_idx = -1;
|
2019-11-06 11:45:40 +01:00
|
|
|
u32_t oldest = UINT32_MAX;
|
2019-03-15 09:35:28 +01:00
|
|
|
int i;
|
|
|
|
|
|
|
|
k_sem_take(&nbr_lock, K_FOREVER);
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
nbr = get_nbr(i);
|
|
|
|
if (!nbr || !nbr->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = net_ipv6_nbr_data(nbr);
|
|
|
|
if (!data || data->is_router ||
|
|
|
|
data->state != NET_IPV6_NBR_STATE_STALE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nbr_idx == -1) {
|
|
|
|
nbr_idx = i;
|
|
|
|
oldest = data->stale_counter;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oldest == MIN(oldest, data->stale_counter)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
nbr_idx = i;
|
|
|
|
oldest = data->stale_counter;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nbr_idx != -1) {
|
|
|
|
nbr = get_nbr(nbr_idx);
|
|
|
|
if (!nbr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_ipv6_nbr_rm(nbr->iface,
|
|
|
|
&net_ipv6_nbr_data(nbr)->addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
k_sem_give(&nbr_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct net_nbr *add_nbr(struct net_if *iface,
|
|
|
|
struct in6_addr *addr,
|
|
|
|
bool is_router,
|
|
|
|
enum net_ipv6_nbr_state state)
|
|
|
|
{
|
|
|
|
struct net_nbr *nbr;
|
|
|
|
|
|
|
|
nbr = nbr_lookup(&net_neighbor.table, iface, addr);
|
|
|
|
if (nbr) {
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
|
|
|
|
nbr = nbr_new(iface, addr, is_router, state);
|
|
|
|
if (nbr) {
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if there are any stale neighbors, delete the oldest
|
|
|
|
* one and try to add new neighbor.
|
|
|
|
*/
|
|
|
|
ipv6_nd_remove_old_stale_nbr();
|
|
|
|
|
|
|
|
nbr = nbr_new(iface, addr, is_router, state);
|
|
|
|
if (!nbr) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_nbr *net_ipv6_nbr_add(struct net_if *iface,
|
|
|
|
struct in6_addr *addr,
|
|
|
|
struct net_linkaddr *lladdr,
|
|
|
|
bool is_router,
|
|
|
|
enum net_ipv6_nbr_state state)
|
|
|
|
{
|
|
|
|
struct net_nbr *nbr;
|
|
|
|
int ret;
|
|
|
|
#if defined(CONFIG_NET_MGMT_EVENT_INFO)
|
|
|
|
struct net_event_ipv6_nbr info;
|
|
|
|
#endif
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
nbr = add_nbr(iface, addr, is_router, state);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (!nbr) {
|
2019-03-15 09:35:28 +01:00
|
|
|
NET_ERR("Could not add router neighbor %s [%s]",
|
|
|
|
log_strdup(net_sprint_ipv6_addr(addr)),
|
|
|
|
log_strdup(net_sprint_ll_addr(lladdr->addr,
|
|
|
|
lladdr->len)));
|
|
|
|
return NULL;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (net_nbr_link(nbr, iface, lladdr) == -EALREADY &&
|
|
|
|
net_ipv6_nbr_data(nbr)->state != NET_IPV6_NBR_STATE_STATIC) {
|
|
|
|
/* Update the lladdr if the node was already known */
|
|
|
|
struct net_linkaddr_storage *cached_lladdr;
|
|
|
|
|
|
|
|
cached_lladdr = net_nbr_get_lladdr(nbr->idx);
|
|
|
|
|
|
|
|
if (memcmp(cached_lladdr->addr, lladdr->addr, lladdr->len)) {
|
|
|
|
dbg_update_neighbor_lladdr(lladdr, cached_lladdr, addr);
|
|
|
|
|
|
|
|
net_linkaddr_set(cached_lladdr, lladdr->addr,
|
|
|
|
lladdr->len);
|
|
|
|
|
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_STALE);
|
|
|
|
} else if (net_ipv6_nbr_data(nbr)->state ==
|
|
|
|
NET_IPV6_NBR_STATE_INCOMPLETE) {
|
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_STALE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (net_ipv6_nbr_data(nbr)->state == NET_IPV6_NBR_STATE_INCOMPLETE) {
|
|
|
|
/* Send NS so that we can verify that the neighbor is
|
|
|
|
* reachable.
|
|
|
|
*/
|
|
|
|
ret = net_ipv6_send_ns(iface, NULL, NULL, NULL, addr, false);
|
|
|
|
if (ret < 0) {
|
|
|
|
NET_DBG("Cannot send NS (%d)", ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NET_DBG("[%d] nbr %p state %d router %d IPv6 %s ll %s iface %p",
|
|
|
|
nbr->idx, nbr, state, is_router,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(addr)),
|
|
|
|
log_strdup(net_sprint_ll_addr(lladdr->addr, lladdr->len)),
|
2018-08-15 12:58:57 +02:00
|
|
|
nbr->iface);
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_MGMT_EVENT_INFO)
|
|
|
|
info.idx = nbr->idx;
|
|
|
|
net_ipaddr_copy(&info.addr, addr);
|
|
|
|
net_mgmt_event_notify_with_info(NET_EVENT_IPV6_NBR_ADD,
|
|
|
|
iface, (void *) &info,
|
|
|
|
sizeof(struct net_event_ipv6_nbr));
|
|
|
|
#else
|
|
|
|
net_mgmt_event_notify(NET_EVENT_IPV6_NBR_ADD, iface);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void net_neighbor_data_remove(struct net_nbr *nbr)
|
|
|
|
{
|
|
|
|
NET_DBG("Neighbor %p removed", nbr);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void net_neighbor_table_clear(struct net_nbr_table *table)
|
|
|
|
{
|
|
|
|
NET_DBG("Neighbor table %p cleared", table);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct in6_addr *net_ipv6_nbr_lookup_by_index(struct net_if *iface,
|
|
|
|
u8_t idx)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (idx == NET_NBR_LLADDR_UNKNOWN) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
struct net_nbr *nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (!nbr->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iface && nbr->iface != iface) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nbr->idx == idx) {
|
|
|
|
return &net_ipv6_nbr_data(nbr)->addr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
const char *net_ipv6_nbr_state2str(enum net_ipv6_nbr_state state)
|
|
|
|
{
|
|
|
|
return "<unknown state>";
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_NBR_CACHE */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_DAD)
|
|
|
|
int net_ipv6_start_dad(struct net_if *iface, struct net_if_addr *ifaddr)
|
|
|
|
{
|
|
|
|
return net_ipv6_send_ns(iface, NULL, NULL, NULL,
|
|
|
|
&ifaddr->address.in6_addr, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool dad_failed(struct net_if *iface, struct in6_addr *addr)
|
|
|
|
{
|
2018-11-02 15:05:58 +01:00
|
|
|
if (net_ipv6_is_ll_addr(addr)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
NET_ERR("DAD failed, no ll IPv6 address!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_if_ipv6_dad_failed(iface, addr);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_DAD */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_NBR_CACHE)
|
|
|
|
static struct in6_addr *check_route(struct net_if *iface,
|
|
|
|
struct in6_addr *dst,
|
|
|
|
bool *try_route)
|
|
|
|
{
|
|
|
|
struct in6_addr *nexthop = NULL;
|
|
|
|
struct net_route_entry *route;
|
|
|
|
struct net_if_router *router;
|
|
|
|
|
|
|
|
route = net_route_lookup(iface, dst);
|
|
|
|
if (route) {
|
|
|
|
nexthop = net_route_get_nexthop(route);
|
|
|
|
|
|
|
|
NET_DBG("Route %p nexthop %s", route,
|
2018-10-02 13:57:55 +02:00
|
|
|
nexthop ? log_strdup(net_sprint_ipv6_addr(nexthop)) :
|
|
|
|
"<unknown>");
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (!nexthop) {
|
|
|
|
net_route_del(route);
|
|
|
|
|
|
|
|
NET_DBG("No route to host %s",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(dst)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* No specific route to this host, use the default
|
|
|
|
* route instead.
|
|
|
|
*/
|
|
|
|
router = net_if_ipv6_router_find_default(NULL, dst);
|
|
|
|
if (!router) {
|
|
|
|
NET_DBG("No default route to %s",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(dst)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* Try to send the packet anyway */
|
|
|
|
nexthop = dst;
|
|
|
|
if (try_route) {
|
|
|
|
*try_route = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nexthop;
|
|
|
|
}
|
|
|
|
|
|
|
|
nexthop = &router->address.in6_addr;
|
|
|
|
|
|
|
|
NET_DBG("Router %p nexthop %s", router,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(nexthop)));
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nexthop;
|
|
|
|
}
|
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
enum net_verdict net_ipv6_prepare_for_send(struct net_pkt *pkt)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
2018-12-21 09:08:19 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(ipv6_access, struct net_ipv6_hdr);
|
2018-08-15 12:58:57 +02:00
|
|
|
struct in6_addr *nexthop = NULL;
|
|
|
|
struct net_if *iface = NULL;
|
2018-12-21 09:08:19 +01:00
|
|
|
struct net_ipv6_hdr *ip_hdr;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_nbr *nbr;
|
|
|
|
int ret;
|
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
NET_ASSERT(pkt && pkt->buffer);
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
ip_hdr = (struct net_ipv6_hdr *)net_pkt_get_data(pkt, &ipv6_access);
|
2018-12-21 09:08:19 +01:00
|
|
|
if (!ip_hdr) {
|
|
|
|
return NET_DROP;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_FRAGMENT)
|
|
|
|
/* If we have already fragmented the packet, the fragment id will
|
|
|
|
* contain a proper value and we can skip other checks.
|
|
|
|
*/
|
2019-03-27 02:57:45 +01:00
|
|
|
if (net_pkt_ipv6_fragment_id(pkt) == 0U) {
|
2019-03-21 14:29:11 +01:00
|
|
|
u16_t mtu = net_if_get_mtu(net_pkt_iface(pkt));
|
2018-08-15 12:58:57 +02:00
|
|
|
size_t pkt_len = net_pkt_get_len(pkt);
|
|
|
|
|
2019-03-21 14:29:11 +01:00
|
|
|
mtu = MAX(NET_IPV6_MTU, mtu);
|
|
|
|
if (mtu < pkt_len) {
|
2018-08-15 12:58:57 +02:00
|
|
|
ret = net_ipv6_send_fragmented_pkt(net_pkt_iface(pkt),
|
|
|
|
pkt, pkt_len);
|
|
|
|
if (ret < 0) {
|
|
|
|
NET_DBG("Cannot fragment IPv6 pkt (%d)", ret);
|
|
|
|
|
|
|
|
if (ret == -ENOMEM) {
|
|
|
|
/* Try to send the packet if we could
|
|
|
|
* not allocate enough network packets
|
|
|
|
* and hope the original large packet
|
|
|
|
* can be sent ok.
|
|
|
|
*/
|
|
|
|
goto ignore_frag_error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We "fake" the sending of the packet here so that
|
|
|
|
* tcp.c:tcp_retry_expired() will increase the ref
|
|
|
|
* count when re-sending the packet. This is crucial
|
|
|
|
* thing to do here and will cause free memory access
|
|
|
|
* if not done.
|
|
|
|
*/
|
2018-12-21 09:08:19 +01:00
|
|
|
if (IS_ENABLED(CONFIG_NET_TCP)) {
|
|
|
|
net_pkt_set_sent(pkt, true);
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* We need to unref here because we simulate the packet
|
|
|
|
* sending.
|
|
|
|
*/
|
|
|
|
net_pkt_unref(pkt);
|
|
|
|
|
|
|
|
/* No need to continue with the sending as the packet
|
|
|
|
* is now split and its fragments will be sent
|
|
|
|
* separately to network.
|
|
|
|
*/
|
2018-12-21 09:08:19 +01:00
|
|
|
return NET_CONTINUE;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ignore_frag_error:
|
|
|
|
#endif /* CONFIG_NET_IPV6_FRAGMENT */
|
|
|
|
|
|
|
|
/* If the IPv6 destination address is not link local, then try to get
|
|
|
|
* the next hop from routing table if we have multi interface routing
|
|
|
|
* enabled. The reason for this is that the neighbor cache will not
|
|
|
|
* contain public IPv6 address information so in that case we should
|
|
|
|
* not enter this branch.
|
|
|
|
*/
|
2018-09-11 09:16:03 +02:00
|
|
|
if ((net_pkt_lladdr_dst(pkt)->addr &&
|
2018-08-15 12:58:57 +02:00
|
|
|
((IS_ENABLED(CONFIG_NET_ROUTING) &&
|
2018-12-21 09:08:19 +01:00
|
|
|
net_ipv6_is_ll_addr(&ip_hdr->dst)) ||
|
2018-08-15 12:58:57 +02:00
|
|
|
!IS_ENABLED(CONFIG_NET_ROUTING))) ||
|
2018-12-21 09:08:19 +01:00
|
|
|
net_ipv6_is_addr_mcast(&ip_hdr->dst) ||
|
|
|
|
/* Workaround Linux bug, see:
|
|
|
|
* https://github.com/zephyrproject-rtos/zephyr/issues/3111
|
|
|
|
*/
|
2019-04-16 09:55:34 +02:00
|
|
|
net_if_flag_is_set(net_pkt_iface(pkt),
|
|
|
|
NET_IF_POINTOPOINT)) {
|
2018-12-21 09:08:19 +01:00
|
|
|
return NET_OK;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
if (net_if_ipv6_addr_onlink(&iface, &ip_hdr->dst)) {
|
|
|
|
nexthop = &ip_hdr->dst;
|
2018-08-15 12:58:57 +02:00
|
|
|
net_pkt_set_iface(pkt, iface);
|
|
|
|
} else {
|
|
|
|
/* We need to figure out where the destination
|
|
|
|
* host is located.
|
|
|
|
*/
|
|
|
|
bool try_route = false;
|
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
nexthop = check_route(NULL, &ip_hdr->dst, &try_route);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (!nexthop) {
|
2018-12-21 09:08:19 +01:00
|
|
|
return NET_DROP;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (try_route) {
|
|
|
|
goto try_send;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!iface) {
|
|
|
|
/* This means that the dst was not onlink, so try to
|
|
|
|
* figure out the interface using nexthop instead.
|
|
|
|
*/
|
|
|
|
if (net_if_ipv6_addr_onlink(&iface, nexthop)) {
|
|
|
|
net_pkt_set_iface(pkt, iface);
|
|
|
|
} else {
|
|
|
|
iface = net_pkt_iface(pkt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the above check returns null, we try to send
|
|
|
|
* the packet and hope for the best.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
try_send:
|
|
|
|
nbr = nbr_lookup(&net_neighbor.table, iface, nexthop);
|
|
|
|
|
|
|
|
NET_DBG("Neighbor lookup %p (%d) iface %p addr %s state %s", nbr,
|
|
|
|
nbr ? nbr->idx : NET_NBR_LLADDR_UNKNOWN, iface,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(nexthop)),
|
2018-08-15 12:58:57 +02:00
|
|
|
nbr ? net_ipv6_nbr_state2str(net_ipv6_nbr_data(nbr)->state) :
|
|
|
|
"-");
|
|
|
|
|
|
|
|
if (nbr && nbr->idx != NET_NBR_LLADDR_UNKNOWN) {
|
|
|
|
struct net_linkaddr_storage *lladdr;
|
|
|
|
|
|
|
|
lladdr = net_nbr_get_lladdr(nbr->idx);
|
|
|
|
|
2018-09-11 09:16:03 +02:00
|
|
|
net_pkt_lladdr_dst(pkt)->addr = lladdr->addr;
|
|
|
|
net_pkt_lladdr_dst(pkt)->len = lladdr->len;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
NET_DBG("Neighbor %p addr %s", nbr,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ll_addr(lladdr->addr,
|
|
|
|
lladdr->len)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* Start the NUD if we are in STALE state.
|
|
|
|
* See RFC 4861 ch 7.3.3 for details.
|
|
|
|
*/
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
if (net_ipv6_nbr_data(nbr)->state == NET_IPV6_NBR_STATE_STALE) {
|
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_DELAY);
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
ipv6_nd_restart_reachable_timer(nbr,
|
|
|
|
DELAY_FIRST_PROBE_TIME);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
#endif
|
2018-12-21 09:08:19 +01:00
|
|
|
return NET_OK;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
/* We need to send NS and wait for NA before sending the packet. */
|
|
|
|
ret = net_ipv6_send_ns(net_pkt_iface(pkt), pkt,
|
2018-12-21 09:08:19 +01:00
|
|
|
&ip_hdr->src, NULL, nexthop, false);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (ret < 0) {
|
|
|
|
/* In case of an error, the NS send function will unref
|
|
|
|
* the pkt.
|
|
|
|
*/
|
|
|
|
NET_DBG("Cannot send NS (%d)", ret);
|
|
|
|
}
|
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
NET_DBG("pkt %p (buffer %p) will be sent later", pkt, pkt->buffer);
|
|
|
|
|
|
|
|
return NET_CONTINUE;
|
2018-08-15 12:58:57 +02:00
|
|
|
#else
|
|
|
|
ARG_UNUSED(ret);
|
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
NET_DBG("pkt %p (buffer %p) cannot be sent, dropping it.",
|
|
|
|
pkt, pkt->buffer);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-21 09:08:19 +01:00
|
|
|
return NET_DROP;
|
2018-08-15 12:58:57 +02:00
|
|
|
#endif /* CONFIG_NET_IPV6_ND */
|
|
|
|
}
|
|
|
|
|
|
|
|
struct net_nbr *net_ipv6_nbr_lookup(struct net_if *iface,
|
|
|
|
struct in6_addr *addr)
|
|
|
|
{
|
|
|
|
return nbr_lookup(&net_neighbor.table, iface, addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct net_nbr *net_ipv6_get_nbr(struct net_if *iface, u8_t idx)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (idx == NET_NBR_LLADDR_UNKNOWN) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
struct net_nbr *nbr = get_nbr(i);
|
|
|
|
|
|
|
|
if (nbr->ref) {
|
|
|
|
if (iface && nbr->iface != iface) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nbr->idx == idx) {
|
|
|
|
return nbr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u8_t get_llao_len(struct net_if *iface)
|
|
|
|
{
|
2019-08-09 10:06:24 +02:00
|
|
|
u8_t total_len = net_if_get_link_addr(iface)->len +
|
|
|
|
sizeof(struct net_icmpv6_nd_opt_hdr);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-08-09 10:06:24 +02:00
|
|
|
return ROUND_UP(total_len, 8U);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
static inline bool set_llao(struct net_pkt *pkt,
|
|
|
|
struct net_linkaddr *lladdr,
|
|
|
|
u8_t llao_len, u8_t type)
|
2018-12-20 10:33:35 +01:00
|
|
|
{
|
|
|
|
struct net_icmpv6_nd_opt_hdr opt_hdr = {
|
|
|
|
.type = type,
|
|
|
|
.len = llao_len >> 3,
|
|
|
|
};
|
|
|
|
|
2019-02-20 10:01:57 +01:00
|
|
|
if (net_pkt_write(pkt, &opt_hdr,
|
|
|
|
sizeof(struct net_icmpv6_nd_opt_hdr)) ||
|
|
|
|
net_pkt_write(pkt, lladdr->addr, lladdr->len) ||
|
2018-12-20 10:33:35 +01:00
|
|
|
net_pkt_memset(pkt, 0, llao_len - lladdr->len - 2)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
static inline struct net_nbr *handle_ns_neighbor(struct net_pkt *pkt,
|
2018-12-20 14:00:05 +01:00
|
|
|
u8_t ll_len)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
|
|
|
struct net_linkaddr_storage lladdr;
|
|
|
|
struct net_linkaddr nbr_lladdr;
|
|
|
|
|
2019-03-28 21:57:54 +01:00
|
|
|
lladdr.len = ll_len * 8U - 2;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-02-20 09:40:48 +01:00
|
|
|
if (net_pkt_read(pkt, lladdr.addr, lladdr.len)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
nbr_lladdr.len = lladdr.len;
|
|
|
|
nbr_lladdr.addr = lladdr.addr;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* IEEE802154 lladdress is 8 bytes long, so it requires
|
|
|
|
* 2 * 8 bytes - 2 - padding.
|
|
|
|
* The formula above needs to be adjusted.
|
|
|
|
*/
|
2018-09-11 09:16:03 +02:00
|
|
|
if (net_pkt_lladdr_src(pkt)->len < nbr_lladdr.len) {
|
|
|
|
nbr_lladdr.len = net_pkt_lladdr_src(pkt)->len;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
return net_ipv6_nbr_add(net_pkt_iface(pkt),
|
|
|
|
&NET_IPV6_HDR(pkt)->src,
|
|
|
|
&nbr_lladdr, false,
|
|
|
|
NET_IPV6_NBR_STATE_INCOMPLETE);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int net_ipv6_send_na(struct net_if *iface, const struct in6_addr *src,
|
|
|
|
const struct in6_addr *dst, const struct in6_addr *tgt,
|
|
|
|
u8_t flags)
|
|
|
|
{
|
2018-12-20 10:33:35 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(na_access,
|
|
|
|
struct net_icmpv6_na_hdr);
|
|
|
|
int ret = -ENOBUFS;
|
|
|
|
struct net_icmpv6_na_hdr *na_hdr;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_pkt *pkt;
|
|
|
|
u8_t llao_len;
|
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
llao_len = get_llao_len(iface);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
pkt = net_pkt_alloc_with_buffer(iface,
|
|
|
|
sizeof(struct net_icmpv6_na_hdr) +
|
|
|
|
llao_len,
|
|
|
|
AF_INET6, IPPROTO_ICMPV6,
|
|
|
|
ND_NET_BUF_TIMEOUT);
|
|
|
|
if (!pkt) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
net_pkt_set_ipv6_hop_limit(pkt, NET_IPV6_ND_HOP_LIMIT);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-02-20 08:58:08 +01:00
|
|
|
if (net_ipv6_create(pkt, src, dst) ||
|
2018-12-20 10:33:35 +01:00
|
|
|
net_icmpv6_create(pkt, NET_ICMPV6_NA, 0)) {
|
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
na_hdr = (struct net_icmpv6_na_hdr *)net_pkt_get_data(pkt, &na_access);
|
2018-12-20 10:33:35 +01:00
|
|
|
if (!na_hdr) {
|
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
/* Let's make sure reserved part is full of 0 */
|
|
|
|
memset(na_hdr, 0, sizeof(struct net_icmpv6_na_hdr));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
na_hdr->flags = flags;
|
|
|
|
net_ipaddr_copy(&na_hdr->tgt, tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
if (net_pkt_set_data(pkt, &na_access)) {
|
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
if (!set_llao(pkt, net_if_get_link_addr(iface),
|
|
|
|
llao_len, NET_ICMPV6_ND_OPT_TLLAO)) {
|
2018-12-20 10:33:35 +01:00
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
net_pkt_cursor_init(pkt);
|
2019-02-01 21:29:39 +01:00
|
|
|
net_ipv6_finalize(pkt, IPPROTO_ICMPV6);
|
2018-12-20 10:33:35 +01:00
|
|
|
|
|
|
|
dbg_addr_sent_tgt("Neighbor Advertisement", src, dst, &na_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (net_send_data(pkt) < 0) {
|
2018-12-20 10:33:35 +01:00
|
|
|
net_stats_update_ipv6_nd_drop(iface);
|
|
|
|
ret = -EINVAL;
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
net_stats_update_ipv6_nd_sent(iface);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
net_pkt_unref(pkt);
|
|
|
|
|
2018-12-20 10:33:35 +01:00
|
|
|
return ret;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ns_routing_info(struct net_pkt *pkt,
|
|
|
|
struct in6_addr *nexthop,
|
|
|
|
struct in6_addr *tgt)
|
|
|
|
{
|
2018-11-30 11:54:56 +01:00
|
|
|
if (CONFIG_NET_IPV6_LOG_LEVEL >= LOG_LEVEL_DBG) {
|
2018-07-06 10:35:07 +02:00
|
|
|
char out[NET_IPV6_ADDR_LEN];
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-07-06 10:35:07 +02:00
|
|
|
snprintk(out, sizeof(out), "%s",
|
|
|
|
net_sprint_ipv6_addr(nexthop));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-07-06 10:35:07 +02:00
|
|
|
if (net_ipv6_addr_cmp(nexthop, tgt)) {
|
2018-10-02 13:57:55 +02:00
|
|
|
NET_DBG("Routing to %s iface %p",
|
|
|
|
log_strdup(out),
|
2018-07-06 10:35:07 +02:00
|
|
|
net_pkt_iface(pkt));
|
|
|
|
} else {
|
|
|
|
NET_DBG("Routing to %s via %s iface %p",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(tgt)),
|
|
|
|
log_strdup(out),
|
2018-07-06 10:35:07 +02:00
|
|
|
net_pkt_iface(pkt));
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
static enum net_verdict handle_ns_input(struct net_pkt *pkt,
|
2018-12-20 12:59:56 +01:00
|
|
|
struct net_ipv6_hdr *ip_hdr,
|
|
|
|
struct net_icmp_hdr *icmp_hdr)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(ns_access,
|
|
|
|
struct net_icmpv6_ns_hdr);
|
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(nd_access, struct net_icmpv6_nd_opt_hdr);
|
|
|
|
u16_t length = net_pkt_get_len(pkt);
|
2018-11-29 20:23:03 +01:00
|
|
|
u8_t flags = 0U;
|
2018-08-15 12:58:57 +02:00
|
|
|
bool routing = false;
|
2018-12-20 14:00:05 +01:00
|
|
|
struct net_icmpv6_nd_opt_hdr *nd_opt_hdr;
|
|
|
|
struct net_icmpv6_ns_hdr *ns_hdr;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_if_addr *ifaddr;
|
|
|
|
const struct in6_addr *src;
|
2018-12-20 14:00:05 +01:00
|
|
|
struct in6_addr *tgt;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
ns_hdr = (struct net_icmpv6_ns_hdr *)net_pkt_get_data(pkt, &ns_access);
|
2018-12-20 14:00:05 +01:00
|
|
|
if (!ns_hdr) {
|
|
|
|
NET_ERR("DROP: NULL NS header");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
dbg_addr_recv_tgt("Neighbor Solicitation",
|
2018-12-20 14:00:05 +01:00
|
|
|
&ip_hdr->src, &ip_hdr->dst, &ns_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
net_stats_update_ipv6_nd_recv(net_pkt_iface(pkt));
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
if (((length < (sizeof(struct net_ipv6_hdr) +
|
2018-08-15 12:58:57 +02:00
|
|
|
sizeof(struct net_icmp_hdr) +
|
|
|
|
sizeof(struct net_icmpv6_ns_hdr))) ||
|
2018-12-20 14:00:05 +01:00
|
|
|
(ip_hdr->hop_limit != NET_IPV6_ND_HOP_LIMIT)) &&
|
2019-03-27 02:57:45 +01:00
|
|
|
(net_ipv6_is_addr_mcast(&ns_hdr->tgt) && icmp_hdr->code != 0U)) {
|
2018-12-20 14:00:05 +01:00
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &ns_access);
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
net_pkt_set_ipv6_ext_opt_len(pkt, sizeof(struct net_icmpv6_ns_hdr));
|
2018-12-20 14:00:05 +01:00
|
|
|
length -= (sizeof(struct net_ipv6_hdr) + sizeof(struct net_icmp_hdr));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
nd_opt_hdr = (struct net_icmpv6_nd_opt_hdr *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &nd_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
while (nd_opt_hdr && nd_opt_hdr->len > 0 &&
|
|
|
|
net_pkt_ipv6_ext_opt_len(pkt) < length) {
|
|
|
|
u8_t prev_opt_len;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &nd_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
switch (nd_opt_hdr->type) {
|
2018-08-15 12:58:57 +02:00
|
|
|
case NET_ICMPV6_ND_OPT_SLLAO:
|
2018-12-18 15:28:40 +01:00
|
|
|
if (net_ipv6_is_addr_unspecified(&ip_hdr->src)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
if (nd_opt_hdr->len > 2) {
|
|
|
|
NET_ERR("DROP: Too long source ll address "
|
2018-08-15 12:58:57 +02:00
|
|
|
"in NS option");
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
if (!handle_ns_neighbor(pkt, nd_opt_hdr->len)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
break;
|
2018-08-15 12:58:57 +02:00
|
|
|
default:
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("Unknown ND option 0x%x", nd_opt_hdr->type);
|
2018-08-15 12:58:57 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_opt_len = net_pkt_ipv6_ext_opt_len(pkt);
|
|
|
|
|
|
|
|
net_pkt_set_ipv6_ext_opt_len(pkt,
|
|
|
|
net_pkt_ipv6_ext_opt_len(pkt) +
|
2018-12-20 14:00:05 +01:00
|
|
|
(nd_opt_hdr->len << 3));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (prev_opt_len >= net_pkt_ipv6_ext_opt_len(pkt)) {
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_ERR("DROP: Corrupted NS message");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
nd_opt_hdr = (struct net_icmpv6_nd_opt_hdr *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &nd_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (IS_ENABLED(CONFIG_NET_ROUTING)) {
|
2018-12-20 14:00:05 +01:00
|
|
|
ifaddr = net_if_ipv6_addr_lookup(&ns_hdr->tgt, NULL);
|
2018-08-15 12:58:57 +02:00
|
|
|
} else {
|
|
|
|
ifaddr = net_if_ipv6_addr_lookup_by_iface(net_pkt_iface(pkt),
|
2018-12-20 14:00:05 +01:00
|
|
|
&ns_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!ifaddr) {
|
|
|
|
if (IS_ENABLED(CONFIG_NET_ROUTING)) {
|
|
|
|
struct in6_addr *nexthop;
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
nexthop = check_route(NULL, &ns_hdr->tgt, NULL);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (nexthop) {
|
2018-12-20 14:00:05 +01:00
|
|
|
ns_routing_info(pkt, nexthop, &ns_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* Note that the target is not the address of
|
|
|
|
* the "nethop" as that is a link-local address
|
|
|
|
* which is not routable.
|
|
|
|
*/
|
2018-12-20 14:00:05 +01:00
|
|
|
tgt = &ns_hdr->tgt;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* Source address must be one of our real
|
|
|
|
* interface address where the packet was
|
|
|
|
* received.
|
|
|
|
*/
|
|
|
|
src = net_if_ipv6_select_src_addr(
|
2018-12-18 15:28:40 +01:00
|
|
|
net_pkt_iface(pkt), &ip_hdr->src);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (!src) {
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: No interface address "
|
|
|
|
"for dst %s iface %p",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(
|
|
|
|
net_sprint_ipv6_addr(
|
2018-12-18 15:28:40 +01:00
|
|
|
&ip_hdr->src)),
|
2018-08-15 12:58:57 +02:00
|
|
|
net_pkt_iface(pkt));
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
routing = true;
|
|
|
|
goto nexthop_found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: No such interface address %s",
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&ns_hdr->tgt)));
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
} else {
|
|
|
|
tgt = &ifaddr->address.in6_addr;
|
|
|
|
|
|
|
|
/* As we swap the addresses later, the source will correctly
|
|
|
|
* have our address.
|
|
|
|
*/
|
2018-12-18 15:28:40 +01:00
|
|
|
src = &ip_hdr->src;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
nexthop_found:
|
|
|
|
|
|
|
|
#if !defined(CONFIG_NET_IPV6_DAD)
|
2018-12-18 15:28:40 +01:00
|
|
|
if (net_ipv6_is_addr_unspecified(&ip_hdr->src)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* CONFIG_NET_IPV6_DAD */
|
|
|
|
|
|
|
|
/* Do DAD */
|
2018-12-18 15:28:40 +01:00
|
|
|
if (net_ipv6_is_addr_unspecified(&ip_hdr->src)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
if (!net_ipv6_is_addr_solicited_node(&ip_hdr->dst)) {
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: Not solicited node addr %s",
|
2018-12-18 15:28:40 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&ip_hdr->dst)));
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ifaddr->addr_state == NET_ADDR_TENTATIVE) {
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: DAD failed for %s iface %p",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(
|
|
|
|
&ifaddr->address.in6_addr)),
|
2018-08-15 12:58:57 +02:00
|
|
|
net_pkt_iface(pkt));
|
|
|
|
|
|
|
|
dad_failed(net_pkt_iface(pkt),
|
|
|
|
&ifaddr->address.in6_addr);
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We reuse the received packet to send the NA */
|
2018-12-18 15:28:40 +01:00
|
|
|
net_ipv6_addr_create_ll_allnodes_mcast(&ip_hdr->dst);
|
|
|
|
net_ipaddr_copy(&ip_hdr->src,
|
2018-08-15 12:58:57 +02:00
|
|
|
net_if_ipv6_select_src_addr(net_pkt_iface(pkt),
|
2018-12-18 15:28:40 +01:00
|
|
|
&ip_hdr->dst));
|
2018-08-15 12:58:57 +02:00
|
|
|
flags = NET_ICMPV6_NA_FLAG_OVERRIDE;
|
|
|
|
goto send_na;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_DAD */
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
if (net_ipv6_is_my_addr(&ip_hdr->src)) {
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: Duplicate IPv6 %s address",
|
2018-12-18 15:28:40 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&ip_hdr->src)));
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Address resolution */
|
2018-12-18 15:28:40 +01:00
|
|
|
if (net_ipv6_is_addr_solicited_node(&ip_hdr->dst)) {
|
|
|
|
net_ipaddr_copy(&ip_hdr->dst, &ip_hdr->src);
|
2018-12-20 14:00:05 +01:00
|
|
|
net_ipaddr_copy(&ip_hdr->src, &ns_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
flags = NET_ICMPV6_NA_FLAG_SOLICITED |
|
|
|
|
NET_ICMPV6_NA_FLAG_OVERRIDE;
|
|
|
|
goto send_na;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (routing) {
|
|
|
|
/* No need to do NUD here when the target is being routed. */
|
|
|
|
goto send_na;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Neighbor Unreachability Detection (NUD) */
|
|
|
|
if (IS_ENABLED(CONFIG_NET_ROUTING)) {
|
2018-12-18 15:28:40 +01:00
|
|
|
ifaddr = net_if_ipv6_addr_lookup(&ip_hdr->dst, NULL);
|
2018-08-15 12:58:57 +02:00
|
|
|
} else {
|
|
|
|
ifaddr = net_if_ipv6_addr_lookup_by_iface(net_pkt_iface(pkt),
|
2018-12-18 15:28:40 +01:00
|
|
|
&ip_hdr->dst);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ifaddr) {
|
2018-12-18 15:28:40 +01:00
|
|
|
net_ipaddr_copy(&ip_hdr->dst, &ip_hdr->src);
|
2018-12-20 14:00:05 +01:00
|
|
|
net_ipaddr_copy(&ip_hdr->src, &ns_hdr->tgt);
|
2018-12-18 15:28:40 +01:00
|
|
|
src = &ip_hdr->src;
|
2018-08-15 12:58:57 +02:00
|
|
|
tgt = &ifaddr->address.in6_addr;
|
|
|
|
flags = NET_ICMPV6_NA_FLAG_SOLICITED |
|
|
|
|
NET_ICMPV6_NA_FLAG_OVERRIDE;
|
|
|
|
goto send_na;
|
|
|
|
} else {
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: NUD failed");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
send_na:
|
2018-12-20 14:00:05 +01:00
|
|
|
if (!net_ipv6_send_na(net_pkt_iface(pkt), src,
|
|
|
|
&ip_hdr->dst, tgt, flags)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
net_pkt_unref(pkt);
|
|
|
|
return NET_OK;
|
|
|
|
}
|
|
|
|
|
2018-12-20 14:00:05 +01:00
|
|
|
NET_DBG("DROP: Cannot send NA");
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
return NET_DROP;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
net_stats_update_ipv6_nd_drop(net_pkt_iface(pkt));
|
|
|
|
|
|
|
|
return NET_DROP;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_NBR_CACHE */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
2019-03-15 09:35:28 +01:00
|
|
|
static void ipv6_nd_restart_reachable_timer(struct net_nbr *nbr, s64_t time)
|
|
|
|
{
|
|
|
|
s64_t remaining;
|
|
|
|
|
|
|
|
if (nbr) {
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable = k_uptime_get();
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable_timeout = time;
|
|
|
|
}
|
|
|
|
|
|
|
|
remaining = k_delayed_work_remaining_get(&ipv6_nd_reachable_timer);
|
|
|
|
if (!remaining || remaining > time) {
|
|
|
|
k_delayed_work_submit(&ipv6_nd_reachable_timer, time);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
static void ipv6_nd_reachable_timeout(struct k_work *work)
|
|
|
|
{
|
|
|
|
s64_t current = k_uptime_get();
|
|
|
|
struct net_nbr *nbr = NULL;
|
|
|
|
struct net_ipv6_nbr_data *data = NULL;
|
|
|
|
int ret;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_NET_IPV6_MAX_NEIGHBORS; i++) {
|
|
|
|
s64_t remaining;
|
|
|
|
|
|
|
|
nbr = get_nbr(i);
|
|
|
|
if (!nbr || !nbr->ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = net_ipv6_nbr_data(nbr);
|
|
|
|
if (!data) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data->reachable) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
remaining = data->reachable + data->reachable_timeout - current;
|
|
|
|
if (remaining > 0) {
|
2019-03-15 09:35:28 +01:00
|
|
|
ipv6_nd_restart_reachable_timer(NULL, remaining);
|
2018-08-15 12:58:57 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->reachable = 0;
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case NET_IPV6_NBR_STATE_STATIC:
|
2020-01-08 08:42:23 +01:00
|
|
|
NET_ASSERT(false, "Static entry shall never timeout");
|
2018-08-15 12:58:57 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_IPV6_NBR_STATE_INCOMPLETE:
|
|
|
|
if (data->ns_count >= MAX_MULTICAST_SOLICIT) {
|
2019-03-15 09:35:28 +01:00
|
|
|
net_ipv6_nbr_rm(nbr->iface, &data->addr);
|
2018-08-15 12:58:57 +02:00
|
|
|
} else {
|
|
|
|
data->ns_count++;
|
|
|
|
|
|
|
|
NET_DBG("nbr %p incomplete count %u", nbr,
|
|
|
|
data->ns_count);
|
|
|
|
|
|
|
|
ret = net_ipv6_send_ns(nbr->iface, NULL, NULL,
|
|
|
|
NULL, &data->addr,
|
|
|
|
false);
|
|
|
|
if (ret < 0) {
|
|
|
|
NET_DBG("Cannot send NS (%d)", ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_IPV6_NBR_STATE_REACHABLE:
|
|
|
|
data->state = NET_IPV6_NBR_STATE_STALE;
|
|
|
|
|
|
|
|
NET_DBG("nbr %p moving %s state to STALE (%d)",
|
2018-10-02 13:57:55 +02:00
|
|
|
nbr,
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&data->addr)),
|
2018-08-15 12:58:57 +02:00
|
|
|
data->state);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_IPV6_NBR_STATE_STALE:
|
|
|
|
NET_DBG("nbr %p removing stale address %s",
|
2018-10-02 13:57:55 +02:00
|
|
|
nbr,
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&data->addr)));
|
2019-03-15 09:35:28 +01:00
|
|
|
net_ipv6_nbr_rm(nbr->iface, &data->addr);
|
2018-08-15 12:58:57 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_IPV6_NBR_STATE_DELAY:
|
|
|
|
data->state = NET_IPV6_NBR_STATE_PROBE;
|
2018-11-29 20:23:03 +01:00
|
|
|
data->ns_count = 0U;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
NET_DBG("nbr %p moving %s state to PROBE (%d)",
|
2018-10-02 13:57:55 +02:00
|
|
|
nbr,
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&data->addr)),
|
2018-08-15 12:58:57 +02:00
|
|
|
data->state);
|
|
|
|
|
|
|
|
/* Intentionally continuing to probe state */
|
|
|
|
|
|
|
|
case NET_IPV6_NBR_STATE_PROBE:
|
|
|
|
if (data->ns_count >= MAX_UNICAST_SOLICIT) {
|
2019-03-15 09:35:28 +01:00
|
|
|
net_ipv6_nbr_rm(nbr->iface, &data->addr);
|
2018-08-15 12:58:57 +02:00
|
|
|
} else {
|
|
|
|
data->ns_count++;
|
|
|
|
|
|
|
|
NET_DBG("nbr %p probe count %u", nbr,
|
|
|
|
data->ns_count);
|
|
|
|
|
|
|
|
ret = net_ipv6_send_ns(nbr->iface, NULL, NULL,
|
|
|
|
NULL, &data->addr,
|
|
|
|
false);
|
|
|
|
if (ret < 0) {
|
|
|
|
NET_DBG("Cannot send NS (%d)", ret);
|
|
|
|
}
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
ipv6_nd_restart_reachable_timer(nbr,
|
|
|
|
RETRANS_TIMER);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void net_ipv6_nbr_set_reachable_timer(struct net_if *iface,
|
|
|
|
struct net_nbr *nbr)
|
|
|
|
{
|
|
|
|
u32_t time;
|
|
|
|
|
|
|
|
time = net_if_ipv6_get_reachable_time(iface);
|
|
|
|
|
2020-01-08 08:42:23 +01:00
|
|
|
NET_ASSERT(time, "Zero reachable timeout!");
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
NET_DBG("Starting reachable timer nbr %p data %p time %d ms",
|
|
|
|
nbr, net_ipv6_nbr_data(nbr), time);
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
ipv6_nd_restart_reachable_timer(nbr, time);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_ND */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_NBR_CACHE)
|
|
|
|
static inline bool handle_na_neighbor(struct net_pkt *pkt,
|
|
|
|
struct net_icmpv6_na_hdr *na_hdr,
|
|
|
|
u16_t tllao_offset)
|
|
|
|
{
|
|
|
|
struct net_linkaddr_storage lladdr = { 0 };
|
2018-12-20 13:30:25 +01:00
|
|
|
bool lladdr_changed = false;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_linkaddr_storage *cached_lladdr;
|
|
|
|
struct net_pkt *pending;
|
2018-12-20 13:30:25 +01:00
|
|
|
struct net_nbr *nbr;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
nbr = nbr_lookup(&net_neighbor.table, net_pkt_iface(pkt), &na_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
NET_DBG("Neighbor lookup %p iface %p addr %s", nbr,
|
|
|
|
net_pkt_iface(pkt),
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&na_hdr->tgt)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (!nbr) {
|
|
|
|
nbr_print();
|
|
|
|
|
|
|
|
NET_DBG("No such neighbor found, msg discarded");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tllao_offset) {
|
2019-06-15 14:52:34 +02:00
|
|
|
lladdr.len = net_pkt_lladdr_src(pkt)->len;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
net_pkt_cursor_init(pkt);
|
|
|
|
|
|
|
|
if (net_pkt_skip(pkt, tllao_offset) ||
|
2019-02-20 09:40:48 +01:00
|
|
|
net_pkt_read(pkt, lladdr.addr, lladdr.len)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nbr->idx == NET_NBR_LLADDR_UNKNOWN) {
|
|
|
|
struct net_linkaddr nbr_lladdr;
|
|
|
|
|
|
|
|
if (!tllao_offset) {
|
|
|
|
NET_DBG("No target link layer address.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
nbr_lladdr.len = lladdr.len;
|
|
|
|
nbr_lladdr.addr = lladdr.addr;
|
|
|
|
|
|
|
|
if (net_nbr_link(nbr, net_pkt_iface(pkt), &nbr_lladdr)) {
|
|
|
|
nbr_free(nbr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
NET_DBG("[%d] nbr %p state %d IPv6 %s ll %s",
|
|
|
|
nbr->idx, nbr, net_ipv6_nbr_data(nbr)->state,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&na_hdr->tgt)),
|
|
|
|
log_strdup(net_sprint_ll_addr(nbr_lladdr.addr,
|
|
|
|
nbr_lladdr.len)));
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
cached_lladdr = net_nbr_get_lladdr(nbr->idx);
|
|
|
|
if (!cached_lladdr) {
|
|
|
|
NET_DBG("No lladdr but index defined");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tllao_offset) {
|
|
|
|
lladdr_changed = memcmp(lladdr.addr,
|
|
|
|
cached_lladdr->addr,
|
|
|
|
cached_lladdr->len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Update the cached address if we do not yet known it */
|
|
|
|
if (net_ipv6_nbr_data(nbr)->state == NET_IPV6_NBR_STATE_INCOMPLETE) {
|
|
|
|
if (!tllao_offset) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lladdr_changed) {
|
|
|
|
dbg_update_neighbor_lladdr_raw(lladdr.addr,
|
|
|
|
cached_lladdr,
|
|
|
|
&na_hdr->tgt);
|
|
|
|
|
|
|
|
net_linkaddr_set(cached_lladdr, lladdr.addr,
|
|
|
|
cached_lladdr->len);
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
if (na_hdr->flags & NET_ICMPV6_NA_FLAG_SOLICITED) {
|
2018-08-15 12:58:57 +02:00
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_REACHABLE);
|
2018-11-29 20:23:03 +01:00
|
|
|
net_ipv6_nbr_data(nbr)->ns_count = 0U;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* We might have active timer from PROBE */
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable = 0;
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable_timeout = 0;
|
|
|
|
|
|
|
|
net_ipv6_nbr_set_reachable_timer(net_pkt_iface(pkt),
|
|
|
|
nbr);
|
|
|
|
} else {
|
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_STALE);
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
net_ipv6_nbr_data(nbr)->is_router =
|
|
|
|
(na_hdr->flags & NET_ICMPV6_NA_FLAG_ROUTER);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
goto send_pending;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We do not update the address if override bit is not set
|
|
|
|
* and we have a valid address in the cache.
|
|
|
|
*/
|
2018-12-20 13:30:25 +01:00
|
|
|
if (!(na_hdr->flags & NET_ICMPV6_NA_FLAG_OVERRIDE) && lladdr_changed) {
|
2018-08-15 12:58:57 +02:00
|
|
|
if (net_ipv6_nbr_data(nbr)->state ==
|
|
|
|
NET_IPV6_NBR_STATE_REACHABLE) {
|
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_STALE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
if (na_hdr->flags & NET_ICMPV6_NA_FLAG_OVERRIDE ||
|
|
|
|
(!(na_hdr->flags & NET_ICMPV6_NA_FLAG_OVERRIDE) &&
|
|
|
|
tllao_offset && !lladdr_changed)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (lladdr_changed) {
|
|
|
|
dbg_update_neighbor_lladdr_raw(
|
|
|
|
lladdr.addr, cached_lladdr, &na_hdr->tgt);
|
|
|
|
|
|
|
|
net_linkaddr_set(cached_lladdr, lladdr.addr,
|
|
|
|
cached_lladdr->len);
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
if (na_hdr->flags & NET_ICMPV6_NA_FLAG_SOLICITED) {
|
2018-08-15 12:58:57 +02:00
|
|
|
ipv6_nbr_set_state(nbr, NET_IPV6_NBR_STATE_REACHABLE);
|
|
|
|
|
|
|
|
/* We might have active timer from PROBE */
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable = 0;
|
|
|
|
net_ipv6_nbr_data(nbr)->reachable_timeout = 0;
|
|
|
|
|
|
|
|
net_ipv6_nbr_set_reachable_timer(net_pkt_iface(pkt),
|
|
|
|
nbr);
|
|
|
|
} else {
|
|
|
|
if (lladdr_changed) {
|
|
|
|
ipv6_nbr_set_state(nbr,
|
|
|
|
NET_IPV6_NBR_STATE_STALE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
if (net_ipv6_nbr_data(nbr)->is_router &&
|
|
|
|
!(na_hdr->flags & NET_ICMPV6_NA_FLAG_ROUTER)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
/* Update the routing if the peer is no longer
|
|
|
|
* a router.
|
|
|
|
*/
|
|
|
|
/* FIXME */
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
net_ipv6_nbr_data(nbr)->is_router =
|
|
|
|
(na_hdr->flags & NET_ICMPV6_NA_FLAG_ROUTER);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
send_pending:
|
|
|
|
/* Next send any pending messages to the peer. */
|
|
|
|
pending = net_ipv6_nbr_data(nbr)->pending;
|
|
|
|
if (pending) {
|
2018-12-20 13:30:25 +01:00
|
|
|
NET_DBG("Sending pending %p to lladdr %s", pending,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ll_addr(cached_lladdr->addr,
|
|
|
|
cached_lladdr->len)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (net_send_data(pending) < 0) {
|
|
|
|
nbr_clear_ns_pending(net_ipv6_nbr_data(nbr));
|
|
|
|
} else {
|
|
|
|
net_ipv6_nbr_data(nbr)->pending = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_pkt_unref(pending);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
static enum net_verdict handle_na_input(struct net_pkt *pkt,
|
2018-12-20 12:59:56 +01:00
|
|
|
struct net_ipv6_hdr *ip_hdr,
|
|
|
|
struct net_icmp_hdr *icmp_hdr)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
2018-12-20 13:30:25 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(na_access,
|
|
|
|
struct net_icmpv6_na_hdr);
|
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(nd_access, struct net_icmpv6_nd_opt_hdr);
|
|
|
|
u16_t length = net_pkt_get_len(pkt);
|
2018-11-29 20:23:03 +01:00
|
|
|
u16_t tllao_offset = 0U;
|
2018-12-20 13:30:25 +01:00
|
|
|
struct net_icmpv6_nd_opt_hdr *nd_opt_hdr;
|
|
|
|
struct net_icmpv6_na_hdr *na_hdr;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_if_addr *ifaddr;
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
na_hdr = (struct net_icmpv6_na_hdr *)net_pkt_get_data(pkt, &na_access);
|
2018-12-20 13:30:25 +01:00
|
|
|
if (!na_hdr) {
|
|
|
|
NET_ERR("DROP: NULL NA header");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
dbg_addr_recv_tgt("Neighbor Advertisement",
|
2018-12-20 13:30:25 +01:00
|
|
|
&ip_hdr->src, &ip_hdr->dst, &na_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
net_stats_update_ipv6_nd_recv(net_pkt_iface(pkt));
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
if (((length < (sizeof(struct net_ipv6_hdr) +
|
|
|
|
sizeof(struct net_icmp_hdr) +
|
|
|
|
sizeof(struct net_icmpv6_na_hdr) +
|
|
|
|
sizeof(struct net_icmpv6_nd_opt_hdr))) ||
|
|
|
|
(ip_hdr->hop_limit != NET_IPV6_ND_HOP_LIMIT) ||
|
|
|
|
net_ipv6_is_addr_mcast(&na_hdr->tgt) ||
|
|
|
|
(na_hdr->flags & NET_ICMPV6_NA_FLAG_SOLICITED &&
|
|
|
|
net_ipv6_is_addr_mcast(&ip_hdr->dst))) &&
|
2019-03-27 02:57:45 +01:00
|
|
|
(icmp_hdr->code != 0U)) {
|
2018-12-20 13:30:25 +01:00
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &na_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
net_pkt_set_ipv6_ext_opt_len(pkt, sizeof(struct net_icmpv6_na_hdr));
|
|
|
|
length -= (sizeof(struct net_ipv6_hdr) + sizeof(struct net_icmp_hdr));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
nd_opt_hdr = (struct net_icmpv6_nd_opt_hdr *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &nd_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
while (nd_opt_hdr && nd_opt_hdr->len &&
|
|
|
|
net_pkt_ipv6_ext_opt_len(pkt) < length) {
|
|
|
|
u8_t prev_opt_len;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
switch (nd_opt_hdr->type) {
|
2018-08-15 12:58:57 +02:00
|
|
|
case NET_ICMPV6_ND_OPT_TLLAO:
|
|
|
|
tllao_offset = net_pkt_ip_hdr_len(pkt) +
|
|
|
|
net_pkt_ipv6_ext_len(pkt) +
|
|
|
|
sizeof(struct net_icmp_hdr) +
|
|
|
|
net_pkt_ipv6_ext_opt_len(pkt) + 1 + 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-12-20 13:30:25 +01:00
|
|
|
NET_DBG("Unknown ND option 0x%x", nd_opt_hdr->type);
|
2018-08-15 12:58:57 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_opt_len = net_pkt_ipv6_ext_opt_len(pkt);
|
|
|
|
|
|
|
|
net_pkt_set_ipv6_ext_opt_len(pkt,
|
|
|
|
net_pkt_ipv6_ext_opt_len(pkt) +
|
2018-12-20 13:30:25 +01:00
|
|
|
(nd_opt_hdr->len << 3));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (prev_opt_len >= net_pkt_ipv6_ext_opt_len(pkt)) {
|
2018-12-20 13:30:25 +01:00
|
|
|
NET_ERR("DROP: Corrupted NA message");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &nd_access);
|
|
|
|
nd_opt_hdr = (struct net_icmpv6_nd_opt_hdr *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &nd_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ifaddr = net_if_ipv6_addr_lookup_by_iface(net_pkt_iface(pkt),
|
2018-12-20 13:30:25 +01:00
|
|
|
&na_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (ifaddr) {
|
|
|
|
NET_DBG("Interface %p already has address %s",
|
|
|
|
net_pkt_iface(pkt),
|
2018-12-20 13:30:25 +01:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&na_hdr->tgt)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_DAD)
|
|
|
|
if (ifaddr->addr_state == NET_ADDR_TENTATIVE) {
|
2018-12-20 13:30:25 +01:00
|
|
|
dad_failed(net_pkt_iface(pkt), &na_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_DAD */
|
|
|
|
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 13:30:25 +01:00
|
|
|
if (!handle_na_neighbor(pkt, na_hdr, tllao_offset)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_stats_update_ipv6_nd_sent(net_pkt_iface(pkt));
|
|
|
|
|
|
|
|
net_pkt_unref(pkt);
|
|
|
|
|
|
|
|
return NET_OK;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
net_stats_update_ipv6_nd_drop(net_pkt_iface(pkt));
|
|
|
|
|
|
|
|
return NET_DROP;
|
|
|
|
}
|
|
|
|
|
|
|
|
int net_ipv6_send_ns(struct net_if *iface,
|
|
|
|
struct net_pkt *pending,
|
2018-12-20 11:08:14 +01:00
|
|
|
const struct in6_addr *src,
|
|
|
|
const struct in6_addr *dst,
|
|
|
|
const struct in6_addr *tgt,
|
2018-08-15 12:58:57 +02:00
|
|
|
bool is_my_address)
|
|
|
|
{
|
2018-12-20 11:08:14 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(ns_access,
|
|
|
|
struct net_icmpv6_ns_hdr);
|
|
|
|
struct net_pkt *pkt = NULL;
|
|
|
|
int ret = -ENOBUFS;
|
|
|
|
struct net_icmpv6_ns_hdr *ns_hdr;
|
|
|
|
struct in6_addr node_dst;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_nbr *nbr;
|
|
|
|
u8_t llao_len;
|
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
if (!dst) {
|
|
|
|
net_ipv6_addr_create_solicited_node(tgt, &node_dst);
|
|
|
|
dst = &node_dst;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
llao_len = get_llao_len(iface);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
if (is_my_address) {
|
|
|
|
src = net_ipv6_unspecified_address();
|
2019-03-27 02:57:45 +01:00
|
|
|
llao_len = 0U;
|
2018-12-20 11:08:14 +01:00
|
|
|
} else {
|
|
|
|
if (!src) {
|
|
|
|
src = net_if_ipv6_select_src_addr(iface, dst);
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
if (net_ipv6_is_addr_unspecified(src)) {
|
|
|
|
NET_DBG("No source address for NS");
|
|
|
|
ret = -EINVAL;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
pkt = net_pkt_alloc_with_buffer(iface,
|
|
|
|
sizeof(struct net_icmpv6_ns_hdr) +
|
|
|
|
llao_len,
|
|
|
|
AF_INET6, IPPROTO_ICMPV6,
|
|
|
|
ND_NET_BUF_TIMEOUT);
|
|
|
|
if (!pkt) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
net_pkt_set_ipv6_hop_limit(pkt, NET_IPV6_ND_HOP_LIMIT);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-02-20 08:58:08 +01:00
|
|
|
if (net_ipv6_create(pkt, src, dst) ||
|
2018-12-20 11:08:14 +01:00
|
|
|
net_icmpv6_create(pkt, NET_ICMPV6_NS, 0)) {
|
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
ns_hdr = (struct net_icmpv6_ns_hdr *)net_pkt_get_data(pkt, &ns_access);
|
2018-12-20 11:08:14 +01:00
|
|
|
if (!ns_hdr) {
|
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
ns_hdr->reserved = 0U;
|
2018-12-20 11:08:14 +01:00
|
|
|
net_ipaddr_copy(&ns_hdr->tgt, tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
if (net_pkt_set_data(pkt, &ns_access)) {
|
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
if (!is_my_address) {
|
2018-12-20 11:24:13 +01:00
|
|
|
if (!set_llao(pkt, net_if_get_link_addr(iface),
|
|
|
|
llao_len, NET_ICMPV6_ND_OPT_SLLAO)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
net_pkt_cursor_init(pkt);
|
2019-02-01 21:29:39 +01:00
|
|
|
net_ipv6_finalize(pkt, IPPROTO_ICMPV6);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
nbr = add_nbr(iface, &ns_hdr->tgt, false,
|
|
|
|
NET_IPV6_NBR_STATE_INCOMPLETE);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (!nbr) {
|
2019-03-15 09:35:28 +01:00
|
|
|
NET_DBG("Could not create new neighbor %s",
|
|
|
|
log_strdup(net_sprint_ipv6_addr(&ns_hdr->tgt)));
|
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pending) {
|
|
|
|
if (!net_ipv6_nbr_data(nbr)->pending) {
|
|
|
|
net_ipv6_nbr_data(nbr)->pending = net_pkt_ref(pending);
|
|
|
|
} else {
|
|
|
|
NET_DBG("Packet %p already pending for "
|
|
|
|
"operation. Discarding pending %p and pkt %p",
|
|
|
|
net_ipv6_nbr_data(nbr)->pending, pending, pkt);
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
NET_DBG("Setting timeout %d for NS", NS_REPLY_TIMEOUT);
|
|
|
|
|
|
|
|
net_ipv6_nbr_data(nbr)->send_ns = k_uptime_get();
|
|
|
|
|
|
|
|
/* Let's start the timer if necessary */
|
|
|
|
if (!k_delayed_work_remaining_get(&ipv6_ns_reply_timer)) {
|
|
|
|
k_delayed_work_submit(&ipv6_ns_reply_timer,
|
|
|
|
NS_REPLY_TIMEOUT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
dbg_addr_sent_tgt("Neighbor Solicitation", src, dst, &ns_hdr->tgt);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (net_send_data(pkt) < 0) {
|
|
|
|
NET_DBG("Cannot send NS %p (pending %p)", pkt, pending);
|
|
|
|
|
|
|
|
if (pending) {
|
|
|
|
nbr_clear_ns_pending(net_ipv6_nbr_data(nbr));
|
2018-12-20 11:08:14 +01:00
|
|
|
pending = NULL;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
net_stats_update_ipv6_nd_sent(iface);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
drop:
|
2018-12-20 11:08:14 +01:00
|
|
|
if (pending) {
|
|
|
|
net_pkt_unref(pending);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pkt) {
|
|
|
|
net_pkt_unref(pkt);
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:08:14 +01:00
|
|
|
net_stats_update_ipv6_nd_drop(iface);
|
|
|
|
|
|
|
|
return ret;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_NBR_CACHE */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
int net_ipv6_send_rs(struct net_if *iface)
|
|
|
|
{
|
2018-11-29 20:23:03 +01:00
|
|
|
u8_t llao_len = 0U;
|
2018-12-20 11:24:13 +01:00
|
|
|
int ret = -ENOBUFS;
|
|
|
|
const struct in6_addr *src;
|
|
|
|
struct in6_addr dst;
|
|
|
|
struct net_pkt *pkt;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
net_ipv6_addr_create_ll_allnodes_mcast(&dst);
|
|
|
|
src = net_if_ipv6_select_src_addr(iface, &dst);
|
|
|
|
|
|
|
|
if (!net_ipv6_is_addr_unspecified(src)) {
|
|
|
|
llao_len = get_llao_len(iface);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
pkt = net_pkt_alloc_with_buffer(iface,
|
|
|
|
sizeof(struct net_icmpv6_rs_hdr) +
|
|
|
|
llao_len,
|
|
|
|
AF_INET6, IPPROTO_ICMPV6,
|
|
|
|
ND_NET_BUF_TIMEOUT);
|
|
|
|
if (!pkt) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
net_pkt_set_ipv6_hop_limit(pkt, NET_IPV6_ND_HOP_LIMIT);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-02-20 08:58:08 +01:00
|
|
|
if (net_ipv6_create(pkt, src, &dst) ||
|
2018-12-20 11:24:13 +01:00
|
|
|
net_icmpv6_create(pkt, NET_ICMPV6_RS, 0) ||
|
|
|
|
net_pkt_memset(pkt, 0, sizeof(struct net_icmpv6_rs_hdr))) {
|
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
if (llao_len > 0) {
|
|
|
|
if (!set_llao(pkt, net_if_get_link_addr(iface),
|
|
|
|
llao_len, NET_ICMPV6_ND_OPT_SLLAO)) {
|
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
net_pkt_cursor_init(pkt);
|
2019-02-01 21:29:39 +01:00
|
|
|
net_ipv6_finalize(pkt, IPPROTO_ICMPV6);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
dbg_addr_sent("Router Solicitation", src, &dst);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (net_send_data(pkt) < 0) {
|
2018-12-20 11:24:13 +01:00
|
|
|
net_stats_update_ipv6_nd_drop(iface);
|
|
|
|
ret = -EINVAL;
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
net_stats_update_ipv6_nd_sent(iface);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
net_pkt_unref(pkt);
|
|
|
|
|
2018-12-20 11:24:13 +01:00
|
|
|
return ret;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int net_ipv6_start_rs(struct net_if *iface)
|
|
|
|
{
|
|
|
|
return net_ipv6_send_rs(iface);
|
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
static inline struct net_nbr *handle_ra_neighbor(struct net_pkt *pkt, u8_t len)
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
{
|
|
|
|
struct net_linkaddr lladdr;
|
|
|
|
struct net_linkaddr_storage llstorage;
|
|
|
|
u8_t padding;
|
|
|
|
|
|
|
|
llstorage.len = NET_LINK_ADDR_MAX_LENGTH;
|
|
|
|
lladdr.addr = llstorage.addr;
|
2018-12-20 15:26:22 +01:00
|
|
|
lladdr.len = NET_LINK_ADDR_MAX_LENGTH;
|
2018-09-11 09:16:03 +02:00
|
|
|
if (net_pkt_lladdr_src(pkt)->len < lladdr.len) {
|
|
|
|
lladdr.len = net_pkt_lladdr_src(pkt)->len;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2019-02-20 09:40:48 +01:00
|
|
|
if (net_pkt_read(pkt, lladdr.addr, lladdr.len)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
padding = len * 8U - 2 - lladdr.len;
|
2018-08-15 12:58:57 +02:00
|
|
|
if (padding) {
|
2018-12-20 15:26:22 +01:00
|
|
|
if (net_pkt_skip(pkt, padding)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-15 09:35:28 +01:00
|
|
|
return net_ipv6_nbr_add(net_pkt_iface(pkt),
|
|
|
|
&NET_IPV6_HDR(pkt)->src,
|
|
|
|
&lladdr, true,
|
|
|
|
NET_IPV6_NBR_STATE_STALE);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void handle_prefix_onlink(struct net_pkt *pkt,
|
|
|
|
struct net_icmpv6_nd_opt_prefix_info *prefix_info)
|
|
|
|
{
|
|
|
|
struct net_if_ipv6_prefix *prefix;
|
|
|
|
|
|
|
|
prefix = net_if_ipv6_prefix_lookup(net_pkt_iface(pkt),
|
|
|
|
&prefix_info->prefix,
|
|
|
|
prefix_info->prefix_len);
|
|
|
|
if (!prefix) {
|
|
|
|
if (!prefix_info->valid_lifetime) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
prefix = net_if_ipv6_prefix_add(net_pkt_iface(pkt),
|
|
|
|
&prefix_info->prefix,
|
|
|
|
prefix_info->prefix_len,
|
|
|
|
prefix_info->valid_lifetime);
|
|
|
|
if (prefix) {
|
|
|
|
NET_DBG("Interface %p add prefix %s/%d lifetime %u",
|
|
|
|
net_pkt_iface(pkt),
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(
|
|
|
|
&prefix_info->prefix)),
|
2018-08-15 12:58:57 +02:00
|
|
|
prefix_info->prefix_len,
|
|
|
|
prefix_info->valid_lifetime);
|
|
|
|
} else {
|
|
|
|
NET_ERR("Prefix %s/%d could not be added to iface %p",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(
|
|
|
|
&prefix_info->prefix)),
|
2018-08-15 12:58:57 +02:00
|
|
|
prefix_info->prefix_len,
|
|
|
|
net_pkt_iface(pkt));
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (prefix_info->valid_lifetime) {
|
|
|
|
case 0:
|
|
|
|
NET_DBG("Interface %p delete prefix %s/%d",
|
|
|
|
net_pkt_iface(pkt),
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&prefix_info->prefix)),
|
2018-08-15 12:58:57 +02:00
|
|
|
prefix_info->prefix_len);
|
|
|
|
|
|
|
|
net_if_ipv6_prefix_rm(net_pkt_iface(pkt),
|
|
|
|
&prefix->prefix,
|
|
|
|
prefix->len);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_IPV6_ND_INFINITE_LIFETIME:
|
|
|
|
NET_DBG("Interface %p prefix %s/%d infinite",
|
|
|
|
net_pkt_iface(pkt),
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&prefix->prefix)),
|
2018-08-15 12:58:57 +02:00
|
|
|
prefix->len);
|
|
|
|
|
|
|
|
net_if_ipv6_prefix_set_lf(prefix, true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
NET_DBG("Interface %p update prefix %s/%u lifetime %u",
|
|
|
|
net_pkt_iface(pkt),
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&prefix_info->prefix)),
|
2018-12-20 15:26:22 +01:00
|
|
|
prefix_info->prefix_len, prefix_info->valid_lifetime);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
net_if_ipv6_prefix_set_lf(prefix, false);
|
|
|
|
net_if_ipv6_prefix_set_timer(prefix,
|
2018-12-20 15:26:22 +01:00
|
|
|
prefix_info->valid_lifetime);
|
2018-08-15 12:58:57 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define TWO_HOURS (2 * 60 * 60)
|
|
|
|
|
2018-08-17 16:04:10 +02:00
|
|
|
static u32_t time_diff(u32_t time1, u32_t time2)
|
|
|
|
{
|
|
|
|
return (u32_t)abs((s32_t)time1 - (s32_t)time2);
|
|
|
|
}
|
|
|
|
|
2018-08-15 12:58:57 +02:00
|
|
|
static inline u32_t remaining_lifetime(struct net_if_addr *ifaddr)
|
|
|
|
{
|
2018-08-17 16:04:10 +02:00
|
|
|
u64_t remaining;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-08-17 16:04:10 +02:00
|
|
|
if (ifaddr->lifetime.timer_timeout == 0) {
|
2018-08-15 12:58:57 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-08-17 16:04:10 +02:00
|
|
|
remaining = (u64_t)ifaddr->lifetime.timer_timeout +
|
|
|
|
(u64_t)ifaddr->lifetime.wrap_counter *
|
|
|
|
(u64_t)NET_TIMEOUT_MAX_VALUE -
|
|
|
|
(u64_t)time_diff(k_uptime_get_32(),
|
|
|
|
ifaddr->lifetime.timer_start);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-08-17 16:04:10 +02:00
|
|
|
return (u32_t)(remaining / K_MSEC(1000));
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void handle_prefix_autonomous(struct net_pkt *pkt,
|
|
|
|
struct net_icmpv6_nd_opt_prefix_info *prefix_info)
|
|
|
|
{
|
|
|
|
struct in6_addr addr = { };
|
|
|
|
struct net_if_addr *ifaddr;
|
|
|
|
|
|
|
|
/* Create IPv6 address using the given prefix and iid. We first
|
|
|
|
* setup link local address, and then copy prefix over first 8
|
|
|
|
* bytes of that address.
|
|
|
|
*/
|
|
|
|
net_ipv6_addr_create_iid(&addr,
|
|
|
|
net_if_get_link_addr(net_pkt_iface(pkt)));
|
|
|
|
memcpy(&addr, &prefix_info->prefix, sizeof(struct in6_addr) / 2);
|
|
|
|
|
|
|
|
ifaddr = net_if_ipv6_addr_lookup(&addr, NULL);
|
|
|
|
if (ifaddr && ifaddr->addr_type == NET_ADDR_AUTOCONF) {
|
|
|
|
if (prefix_info->valid_lifetime ==
|
|
|
|
NET_IPV6_ND_INFINITE_LIFETIME) {
|
|
|
|
net_if_addr_set_lf(ifaddr, true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* RFC 4862 ch 5.5.3 */
|
|
|
|
if ((prefix_info->valid_lifetime > TWO_HOURS) ||
|
|
|
|
(prefix_info->valid_lifetime >
|
|
|
|
remaining_lifetime(ifaddr))) {
|
|
|
|
NET_DBG("Timer updating for address %s "
|
|
|
|
"long lifetime %u secs",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&addr)),
|
2018-08-15 12:58:57 +02:00
|
|
|
prefix_info->valid_lifetime);
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
net_if_ipv6_addr_update_lifetime(
|
|
|
|
ifaddr, prefix_info->valid_lifetime);
|
2018-08-15 12:58:57 +02:00
|
|
|
} else {
|
|
|
|
NET_DBG("Timer updating for address %s "
|
|
|
|
"lifetime %u secs",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&addr)),
|
|
|
|
TWO_HOURS);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
net_if_ipv6_addr_update_lifetime(ifaddr, TWO_HOURS);
|
|
|
|
}
|
|
|
|
|
|
|
|
net_if_addr_set_lf(ifaddr, false);
|
|
|
|
} else {
|
|
|
|
if (prefix_info->valid_lifetime ==
|
|
|
|
NET_IPV6_ND_INFINITE_LIFETIME) {
|
|
|
|
net_if_ipv6_addr_add(net_pkt_iface(pkt),
|
|
|
|
&addr, NET_ADDR_AUTOCONF, 0);
|
|
|
|
} else {
|
|
|
|
net_if_ipv6_addr_add(net_pkt_iface(pkt),
|
|
|
|
&addr, NET_ADDR_AUTOCONF,
|
|
|
|
prefix_info->valid_lifetime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
static inline bool handle_ra_prefix(struct net_pkt *pkt)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(rapfx_access,
|
|
|
|
struct net_icmpv6_nd_opt_prefix_info);
|
|
|
|
struct net_icmpv6_nd_opt_prefix_info *pfx_info;
|
2019-06-25 09:14:53 +02:00
|
|
|
u32_t valid_lifetime, preferred_lifetime;
|
2018-12-20 15:26:22 +01:00
|
|
|
|
|
|
|
pfx_info = (struct net_icmpv6_nd_opt_prefix_info *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &rapfx_access);
|
2018-12-20 15:26:22 +01:00
|
|
|
if (!pfx_info) {
|
|
|
|
return false;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &rapfx_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-06-25 09:14:53 +02:00
|
|
|
valid_lifetime = ntohl(pfx_info->valid_lifetime);
|
|
|
|
preferred_lifetime = ntohl(pfx_info->preferred_lifetime);
|
2018-12-20 15:26:22 +01:00
|
|
|
|
2019-06-25 09:14:53 +02:00
|
|
|
if (valid_lifetime >= preferred_lifetime &&
|
2018-12-20 15:26:22 +01:00
|
|
|
!net_ipv6_is_ll_addr(&pfx_info->prefix)) {
|
|
|
|
if (pfx_info->flags & NET_ICMPV6_RA_FLAG_ONLINK) {
|
|
|
|
handle_prefix_onlink(pkt, pfx_info);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
if ((pfx_info->flags & NET_ICMPV6_RA_FLAG_AUTONOMOUS) &&
|
2019-06-25 09:14:53 +02:00
|
|
|
valid_lifetime &&
|
2018-12-20 15:26:22 +01:00
|
|
|
(pfx_info->prefix_len == NET_IPV6_DEFAULT_PREFIX_LEN)) {
|
|
|
|
handle_prefix_autonomous(pkt, pfx_info);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
return true;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_6LO_CONTEXT)
|
|
|
|
/* 6lowpan Context Option RFC 6775, 4.2 */
|
2018-12-20 15:26:22 +01:00
|
|
|
static inline bool handle_ra_6co(struct net_pkt *pkt, u8_t len)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(ctx_access, struct net_icmpv6_nd_opt_6co);
|
|
|
|
struct net_icmpv6_nd_opt_6co *context;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
context = (struct net_icmpv6_nd_opt_6co *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &ctx_access);
|
2018-12-20 15:26:22 +01:00
|
|
|
if (!context) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* RFC 6775, 4.2
|
|
|
|
* Context Length: 8-bit unsigned integer. The number of leading
|
|
|
|
* bits in the Context Prefix field that are valid. The value ranges
|
|
|
|
* from 0 to 128. If it is more than 64, then the Length MUST be 3.
|
|
|
|
*/
|
2019-03-27 02:57:45 +01:00
|
|
|
if ((context->context_len > 64 && len != 3U) ||
|
|
|
|
(context->context_len <= 64U && len != 2U)) {
|
2018-12-20 15:26:22 +01:00
|
|
|
return false;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
context->context_len = context->context_len / 8U;
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
/* context_len: The number of leading bits in the Context Prefix
|
2018-12-20 15:26:22 +01:00
|
|
|
* field that are valid. Rest must be set to 0 by the sender and
|
|
|
|
* ignored by the receiver. But since there is no way to make sure
|
|
|
|
* the sender followed the rule, let's make sure rest is set to 0.
|
2018-08-15 12:58:57 +02:00
|
|
|
*/
|
2018-12-20 15:26:22 +01:00
|
|
|
if (context->context_len != sizeof(struct in6_addr)) {
|
|
|
|
(void)memset(context->prefix.s6_addr + context->context_len, 0,
|
|
|
|
sizeof(struct in6_addr) - context->context_len);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
net_6lo_set_context(net_pkt_iface(pkt), context);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
return true;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
static enum net_verdict handle_ra_input(struct net_pkt *pkt,
|
2018-12-20 12:59:56 +01:00
|
|
|
struct net_ipv6_hdr *ip_hdr,
|
|
|
|
struct net_icmp_hdr *icmp_hdr)
|
2018-08-15 12:58:57 +02:00
|
|
|
{
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(ra_access,
|
|
|
|
struct net_icmpv6_ra_hdr);
|
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(nd_access, struct net_icmpv6_nd_opt_hdr);
|
|
|
|
u16_t length = net_pkt_get_len(pkt);
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_nbr *nbr = NULL;
|
2018-12-20 15:26:22 +01:00
|
|
|
struct net_icmpv6_nd_opt_hdr *nd_opt_hdr;
|
|
|
|
struct net_icmpv6_ra_hdr *ra_hdr;
|
2018-08-15 12:58:57 +02:00
|
|
|
struct net_if_router *router;
|
2019-06-25 09:14:53 +02:00
|
|
|
u32_t mtu, reachable_time, retrans_timer;
|
|
|
|
u16_t router_lifetime;
|
2018-12-20 15:26:22 +01:00
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
ra_hdr = (struct net_icmpv6_ra_hdr *)net_pkt_get_data(pkt, &ra_access);
|
2018-12-20 15:26:22 +01:00
|
|
|
if (!ra_hdr) {
|
|
|
|
NET_ERR("DROP: NULL RA header");
|
|
|
|
goto drop;
|
|
|
|
}
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
dbg_addr_recv("Router Advertisement", &ip_hdr->src, &ip_hdr->dst);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
net_stats_update_ipv6_nd_recv(net_pkt_iface(pkt));
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
if (((length < (sizeof(struct net_ipv6_hdr) +
|
|
|
|
sizeof(struct net_icmp_hdr) +
|
|
|
|
sizeof(struct net_icmpv6_ra_hdr) +
|
|
|
|
sizeof(struct net_icmpv6_nd_opt_hdr))) ||
|
|
|
|
(ip_hdr->hop_limit != NET_IPV6_ND_HOP_LIMIT) ||
|
2019-03-27 02:57:45 +01:00
|
|
|
!net_ipv6_is_ll_addr(&ip_hdr->src)) &&
|
|
|
|
icmp_hdr->code != 0U) {
|
2018-12-20 15:26:22 +01:00
|
|
|
goto drop;
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
net_pkt_acknowledge_data(pkt, &ra_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2019-06-25 09:14:53 +02:00
|
|
|
router_lifetime = ntohs(ra_hdr->router_lifetime);
|
|
|
|
reachable_time = ntohl(ra_hdr->reachable_time);
|
|
|
|
retrans_timer = ntohl(ra_hdr->retrans_timer);
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
if (ra_hdr->cur_hop_limit) {
|
|
|
|
net_ipv6_set_hop_limit(net_pkt_iface(pkt),
|
|
|
|
ra_hdr->cur_hop_limit);
|
2018-08-15 12:58:57 +02:00
|
|
|
NET_DBG("New hop limit %d",
|
|
|
|
net_if_ipv6_get_hop_limit(net_pkt_iface(pkt)));
|
|
|
|
}
|
|
|
|
|
2019-06-25 09:14:53 +02:00
|
|
|
if (reachable_time && reachable_time <= MAX_REACHABLE_TIME &&
|
2018-08-15 12:58:57 +02:00
|
|
|
(net_if_ipv6_get_reachable_time(net_pkt_iface(pkt)) !=
|
2019-06-25 09:14:53 +02:00
|
|
|
reachable_time)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
net_if_ipv6_set_base_reachable_time(net_pkt_iface(pkt),
|
2019-06-25 09:14:53 +02:00
|
|
|
reachable_time);
|
2018-08-15 12:58:57 +02:00
|
|
|
net_if_ipv6_set_reachable_time(
|
|
|
|
net_pkt_iface(pkt)->config.ip.ipv6);
|
|
|
|
}
|
|
|
|
|
2019-06-25 09:14:53 +02:00
|
|
|
if (retrans_timer) {
|
2018-08-15 12:58:57 +02:00
|
|
|
net_if_ipv6_set_retrans_timer(net_pkt_iface(pkt),
|
2018-12-20 15:26:22 +01:00
|
|
|
ra_hdr->retrans_timer);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
net_pkt_set_ipv6_ext_opt_len(pkt, sizeof(struct net_icmpv6_ra_hdr));
|
|
|
|
length -= (sizeof(struct net_ipv6_hdr) + sizeof(struct net_icmp_hdr));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
nd_opt_hdr = (struct net_icmpv6_nd_opt_hdr *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &nd_access);
|
2018-12-20 15:26:22 +01:00
|
|
|
while (nd_opt_hdr) {
|
|
|
|
net_pkt_acknowledge_data(pkt, &nd_access);
|
|
|
|
|
|
|
|
switch (nd_opt_hdr->type) {
|
2018-08-15 12:58:57 +02:00
|
|
|
case NET_ICMPV6_ND_OPT_SLLAO:
|
2018-12-20 15:26:22 +01:00
|
|
|
nbr = handle_ra_neighbor(pkt, nd_opt_hdr->len);
|
|
|
|
if (!nbr) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case NET_ICMPV6_ND_OPT_MTU:
|
|
|
|
/* MTU has reserved 2 bytes, so skip it. */
|
2018-12-20 15:26:22 +01:00
|
|
|
if (net_pkt_skip(pkt, 2) ||
|
2019-02-20 09:40:48 +01:00
|
|
|
net_pkt_read_be32(pkt, &mtu)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mtu < MIN_IPV6_MTU || mtu > MAX_IPV6_MTU) {
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_ERR("DROP: Unsupported MTU %u, min is %u, "
|
2018-08-15 12:58:57 +02:00
|
|
|
"max is %u",
|
|
|
|
mtu, MIN_IPV6_MTU, MAX_IPV6_MTU);
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_if_set_mtu(net_pkt_iface(pkt), mtu);
|
|
|
|
|
|
|
|
break;
|
|
|
|
case NET_ICMPV6_ND_OPT_PREFIX_INFO:
|
2018-12-20 15:26:22 +01:00
|
|
|
if (!handle_ra_prefix(pkt)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
#if defined(CONFIG_NET_6LO_CONTEXT)
|
|
|
|
case NET_ICMPV6_ND_OPT_6CO:
|
|
|
|
/* RFC 6775, 4.2 (Length)*/
|
2019-03-27 02:57:45 +01:00
|
|
|
if (!(nd_opt_hdr->len == 2U || nd_opt_hdr->len == 3U)) {
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_ERR("DROP: Invalid 6CO length %d",
|
|
|
|
nd_opt_hdr->len);
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
if (!handle_ra_6co(pkt, nd_opt_hdr->len)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case NET_ICMPV6_ND_OPT_ROUTE:
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_DBG("Route option skipped");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto skip;
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_RA_RDNSS)
|
|
|
|
case NET_ICMPV6_ND_OPT_RDNSS:
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_DBG("RDNSS option skipped");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto skip;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
case NET_ICMPV6_ND_OPT_DNSSL:
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_DBG("DNSSL option skipped");
|
2018-08-15 12:58:57 +02:00
|
|
|
goto skip;
|
|
|
|
|
|
|
|
default:
|
2018-12-20 15:26:22 +01:00
|
|
|
NET_DBG("Unknown ND option 0x%x", nd_opt_hdr->type);
|
2018-08-15 12:58:57 +02:00
|
|
|
skip:
|
2019-03-27 02:57:45 +01:00
|
|
|
if (net_pkt_skip(pkt, nd_opt_hdr->len * 8U - 2)) {
|
2018-08-15 12:58:57 +02:00
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2018-12-20 15:26:22 +01:00
|
|
|
|
|
|
|
nd_opt_hdr = (struct net_icmpv6_nd_opt_hdr *)
|
2019-02-20 09:28:18 +01:00
|
|
|
net_pkt_get_data(pkt, &nd_access);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
2018-12-18 15:28:40 +01:00
|
|
|
router = net_if_ipv6_router_lookup(net_pkt_iface(pkt), &ip_hdr->src);
|
2018-08-15 12:58:57 +02:00
|
|
|
if (router) {
|
2019-06-25 09:14:53 +02:00
|
|
|
if (!router_lifetime) {
|
2018-08-15 12:58:57 +02:00
|
|
|
/* TODO: Start rs_timer on iface if no routers
|
|
|
|
* at all available on iface.
|
|
|
|
*/
|
|
|
|
net_if_ipv6_router_rm(router);
|
|
|
|
} else {
|
|
|
|
if (nbr) {
|
|
|
|
net_ipv6_nbr_data(nbr)->is_router = true;
|
|
|
|
}
|
|
|
|
|
2018-12-20 15:26:22 +01:00
|
|
|
net_if_ipv6_router_update_lifetime(
|
2019-06-25 09:14:53 +02:00
|
|
|
router, router_lifetime);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
net_if_ipv6_router_add(net_pkt_iface(pkt),
|
2019-06-25 09:14:53 +02:00
|
|
|
&ip_hdr->src, router_lifetime);
|
2018-08-15 12:58:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (nbr && net_ipv6_nbr_data(nbr)->pending) {
|
|
|
|
NET_DBG("Sending pending pkt %p to %s",
|
|
|
|
net_ipv6_nbr_data(nbr)->pending,
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_sprint_ipv6_addr(&NET_IPV6_HDR(
|
|
|
|
net_ipv6_nbr_data(nbr)->pending)->dst)));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
if (net_send_data(net_ipv6_nbr_data(nbr)->pending) < 0) {
|
|
|
|
net_pkt_unref(net_ipv6_nbr_data(nbr)->pending);
|
|
|
|
}
|
|
|
|
|
|
|
|
nbr_clear_ns_pending(net_ipv6_nbr_data(nbr));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Cancel the RS timer on iface */
|
2019-05-21 12:07:08 +02:00
|
|
|
net_if_stop_rs(net_pkt_iface(pkt));
|
2018-08-15 12:58:57 +02:00
|
|
|
|
|
|
|
net_pkt_unref(pkt);
|
|
|
|
|
|
|
|
return NET_OK;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
net_stats_update_ipv6_nd_drop(net_pkt_iface(pkt));
|
|
|
|
|
|
|
|
return NET_DROP;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6_ND */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_NBR_CACHE)
|
|
|
|
static struct net_icmpv6_handler ns_input_handler = {
|
|
|
|
.type = NET_ICMPV6_NS,
|
|
|
|
.code = 0,
|
|
|
|
.handler = handle_ns_input,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct net_icmpv6_handler na_input_handler = {
|
|
|
|
.type = NET_ICMPV6_NA,
|
|
|
|
.code = 0,
|
|
|
|
.handler = handle_na_input,
|
|
|
|
};
|
|
|
|
#endif /* CONFIG_NET_IPV6_NBR_CACHE */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
static struct net_icmpv6_handler ra_input_handler = {
|
|
|
|
.type = NET_ICMPV6_RA,
|
|
|
|
.code = 0,
|
|
|
|
.handler = handle_ra_input,
|
|
|
|
};
|
|
|
|
#endif /* CONFIG_NET_IPV6_ND */
|
|
|
|
|
|
|
|
void net_ipv6_nbr_init(void)
|
|
|
|
{
|
|
|
|
#if defined(CONFIG_NET_IPV6_NBR_CACHE)
|
|
|
|
net_icmpv6_register_handler(&ns_input_handler);
|
|
|
|
net_icmpv6_register_handler(&na_input_handler);
|
|
|
|
k_delayed_work_init(&ipv6_ns_reply_timer, ipv6_ns_reply_timeout);
|
|
|
|
#endif
|
|
|
|
#if defined(CONFIG_NET_IPV6_ND)
|
|
|
|
net_icmpv6_register_handler(&ra_input_handler);
|
|
|
|
k_delayed_work_init(&ipv6_nd_reachable_timer,
|
|
|
|
ipv6_nd_reachable_timeout);
|
2019-03-15 09:35:28 +01:00
|
|
|
k_sem_init(&nbr_lock, 1, UINT_MAX);
|
2018-08-15 12:58:57 +02:00
|
|
|
#endif
|
|
|
|
}
|