1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * net/sched/cls_basic.c Basic Packet Classifier.
5 * Authors: Thomas Graf <tgraf@suug.ch>
8 #include <linux/module.h>
9 #include <linux/slab.h>
10 #include <linux/types.h>
11 #include <linux/kernel.h>
12 #include <linux/string.h>
13 #include <linux/errno.h>
14 #include <linux/rtnetlink.h>
15 #include <linux/skbuff.h>
16 #include <linux/idr.h>
17 #include <linux/percpu.h>
18 #include <net/netlink.h>
19 #include <net/act_api.h>
20 #include <net/pkt_cls.h>
23 struct list_head flist
;
24 struct idr handle_idr
;
31 struct tcf_ematch_tree ematches
;
32 struct tcf_result res
;
34 struct list_head link
;
35 struct tc_basic_pcnt __percpu
*pf
;
36 struct rcu_work rwork
;
39 static int basic_classify(struct sk_buff
*skb
, const struct tcf_proto
*tp
,
40 struct tcf_result
*res
)
43 struct basic_head
*head
= rcu_dereference_bh(tp
->root
);
44 struct basic_filter
*f
;
46 list_for_each_entry_rcu(f
, &head
->flist
, link
) {
47 __this_cpu_inc(f
->pf
->rcnt
);
48 if (!tcf_em_tree_match(skb
, &f
->ematches
, NULL
))
50 __this_cpu_inc(f
->pf
->rhit
);
52 r
= tcf_exts_exec(skb
, &f
->exts
, res
);
60 static void *basic_get(struct tcf_proto
*tp
, u32 handle
)
62 struct basic_head
*head
= rtnl_dereference(tp
->root
);
63 struct basic_filter
*f
;
65 list_for_each_entry(f
, &head
->flist
, link
) {
66 if (f
->handle
== handle
) {
74 static int basic_init(struct tcf_proto
*tp
)
76 struct basic_head
*head
;
78 head
= kzalloc(sizeof(*head
), GFP_KERNEL
);
81 INIT_LIST_HEAD(&head
->flist
);
82 idr_init(&head
->handle_idr
);
83 rcu_assign_pointer(tp
->root
, head
);
87 static void __basic_delete_filter(struct basic_filter
*f
)
89 tcf_exts_destroy(&f
->exts
);
90 tcf_em_tree_destroy(&f
->ematches
);
91 tcf_exts_put_net(&f
->exts
);
96 static void basic_delete_filter_work(struct work_struct
*work
)
98 struct basic_filter
*f
= container_of(to_rcu_work(work
),
102 __basic_delete_filter(f
);
106 static void basic_destroy(struct tcf_proto
*tp
, bool rtnl_held
,
107 struct netlink_ext_ack
*extack
)
109 struct basic_head
*head
= rtnl_dereference(tp
->root
);
110 struct basic_filter
*f
, *n
;
112 list_for_each_entry_safe(f
, n
, &head
->flist
, link
) {
113 list_del_rcu(&f
->link
);
114 tcf_unbind_filter(tp
, &f
->res
);
115 idr_remove(&head
->handle_idr
, f
->handle
);
116 if (tcf_exts_get_net(&f
->exts
))
117 tcf_queue_work(&f
->rwork
, basic_delete_filter_work
);
119 __basic_delete_filter(f
);
121 idr_destroy(&head
->handle_idr
);
122 kfree_rcu(head
, rcu
);
125 static int basic_delete(struct tcf_proto
*tp
, void *arg
, bool *last
,
126 bool rtnl_held
, struct netlink_ext_ack
*extack
)
128 struct basic_head
*head
= rtnl_dereference(tp
->root
);
129 struct basic_filter
*f
= arg
;
131 list_del_rcu(&f
->link
);
132 tcf_unbind_filter(tp
, &f
->res
);
133 idr_remove(&head
->handle_idr
, f
->handle
);
134 tcf_exts_get_net(&f
->exts
);
135 tcf_queue_work(&f
->rwork
, basic_delete_filter_work
);
136 *last
= list_empty(&head
->flist
);
140 static const struct nla_policy basic_policy
[TCA_BASIC_MAX
+ 1] = {
141 [TCA_BASIC_CLASSID
] = { .type
= NLA_U32
},
142 [TCA_BASIC_EMATCHES
] = { .type
= NLA_NESTED
},
145 static int basic_set_parms(struct net
*net
, struct tcf_proto
*tp
,
146 struct basic_filter
*f
, unsigned long base
,
148 struct nlattr
*est
, bool ovr
,
149 struct netlink_ext_ack
*extack
)
153 err
= tcf_exts_validate(net
, tp
, tb
, est
, &f
->exts
, ovr
, true, extack
);
157 err
= tcf_em_tree_validate(tp
, tb
[TCA_BASIC_EMATCHES
], &f
->ematches
);
161 if (tb
[TCA_BASIC_CLASSID
]) {
162 f
->res
.classid
= nla_get_u32(tb
[TCA_BASIC_CLASSID
]);
163 tcf_bind_filter(tp
, &f
->res
, base
);
170 static int basic_change(struct net
*net
, struct sk_buff
*in_skb
,
171 struct tcf_proto
*tp
, unsigned long base
, u32 handle
,
172 struct nlattr
**tca
, void **arg
, bool ovr
,
173 bool rtnl_held
, struct netlink_ext_ack
*extack
)
176 struct basic_head
*head
= rtnl_dereference(tp
->root
);
177 struct nlattr
*tb
[TCA_BASIC_MAX
+ 1];
178 struct basic_filter
*fold
= (struct basic_filter
*) *arg
;
179 struct basic_filter
*fnew
;
181 if (tca
[TCA_OPTIONS
] == NULL
)
184 err
= nla_parse_nested_deprecated(tb
, TCA_BASIC_MAX
, tca
[TCA_OPTIONS
],
190 if (handle
&& fold
->handle
!= handle
)
194 fnew
= kzalloc(sizeof(*fnew
), GFP_KERNEL
);
198 err
= tcf_exts_init(&fnew
->exts
, net
, TCA_BASIC_ACT
, TCA_BASIC_POLICE
);
204 err
= idr_alloc_u32(&head
->handle_idr
, fnew
, &handle
,
205 INT_MAX
, GFP_KERNEL
);
207 err
= idr_alloc_u32(&head
->handle_idr
, fnew
, &handle
,
212 fnew
->handle
= handle
;
213 fnew
->pf
= alloc_percpu(struct tc_basic_pcnt
);
219 err
= basic_set_parms(net
, tp
, fnew
, base
, tb
, tca
[TCA_RATE
], ovr
,
223 idr_remove(&head
->handle_idr
, fnew
->handle
);
230 idr_replace(&head
->handle_idr
, fnew
, fnew
->handle
);
231 list_replace_rcu(&fold
->link
, &fnew
->link
);
232 tcf_unbind_filter(tp
, &fold
->res
);
233 tcf_exts_get_net(&fold
->exts
);
234 tcf_queue_work(&fold
->rwork
, basic_delete_filter_work
);
236 list_add_rcu(&fnew
->link
, &head
->flist
);
241 free_percpu(fnew
->pf
);
242 tcf_exts_destroy(&fnew
->exts
);
247 static void basic_walk(struct tcf_proto
*tp
, struct tcf_walker
*arg
,
250 struct basic_head
*head
= rtnl_dereference(tp
->root
);
251 struct basic_filter
*f
;
253 list_for_each_entry(f
, &head
->flist
, link
) {
254 if (arg
->count
< arg
->skip
)
257 if (arg
->fn(tp
, f
, arg
) < 0) {
266 static void basic_bind_class(void *fh
, u32 classid
, unsigned long cl
)
268 struct basic_filter
*f
= fh
;
270 if (f
&& f
->res
.classid
== classid
)
274 static int basic_dump(struct net
*net
, struct tcf_proto
*tp
, void *fh
,
275 struct sk_buff
*skb
, struct tcmsg
*t
, bool rtnl_held
)
277 struct tc_basic_pcnt gpf
= {};
278 struct basic_filter
*f
= fh
;
285 t
->tcm_handle
= f
->handle
;
287 nest
= nla_nest_start_noflag(skb
, TCA_OPTIONS
);
289 goto nla_put_failure
;
291 if (f
->res
.classid
&&
292 nla_put_u32(skb
, TCA_BASIC_CLASSID
, f
->res
.classid
))
293 goto nla_put_failure
;
295 for_each_possible_cpu(cpu
) {
296 struct tc_basic_pcnt
*pf
= per_cpu_ptr(f
->pf
, cpu
);
298 gpf
.rcnt
+= pf
->rcnt
;
299 gpf
.rhit
+= pf
->rhit
;
302 if (nla_put_64bit(skb
, TCA_BASIC_PCNT
,
303 sizeof(struct tc_basic_pcnt
),
304 &gpf
, TCA_BASIC_PAD
))
305 goto nla_put_failure
;
307 if (tcf_exts_dump(skb
, &f
->exts
) < 0 ||
308 tcf_em_tree_dump(skb
, &f
->ematches
, TCA_BASIC_EMATCHES
) < 0)
309 goto nla_put_failure
;
311 nla_nest_end(skb
, nest
);
313 if (tcf_exts_dump_stats(skb
, &f
->exts
) < 0)
314 goto nla_put_failure
;
319 nla_nest_cancel(skb
, nest
);
323 static struct tcf_proto_ops cls_basic_ops __read_mostly
= {
325 .classify
= basic_classify
,
327 .destroy
= basic_destroy
,
329 .change
= basic_change
,
330 .delete = basic_delete
,
333 .bind_class
= basic_bind_class
,
334 .owner
= THIS_MODULE
,
337 static int __init
init_basic(void)
339 return register_tcf_proto_ops(&cls_basic_ops
);
342 static void __exit
exit_basic(void)
344 unregister_tcf_proto_ops(&cls_basic_ops
);
347 module_init(init_basic
)
348 module_exit(exit_basic
)
349 MODULE_LICENSE("GPL");