Skip to content

Commit

Permalink
lib: Fix memory issues with hash map
Browse files Browse the repository at this point in the history
Each entry in hash map must be allocated memory
on the heap.

In addition, much check that head is initialized
prior to calling _add() and _del() functions.

Also made corrections to compare and hash functions.

Signed-off-by: Nathan Duddles <[email protected]>
  • Loading branch information
nathan-duddles committed Mar 20, 2022
1 parent 399a33c commit 9dd3347
Showing 1 changed file with 90 additions and 54 deletions.
144 changes: 90 additions & 54 deletions lib/vrf.c
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@
#include "lib_errors.h"
#include "northbound.h"
#include "northbound_cli.h"
#include "jhash.h"

/* default VRF name value used when VRF backend is not NETNS */
#define VRF_DEFAULT_NAME_INTERNAL "default"
Expand All @@ -58,79 +59,102 @@ static int vrf_backend;
static int vrf_backend_configured;
static char vrf_default_name[VRF_NAMSIZ] = VRF_DEFAULT_NAME_INTERNAL;

/*
* Turn on/off debug code
* for vrf.
*/
static int debug_vrf = 0;

/* Holding VRF hooks */
static struct vrf_master {
int (*vrf_new_hook)(struct vrf *);
int (*vrf_delete_hook)(struct vrf *);
int (*vrf_enable_hook)(struct vrf *);
int (*vrf_disable_hook)(struct vrf *);
} vrf_master = {
0,
};

static int vrf_is_enabled(struct vrf *vrf);

/*Hash Table for table_id based lookup of vrf_id*/
DEFINE_MTYPE_STATIC(LIB, HASH_ENTRY, "VRF IDs by Table Hash Entry");

PREDECL_HASH(vrf_ids_by_table);

struct vrf_ids_by_table_proxy {
struct vrf_ids_by_table_item vrf_ids_by_table_hash_item;
struct vrf_ids_by_table_entry {
struct vrf_ids_by_table_item hash_item;
uint32_t table_id;
vrf_id_t vrf_id;
};

static struct vrf_ids_by_table_head vrf_ids_by_table_head;

static int vrf_ids_by_table_cmp(const struct vrf_ids_by_table_proxy *item_a,
const struct vrf_ids_by_table_proxy *item_b)
static int vrf_ids_by_table_cmp(const struct vrf_ids_by_table_entry *a,
const struct vrf_ids_by_table_entry *b)
{
return (item_a->table_id == item_b->table_id);
return numcmp(a->table_id, b->table_id);
}

static uint32_t vrf_ids_by_table_hash(const struct vrf_ids_by_table_proxy *item)
static uint32_t vrf_ids_by_table_hash(const struct vrf_ids_by_table_entry *item)
{
return item->table_id;
return jhash_1word(item->table_id, 0xa010bf92);
}

DECLARE_HASH(vrf_ids_by_table, struct vrf_ids_by_table_proxy,
vrf_ids_by_table_hash_item, vrf_ids_by_table_cmp,
DECLARE_HASH(vrf_ids_by_table, struct vrf_ids_by_table_entry,
hash_item, vrf_ids_by_table_cmp,
vrf_ids_by_table_hash);

static struct vrf_ids_by_table_head vrf_ids_by_table_head;

static int vrf_ids_by_table_is_initialized;

/* Helper functions for vrf_ids_by_table hashmap */
static void vrf_ids_by_table_delete(uint32_t table_id, vrf_id_t vrf_id)
static struct vrf_ids_by_table_entry *vrf_ids_by_table_entry_new(void)
{
struct vrf_ids_by_table_proxy ref = {.table_id = table_id};
struct vrf_ids_by_table_proxy *item =
vrf_ids_by_table_find(&vrf_ids_by_table_head, &ref);
struct vrf_ids_by_table_entry *new_entry;
new_entry = XCALLOC(MTYPE_HASH_ENTRY, sizeof(struct vrf_ids_by_table_entry));
return new_entry;
}

/* If entry contains both table_id and vrf_id, remove from hashmap */
if (item != NULL && item->vrf_id == vrf_id) {
vrf_ids_by_table_del(&vrf_ids_by_table_head, &ref);
}
static void vrf_ids_by_table_delete(uint32_t table_id)
{
if(!vrf_ids_by_table_is_initialized)
return;

struct vrf_ids_by_table_entry lookup = {.table_id = table_id};
struct vrf_ids_by_table_entry *entry =
vrf_ids_by_table_find(&vrf_ids_by_table_head, &lookup);

/* If entry contains both table_id and vrf_id,
* remove from hashmap
*/
if (entry) {
vrf_ids_by_table_del(&vrf_ids_by_table_head, entry);
XFREE(MTYPE_HASH_ENTRY, entry);
}
}

static void vrf_ids_by_table_update(uint32_t table_id, vrf_id_t new_vrf_id)
{
struct vrf_ids_by_table_proxy ref = {.table_id = table_id,
.vrf_id = new_vrf_id};

/* If not already in hashmap, adds and returns null;
* otherwise returns current entry
*/
struct vrf_ids_by_table_proxy *item =
vrf_ids_by_table_add(&vrf_ids_by_table_head, &ref);
if (new_vrf_id == VRF_UNKNOWN)
return;

/* If current entry with table_id, then update */
if (item != NULL) {
item->vrf_id = new_vrf_id;
if (!vrf_ids_by_table_is_initialized) {
vrf_ids_by_table_init(&vrf_ids_by_table_head);
vrf_ids_by_table_is_initialized = 1;
}
}

/*
* Turn on/off debug code
* for vrf.
*/
static int debug_vrf = 0;
struct vrf_ids_by_table_entry lookup = {.table_id = table_id};
struct vrf_ids_by_table_entry *entry =
vrf_ids_by_table_find(&vrf_ids_by_table_head, &lookup);

/* Holding VRF hooks */
static struct vrf_master {
int (*vrf_new_hook)(struct vrf *);
int (*vrf_delete_hook)(struct vrf *);
int (*vrf_enable_hook)(struct vrf *);
int (*vrf_disable_hook)(struct vrf *);
} vrf_master = {
0,
};
if (!entry) {
entry = vrf_ids_by_table_entry_new();
entry->table_id = table_id;
vrf_ids_by_table_add(&vrf_ids_by_table_head, entry);
}

static int vrf_is_enabled(struct vrf *vrf);
entry->vrf_id = new_vrf_id;
}

/* VRF list existance check by name. */
struct vrf *vrf_lookup_by_name(const char *name)
Expand Down Expand Up @@ -306,6 +330,8 @@ struct vrf *vrf_update(vrf_id_t new_vrf_id, const char *name)
*/
void vrf_update_table_id(struct vrf *vrf, uint32_t new_table_id)
{
if (!vrf)
return;

/* If new and old table ids are same */
if (new_table_id == vrf->data.l.table_id) {
Expand All @@ -319,7 +345,9 @@ void vrf_update_table_id(struct vrf *vrf, uint32_t new_table_id)
/* New and old table ids are different
* so remove any entry with old table_id and vrf_id, and add new entry
*/
vrf_ids_by_table_delete(vrf->data.l.table_id, vrf->vrf_id);
vrf_ids_by_table_delete(vrf->data.l.table_id);

// segmentation fault occurs here, called from zclient, zclient_vrf_add
vrf_ids_by_table_update(new_table_id, vrf->vrf_id);

/*Update table_id in vrf struct */
Expand Down Expand Up @@ -358,7 +386,7 @@ void vrf_delete(struct vrf *vrf)
(*vrf_master.vrf_delete_hook)(vrf);

/* remove from vrf_ids_by_table hash map*/
vrf_ids_by_table_delete(vrf->data.l.table_id, vrf->vrf_id);
vrf_ids_by_table_delete(vrf->data.l.table_id);

QOBJ_UNREG(vrf);

Expand Down Expand Up @@ -409,13 +437,12 @@ vrf_id_t vrf_lookup_by_table(uint32_t table_id, ns_id_t ns_id)
/* case vrf with VRF_BACKEND_VRF_LITE : match the table_id */
} else if (vrf_get_backend() == VRF_BACKEND_VRF_LITE) {

struct vrf_ids_by_table_proxy ref = {.table_id = table_id};
struct vrf_ids_by_table_entry lookup = {.table_id = table_id};

struct vrf_ids_by_table_proxy *found_entry =
vrf_ids_by_table_find(&vrf_ids_by_table_head, &ref);
if (found_entry != NULL) {
return found_entry->vrf_id;
}
struct vrf_ids_by_table_entry *entry =
vrf_ids_by_table_find(&vrf_ids_by_table_head, &lookup);
if (entry != NULL)
return entry->vrf_id;
}

return VRF_DEFAULT;
Expand Down Expand Up @@ -640,6 +667,10 @@ void vrf_init(int (*create)(struct vrf *), int (*enable)(struct vrf *),
vrf_master.vrf_disable_hook = disable;
vrf_master.vrf_delete_hook = destroy;

/* Initialize vrf_ids_by_table */
vrf_ids_by_table_init(&vrf_ids_by_table_head);
vrf_ids_by_table_is_initialized = 1;

/* The default VRF always exists. */
default_vrf = vrf_get(VRF_DEFAULT, VRF_DEFAULT_NAME);
if (!default_vrf) {
Expand Down Expand Up @@ -701,6 +732,11 @@ void vrf_terminate(void)
vrf = vrf_lookup_by_id(VRF_DEFAULT);
if (vrf)
vrf_terminate_single(vrf);

/* Clean up vrf_ids_by_table hashmap */
vrf_ids_by_table_fini(&vrf_ids_by_table_head);

vrf_ids_by_table_is_initialized = 0;
}

int vrf_socket(int domain, int type, int protocol, vrf_id_t vrf_id,
Expand Down

0 comments on commit 9dd3347

Please sign in to comment.