diff --git a/lib/route.h b/lib/route.h index eae251e7..50e62440 100644 --- a/lib/route.h +++ b/lib/route.h @@ -265,6 +265,7 @@ struct ea_storage { #define EALF_SORTED 1 /* Attributes are sorted by code */ #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 */ struct ea_class { #define EA_CLASS_INSIDE \ diff --git a/nest/rt-attr.c b/nest/rt-attr.c index 471209ee..b41e56c2 100644 --- a/nest/rt-attr.c +++ b/nest/rt-attr.c @@ -182,6 +182,10 @@ DOMAIN(attrs) attrs_domain; pool *rta_pool; +/* Assuming page size of 4096, these are magic values for slab allocation */ +static const uint ea_slab_sizes[] = { 56, 112, 168, 288, 448, 800, 1344 }; +static slab *ea_slab[ARRAY_SIZE(ea_slab_sizes)]; + static slab *rte_src_slab; static struct idm src_ids; @@ -1485,11 +1489,22 @@ ea_lookup(ea_list *o, int overlay) } uint elen = ea_list_size(o); - r = mb_alloc(rta_pool, elen + sizeof(struct ea_storage)); + uint sz = elen + sizeof(struct ea_storage); + for (uint i=0; il, o, elen); ea_list_ref(r->l); - r->l->flags |= EALF_CACHED; + r->l->flags |= EALF_CACHED | huge; r->hash_key = h; r->uc = 1; @@ -1516,7 +1531,10 @@ ea_free_locked(struct ea_storage *a) a->next_hash->pprev_hash = a->pprev_hash; ea_list_unref(a->l); - mb_free(a); + if (a->l->flags & EALF_HUGE) + mb_free(a); + else + sl_free(a); } static void @@ -1580,6 +1598,9 @@ rta_init(void) rta_pool = rp_new(&root_pool, "Attributes"); + for (uint i=0; i