From ae82142cf020e8fc2c6fe6b43fac64b0d6950f95 Mon Sep 17 00:00:00 2001 From: anb <anb@dev.null> Date: Fri, 12 Jan 2018 19:08:57 +0000 Subject: [PATCH 1/2] lib: make map_contains recongize null value Without changing the interface, map_contains is able to tell whether the item exist in map or not. --- lib/generic/map.c | 52 ++++++++++++++++++++++++++++------------------- tests/test_map.c | 14 ++++++++++++- 2 files changed, 44 insertions(+), 22 deletions(-) diff --git a/lib/generic/map.c b/lib/generic/map.c index fcf026b33..4920bb2a9 100644 --- a/lib/generic/map.c +++ b/lib/generic/map.c @@ -118,24 +118,7 @@ static cb_data_t *cbt_make_data(map_t *map, const uint8_t *str, size_t len, void return x; } -/*! Creates a new, empty critbit map */ -EXPORT map_t map_make(void) -{ - map_t map; - map.root = NULL; - map.malloc = &malloc_std; - map.free = &free_std; - map.baton = NULL; - return map; -} - -/*! Returns non-zero if map contains str */ -EXPORT int map_contains(map_t *map, const char *str) -{ - return map_get(map, str) != NULL; -} - -EXPORT void *map_get(map_t *map, const char *str) +static int cbt_get(map_t *map, const char *str, void **value) { const uint8_t *ubytes = (void *)str; const size_t ulen = strlen(str); @@ -143,7 +126,7 @@ EXPORT void *map_get(map_t *map, const char *str) cb_data_t *x = NULL; if (p == NULL) { - return NULL; + return 0; } while (ref_is_internal(p)) { @@ -161,10 +144,37 @@ EXPORT void *map_get(map_t *map, const char *str) x = (cb_data_t *)p; if (strcmp(str, (const char *)x->key) == 0) { - return x->value; + if (value != NULL) { + *value = x->value; + } + return 1; } - return NULL; + return 0; +} + +/*! Creates a new, empty critbit map */ +EXPORT map_t map_make(void) +{ + map_t map; + map.root = NULL; + map.malloc = &malloc_std; + map.free = &free_std; + map.baton = NULL; + return map; +} + +/*! Returns non-zero if map contains str */ +EXPORT int map_contains(map_t *map, const char *str) +{ + return cbt_get(map, str, NULL); +} + +EXPORT void *map_get(map_t *map, const char *str) +{ + void *v = NULL; + cbt_get(map, str, &v); + return v; } /*! Inserts str into map, returns 0 on success */ diff --git a/tests/test_map.c b/tests/test_map.c index 6072ee169..baf16e967 100644 --- a/tests/test_map.c +++ b/tests/test_map.c @@ -90,6 +90,17 @@ static void test_delete(void **state) assert_int_equal(map_del(tree, "most likely not in tree"), 1); } +/* Test null value existence */ +static void test_null_value(void **state) +{ + map_t *tree = *state; + char *key = "foo"; + + assert_int_equal(map_set(tree, key, (void *)0), 0); + assert_true(map_contains(tree, key)); + assert_int_equal(map_del(tree, key), 0); +} + static void test_init(void **state) { static map_t tree; @@ -112,8 +123,9 @@ int main(int argc, char **argv) unit_test(test_insert), unit_test(test_get), unit_test(test_delete), + unit_test(test_null_value), group_test_teardown(test_deinit) }; return run_group_tests(tests); -} \ No newline at end of file +} -- GitLab From 7b89e3474d350c8ff3da5ed5866b0bb08395473b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Vladim=C3=ADr=20=C4=8Cun=C3=A1t?= <vladimir.cunat@nic.cz> Date: Mon, 22 Jan 2018 15:58:24 +0100 Subject: [PATCH 2/2] lib map: comment around the change in parent commit --- lib/generic/map.c | 1 + lib/generic/map.h | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/lib/generic/map.c b/lib/generic/map.c index 4920bb2a9..73de090a3 100644 --- a/lib/generic/map.c +++ b/lib/generic/map.c @@ -118,6 +118,7 @@ static cb_data_t *cbt_make_data(map_t *map, const uint8_t *str, size_t len, void return x; } +/*! Like map_contains, but also set the value, if passed and found. */ static int cbt_get(map_t *map, const char *str, void **value) { const uint8_t *ubytes = (void *)str; diff --git a/lib/generic/map.h b/lib/generic/map.h index dc459f5ef..9a4c3c08d 100644 --- a/lib/generic/map.h +++ b/lib/generic/map.h @@ -76,7 +76,7 @@ map_t map_make(void); /** Returns non-zero if map contains str */ int map_contains(map_t *map, const char *str); -/** Returns value if map contains str */ +/** Returns value if map contains str. Note: NULL may mean two different things. */ void *map_get(map_t *map, const char *str); /** Inserts str into map, returns 0 on suceess */ -- GitLab