Vladimir Medvedkin 5a5793a5ff rib: add RIB library
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>
2019-11-06 00:08:56 +01:00

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_ */