blob: f3ec1f90bf6f72cdbd0c7799b8f843d026f3fa4e [file] [log] [blame]
/*
* Copyright (c) 2017-2019 Cisco and/or its affiliates.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <vnet/session/session_table.h>
#include <vnet/session/session.h>
#include <vnet/session/session_rules_table.h>
/**
* Pool of session tables
*/
static session_table_t *lookup_tables;
session_table_t *
_get_session_tables (void)
{
return lookup_tables;
}
session_table_t *
session_table_alloc (void)
{
session_table_t *slt;
pool_get_aligned (lookup_tables, slt, CLIB_CACHE_LINE_BYTES);
clib_memset (slt, 0, sizeof (*slt));
return slt;
}
u32
session_table_index (session_table_t * slt)
{
return (slt - lookup_tables);
}
session_table_t *
session_table_get (u32 table_index)
{
if (pool_is_free_index (lookup_tables, table_index))
return 0;
return pool_elt_at_index (lookup_tables, table_index);
}
#define foreach_hash_table_parameter \
_(v4,session,buckets,20000) \
_(v4,session,memory,(64<<20)) \
_(v6,session,buckets,20000) \
_(v6,session,memory,(64<<20)) \
_(v4,halfopen,buckets,20000) \
_(v4,halfopen,memory,(64<<20)) \
_(v6,halfopen,buckets,20000) \
_(v6,halfopen,memory,(64<<20))
void
session_table_free (session_table_t *slt, u8 fib_proto)
{
u8 all = fib_proto > FIB_PROTOCOL_IP6 ? 1 : 0;
session_rules_table_free (slt, fib_proto);
if (fib_proto == FIB_PROTOCOL_IP4 || all)
{
clib_bihash_free_16_8 (&slt->v4_session_hash);
clib_bihash_free_16_8 (&slt->v4_half_open_hash);
}
if (fib_proto == FIB_PROTOCOL_IP6 || all)
{
clib_bihash_free_48_8 (&slt->v6_session_hash);
clib_bihash_free_48_8 (&slt->v6_half_open_hash);
}
vec_free (slt->appns_index);
pool_put (lookup_tables, slt);
}
/**
* Initialize session table hash tables
*
* If vpp configured with set of table parameters it uses them,
* otherwise it uses defaults above.
*/
void
session_table_init (session_table_t *slt, u8 fib_proto)
{
u8 all = fib_proto > FIB_PROTOCOL_IP6 ? 1 : 0;
#define _(af,table,parm,value) \
u32 configured_##af##_##table##_table_##parm = value;
foreach_hash_table_parameter;
#undef _
#define _(af,table,parm,value) \
if (session_main.configured_##af##_##table##_table_##parm) \
configured_##af##_##table##_table_##parm = \
session_main.configured_##af##_##table##_table_##parm;
foreach_hash_table_parameter;
#undef _
slt->srtg_handle = SESSION_SRTG_HANDLE_INVALID;
if (fib_proto == FIB_PROTOCOL_IP4 || all)
{
clib_bihash_init2_args_16_8_t _a, *a = &_a;
memset (a, 0, sizeof (*a));
a->h = &slt->v4_session_hash;
a->name = "v4 session table";
a->nbuckets = configured_v4_session_table_buckets;
a->memory_size = configured_v4_session_table_memory;
a->dont_add_to_all_bihash_list = 1;
a->instantiate_immediately = 1;
clib_bihash_init2_16_8 (a);
memset (a, 0, sizeof (*a));
a->h = &slt->v4_half_open_hash;
a->name = "v4 half-open table";
a->nbuckets = configured_v4_halfopen_table_buckets;
a->memory_size = configured_v4_halfopen_table_memory;
a->dont_add_to_all_bihash_list = 1;
a->instantiate_immediately = 1;
clib_bihash_init2_16_8 (a);
}
if (fib_proto == FIB_PROTOCOL_IP6 || all)
{
clib_bihash_init2_args_48_8_t _a, *a = &_a;
memset (a, 0, sizeof (*a));
a->h = &slt->v6_session_hash;
a->name = "v6 session table";
a->nbuckets = configured_v6_session_table_buckets;
a->memory_size = configured_v6_session_table_memory;
a->dont_add_to_all_bihash_list = 1;
a->instantiate_immediately = 1;
clib_bihash_init2_48_8 (a);
memset (a, 0, sizeof (*a));
a->h = &slt->v6_half_open_hash;
a->name = "v6 half-open table";
a->nbuckets = configured_v6_halfopen_table_buckets;
a->memory_size = configured_v6_halfopen_table_memory;
a->dont_add_to_all_bihash_list = 1;
a->instantiate_immediately = 1;
clib_bihash_init2_48_8 (a);
}
}
typedef struct _ip4_session_table_walk_ctx_t
{
ip4_session_table_walk_fn_t fn;
void *ctx;
} ip4_session_table_walk_ctx_t;
static int
ip4_session_table_walk_cb (clib_bihash_kv_16_8_t * kvp, void *arg)
{
ip4_session_table_walk_ctx_t *ctx = arg;
ctx->fn (kvp, ctx->ctx);
return (BIHASH_WALK_CONTINUE);
}
void
ip4_session_table_walk (clib_bihash_16_8_t * hash,
ip4_session_table_walk_fn_t fn, void *arg)
{
ip4_session_table_walk_ctx_t ctx = {
.fn = fn,
.ctx = arg,
};
clib_bihash_foreach_key_value_pair_16_8 (hash, ip4_session_table_walk_cb,
&ctx);
}
u32
session_table_memory_size (session_table_t *st)
{
u64 total_size = 0;
if (clib_bihash_is_initialised_16_8 (&st->v4_session_hash))
{
clib_bihash_alloc_chunk_16_8_t *c = st->v4_session_hash.chunks;
while (c)
{
total_size += c->size;
c = c->next;
}
c = st->v4_half_open_hash.chunks;
while (c)
{
total_size += c->size;
c = c->next;
}
}
if (clib_bihash_is_initialised_48_8 (&st->v6_session_hash))
{
clib_bihash_alloc_chunk_48_8_t *c = st->v6_session_hash.chunks;
while (c)
{
total_size += c->size;
c = c->next;
}
c = st->v6_half_open_hash.chunks;
while (c)
{
total_size += c->size;
c = c->next;
}
}
return total_size;
}
u8 *
format_session_table (u8 *s, va_list *args)
{
session_table_t *st = va_arg (*args, session_table_t *);
u32 appns_index, i;
s = format (s, "appns index: ");
vec_foreach_index (i, st->appns_index)
{
appns_index = *vec_elt_at_index (st->appns_index, i);
if (i > 0)
s = format (s, ", ");
s = format (s, "%d", appns_index);
}
s = format (s, "\n");
if (clib_bihash_is_initialised_16_8 (&st->v4_session_hash))
{
s = format (s, "%U", format_bihash_16_8, &st->v4_session_hash, 0);
s = format (s, "%U", format_bihash_16_8, &st->v4_half_open_hash, 0);
}
if (clib_bihash_is_initialised_48_8 (&st->v6_session_hash))
{
s = format (s, "%U", format_bihash_48_8, &st->v6_session_hash, 0);
s = format (s, "%U", format_bihash_48_8, &st->v6_half_open_hash, 0);
}
return s;
}
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/