mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-08 18:11:54 +00:00
Cached route attributes now have explicitly marked layers
Also the rta_* functions renamed to ea_* functions
This commit is contained in:
parent
ba3b8607fb
commit
c872b8f826
52
lib/route.h
52
lib/route.h
@ -235,11 +235,22 @@ typedef struct eattr {
|
|||||||
typedef struct ea_list {
|
typedef struct ea_list {
|
||||||
struct ea_list *next; /* In case we have an override list */
|
struct ea_list *next; /* In case we have an override list */
|
||||||
byte flags; /* Flags: EALF_... */
|
byte flags; /* Flags: EALF_... */
|
||||||
byte rfu;
|
byte stored:5; /* enum ea_stored */
|
||||||
|
byte rfu:3;
|
||||||
word count; /* Number of attributes */
|
word count; /* Number of attributes */
|
||||||
eattr attrs[0]; /* Attribute definitions themselves */
|
eattr attrs[0]; /* Attribute definitions themselves */
|
||||||
} ea_list;
|
} ea_list;
|
||||||
|
|
||||||
|
enum ea_stored {
|
||||||
|
EALS_NONE = 0, /* This is a temporary ea_list */
|
||||||
|
EALS_PREIMPORT = 1, /* State when route entered rte_update() */
|
||||||
|
EALS_FILTERED = 2, /* State after filters */
|
||||||
|
EALS_IN_TABLE = 3, /* State in table */
|
||||||
|
EALS_KEY = 4, /* EA list used as key */
|
||||||
|
EALS_CUSTOM = 0x10, /* OR this with custom values */
|
||||||
|
EALS_MAX = 0x20,
|
||||||
|
};
|
||||||
|
|
||||||
struct ea_storage {
|
struct ea_storage {
|
||||||
struct ea_storage *next_hash; /* Next in hash chain */
|
struct ea_storage *next_hash; /* Next in hash chain */
|
||||||
struct ea_storage **pprev_hash; /* Previous in hash chain */
|
struct ea_storage **pprev_hash; /* Previous in hash chain */
|
||||||
@ -250,7 +261,6 @@ struct ea_storage {
|
|||||||
|
|
||||||
#define EALF_SORTED 1 /* Attributes are sorted by code */
|
#define EALF_SORTED 1 /* Attributes are sorted by code */
|
||||||
#define EALF_BISECT 2 /* Use interval bisection for searching */
|
#define EALF_BISECT 2 /* Use interval bisection for searching */
|
||||||
#define EALF_CACHED 4 /* List is cached */
|
|
||||||
#define EALF_HUGE 8 /* List is too big to fit into slab */
|
#define EALF_HUGE 8 /* List is too big to fit into slab */
|
||||||
|
|
||||||
struct ea_class {
|
struct ea_class {
|
||||||
@ -333,12 +343,12 @@ static inline eattr *ea_find_by_name(ea_list *l, const char *name)
|
|||||||
eattr *ea_walk(struct ea_walk_state *s, uint id, uint max);
|
eattr *ea_walk(struct ea_walk_state *s, uint id, uint max);
|
||||||
void ea_dump(ea_list *);
|
void ea_dump(ea_list *);
|
||||||
int ea_same(ea_list *x, ea_list *y); /* Test whether two ea_lists are identical */
|
int ea_same(ea_list *x, ea_list *y); /* Test whether two ea_lists are identical */
|
||||||
uint ea_hash(ea_list *e); /* Calculate 16-bit hash value */
|
uint ea_hash(ea_list *e); /* Calculate attributes hash value */
|
||||||
ea_list *ea_append(ea_list *to, ea_list *what);
|
ea_list *ea_append(ea_list *to, ea_list *what);
|
||||||
void ea_format_bitfield(const struct eattr *a, byte *buf, int bufsize, const char **names, int min, int max);
|
void ea_format_bitfield(const struct eattr *a, byte *buf, int bufsize, const char **names, int min, int max);
|
||||||
|
|
||||||
/* Normalize ea_list; allocates the result from tmp_linpool */
|
/* Normalize ea_list; allocates the result from tmp_linpool */
|
||||||
ea_list *ea_normalize(ea_list *e, int overlay);
|
ea_list *ea_normalize(ea_list *e, u32 upto);
|
||||||
|
|
||||||
uint ea_list_size(ea_list *);
|
uint ea_list_size(ea_list *);
|
||||||
void ea_list_copy(ea_list *dest, ea_list *src, uint size);
|
void ea_list_copy(ea_list *dest, ea_list *src, uint size);
|
||||||
@ -523,18 +533,39 @@ static inline int rte_dest(const rte *r)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void rta_init(void);
|
void rta_init(void);
|
||||||
ea_list *ea_lookup(ea_list *, int overlay); /* Get a cached (and normalized) variant of this attribute list */
|
|
||||||
static inline int ea_is_cached(const ea_list *r) { return r->flags & EALF_CACHED; }
|
ea_list *ea_lookup_slow(ea_list *r, u32 squash_upto, enum ea_stored oid);
|
||||||
|
|
||||||
static inline struct ea_storage *ea_get_storage(ea_list *r)
|
static inline struct ea_storage *ea_get_storage(ea_list *r)
|
||||||
{
|
{
|
||||||
ASSERT_DIE(ea_is_cached(r));
|
ASSERT_DIE(r->stored);
|
||||||
return SKIP_BACK(struct ea_storage, l[0], r);
|
return SKIP_BACK(struct ea_storage, l[0], r);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline ea_list *ea_clone(ea_list *r) {
|
static inline ea_list *ea_ref(ea_list *r)
|
||||||
|
{
|
||||||
ASSERT_DIE(0 < atomic_fetch_add_explicit(&ea_get_storage(r)->uc, 1, memory_order_acq_rel));
|
ASSERT_DIE(0 < atomic_fetch_add_explicit(&ea_get_storage(r)->uc, 1, memory_order_acq_rel));
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline ea_list *ea_lookup(ea_list *r, u32 squash_upto, enum ea_stored oid)
|
||||||
|
{
|
||||||
|
ASSERT_DIE(oid);
|
||||||
|
if ((r->stored == oid) || BIT32_TEST(&squash_upto, r->stored))
|
||||||
|
return ea_ref(r);
|
||||||
|
else
|
||||||
|
return ea_lookup_slow(r, squash_upto, oid);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline ea_list *ea_strip_to(ea_list *r, u32 strip_to)
|
||||||
|
{
|
||||||
|
ASSERT_DIE(strip_to);
|
||||||
|
while (r && !BIT32_TEST(&strip_to, r->stored))
|
||||||
|
r = r->next;
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
void ea__free(struct ea_storage *r);
|
void ea__free(struct ea_storage *r);
|
||||||
static inline void ea_free(ea_list *l) {
|
static inline void ea_free(ea_list *l) {
|
||||||
if (!l) return;
|
if (!l) return;
|
||||||
@ -546,9 +577,4 @@ void ea_dump(ea_list *);
|
|||||||
void ea_dump_all(void);
|
void ea_dump_all(void);
|
||||||
void ea_show_list(struct cli *, ea_list *);
|
void ea_show_list(struct cli *, ea_list *);
|
||||||
|
|
||||||
#define rta_lookup ea_lookup
|
|
||||||
#define rta_is_cached ea_is_cached
|
|
||||||
#define rta_clone ea_clone
|
|
||||||
#define rta_free ea_free
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -1225,7 +1225,7 @@ mpls_get_key_attrs(struct mpls_fec_map *m, ea_list *src)
|
|||||||
ea.a[ea.l.count++] = EA_LITERAL_EMBEDDED(&ea_gen_source, 0, m->mpls_rts);
|
ea.a[ea.l.count++] = EA_LITERAL_EMBEDDED(&ea_gen_source, 0, m->mpls_rts);
|
||||||
PUT_ATTR(&ea_gen_mpls_class);
|
PUT_ATTR(&ea_gen_mpls_class);
|
||||||
|
|
||||||
return ea_get_storage(ea_lookup(&ea.l, 0));
|
return ea_get_storage(ea_lookup(&ea.l, 0, EALS_KEY));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -1027,7 +1027,7 @@ ea_sort(ea_list *e)
|
|||||||
* a given &ea_list after merging with ea_merge().
|
* a given &ea_list after merging with ea_merge().
|
||||||
*/
|
*/
|
||||||
static unsigned
|
static unsigned
|
||||||
ea_scan(const ea_list *e, int overlay)
|
ea_scan(const ea_list *e, u32 upto)
|
||||||
{
|
{
|
||||||
unsigned cnt = 0;
|
unsigned cnt = 0;
|
||||||
|
|
||||||
@ -1035,7 +1035,7 @@ ea_scan(const ea_list *e, int overlay)
|
|||||||
{
|
{
|
||||||
cnt += e->count;
|
cnt += e->count;
|
||||||
e = e->next;
|
e = e->next;
|
||||||
if (e && overlay && ea_is_cached(e))
|
if (e && BIT32_TEST(&upto, e->stored))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return sizeof(ea_list) + sizeof(eattr)*cnt;
|
return sizeof(ea_list) + sizeof(eattr)*cnt;
|
||||||
@ -1056,7 +1056,7 @@ ea_scan(const ea_list *e, int overlay)
|
|||||||
* by calling ea_sort().
|
* by calling ea_sort().
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
ea_merge(ea_list *e, ea_list *t, int overlay)
|
ea_merge(ea_list *e, ea_list *t, u32 upto)
|
||||||
{
|
{
|
||||||
eattr *d = t->attrs;
|
eattr *d = t->attrs;
|
||||||
|
|
||||||
@ -1070,7 +1070,7 @@ ea_merge(ea_list *e, ea_list *t, int overlay)
|
|||||||
d += e->count;
|
d += e->count;
|
||||||
e = e->next;
|
e = e->next;
|
||||||
|
|
||||||
if (e && overlay && ea_is_cached(e))
|
if (e && BIT32_TEST(&upto, e->stored))
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1078,22 +1078,22 @@ ea_merge(ea_list *e, ea_list *t, int overlay)
|
|||||||
}
|
}
|
||||||
|
|
||||||
ea_list *
|
ea_list *
|
||||||
ea_normalize(ea_list *e, int overlay)
|
ea_normalize(ea_list *e, u32 upto)
|
||||||
{
|
{
|
||||||
#if 0
|
#if 0
|
||||||
debug("(normalize)");
|
debug("(normalize)");
|
||||||
ea_dump(e);
|
ea_dump(e);
|
||||||
debug(" ----> ");
|
debug(" ----> ");
|
||||||
#endif
|
#endif
|
||||||
ea_list *t = tmp_alloc(ea_scan(e, overlay));
|
ea_list *t = tmp_allocz(ea_scan(e, upto));
|
||||||
ea_merge(e, t, overlay);
|
ea_merge(e, t, upto);
|
||||||
ea_sort(t);
|
ea_sort(t);
|
||||||
#if 0
|
#if 0
|
||||||
ea_dump(t);
|
ea_dump(t);
|
||||||
debug("\n");
|
debug("\n");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return t->count ? t : t->next;
|
return t;
|
||||||
}
|
}
|
||||||
|
|
||||||
static _Bool
|
static _Bool
|
||||||
@ -1214,10 +1214,7 @@ ea_list_ref(ea_list *l)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (l->next)
|
if (l->next)
|
||||||
{
|
ea_ref(l->next);
|
||||||
ASSERT_DIE(ea_is_cached(l->next));
|
|
||||||
ea_clone(l->next);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ea_free_nested(ea_list *l);
|
static void ea_free_nested(ea_list *l);
|
||||||
@ -1511,11 +1508,11 @@ ea_dump(ea_list *e)
|
|||||||
}
|
}
|
||||||
while (e)
|
while (e)
|
||||||
{
|
{
|
||||||
struct ea_storage *s = ea_is_cached(e) ? ea_get_storage(e) : NULL;
|
struct ea_storage *s = e->stored ? ea_get_storage(e) : NULL;
|
||||||
debug("[%c%c%c] uc=%d h=%08x",
|
debug("[%c%c] overlay=%d uc=%d h=%08x",
|
||||||
(e->flags & EALF_SORTED) ? 'S' : 's',
|
(e->flags & EALF_SORTED) ? 'S' : 's',
|
||||||
(e->flags & EALF_BISECT) ? 'B' : 'b',
|
(e->flags & EALF_BISECT) ? 'B' : 'b',
|
||||||
(e->flags & EALF_CACHED) ? 'C' : 'c',
|
e->stored,
|
||||||
s ? atomic_load_explicit(&s->uc, memory_order_relaxed) : 0,
|
s ? atomic_load_explicit(&s->uc, memory_order_relaxed) : 0,
|
||||||
s ? s->hash_key : 0);
|
s ? s->hash_key : 0);
|
||||||
for(i=0; i<e->count; i++)
|
for(i=0; i<e->count; i++)
|
||||||
@ -1561,7 +1558,7 @@ ea_dump(ea_list *e)
|
|||||||
* ea_hash() takes an extended attribute list and calculated a hopefully
|
* ea_hash() takes an extended attribute list and calculated a hopefully
|
||||||
* uniformly distributed hash value from its contents.
|
* uniformly distributed hash value from its contents.
|
||||||
*/
|
*/
|
||||||
inline uint
|
uint
|
||||||
ea_hash(ea_list *e)
|
ea_hash(ea_list *e)
|
||||||
{
|
{
|
||||||
const u64 mul = 0x68576150f3d6847;
|
const u64 mul = 0x68576150f3d6847;
|
||||||
@ -1678,19 +1675,22 @@ rta_rehash(void)
|
|||||||
* converted to the normalized form.
|
* converted to the normalized form.
|
||||||
*/
|
*/
|
||||||
ea_list *
|
ea_list *
|
||||||
ea_lookup(ea_list *o, int overlay)
|
ea_lookup_slow(ea_list *o, u32 squash_upto, enum ea_stored oid)
|
||||||
{
|
{
|
||||||
struct ea_storage *r;
|
struct ea_storage *r;
|
||||||
uint h;
|
uint h;
|
||||||
|
|
||||||
ASSERT(!ea_is_cached(o));
|
ASSERT(o->stored != oid);
|
||||||
o = ea_normalize(o, overlay);
|
ASSERT(oid);
|
||||||
|
o = ea_normalize(o, squash_upto);
|
||||||
h = ea_hash(o);
|
h = ea_hash(o);
|
||||||
|
|
||||||
|
squash_upto |= BIT32_VAL(oid);
|
||||||
|
|
||||||
RTA_LOCK;
|
RTA_LOCK;
|
||||||
|
|
||||||
for(r=rta_hash_table[h & rta_cache_mask]; r; r=r->next_hash)
|
for(r=rta_hash_table[h & rta_cache_mask]; r; r=r->next_hash)
|
||||||
if (r->hash_key == h && ea_same(r->l, o))
|
if (r->hash_key == h && ea_same(r->l, o) && BIT32_TEST(&squash_upto, r->l->stored))
|
||||||
{
|
{
|
||||||
atomic_fetch_add_explicit(&r->uc, 1, memory_order_acq_rel);
|
atomic_fetch_add_explicit(&r->uc, 1, memory_order_acq_rel);
|
||||||
RTA_UNLOCK;
|
RTA_UNLOCK;
|
||||||
@ -1713,7 +1713,8 @@ ea_lookup(ea_list *o, int overlay)
|
|||||||
ea_list_copy(r->l, o, elen);
|
ea_list_copy(r->l, o, elen);
|
||||||
ea_list_ref(r->l);
|
ea_list_ref(r->l);
|
||||||
|
|
||||||
r->l->flags |= EALF_CACHED | huge;
|
r->l->flags |= huge;
|
||||||
|
r->l->stored = oid;
|
||||||
r->hash_key = h;
|
r->hash_key = h;
|
||||||
atomic_store_explicit(&r->uc, 1, memory_order_release);
|
atomic_store_explicit(&r->uc, 1, memory_order_release);
|
||||||
|
|
||||||
|
@ -54,9 +54,9 @@ rt_show_rte(struct cli *c, byte *ia, rte *e, struct rt_show_data *d, int primary
|
|||||||
else
|
else
|
||||||
from[0] = 0;
|
from[0] = 0;
|
||||||
|
|
||||||
/* Need to normalize the extended attributes */
|
/* Need to normalize the attributes for dumping */
|
||||||
if (d->verbose && !rta_is_cached(a) && a)
|
if (d->verbose && !a->stored)
|
||||||
a = ea_normalize(a, 0);
|
a = ea_normalize(a, EALS_NONE);
|
||||||
|
|
||||||
get_route_info = e->src->owner->class ? e->src->owner->class->get_route_info : NULL;
|
get_route_info = e->src->owner->class ? e->src->owner->class->get_route_info : NULL;
|
||||||
if (get_route_info)
|
if (get_route_info)
|
||||||
|
@ -494,10 +494,7 @@ rte_store(const rte *r, struct netindex *i, struct rtable_private *tab)
|
|||||||
|
|
||||||
rt_lock_source(e->src);
|
rt_lock_source(e->src);
|
||||||
|
|
||||||
if (ea_is_cached(e->attrs))
|
e->attrs = ea_lookup(e->attrs, BIT32_ALL(EALS_PREIMPORT, EALS_FILTERED), EALS_IN_TABLE);
|
||||||
e->attrs = rta_clone(e->attrs);
|
|
||||||
else
|
|
||||||
e->attrs = rta_lookup(e->attrs, 1);
|
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
debug("(store) %N ", i->addr);
|
debug("(store) %N ", i->addr);
|
||||||
@ -524,7 +521,7 @@ rte_free(struct rte_storage *e, struct rtable_private *tab)
|
|||||||
|
|
||||||
rt_unlock_source(e->rte.src);
|
rt_unlock_source(e->rte.src);
|
||||||
|
|
||||||
rta_free(e->rte.attrs);
|
ea_free(e->rte.attrs);
|
||||||
sl_free(e);
|
sl_free(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1453,7 +1450,7 @@ rte_same(const rte *x, const rte *y)
|
|||||||
return
|
return
|
||||||
(x == y) || (
|
(x == y) || (
|
||||||
(x->attrs == y->attrs) ||
|
(x->attrs == y->attrs) ||
|
||||||
((!(x->attrs->flags & EALF_CACHED) || !(y->attrs->flags & EALF_CACHED)) && ea_same(x->attrs, y->attrs))
|
((!x->attrs->stored || !y->attrs->stored) && ea_same(x->attrs, y->attrs))
|
||||||
) &&
|
) &&
|
||||||
x->src == y->src &&
|
x->src == y->src &&
|
||||||
rte_is_filtered(x) == rte_is_filtered(y);
|
rte_is_filtered(x) == rte_is_filtered(y);
|
||||||
@ -1716,14 +1713,11 @@ rte_update(struct channel *c, const net_addr *n, rte *new, struct rte_src *src)
|
|||||||
|
|
||||||
ASSERT(c->channel_state == CS_UP);
|
ASSERT(c->channel_state == CS_UP);
|
||||||
|
|
||||||
ea_list *ea_tmp[2] = {};
|
ea_list *ea_prefilter = NULL, *ea_postfilter = NULL;
|
||||||
|
|
||||||
/* The import reloader requires prefilter routes to be the first layer */
|
/* Storing prefilter routes as an explicit layer */
|
||||||
if (new && (c->in_keep & RIK_PREFILTER))
|
if (new && (c->in_keep & RIK_PREFILTER))
|
||||||
ea_tmp[0] = new->attrs =
|
ea_prefilter = new->attrs = ea_lookup(new->attrs, 0, EALS_PREIMPORT);
|
||||||
(ea_is_cached(new->attrs) && !new->attrs->next) ?
|
|
||||||
ea_clone(new->attrs) :
|
|
||||||
ea_lookup(new->attrs, 0);
|
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
debug("%s.%s -(prefilter)-> %s: %N ", c->proto->name, c->name, c->table->name, n);
|
debug("%s.%s -(prefilter)-> %s: %N ", c->proto->name, c->name, c->table->name, n);
|
||||||
@ -1766,8 +1760,8 @@ rte_update(struct channel *c, const net_addr *n, rte *new, struct rte_src *src)
|
|||||||
|
|
||||||
if (new)
|
if (new)
|
||||||
{
|
{
|
||||||
ea_tmp[1] = new->attrs =
|
ea_postfilter = new->attrs = ea_lookup(new->attrs,
|
||||||
ea_is_cached(new->attrs) ? ea_clone(new->attrs) : ea_lookup(new->attrs, !!ea_tmp[0]);
|
ea_prefilter ? BIT32_ALL(EALS_PREIMPORT) : 0, EALS_FILTERED);
|
||||||
|
|
||||||
if (net_is_flow(n))
|
if (net_is_flow(n))
|
||||||
rt_flowspec_resolve_rte(new, c);
|
rt_flowspec_resolve_rte(new, c);
|
||||||
@ -1795,9 +1789,8 @@ rte_update(struct channel *c, const net_addr *n, rte *new, struct rte_src *src)
|
|||||||
|
|
||||||
/* Now the route attributes are kept by the in-table cached version
|
/* Now the route attributes are kept by the in-table cached version
|
||||||
* and we may drop the local handles */
|
* and we may drop the local handles */
|
||||||
for (uint k = 0; k < ARRAY_SIZE(ea_tmp); k++)
|
ea_free(ea_prefilter);
|
||||||
if (ea_tmp[k])
|
ea_free(ea_postfilter);
|
||||||
ea_free(ea_tmp[k]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -3135,7 +3128,10 @@ rt_next_hop_update_rte(const rte *old, rte *new)
|
|||||||
if (!head)
|
if (!head)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
/* Get the state of the route just before nexthop was resolved */
|
||||||
*new = *old;
|
*new = *old;
|
||||||
|
new->attrs = ea_strip_to(new->attrs, BIT32_ALL(EALS_PREIMPORT, EALS_FILTERED));
|
||||||
|
|
||||||
RT_LOCKED(head->he->owner, tab)
|
RT_LOCKED(head->he->owner, tab)
|
||||||
rta_apply_hostentry(tab, &new->attrs, head);
|
rta_apply_hostentry(tab, &new->attrs, head);
|
||||||
return 1;
|
return 1;
|
||||||
@ -3226,7 +3222,7 @@ rt_flowspec_check(rtable *tab_ip, rtable *tab_flow, const net_addr *n, ea_list *
|
|||||||
if (nb)
|
if (nb)
|
||||||
{
|
{
|
||||||
rb = RTE_COPY_VALID(RTE_OR_NULL(nb->routes));
|
rb = RTE_COPY_VALID(RTE_OR_NULL(nb->routes));
|
||||||
rta_clone(rb.attrs);
|
ea_ref(rb.attrs);
|
||||||
net_copy(&nau.n, nb->routes->rte.net);
|
net_copy(&nau.n, nb->routes->rte.net);
|
||||||
rb.net = &nau.n;
|
rb.net = &nau.n;
|
||||||
}
|
}
|
||||||
@ -3307,6 +3303,7 @@ rt_flowspec_update_rte(rtable *tab, const rte *r, rte *new)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
*new = *r;
|
*new = *r;
|
||||||
|
new->attrs = ea_strip_to(new->attrs, BIT32_ALL(EALS_PREIMPORT, EALS_FILTERED));
|
||||||
ea_set_attr_u32(&new->attrs, &ea_gen_flowspec_valid, 0, valid);
|
ea_set_attr_u32(&new->attrs, &ea_gen_flowspec_valid, 0, valid);
|
||||||
return 1;
|
return 1;
|
||||||
#else
|
#else
|
||||||
@ -4141,8 +4138,7 @@ void channel_reload_export_bulk(struct rt_export_request *req, const net_addr *n
|
|||||||
rte new = rte_init_from(feed[i]);
|
rte new = rte_init_from(feed[i]);
|
||||||
|
|
||||||
/* Strip the later attribute layers */
|
/* Strip the later attribute layers */
|
||||||
while (new.attrs->next)
|
new.attrs = ea_strip_to(new.attrs, BIT32_ALL(EALS_PREIMPORT));
|
||||||
new.attrs = new.attrs->next;
|
|
||||||
|
|
||||||
/* And reload the route */
|
/* And reload the route */
|
||||||
rte_update(c, net, &new, new.src);
|
rte_update(c, net, &new, new.src);
|
||||||
@ -4243,7 +4239,7 @@ hc_new_hostentry(struct hostcache *hc, pool *p, ip_addr a, ip_addr ll, rtable *d
|
|||||||
static void
|
static void
|
||||||
hc_delete_hostentry(struct hostcache *hc, pool *p, struct hostentry *he)
|
hc_delete_hostentry(struct hostcache *hc, pool *p, struct hostentry *he)
|
||||||
{
|
{
|
||||||
rta_free(he->src);
|
ea_free(he->src);
|
||||||
|
|
||||||
rem_node(&he->ln);
|
rem_node(&he->ln);
|
||||||
hc_remove(hc, he);
|
hc_remove(hc, he);
|
||||||
@ -4328,7 +4324,7 @@ rt_free_hostcache(struct rtable_private *tab)
|
|||||||
WALK_LIST(n, hc->hostentries)
|
WALK_LIST(n, hc->hostentries)
|
||||||
{
|
{
|
||||||
struct hostentry *he = SKIP_BACK(struct hostentry, ln, n);
|
struct hostentry *he = SKIP_BACK(struct hostentry, ln, n);
|
||||||
rta_free(he->src);
|
ea_free(he->src);
|
||||||
|
|
||||||
if (!lfuc_finished(&he->uc))
|
if (!lfuc_finished(&he->uc))
|
||||||
log(L_ERR "Hostcache is not empty in table %s", tab->name);
|
log(L_ERR "Hostcache is not empty in table %s", tab->name);
|
||||||
@ -4424,7 +4420,7 @@ rt_update_hostentry(struct rtable_private *tab, struct hostentry *he)
|
|||||||
direct++;
|
direct++;
|
||||||
}
|
}
|
||||||
|
|
||||||
he->src = rta_clone(a);
|
he->src = ea_ref(a);
|
||||||
he->nexthop_linkable = !direct;
|
he->nexthop_linkable = !direct;
|
||||||
he->igp_metric = rt_get_igp_metric(&e->rte);
|
he->igp_metric = rt_get_igp_metric(&e->rte);
|
||||||
}
|
}
|
||||||
@ -4433,7 +4429,7 @@ done:
|
|||||||
/* Add a prefix range to the trie */
|
/* Add a prefix range to the trie */
|
||||||
trie_add_prefix(tab->hostcache->trie, &he_addr, pxlen, he_addr.pxlen);
|
trie_add_prefix(tab->hostcache->trie, &he_addr, pxlen, he_addr.pxlen);
|
||||||
|
|
||||||
rta_free(old_src);
|
ea_free(old_src);
|
||||||
return old_src != he->src;
|
return old_src != he->src;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -265,10 +265,7 @@ aggregator_rt_notify(struct proto *P, struct channel *src_ch, const net_addr *ne
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Store the route attributes */
|
/* Store the route attributes */
|
||||||
if (rta_is_cached(new->attrs))
|
new->attrs = ea_lookup(new->attrs, 0, EALS_KEY);
|
||||||
rta_clone(new->attrs);
|
|
||||||
else
|
|
||||||
new->attrs = rta_lookup(new->attrs, 0);
|
|
||||||
|
|
||||||
/* Insert the new route into the bucket */
|
/* Insert the new route into the bucket */
|
||||||
struct aggregator_route *arte = sl_alloc(p->route_slab);
|
struct aggregator_route *arte = sl_alloc(p->route_slab);
|
||||||
@ -296,7 +293,7 @@ aggregator_rt_notify(struct proto *P, struct channel *src_ch, const net_addr *ne
|
|||||||
|
|
||||||
old_bucket->count--;
|
old_bucket->count--;
|
||||||
HASH_REMOVE2(p->routes, AGGR_RTE, p->p.pool, old_route);
|
HASH_REMOVE2(p->routes, AGGR_RTE, p->p.pool, old_route);
|
||||||
rta_free(old_route->rte.attrs);
|
ea_free(old_route->rte.attrs);
|
||||||
sl_free(old_route);
|
sl_free(old_route);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -408,7 +405,7 @@ aggregator_shutdown(struct proto *P)
|
|||||||
b->rte = arte->next_rte;
|
b->rte = arte->next_rte;
|
||||||
b->count--;
|
b->count--;
|
||||||
HASH_REMOVE(p->routes, AGGR_RTE, arte);
|
HASH_REMOVE(p->routes, AGGR_RTE, arte);
|
||||||
rta_free(arte->rte.attrs);
|
ea_free(arte->rte.attrs);
|
||||||
sl_free(arte);
|
sl_free(arte);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1535,7 +1535,7 @@ bgp_rte_update(struct bgp_parse_state *s, const net_addr *n, u32 path_id, ea_lis
|
|||||||
|
|
||||||
/* Prepare cached route attributes */
|
/* Prepare cached route attributes */
|
||||||
if (!s->mpls && (s->cached_ea == NULL))
|
if (!s->mpls && (s->cached_ea == NULL))
|
||||||
a0 = s->cached_ea = ea_lookup(a0, 0);
|
a0 = s->cached_ea = ea_lookup(a0, 0, EALS_CUSTOM);
|
||||||
|
|
||||||
rte e0 = {
|
rte e0 = {
|
||||||
.attrs = a0,
|
.attrs = a0,
|
||||||
@ -1594,7 +1594,7 @@ bgp_decode_mpls_labels(struct bgp_parse_state *s, byte **pos, uint *len, uint *p
|
|||||||
bgp_apply_mpls_labels(s, to, lnum, labels);
|
bgp_apply_mpls_labels(s, to, lnum, labels);
|
||||||
|
|
||||||
/* Attributes were changed, invalidate cached entry */
|
/* Attributes were changed, invalidate cached entry */
|
||||||
rta_free(s->cached_ea);
|
ea_free(s->cached_ea);
|
||||||
s->cached_ea = NULL;
|
s->cached_ea = NULL;
|
||||||
|
|
||||||
return;
|
return;
|
||||||
@ -2711,7 +2711,7 @@ bgp_decode_nlri(struct bgp_parse_state *s, u32 afi, byte *nlri, uint len, ea_lis
|
|||||||
|
|
||||||
c->desc->decode_nlri(s, nlri, len, ea);
|
c->desc->decode_nlri(s, nlri, len, ea);
|
||||||
|
|
||||||
rta_free(s->cached_ea);
|
ea_free(s->cached_ea);
|
||||||
s->cached_ea = NULL;
|
s->cached_ea = NULL;
|
||||||
|
|
||||||
rt_unlock_source(s->last_src);
|
rt_unlock_source(s->last_src);
|
||||||
@ -2820,7 +2820,7 @@ bgp_rx_update(struct bgp_conn *conn, byte *pkt, uint len)
|
|||||||
ea, s.mp_next_hop_data, s.mp_next_hop_len);
|
ea, s.mp_next_hop_data, s.mp_next_hop_len);
|
||||||
|
|
||||||
done:
|
done:
|
||||||
rta_free(s.cached_ea);
|
ea_free(s.cached_ea);
|
||||||
lp_restore(tmp_linpool, tmpp);
|
lp_restore(tmp_linpool, tmpp);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -2088,7 +2088,7 @@ again1:
|
|||||||
|
|
||||||
ASSERT_DIE(ARRAY_SIZE(eattrs.a) >= eattrs.l.count);
|
ASSERT_DIE(ARRAY_SIZE(eattrs.a) >= eattrs.l.count);
|
||||||
|
|
||||||
ea_list *eal = ea_lookup(&eattrs.l, 0);
|
ea_list *eal = ea_lookup(&eattrs.l, 0, EALS_CUSTOM);
|
||||||
ea_free(nf->old_ea);
|
ea_free(nf->old_ea);
|
||||||
nf->old_ea = eal;
|
nf->old_ea = eal;
|
||||||
|
|
||||||
@ -2107,7 +2107,7 @@ again1:
|
|||||||
else if (nf->old_ea)
|
else if (nf->old_ea)
|
||||||
{
|
{
|
||||||
/* Remove the route */
|
/* Remove the route */
|
||||||
rta_free(nf->old_ea);
|
ea_free(nf->old_ea);
|
||||||
nf->old_ea = NULL;
|
nf->old_ea = NULL;
|
||||||
|
|
||||||
rte_update(p->p.main_channel, nf->fn.addr, NULL, p->p.main_source);
|
rte_update(p->p.main_channel, nf->fn.addr, NULL, p->p.main_source);
|
||||||
|
@ -156,10 +156,10 @@ perf_loop(void *data)
|
|||||||
ea_set_attr_data(&ea, &ea_gen_nexthop, 0,
|
ea_set_attr_data(&ea, &ea_gen_nexthop, 0,
|
||||||
&nhad.ad.data, sizeof nhad - sizeof nhad.ad);
|
&nhad.ad.data, sizeof nhad - sizeof nhad.ad);
|
||||||
|
|
||||||
p->data[i].a = rta_lookup(ea, 0);
|
p->data[i].a = ea_lookup(ea, 0, EALS_CUSTOM);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
p->data[i].a = rta_clone(p->data[i-1].a);
|
p->data[i].a = ea_ref(p->data[i-1].a);
|
||||||
}
|
}
|
||||||
|
|
||||||
clock_gettime(CLOCK_MONOTONIC, &ts_generated);
|
clock_gettime(CLOCK_MONOTONIC, &ts_generated);
|
||||||
|
Loading…
Reference in New Issue
Block a user