/* * 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 #include /** * 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; int i; for (i = 0; i < TRANSPORT_N_PROTOS; i++) session_rules_table_free (&slt->session_rules[i]); vec_free (slt->session_rules); 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); } 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; int i; #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 _ 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); } vec_validate (slt->session_rules, TRANSPORT_N_PROTOS - 1); for (i = 0; i < TRANSPORT_N_PROTOS; i++) session_rules_table_init (&slt->session_rules[i]); } 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 *); 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: */