5a5793a5ff
Add RIB (Routing Information Base) library. This library implements an IPv4 routing table optimized for control plane operations. It implements a control plane struct containing routes in a tree and provides fast add/del operations for routes. Also it allows to perform fast subtree traversals (i.e. retrieve existing subroutes for a given prefix). This structure will be used as a control plane helper structure for FIB implementation. Also it might be used standalone in other different places such as bitmaps for example. Internal implementation is level compressed binary trie. Signed-off-by: Vladimir Medvedkin <vladimir.medvedkin@intel.com>
278 lines
5.8 KiB
C
278 lines
5.8 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright(c) 2018 Vladimir Medvedkin <medvedkinv@gmail.com>
|
|
* Copyright(c) 2019 Intel Corporation
|
|
*/
|
|
|
|
#ifndef _RTE_RIB_H_
|
|
#define _RTE_RIB_H_
|
|
|
|
/**
|
|
* @file
|
|
* Level compressed tree implementation for IPv4 Longest Prefix Match
|
|
*/
|
|
|
|
#include <rte_compat.h>
|
|
|
|
/**
|
|
* rte_rib_get_nxt() flags
|
|
*/
|
|
enum {
|
|
/** flag to get all subroutes in a RIB tree */
|
|
RTE_RIB_GET_NXT_ALL,
|
|
/** flag to get first matched subroutes in a RIB tree */
|
|
RTE_RIB_GET_NXT_COVER
|
|
};
|
|
|
|
struct rte_rib;
|
|
struct rte_rib_node;
|
|
|
|
/** RIB configuration structure */
|
|
struct rte_rib_conf {
|
|
/**
|
|
* Size of extension block inside rte_rib_node.
|
|
* This space could be used to store additional user
|
|
* defined data.
|
|
*/
|
|
size_t ext_sz;
|
|
/* size of rte_rib_node's pool */
|
|
int max_nodes;
|
|
};
|
|
|
|
/**
|
|
* Get an IPv4 mask from prefix length
|
|
* It is caller responsibility to make sure depth is not bigger than 32
|
|
*
|
|
* @param depth
|
|
* prefix length
|
|
* @return
|
|
* IPv4 mask
|
|
*/
|
|
static inline uint32_t
|
|
rte_rib_depth_to_mask(uint8_t depth)
|
|
{
|
|
return (uint32_t)(UINT64_MAX << (32 - depth));
|
|
}
|
|
|
|
/**
|
|
* Lookup an IP into the RIB structure
|
|
*
|
|
* @param rib
|
|
* RIB object handle
|
|
* @param ip
|
|
* IP to be looked up in the RIB
|
|
* @return
|
|
* pointer to struct rte_rib_node on success
|
|
* NULL otherwise
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib_node *
|
|
rte_rib_lookup(struct rte_rib *rib, uint32_t ip);
|
|
|
|
/**
|
|
* Lookup less specific route into the RIB structure
|
|
*
|
|
* @param ent
|
|
* Pointer to struct rte_rib_node that represents target route
|
|
* @return
|
|
* pointer to struct rte_rib_node that represents
|
|
* less specific route on success
|
|
* NULL otherwise
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib_node *
|
|
rte_rib_lookup_parent(struct rte_rib_node *ent);
|
|
|
|
/**
|
|
* Lookup prefix into the RIB structure
|
|
*
|
|
* @param rib
|
|
* RIB object handle
|
|
* @param ip
|
|
* net to be looked up in the RIB
|
|
* @param depth
|
|
* prefix length
|
|
* @return
|
|
* pointer to struct rte_rib_node on success
|
|
* NULL otherwise
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib_node *
|
|
rte_rib_lookup_exact(struct rte_rib *rib, uint32_t ip, uint8_t depth);
|
|
|
|
/**
|
|
* Retrieve next more specific prefix from the RIB
|
|
* that is covered by ip/depth supernet in an ascending order
|
|
*
|
|
* @param rib
|
|
* RIB object handle
|
|
* @param ip
|
|
* net address of supernet prefix that covers returned more specific prefixes
|
|
* @param depth
|
|
* supernet prefix length
|
|
* @param last
|
|
* pointer to the last returned prefix to get next prefix
|
|
* or
|
|
* NULL to get first more specific prefix
|
|
* @param flag
|
|
* -RTE_RIB_GET_NXT_ALL
|
|
* get all prefixes from subtrie
|
|
* -RTE_RIB_GET_NXT_COVER
|
|
* get only first more specific prefix even if it have more specifics
|
|
* @return
|
|
* pointer to the next more specific prefix
|
|
* NULL if there is no prefixes left
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib_node *
|
|
rte_rib_get_nxt(struct rte_rib *rib, uint32_t ip, uint8_t depth,
|
|
struct rte_rib_node *last, int flag);
|
|
|
|
/**
|
|
* Remove prefix from the RIB
|
|
*
|
|
* @param rib
|
|
* RIB object handle
|
|
* @param ip
|
|
* net to be removed from the RIB
|
|
* @param depth
|
|
* prefix length
|
|
*/
|
|
__rte_experimental
|
|
void
|
|
rte_rib_remove(struct rte_rib *rib, uint32_t ip, uint8_t depth);
|
|
|
|
/**
|
|
* Insert prefix into the RIB
|
|
*
|
|
* @param rib
|
|
* RIB object handle
|
|
* @param ip
|
|
* net to be inserted to the RIB
|
|
* @param depth
|
|
* prefix length
|
|
* @return
|
|
* pointer to new rte_rib_node on success
|
|
* NULL otherwise
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib_node *
|
|
rte_rib_insert(struct rte_rib *rib, uint32_t ip, uint8_t depth);
|
|
|
|
/**
|
|
* Get an ip from rte_rib_node
|
|
*
|
|
* @param node
|
|
* pointer to the rib node
|
|
* @param ip
|
|
* pointer to the ip to save
|
|
* @return
|
|
* 0 on success.
|
|
* -1 on failure with rte_errno indicating reason for failure.
|
|
*/
|
|
__rte_experimental
|
|
int
|
|
rte_rib_get_ip(struct rte_rib_node *node, uint32_t *ip);
|
|
|
|
/**
|
|
* Get a depth from rte_rib_node
|
|
*
|
|
* @param node
|
|
* pointer to the rib node
|
|
* @param depth
|
|
* pointer to the depth to save
|
|
* @return
|
|
* 0 on success.
|
|
* -1 on failure with rte_errno indicating reason for failure.
|
|
*/
|
|
__rte_experimental
|
|
int
|
|
rte_rib_get_depth(struct rte_rib_node *node, uint8_t *depth);
|
|
|
|
/**
|
|
* Get ext field from the rib node
|
|
* It is caller responsibility to make sure there are necessary space
|
|
* for the ext field inside rib node.
|
|
*
|
|
* @param node
|
|
* pointer to the rib node
|
|
* @return
|
|
* pointer to the ext
|
|
*/
|
|
__rte_experimental
|
|
void *
|
|
rte_rib_get_ext(struct rte_rib_node *node);
|
|
|
|
/**
|
|
* Get nexthop from the rib node
|
|
*
|
|
* @param node
|
|
* pointer to the rib node
|
|
* @param nh
|
|
* pointer to the nexthop to save
|
|
* @return
|
|
* 0 on success.
|
|
* -1 on failure with rte_errno indicating reason for failure.
|
|
*/
|
|
__rte_experimental
|
|
int
|
|
rte_rib_get_nh(struct rte_rib_node *node, uint64_t *nh);
|
|
|
|
/**
|
|
* Set nexthop into the rib node
|
|
*
|
|
* @param node
|
|
* pointer to the rib node
|
|
* @param nh
|
|
* nexthop value to set to the rib node
|
|
* @return
|
|
* 0 on success.
|
|
* -1 on failure with rte_errno indicating reason for failure.
|
|
*/
|
|
__rte_experimental
|
|
int
|
|
rte_rib_set_nh(struct rte_rib_node *node, uint64_t nh);
|
|
|
|
/**
|
|
* Create RIB
|
|
*
|
|
* @param name
|
|
* RIB name
|
|
* @param socket_id
|
|
* NUMA socket ID for RIB table memory allocation
|
|
* @param conf
|
|
* Structure containing the configuration
|
|
* @return
|
|
* Handle to RIB object on success
|
|
* NULL otherwise with rte_errno indicating reason for failure.
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib *
|
|
rte_rib_create(const char *name, int socket_id, struct rte_rib_conf *conf);
|
|
|
|
/**
|
|
* Find an existing RIB object and return a pointer to it.
|
|
*
|
|
* @param name
|
|
* Name of the rib object as passed to rte_rib_create()
|
|
* @return
|
|
* Pointer to RIB object on success
|
|
* NULL otherwise with rte_errno indicating reason for failure.
|
|
*/
|
|
__rte_experimental
|
|
struct rte_rib *
|
|
rte_rib_find_existing(const char *name);
|
|
|
|
/**
|
|
* Free an RIB object.
|
|
*
|
|
* @param rib
|
|
* RIB object handle
|
|
* @return
|
|
* None
|
|
*/
|
|
__rte_experimental
|
|
void
|
|
rte_rib_free(struct rte_rib *rib);
|
|
|
|
#endif /* _RTE_RIB_H_ */
|