61dca2b8bf
In order to save memory, a single multicast routing entry now contains configurable number of network interfaces. Signed-off-by: Konrad Derda <konrad.derda@nordicsemi.no>
342 lines
8.7 KiB
C
342 lines
8.7 KiB
C
/** @file
|
|
* @brief Route handler
|
|
*
|
|
* This is not to be included by the application.
|
|
*/
|
|
|
|
/*
|
|
* Copyright (c) 2016 Intel Corporation
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#ifndef __ROUTE_H
|
|
#define __ROUTE_H
|
|
|
|
#include <zephyr/kernel.h>
|
|
#include <zephyr/sys/slist.h>
|
|
|
|
#include <zephyr/net/net_ip.h>
|
|
#include <zephyr/net/net_timeout.h>
|
|
|
|
#include "nbr.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/**
|
|
* @brief Next hop entry for a given route.
|
|
*/
|
|
struct net_route_nexthop {
|
|
/** Pointer to nexthop that has same route to a specific
|
|
* neighbor.
|
|
*/
|
|
sys_snode_t node;
|
|
|
|
/** Next hop neighbor */
|
|
struct net_nbr *nbr;
|
|
};
|
|
|
|
/**
|
|
* @brief Route entry to a specific neighbor.
|
|
*/
|
|
struct net_route_entry {
|
|
/** Node information. The routes are also in separate list in
|
|
* order to keep track which one of them is the oldest so that
|
|
* we can remove it if we run out of available routes.
|
|
* The oldest one is the last entry in the list.
|
|
*/
|
|
sys_snode_t node;
|
|
|
|
/** List of neighbors that the routes go through. */
|
|
sys_slist_t nexthop;
|
|
|
|
/** Network interface for the route. */
|
|
struct net_if *iface;
|
|
|
|
/** Route lifetime timer. */
|
|
struct net_timeout lifetime;
|
|
|
|
/** IPv6 address/prefix of the route. */
|
|
struct in6_addr addr;
|
|
|
|
/** IPv6 address/prefix length. */
|
|
uint8_t prefix_len;
|
|
|
|
uint8_t preference : 2;
|
|
|
|
/** Is the route valid forever */
|
|
uint8_t is_infinite : 1;
|
|
};
|
|
|
|
/* Route preference values, as defined in RFC 4191 */
|
|
#define NET_ROUTE_PREFERENCE_HIGH 0x01
|
|
#define NET_ROUTE_PREFERENCE_MEDIUM 0x00
|
|
#define NET_ROUTE_PREFERENCE_LOW 0x03 /* -1 if treated as 2 bit signed int */
|
|
#define NET_ROUTE_PREFERENCE_RESERVED 0x02
|
|
|
|
/**
|
|
* @brief Lookup route to a given destination.
|
|
*
|
|
* @param iface Network interface. If NULL, then check against all interfaces.
|
|
* @param dst Destination IPv6 address.
|
|
*
|
|
* @return Return route entry related to a given destination address, NULL
|
|
* if not found.
|
|
*/
|
|
#if defined(CONFIG_NET_NATIVE)
|
|
struct net_route_entry *net_route_lookup(struct net_if *iface,
|
|
struct in6_addr *dst);
|
|
#else
|
|
static inline struct net_route_entry *net_route_lookup(struct net_if *iface,
|
|
struct in6_addr *dst)
|
|
{
|
|
ARG_UNUSED(iface);
|
|
ARG_UNUSED(dst);
|
|
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* @brief Add a route to routing table.
|
|
*
|
|
* @param iface Network interface that this route is tied to.
|
|
* @param addr IPv6 address.
|
|
* @param prefix_len Length of the IPv6 address/prefix.
|
|
* @param nexthop IPv6 address of the Next hop device.
|
|
* @param lifetime Route lifetime in seconds.
|
|
* @param preference Route preference.
|
|
*
|
|
* @return Return created route entry, NULL if could not be created.
|
|
*/
|
|
struct net_route_entry *net_route_add(struct net_if *iface,
|
|
struct in6_addr *addr,
|
|
uint8_t prefix_len,
|
|
struct in6_addr *nexthop,
|
|
uint32_t lifetime,
|
|
uint8_t preference);
|
|
|
|
/**
|
|
* @brief Delete a route from routing table.
|
|
*
|
|
* @param entry Existing route entry.
|
|
*
|
|
* @return 0 if ok, <0 if error
|
|
*/
|
|
int net_route_del(struct net_route_entry *entry);
|
|
|
|
/**
|
|
* @brief Delete a route from routing table by nexthop.
|
|
*
|
|
* @param iface Network interface to use.
|
|
* @param nexthop IPv6 address of the nexthop device.
|
|
*
|
|
* @return number of routes deleted, <0 if error
|
|
*/
|
|
int net_route_del_by_nexthop(struct net_if *iface,
|
|
struct in6_addr *nexthop);
|
|
|
|
/**
|
|
* @brief Update the route lifetime.
|
|
*
|
|
* @param route Pointer to routing entry.
|
|
* @param lifetime Route lifetime in seconds.
|
|
*
|
|
* @return 0 if ok, <0 if error
|
|
*/
|
|
void net_route_update_lifetime(struct net_route_entry *route, uint32_t lifetime);
|
|
|
|
/**
|
|
* @brief Get nexthop IPv6 address tied to this route.
|
|
*
|
|
* There can be multiple routes to a host but this function
|
|
* will only return the first one in this version.
|
|
*
|
|
* @param entry Route entry to use.
|
|
*
|
|
* @return IPv6 address of the nexthop, NULL if not found.
|
|
*/
|
|
struct in6_addr *net_route_get_nexthop(struct net_route_entry *entry);
|
|
|
|
/**
|
|
* @brief Get generic neighbor entry from route entry.
|
|
*
|
|
* @param route Pointer to routing entry.
|
|
*
|
|
* @return Generic neighbor entry.
|
|
*/
|
|
struct net_nbr *net_route_get_nbr(struct net_route_entry *route);
|
|
|
|
typedef void (*net_route_cb_t)(struct net_route_entry *entry,
|
|
void *user_data);
|
|
|
|
/**
|
|
* @brief Go through all the routing entries and call callback
|
|
* for each entry that is in use.
|
|
*
|
|
* @param cb User supplied callback function to call.
|
|
* @param user_data User specified data.
|
|
*
|
|
* @return Total number of routing entries found.
|
|
*/
|
|
int net_route_foreach(net_route_cb_t cb, void *user_data);
|
|
|
|
#if defined(CONFIG_NET_ROUTE_MCAST)
|
|
/**
|
|
* @brief Multicast route entry.
|
|
*/
|
|
struct net_route_entry_mcast {
|
|
/** Network interfaces for the route. */
|
|
struct net_if *ifaces[CONFIG_NET_MCAST_ROUTE_MAX_IFACES];
|
|
|
|
/** Extra routing engine specific data */
|
|
void *data;
|
|
|
|
/** IPv6 multicast group of the route. */
|
|
struct in6_addr group;
|
|
|
|
/** Routing entry lifetime in seconds. */
|
|
uint32_t lifetime;
|
|
|
|
/** Is this entry in use or not */
|
|
bool is_used;
|
|
|
|
/** IPv6 multicast group prefix length. */
|
|
uint8_t prefix_len;
|
|
};
|
|
#else
|
|
struct net_route_entry_mcast;
|
|
#endif
|
|
|
|
typedef void (*net_route_mcast_cb_t)(struct net_route_entry_mcast *entry,
|
|
void *user_data);
|
|
|
|
/**
|
|
* @brief Forwards a multicast packet by checking the local multicast
|
|
* routing table
|
|
*
|
|
* @param pkt The original received ipv6 packet to forward
|
|
* @param hdr The IPv6 header of the packet
|
|
*
|
|
* @return Number of interfaces which forwarded the packet, or a negative
|
|
* value in case of an error.
|
|
*/
|
|
int net_route_mcast_forward_packet(struct net_pkt *pkt,
|
|
const struct net_ipv6_hdr *hdr);
|
|
|
|
/**
|
|
* @brief Go through all the multicast routing entries and call callback
|
|
* for each entry that is in use.
|
|
*
|
|
* @param cb User supplied callback function to call.
|
|
* @param skip Do not call callback for this address.
|
|
* @param user_data User specified data.
|
|
*
|
|
* @return Total number of multicast routing entries that are in use.
|
|
*/
|
|
int net_route_mcast_foreach(net_route_mcast_cb_t cb,
|
|
struct in6_addr *skip,
|
|
void *user_data);
|
|
|
|
/**
|
|
* @brief Add a multicast routing entry.
|
|
*
|
|
* @param iface Network interface to use.
|
|
* @param group IPv6 multicast address.
|
|
* @param prefix_len Length of the IPv6 group that must match.
|
|
*
|
|
* @return Multicast routing entry.
|
|
*/
|
|
struct net_route_entry_mcast *net_route_mcast_add(struct net_if *iface,
|
|
struct in6_addr *group,
|
|
uint8_t prefix_len);
|
|
|
|
/**
|
|
* @brief Delete a multicast routing entry.
|
|
*
|
|
* @param route Multicast routing entry.
|
|
*
|
|
* @return True if entry was deleted, false otherwise.
|
|
*/
|
|
bool net_route_mcast_del(struct net_route_entry_mcast *route);
|
|
|
|
/**
|
|
* @brief Lookup a multicast routing entry.
|
|
*
|
|
* @param group IPv6 multicast group address
|
|
*
|
|
* @return Routing entry corresponding this multicast group.
|
|
*/
|
|
struct net_route_entry_mcast *
|
|
net_route_mcast_lookup(struct in6_addr *group);
|
|
|
|
/**
|
|
* @brief Add an interface to multicast routing entry.
|
|
*
|
|
* @param entry Multicast routing entry.
|
|
* @param iface Network interface to be added.
|
|
*
|
|
* @return True if the interface was added or found on the
|
|
* list, false otherwise.
|
|
*/
|
|
bool net_route_mcast_iface_add(struct net_route_entry_mcast *entry, struct net_if *iface);
|
|
|
|
/**
|
|
* @brief Delete an interface from multicast routing entry.
|
|
*
|
|
* @param entry Multicast routing entry.
|
|
* @param iface Network interface to be deleted.
|
|
*
|
|
* @return True if entry was deleted, false otherwise.
|
|
*/
|
|
bool net_route_mcast_iface_del(struct net_route_entry_mcast *entry, struct net_if *iface);
|
|
|
|
/**
|
|
* @brief Return a route to destination via some intermediate host.
|
|
*
|
|
* @param iface Network interface
|
|
* @param dst Destination IPv6 address
|
|
* @param route Route entry to destination is returned.
|
|
* @param nexthop Next hop neighbor IPv6 address is returned.
|
|
*
|
|
* @return True if there is a route to the destination, False otherwise
|
|
*/
|
|
bool net_route_get_info(struct net_if *iface,
|
|
struct in6_addr *dst,
|
|
struct net_route_entry **route,
|
|
struct in6_addr **nexthop);
|
|
|
|
/**
|
|
* @brief Send the network packet to network via some intermediate host.
|
|
*
|
|
* @param pkt Network packet to send.
|
|
* @param nexthop Next hop neighbor IPv6 address.
|
|
*
|
|
* @return 0 if there was no error, <0 if the packet could not be sent.
|
|
*/
|
|
int net_route_packet(struct net_pkt *pkt, struct in6_addr *nexthop);
|
|
|
|
/**
|
|
* @brief Send the network packet to network via the given interface.
|
|
*
|
|
* @param pkt Network packet to send.
|
|
* @param iface The network interface the packet should be sent on.
|
|
*
|
|
* @return 0 if there was no error, <0 if the packet could not be sent.
|
|
*/
|
|
int net_route_packet_if(struct net_pkt *pkt, struct net_if *iface);
|
|
|
|
#if defined(CONFIG_NET_ROUTE) && defined(CONFIG_NET_NATIVE)
|
|
void net_route_init(void);
|
|
#else
|
|
#define net_route_init(...)
|
|
#endif /* CONFIG_NET_ROUTE */
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* __ROUTE_H */
|