Blame view

kernel/bpf/arraymap.c 4.07 KB
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
  /* Copyright (c) 2011-2014 PLUMgrid, http://plumgrid.com
   *
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of version 2 of the GNU General Public
   * License as published by the Free Software Foundation.
   *
   * This program is distributed in the hope that it will be useful, but
   * WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
   * General Public License for more details.
   */
  #include <linux/bpf.h>
  #include <linux/err.h>
  #include <linux/vmalloc.h>
  #include <linux/slab.h>
  #include <linux/mm.h>
  
  struct bpf_array {
  	struct bpf_map map;
  	u32 elem_size;
  	char value[0] __aligned(8);
  };
  
  /* Called from syscall */
  static struct bpf_map *array_map_alloc(union bpf_attr *attr)
  {
  	struct bpf_array *array;
daaf427c6   Alexei Starovoitov   bpf: fix arraymap...
28
  	u32 elem_size, array_size;
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
29
30
31
32
33
34
35
  
  	/* check sanity of attributes */
  	if (attr->max_entries == 0 || attr->key_size != 4 ||
  	    attr->value_size == 0)
  		return ERR_PTR(-EINVAL);
  
  	elem_size = round_up(attr->value_size, 8);
daaf427c6   Alexei Starovoitov   bpf: fix arraymap...
36
37
38
39
40
41
  	/* 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;
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
42
  	/* allocate all map elements and zero-initialize them */
daaf427c6   Alexei Starovoitov   bpf: fix arraymap...
43
  	array = kzalloc(array_size, GFP_USER | __GFP_NOWARN);
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
44
  	if (!array) {
daaf427c6   Alexei Starovoitov   bpf: fix arraymap...
45
  		array = vzalloc(array_size);
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
46
47
48
49
50
51
52
53
54
55
56
57
  		if (!array)
  			return ERR_PTR(-ENOMEM);
  	}
  
  	/* copy mandatory map attributes */
  	array->map.key_size = attr->key_size;
  	array->map.value_size = attr->value_size;
  	array->map.max_entries = attr->max_entries;
  
  	array->elem_size = elem_size;
  
  	return &array->map;
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
  }
  
  /* Called from syscall or from eBPF program */
  static void *array_map_lookup_elem(struct bpf_map *map, void *key)
  {
  	struct bpf_array *array = container_of(map, struct bpf_array, map);
  	u32 index = *(u32 *)key;
  
  	if (index >= array->map.max_entries)
  		return NULL;
  
  	return array->value + array->elem_size * index;
  }
  
  /* Called from syscall */
  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);
  	u32 index = *(u32 *)key;
  	u32 *next = (u32 *)next_key;
  
  	if (index >= array->map.max_entries) {
  		*next = 0;
  		return 0;
  	}
  
  	if (index == array->map.max_entries - 1)
  		return -ENOENT;
  
  	*next = index + 1;
  	return 0;
  }
  
  /* Called from syscall or from eBPF program */
  static int array_map_update_elem(struct bpf_map *map, void *key, void *value,
  				 u64 map_flags)
  {
  	struct bpf_array *array = container_of(map, struct bpf_array, map);
  	u32 index = *(u32 *)key;
  
  	if (map_flags > BPF_EXIST)
  		/* unknown flags */
  		return -EINVAL;
  
  	if (index >= array->map.max_entries)
  		/* all elements were pre-allocated, cannot insert a new one */
  		return -E2BIG;
  
  	if (map_flags == BPF_NOEXIST)
daaf427c6   Alexei Starovoitov   bpf: fix arraymap...
107
  		/* all elements already exist */
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
108
  		return -EEXIST;
879e7d33b   Daniel Borkmann   bpf, array: fix h...
109
  	memcpy(array->value + array->elem_size * index, value, map->value_size);
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
  	return 0;
  }
  
  /* Called from syscall or from eBPF program */
  static int array_map_delete_elem(struct bpf_map *map, void *key)
  {
  	return -EINVAL;
  }
  
  /* Called when map->refcnt goes to zero, either from workqueue or from syscall */
  static void array_map_free(struct bpf_map *map)
  {
  	struct bpf_array *array = container_of(map, struct bpf_array, map);
  
  	/* at this point bpf_prog->aux->refcnt == 0 and this map->refcnt == 0,
  	 * so the programs (can be more than one that used this map) were
  	 * disconnected from events. Wait for outstanding programs to complete
  	 * and free the array
  	 */
  	synchronize_rcu();
  
  	kvfree(array);
  }
a2c83fff5   Daniel Borkmann   ebpf: constify va...
133
  static const struct bpf_map_ops array_ops = {
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
134
135
136
137
138
139
140
  	.map_alloc = array_map_alloc,
  	.map_free = array_map_free,
  	.map_get_next_key = array_map_get_next_key,
  	.map_lookup_elem = array_map_lookup_elem,
  	.map_update_elem = array_map_update_elem,
  	.map_delete_elem = array_map_delete_elem,
  };
a2c83fff5   Daniel Borkmann   ebpf: constify va...
141
  static struct bpf_map_type_list array_type __read_mostly = {
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
142
143
144
145
146
147
  	.ops = &array_ops,
  	.type = BPF_MAP_TYPE_ARRAY,
  };
  
  static int __init register_array_map(void)
  {
a2c83fff5   Daniel Borkmann   ebpf: constify va...
148
  	bpf_register_map_type(&array_type);
28fbcfa08   Alexei Starovoitov   bpf: add array ty...
149
150
151
  	return 0;
  }
  late_initcall(register_array_map);