Commit daaf427c authored by Alexei Starovoitov's avatar Alexei Starovoitov Committed by David S. Miller

bpf: fix arraymap NULL deref and missing overflow and zero size checks

- fix NULL pointer dereference:
kernel/bpf/arraymap.c:41 array_map_alloc() error: potential null dereference 'array'.  (kzalloc returns null)
kernel/bpf/arraymap.c:41 array_map_alloc() error: we previously assumed 'array' could be null (see line 40)

- integer overflow check was missing in arraymap
(hashmap checks for overflow via kmalloc_array())

- arraymap can round_up(value_size, 8) to zero. check was missing.

- hashmap was missing zero size check as well, since roundup_pow_of_two() can
truncate into zero

- found a typo in the arraymap comment and unnecessary empty line

Fix all of these issues and make both overflow checks explicit U32 in size.
Reported-by: default avatarkbuild test robot <fengguang.wu@intel.com>
Signed-off-by: default avatarAlexei Starovoitov <ast@plumgrid.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent fcd4d35e
...@@ -25,7 +25,7 @@ struct bpf_array { ...@@ -25,7 +25,7 @@ struct bpf_array {
static struct bpf_map *array_map_alloc(union bpf_attr *attr) static struct bpf_map *array_map_alloc(union bpf_attr *attr)
{ {
struct bpf_array *array; struct bpf_array *array;
u32 elem_size; u32 elem_size, array_size;
/* check sanity of attributes */ /* check sanity of attributes */
if (attr->max_entries == 0 || attr->key_size != 4 || if (attr->max_entries == 0 || attr->key_size != 4 ||
...@@ -34,11 +34,17 @@ static struct bpf_map *array_map_alloc(union bpf_attr *attr) ...@@ -34,11 +34,17 @@ static struct bpf_map *array_map_alloc(union bpf_attr *attr)
elem_size = round_up(attr->value_size, 8); elem_size = round_up(attr->value_size, 8);
/* check round_up into zero and u32 overflow */
if (elem_size == 0 ||
attr->max_entries > (U32_MAX - sizeof(*array)) / elem_size)
return ERR_PTR(-ENOMEM);
array_size = sizeof(*array) + attr->max_entries * elem_size;
/* allocate all map elements and zero-initialize them */ /* allocate all map elements and zero-initialize them */
array = kzalloc(sizeof(*array) + attr->max_entries * elem_size, array = kzalloc(array_size, GFP_USER | __GFP_NOWARN);
GFP_USER | __GFP_NOWARN);
if (!array) { if (!array) {
array = vzalloc(array->map.max_entries * array->elem_size); array = vzalloc(array_size);
if (!array) if (!array)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
} }
...@@ -51,7 +57,6 @@ static struct bpf_map *array_map_alloc(union bpf_attr *attr) ...@@ -51,7 +57,6 @@ static struct bpf_map *array_map_alloc(union bpf_attr *attr)
array->elem_size = elem_size; array->elem_size = elem_size;
return &array->map; return &array->map;
} }
/* Called from syscall or from eBPF program */ /* Called from syscall or from eBPF program */
...@@ -101,7 +106,7 @@ static int array_map_update_elem(struct bpf_map *map, void *key, void *value, ...@@ -101,7 +106,7 @@ static int array_map_update_elem(struct bpf_map *map, void *key, void *value,
return -E2BIG; return -E2BIG;
if (map_flags == BPF_NOEXIST) if (map_flags == BPF_NOEXIST)
/* all elemenets already exist */ /* all elements already exist */
return -EEXIST; return -EEXIST;
memcpy(array->value + array->elem_size * index, value, array->elem_size); memcpy(array->value + array->elem_size * index, value, array->elem_size);
......
...@@ -65,6 +65,11 @@ static struct bpf_map *htab_map_alloc(union bpf_attr *attr) ...@@ -65,6 +65,11 @@ static struct bpf_map *htab_map_alloc(union bpf_attr *attr)
goto free_htab; goto free_htab;
err = -ENOMEM; err = -ENOMEM;
/* prevent zero size kmalloc and check for u32 overflow */
if (htab->n_buckets == 0 ||
htab->n_buckets > U32_MAX / sizeof(struct hlist_head))
goto free_htab;
htab->buckets = kmalloc_array(htab->n_buckets, sizeof(struct hlist_head), htab->buckets = kmalloc_array(htab->n_buckets, sizeof(struct hlist_head),
GFP_USER | __GFP_NOWARN); GFP_USER | __GFP_NOWARN);
......
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