Commit ffc1e480 authored by Teng Qin's avatar Teng Qin Committed by Stefan Bader

bpf: map_get_next_key to return first key on NULL

BugLink: http://bugs.launchpad.net/bugs/1774173

commit 8fe45924 upstream.

When iterating through a map, we need to find a key that does not exist
in the map so map_get_next_key will give us the first key of the map.
This often requires a lot of guessing in production systems.

This patch makes map_get_next_key return the first key when the key
pointer in the parameter is NULL.
Signed-off-by: default avatarTeng Qin <qinteng@fb.com>
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
Acked-by: default avatarDaniel Borkmann <daniel@iogearbox.net>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarChenbo Feng <fengc@google.com>
Cc: Lorenzo Colitti <lorenzo@google.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarJuerg Haefliger <juergh@canonical.com>
Signed-off-by: default avatarStefan Bader <stefan.bader@canonical.com>
parent 8f802e8a
...@@ -102,7 +102,7 @@ static void *array_map_lookup_elem(struct bpf_map *map, void *key) ...@@ -102,7 +102,7 @@ static void *array_map_lookup_elem(struct bpf_map *map, void *key)
static int array_map_get_next_key(struct bpf_map *map, void *key, void *next_key) static int array_map_get_next_key(struct bpf_map *map, void *key, void *next_key)
{ {
struct bpf_array *array = container_of(map, struct bpf_array, map); struct bpf_array *array = container_of(map, struct bpf_array, map);
u32 index = *(u32 *)key; u32 index = key ? *(u32 *)key : U32_MAX;
u32 *next = (u32 *)next_key; u32 *next = (u32 *)next_key;
if (index >= array->map.max_entries) { if (index >= array->map.max_entries) {
......
...@@ -169,12 +169,15 @@ static int htab_map_get_next_key(struct bpf_map *map, void *key, void *next_key) ...@@ -169,12 +169,15 @@ static int htab_map_get_next_key(struct bpf_map *map, void *key, void *next_key)
struct hlist_head *head; struct hlist_head *head;
struct htab_elem *l, *next_l; struct htab_elem *l, *next_l;
u32 hash, key_size; u32 hash, key_size;
int i; int i = 0;
WARN_ON_ONCE(!rcu_read_lock_held()); WARN_ON_ONCE(!rcu_read_lock_held());
key_size = map->key_size; key_size = map->key_size;
if (!key)
goto find_first_elem;
hash = htab_map_hash(key, key_size); hash = htab_map_hash(key, key_size);
head = select_bucket(htab, hash); head = select_bucket(htab, hash);
...@@ -182,10 +185,8 @@ static int htab_map_get_next_key(struct bpf_map *map, void *key, void *next_key) ...@@ -182,10 +185,8 @@ static int htab_map_get_next_key(struct bpf_map *map, void *key, void *next_key)
/* lookup the key */ /* lookup the key */
l = lookup_elem_raw(head, hash, key, key_size); l = lookup_elem_raw(head, hash, key, key_size);
if (!l) { if (!l)
i = 0;
goto find_first_elem; goto find_first_elem;
}
/* key was found, get next key in the same bucket */ /* key was found, get next key in the same bucket */
next_l = hlist_entry_safe(rcu_dereference_raw(hlist_next_rcu(&l->hash_node)), next_l = hlist_entry_safe(rcu_dereference_raw(hlist_next_rcu(&l->hash_node)),
......
...@@ -390,14 +390,18 @@ static int map_get_next_key(union bpf_attr *attr) ...@@ -390,14 +390,18 @@ static int map_get_next_key(union bpf_attr *attr)
if (IS_ERR(map)) if (IS_ERR(map))
return PTR_ERR(map); return PTR_ERR(map);
err = -ENOMEM; if (ukey) {
key = kmalloc(map->key_size, GFP_USER); err = -ENOMEM;
if (!key) key = kmalloc(map->key_size, GFP_USER);
goto err_put; if (!key)
goto err_put;
err = -EFAULT;
if (copy_from_user(key, ukey, map->key_size) != 0) err = -EFAULT;
goto free_key; if (copy_from_user(key, ukey, map->key_size) != 0)
goto free_key;
} else {
key = NULL;
}
err = -ENOMEM; err = -ENOMEM;
next_key = kmalloc(map->key_size, GFP_USER); next_key = kmalloc(map->key_size, GFP_USER);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment