mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-03 15:41:54 +00:00
Nest: Use bitmask of pxlens to speed-up net_route()
Use bitmask to keep track of pxlens that might be in rtable to avoid unnecessary lookups during net_route().
This commit is contained in:
parent
5c73cd94a5
commit
7d1c2ea5c0
@ -166,6 +166,8 @@ typedef struct rtable {
|
|||||||
uint route_count; /* Number of routes in the table */
|
uint route_count; /* Number of routes in the table */
|
||||||
uint route_updates; /* Number of accepted route updates */
|
uint route_updates; /* Number of accepted route updates */
|
||||||
uint route_withdraws; /* Number of accepted route withdraws */
|
uint route_withdraws; /* Number of accepted route withdraws */
|
||||||
|
u32 pxlens[5]; /* Bitmask of pxlens that might be in rtable */
|
||||||
|
u32 pxlens_new[5]; /* The above bitmask, under recalculation */
|
||||||
struct hostcache *hostcache;
|
struct hostcache *hostcache;
|
||||||
struct rtable_config *config; /* Configuration of this table */
|
struct rtable_config *config; /* Configuration of this table */
|
||||||
struct config *deleted; /* Table doesn't exist in current configuration,
|
struct config *deleted; /* Table doesn't exist in current configuration,
|
||||||
@ -301,10 +303,8 @@ void rt_lock_table(rtable *);
|
|||||||
void rt_unlock_table(rtable *);
|
void rt_unlock_table(rtable *);
|
||||||
void rt_setup(pool *, rtable *, struct rtable_config *);
|
void rt_setup(pool *, rtable *, struct rtable_config *);
|
||||||
static inline net *net_find(rtable *tab, const net_addr *addr) { return (net *) fib_find(&tab->fib, addr); }
|
static inline net *net_find(rtable *tab, const net_addr *addr) { return (net *) fib_find(&tab->fib, addr); }
|
||||||
static inline net *net_find_valid(rtable *tab, const net_addr *addr)
|
|
||||||
{ net *n = net_find(tab, addr); return (n && rte_is_valid(n->routes)) ? n : NULL; }
|
|
||||||
static inline net *net_get(rtable *tab, const net_addr *addr) { return (net *) fib_get(&tab->fib, addr); }
|
static inline net *net_get(rtable *tab, const net_addr *addr) { return (net *) fib_get(&tab->fib, addr); }
|
||||||
void *net_route(rtable *tab, const net_addr *n);
|
net *net_route(rtable *tab, const net_addr *n);
|
||||||
int net_roa_check(rtable *tab, const net_addr *n, u32 asn);
|
int net_roa_check(rtable *tab, const net_addr *n, u32 asn);
|
||||||
rte *rte_find(net *net, struct rte_src *src);
|
rte *rte_find(net *net, struct rte_src *src);
|
||||||
rte *rte_get_temp(struct rta *);
|
rte *rte_get_temp(struct rta *);
|
||||||
|
@ -58,35 +58,51 @@ static inline void rt_prune_table(rtable *tab);
|
|||||||
|
|
||||||
|
|
||||||
/* Like fib_route(), but skips empty net entries */
|
/* Like fib_route(), but skips empty net entries */
|
||||||
static inline void *
|
static inline net *
|
||||||
net_route_ip4(rtable *t, net_addr_ip4 *n)
|
net_route_ip4(rtable *t, net_addr_ip4 *a)
|
||||||
{
|
{
|
||||||
net *r;
|
while (1)
|
||||||
|
|
||||||
while (r = net_find_valid(t, (net_addr *) n), (!r) && (n->pxlen > 0))
|
|
||||||
{
|
{
|
||||||
n->pxlen--;
|
if (!BIT32_TEST(t->pxlens, a->pxlen))
|
||||||
ip4_clrbit(&n->prefix, n->pxlen);
|
goto next;
|
||||||
}
|
|
||||||
|
|
||||||
return r;
|
net *n = net_find(t, (net_addr *) a);
|
||||||
|
|
||||||
|
if (n && rte_is_valid(n->routes))
|
||||||
|
return n;
|
||||||
|
|
||||||
|
next:
|
||||||
|
if (!a->pxlen)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
a->pxlen--;
|
||||||
|
ip4_clrbit(&a->prefix, a->pxlen);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void *
|
static inline net *
|
||||||
net_route_ip6(rtable *t, net_addr_ip6 *n)
|
net_route_ip6(rtable *t, net_addr_ip6 *a)
|
||||||
{
|
{
|
||||||
net *r;
|
while (1)
|
||||||
|
|
||||||
while (r = net_find_valid(t, (net_addr *) n), (!r) && (n->pxlen > 0))
|
|
||||||
{
|
{
|
||||||
n->pxlen--;
|
if (!BIT32_TEST(t->pxlens, a->pxlen))
|
||||||
ip6_clrbit(&n->prefix, n->pxlen);
|
goto next;
|
||||||
}
|
|
||||||
|
|
||||||
return r;
|
net *n = net_find(t, (net_addr *) a);
|
||||||
|
|
||||||
|
if (n && rte_is_valid(n->routes))
|
||||||
|
return n;
|
||||||
|
|
||||||
|
next:
|
||||||
|
if (!a->pxlen)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
a->pxlen--;
|
||||||
|
ip6_clrbit(&a->prefix, a->pxlen);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void *
|
static inline net *
|
||||||
net_route_ip6_sadr(rtable *t, net_addr_ip6_sadr *n)
|
net_route_ip6_sadr(rtable *t, net_addr_ip6_sadr *n)
|
||||||
{
|
{
|
||||||
struct fib_node *fn;
|
struct fib_node *fn;
|
||||||
@ -96,6 +112,9 @@ net_route_ip6_sadr(rtable *t, net_addr_ip6_sadr *n)
|
|||||||
net *best = NULL;
|
net *best = NULL;
|
||||||
int best_pxlen = 0;
|
int best_pxlen = 0;
|
||||||
|
|
||||||
|
if (!BIT32_TEST(t->pxlens, n->dst_pxlen))
|
||||||
|
goto next;
|
||||||
|
|
||||||
/* We need to do dst first matching. Since sadr addresses are hashed on dst
|
/* We need to do dst first matching. Since sadr addresses are hashed on dst
|
||||||
prefix only, find the hash table chain and go through it to find the
|
prefix only, find the hash table chain and go through it to find the
|
||||||
match with the smallest matching src prefix. */
|
match with the smallest matching src prefix. */
|
||||||
@ -115,17 +134,16 @@ net_route_ip6_sadr(rtable *t, net_addr_ip6_sadr *n)
|
|||||||
if (best)
|
if (best)
|
||||||
return best;
|
return best;
|
||||||
|
|
||||||
|
next:
|
||||||
if (!n->dst_pxlen)
|
if (!n->dst_pxlen)
|
||||||
break;
|
return NULL;
|
||||||
|
|
||||||
n->dst_pxlen--;
|
n->dst_pxlen--;
|
||||||
ip6_clrbit(&n->dst_prefix, n->dst_pxlen);
|
ip6_clrbit(&n->dst_prefix, n->dst_pxlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void *
|
net *
|
||||||
net_route(rtable *tab, const net_addr *n)
|
net_route(rtable *tab, const net_addr *n)
|
||||||
{
|
{
|
||||||
ASSERT(tab->addr_type == n->type);
|
ASSERT(tab->addr_type == n->type);
|
||||||
@ -1415,6 +1433,9 @@ rte_update2(struct channel *c, const net_addr *n, rte *new, struct rte_src *src)
|
|||||||
if (!rta_is_cached(new->attrs)) /* Need to copy attributes */
|
if (!rta_is_cached(new->attrs)) /* Need to copy attributes */
|
||||||
new->attrs = rta_lookup(new->attrs);
|
new->attrs = rta_lookup(new->attrs);
|
||||||
new->flags |= REF_COW;
|
new->flags |= REF_COW;
|
||||||
|
|
||||||
|
BIT32_SET(c->table->pxlens, net_pxlen(n));
|
||||||
|
BIT32_SET(c->table->pxlens_new, net_pxlen(n));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -1727,6 +1748,8 @@ rt_prune_table(rtable *tab)
|
|||||||
|
|
||||||
FIB_ITERATE_INIT(fit, &tab->fib);
|
FIB_ITERATE_INIT(fit, &tab->fib);
|
||||||
tab->prune_state = 2;
|
tab->prune_state = 2;
|
||||||
|
|
||||||
|
memset(tab->pxlens_new, 0, sizeof(tab->pxlens));
|
||||||
}
|
}
|
||||||
|
|
||||||
again:
|
again:
|
||||||
@ -1757,6 +1780,8 @@ again:
|
|||||||
fib_delete(&tab->fib, n);
|
fib_delete(&tab->fib, n);
|
||||||
goto again;
|
goto again;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BIT32_SET(tab->pxlens_new, net_pxlen(n->n.addr));
|
||||||
}
|
}
|
||||||
FIB_ITERATE_END;
|
FIB_ITERATE_END;
|
||||||
|
|
||||||
@ -1767,6 +1792,8 @@ again:
|
|||||||
tab->gc_counter = 0;
|
tab->gc_counter = 0;
|
||||||
tab->gc_time = current_time();
|
tab->gc_time = current_time();
|
||||||
|
|
||||||
|
memcpy(tab->pxlens, tab->pxlens_new, sizeof(tab->pxlens));
|
||||||
|
|
||||||
/* state change 2->0, 3->1 */
|
/* state change 2->0, 3->1 */
|
||||||
tab->prune_state &= 1;
|
tab->prune_state &= 1;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user