blob: b0bc475f641e3b2c808f78f5be54283de6866d25 [file] [log] [blame]
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +02001/* Copyright (C) 2013 Jozsef Kadlecsik <kadlec@blackhole.kfki.hu>
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 as
5 * published by the Free Software Foundation.
6 */
7
8#ifndef __IP_SET_BITMAP_IP_GEN_H
9#define __IP_SET_BITMAP_IP_GEN_H
10
Jozsef Kadlecsik35b8dcf2013-04-30 23:02:43 +020011#define mtype_do_test IPSET_TOKEN(MTYPE, _do_test)
12#define mtype_gc_test IPSET_TOKEN(MTYPE, _gc_test)
13#define mtype_is_filled IPSET_TOKEN(MTYPE, _is_filled)
14#define mtype_do_add IPSET_TOKEN(MTYPE, _do_add)
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +020015#define mtype_ext_cleanup IPSET_TOKEN(MTYPE, _ext_cleanup)
Jozsef Kadlecsik35b8dcf2013-04-30 23:02:43 +020016#define mtype_do_del IPSET_TOKEN(MTYPE, _do_del)
17#define mtype_do_list IPSET_TOKEN(MTYPE, _do_list)
18#define mtype_do_head IPSET_TOKEN(MTYPE, _do_head)
19#define mtype_adt_elem IPSET_TOKEN(MTYPE, _adt_elem)
20#define mtype_add_timeout IPSET_TOKEN(MTYPE, _add_timeout)
21#define mtype_gc_init IPSET_TOKEN(MTYPE, _gc_init)
22#define mtype_kadt IPSET_TOKEN(MTYPE, _kadt)
23#define mtype_uadt IPSET_TOKEN(MTYPE, _uadt)
24#define mtype_destroy IPSET_TOKEN(MTYPE, _destroy)
25#define mtype_flush IPSET_TOKEN(MTYPE, _flush)
26#define mtype_head IPSET_TOKEN(MTYPE, _head)
27#define mtype_same_set IPSET_TOKEN(MTYPE, _same_set)
28#define mtype_elem IPSET_TOKEN(MTYPE, _elem)
29#define mtype_test IPSET_TOKEN(MTYPE, _test)
30#define mtype_add IPSET_TOKEN(MTYPE, _add)
31#define mtype_del IPSET_TOKEN(MTYPE, _del)
32#define mtype_list IPSET_TOKEN(MTYPE, _list)
33#define mtype_gc IPSET_TOKEN(MTYPE, _gc)
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020034#define mtype MTYPE
35
Jozsef Kadlecsik95ad1f42015-11-07 11:21:47 +010036#define get_ext(set, map, id) ((map)->extensions + ((set)->dsize * (id)))
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020037
38static void
39mtype_gc_init(struct ip_set *set, void (*gc)(unsigned long ul_set))
40{
41 struct mtype *map = set->data;
42
43 init_timer(&map->gc);
Jozsef Kadlecsikca0f6a52015-06-13 19:45:33 +020044 map->gc.data = (unsigned long)set;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020045 map->gc.function = gc;
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +020046 map->gc.expires = jiffies + IPSET_GC_PERIOD(set->timeout) * HZ;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020047 add_timer(&map->gc);
48}
49
50static void
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +020051mtype_ext_cleanup(struct ip_set *set)
52{
53 struct mtype *map = set->data;
54 u32 id;
55
56 for (id = 0; id < map->elements; id++)
57 if (test_bit(id, map->members))
58 ip_set_ext_destroy(set, get_ext(set, map, id));
59}
60
61static void
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020062mtype_destroy(struct ip_set *set)
63{
64 struct mtype *map = set->data;
65
66 if (SET_WITH_TIMEOUT(set))
67 del_timer_sync(&map->gc);
68
69 ip_set_free(map->members);
Jozsef Kadlecsik95ad1f42015-11-07 11:21:47 +010070 if (set->dsize && set->extensions & IPSET_EXT_DESTROY)
71 mtype_ext_cleanup(set);
72 ip_set_free(map);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020073
74 set->data = NULL;
75}
76
77static void
78mtype_flush(struct ip_set *set)
79{
80 struct mtype *map = set->data;
81
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +020082 if (set->extensions & IPSET_EXT_DESTROY)
83 mtype_ext_cleanup(set);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020084 memset(map->members, 0, map->memsize);
85}
86
87static int
88mtype_head(struct ip_set *set, struct sk_buff *skb)
89{
90 const struct mtype *map = set->data;
91 struct nlattr *nested;
Jozsef Kadlecsik95ad1f42015-11-07 11:21:47 +010092 size_t memsize = sizeof(*map) + map->memsize;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +020093
94 nested = ipset_nest_start(skb, IPSET_ATTR_DATA);
95 if (!nested)
96 goto nla_put_failure;
97 if (mtype_do_head(skb, map) ||
98 nla_put_net32(skb, IPSET_ATTR_REFERENCES, htonl(set->ref - 1)) ||
Jozsef Kadlecsik95ad1f42015-11-07 11:21:47 +010099 nla_put_net32(skb, IPSET_ATTR_MEMSIZE, htonl(memsize)))
Oliver Smithb90cb8b2013-09-22 20:56:32 +0200100 goto nla_put_failure;
101 if (unlikely(ip_set_put_flags(skb, set)))
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200102 goto nla_put_failure;
103 ipset_nest_end(skb, nested);
104
105 return 0;
106nla_put_failure:
107 return -EMSGSIZE;
108}
109
110static int
111mtype_test(struct ip_set *set, void *value, const struct ip_set_ext *ext,
112 struct ip_set_ext *mext, u32 flags)
113{
114 struct mtype *map = set->data;
115 const struct mtype_adt_elem *e = value;
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200116 void *x = get_ext(set, map, e->id);
117 int ret = mtype_do_test(e, map, set->dsize);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200118
119 if (ret <= 0)
120 return ret;
121 if (SET_WITH_TIMEOUT(set) &&
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200122 ip_set_timeout_expired(ext_timeout(x, set)))
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200123 return 0;
Jozsef Kadlecsikf48d19d2013-04-08 23:10:22 +0200124 if (SET_WITH_COUNTER(set))
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200125 ip_set_update_counter(ext_counter(x, set), ext, mext, flags);
Anton Danilov39d1ecf2014-08-28 10:11:28 +0400126 if (SET_WITH_SKBINFO(set))
127 ip_set_get_skbinfo(ext_skbinfo(x, set), ext, mext, flags);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200128 return 1;
129}
130
131static int
132mtype_add(struct ip_set *set, void *value, const struct ip_set_ext *ext,
133 struct ip_set_ext *mext, u32 flags)
134{
135 struct mtype *map = set->data;
136 const struct mtype_adt_elem *e = value;
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200137 void *x = get_ext(set, map, e->id);
138 int ret = mtype_do_add(e, map, flags, set->dsize);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200139
140 if (ret == IPSET_ADD_FAILED) {
141 if (SET_WITH_TIMEOUT(set) &&
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200142 ip_set_timeout_expired(ext_timeout(x, set))) {
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200143 ret = 0;
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200144 } else if (!(flags & IPSET_FLAG_EXIST)) {
145 set_bit(e->id, map->members);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200146 return -IPSET_ERR_EXIST;
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200147 }
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +0200148 /* Element is re-added, cleanup extensions */
149 ip_set_ext_destroy(set, x);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200150 }
151
152 if (SET_WITH_TIMEOUT(set))
153#ifdef IP_SET_BITMAP_STORED_TIMEOUT
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200154 mtype_add_timeout(ext_timeout(x, set), e, ext, set, map, ret);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200155#else
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200156 ip_set_timeout_set(ext_timeout(x, set), ext->timeout);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200157#endif
158
Jozsef Kadlecsikf48d19d2013-04-08 23:10:22 +0200159 if (SET_WITH_COUNTER(set))
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200160 ip_set_init_counter(ext_counter(x, set), ext);
Oliver Smithb90cb8b2013-09-22 20:56:32 +0200161 if (SET_WITH_COMMENT(set))
162 ip_set_init_comment(ext_comment(x, set), ext);
Anton Danilov39d1ecf2014-08-28 10:11:28 +0400163 if (SET_WITH_SKBINFO(set))
164 ip_set_init_skbinfo(ext_skbinfo(x, set), ext);
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200165
166 /* Activate element */
167 set_bit(e->id, map->members);
168
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200169 return 0;
170}
171
172static int
173mtype_del(struct ip_set *set, void *value, const struct ip_set_ext *ext,
174 struct ip_set_ext *mext, u32 flags)
175{
176 struct mtype *map = set->data;
177 const struct mtype_adt_elem *e = value;
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +0200178 void *x = get_ext(set, map, e->id);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200179
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +0200180 if (mtype_do_del(e, map))
181 return -IPSET_ERR_EXIST;
182
183 ip_set_ext_destroy(set, x);
184 if (SET_WITH_TIMEOUT(set) &&
185 ip_set_timeout_expired(ext_timeout(x, set)))
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200186 return -IPSET_ERR_EXIST;
187
188 return 0;
189}
190
Jozsef Kadlecsik3fd986b2013-09-25 17:44:35 +0200191#ifndef IP_SET_BITMAP_STORED_TIMEOUT
192static inline bool
193mtype_is_filled(const struct mtype_elem *x)
194{
195 return true;
196}
197#endif
198
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200199static int
200mtype_list(const struct ip_set *set,
201 struct sk_buff *skb, struct netlink_callback *cb)
202{
203 struct mtype *map = set->data;
204 struct nlattr *adt, *nested;
205 void *x;
Jozsef Kadlecsik93302882013-10-18 11:41:55 +0200206 u32 id, first = cb->args[IPSET_CB_ARG0];
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200207 int ret = 0;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200208
209 adt = ipset_nest_start(skb, IPSET_ATTR_ADT);
210 if (!adt)
211 return -EMSGSIZE;
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200212 /* Extensions may be replaced */
213 rcu_read_lock();
Jozsef Kadlecsik93302882013-10-18 11:41:55 +0200214 for (; cb->args[IPSET_CB_ARG0] < map->elements;
215 cb->args[IPSET_CB_ARG0]++) {
216 id = cb->args[IPSET_CB_ARG0];
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200217 x = get_ext(set, map, id);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200218 if (!test_bit(id, map->members) ||
219 (SET_WITH_TIMEOUT(set) &&
220#ifdef IP_SET_BITMAP_STORED_TIMEOUT
Jozsef Kadlecsikca0f6a52015-06-13 19:45:33 +0200221 mtype_is_filled((const struct mtype_elem *)x) &&
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200222#endif
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200223 ip_set_timeout_expired(ext_timeout(x, set))))
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200224 continue;
225 nested = ipset_nest_start(skb, IPSET_ATTR_DATA);
226 if (!nested) {
227 if (id == first) {
228 nla_nest_cancel(skb, adt);
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200229 ret = -EMSGSIZE;
230 goto out;
231 }
232
233 goto nla_put_failure;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200234 }
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200235 if (mtype_do_list(skb, map, id, set->dsize))
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200236 goto nla_put_failure;
Jozsef Kadlecsik3fd986b2013-09-25 17:44:35 +0200237 if (ip_set_put_extensions(skb, set, x,
Jozsef Kadlecsikca0f6a52015-06-13 19:45:33 +0200238 mtype_is_filled((const struct mtype_elem *)x)))
Oliver Smithb90cb8b2013-09-22 20:56:32 +0200239 goto nla_put_failure;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200240 ipset_nest_end(skb, nested);
241 }
242 ipset_nest_end(skb, adt);
243
244 /* Set listing finished */
Jozsef Kadlecsik93302882013-10-18 11:41:55 +0200245 cb->args[IPSET_CB_ARG0] = 0;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200246
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200247 goto out;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200248
249nla_put_failure:
250 nla_nest_cancel(skb, nested);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200251 if (unlikely(id == first)) {
Jozsef Kadlecsik93302882013-10-18 11:41:55 +0200252 cb->args[IPSET_CB_ARG0] = 0;
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200253 ret = -EMSGSIZE;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200254 }
Jozsef Kadlecsik122ebbf2013-04-27 21:02:59 +0200255 ipset_nest_end(skb, adt);
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200256out:
257 rcu_read_unlock();
258 return ret;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200259}
260
261static void
262mtype_gc(unsigned long ul_set)
263{
Jozsef Kadlecsikca0f6a52015-06-13 19:45:33 +0200264 struct ip_set *set = (struct ip_set *)ul_set;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200265 struct mtype *map = set->data;
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +0200266 void *x;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200267 u32 id;
268
269 /* We run parallel with other readers (test element)
Jozsef Kadlecsikca0f6a52015-06-13 19:45:33 +0200270 * but adding/deleting new entries is locked out
271 */
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200272 spin_lock_bh(&set->lock);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200273 for (id = 0; id < map->elements; id++)
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200274 if (mtype_gc_test(id, map, set->dsize)) {
275 x = get_ext(set, map, id);
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +0200276 if (ip_set_timeout_expired(ext_timeout(x, set))) {
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200277 clear_bit(id, map->members);
Jozsef Kadlecsik40cd63b2013-09-09 14:44:29 +0200278 ip_set_ext_destroy(set, x);
279 }
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200280 }
Jozsef Kadlecsik96f51422015-06-13 14:39:59 +0200281 spin_unlock_bh(&set->lock);
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200282
Jozsef Kadlecsikca134ce2013-09-07 00:10:07 +0200283 map->gc.expires = jiffies + IPSET_GC_PERIOD(set->timeout) * HZ;
Jozsef Kadlecsik4d73de32013-04-08 21:00:52 +0200284 add_timer(&map->gc);
285}
286
287static const struct ip_set_type_variant mtype = {
288 .kadt = mtype_kadt,
289 .uadt = mtype_uadt,
290 .adt = {
291 [IPSET_ADD] = mtype_add,
292 [IPSET_DEL] = mtype_del,
293 [IPSET_TEST] = mtype_test,
294 },
295 .destroy = mtype_destroy,
296 .flush = mtype_flush,
297 .head = mtype_head,
298 .list = mtype_list,
299 .same_set = mtype_same_set,
300};
301
302#endif /* __IP_SET_BITMAP_IP_GEN_H */