1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * net/sched/act_connmark.c netfilter connmark retriever action
4 * skb mark is over-written
6 * Copyright (c) 2011 Felix Fietkau <nbd@openwrt.org>
9 #include <linux/module.h>
10 #include <linux/init.h>
11 #include <linux/kernel.h>
12 #include <linux/skbuff.h>
13 #include <linux/rtnetlink.h>
14 #include <linux/pkt_cls.h>
16 #include <linux/ipv6.h>
17 #include <net/netlink.h>
18 #include <net/pkt_sched.h>
19 #include <net/act_api.h>
20 #include <net/pkt_cls.h>
21 #include <uapi/linux/tc_act/tc_connmark.h>
22 #include <net/tc_act/tc_connmark.h>
23 #include <net/tc_wrapper.h>
25 #include <net/netfilter/nf_conntrack.h>
26 #include <net/netfilter/nf_conntrack_core.h>
27 #include <net/netfilter/nf_conntrack_zones.h>
29 static struct tc_action_ops act_connmark_ops
;
31 TC_INDIRECT_SCOPE
int tcf_connmark_act(struct sk_buff
*skb
,
32 const struct tc_action
*a
,
33 struct tcf_result
*res
)
35 const struct nf_conntrack_tuple_hash
*thash
;
36 struct nf_conntrack_tuple tuple
;
37 enum ip_conntrack_info ctinfo
;
38 struct tcf_connmark_info
*ca
= to_connmark(a
);
39 struct tcf_connmark_parms
*parms
;
40 struct nf_conntrack_zone zone
;
44 tcf_lastuse_update(&ca
->tcf_tm
);
45 tcf_action_update_bstats(&ca
->common
, skb
);
47 parms
= rcu_dereference_bh(ca
->parms
);
49 switch (skb_protocol(skb
, true)) {
51 if (skb
->len
< sizeof(struct iphdr
))
56 case htons(ETH_P_IPV6
):
57 if (skb
->len
< sizeof(struct ipv6hdr
))
66 c
= nf_ct_get(skb
, &ctinfo
);
68 skb
->mark
= READ_ONCE(c
->mark
);
72 if (!nf_ct_get_tuplepr(skb
, skb_network_offset(skb
), proto
, parms
->net
,
76 zone
.id
= parms
->zone
;
77 zone
.dir
= NF_CT_DEFAULT_ZONE_DIR
;
79 thash
= nf_conntrack_find_get(parms
->net
, &zone
, &tuple
);
83 c
= nf_ct_tuplehash_to_ctrack(thash
);
84 skb
->mark
= READ_ONCE(c
->mark
);
88 /* using overlimits stats to count how many packets marked */
89 tcf_action_inc_overlimit_qstats(&ca
->common
);
91 return READ_ONCE(ca
->tcf_action
);
94 static const struct nla_policy connmark_policy
[TCA_CONNMARK_MAX
+ 1] = {
95 [TCA_CONNMARK_PARMS
] = { .len
= sizeof(struct tc_connmark
) },
98 static int tcf_connmark_init(struct net
*net
, struct nlattr
*nla
,
99 struct nlattr
*est
, struct tc_action
**a
,
100 struct tcf_proto
*tp
, u32 flags
,
101 struct netlink_ext_ack
*extack
)
103 struct tc_action_net
*tn
= net_generic(net
, act_connmark_ops
.net_id
);
104 struct tcf_connmark_parms
*nparms
, *oparms
;
105 struct nlattr
*tb
[TCA_CONNMARK_MAX
+ 1];
106 bool bind
= flags
& TCA_ACT_FLAGS_BIND
;
107 struct tcf_chain
*goto_ch
= NULL
;
108 struct tcf_connmark_info
*ci
;
109 struct tc_connmark
*parm
;
116 ret
= nla_parse_nested_deprecated(tb
, TCA_CONNMARK_MAX
, nla
,
117 connmark_policy
, NULL
);
121 if (!tb
[TCA_CONNMARK_PARMS
])
124 nparms
= kzalloc(sizeof(*nparms
), GFP_KERNEL
);
128 parm
= nla_data(tb
[TCA_CONNMARK_PARMS
]);
130 ret
= tcf_idr_check_alloc(tn
, &index
, a
, bind
);
132 ret
= tcf_idr_create_from_flags(tn
, index
, est
, a
,
133 &act_connmark_ops
, bind
, flags
);
135 tcf_idr_cleanup(tn
, index
);
140 ci
= to_connmark(*a
);
143 nparms
->zone
= parm
->zone
;
146 } else if (ret
> 0) {
147 ci
= to_connmark(*a
);
152 if (!(flags
& TCA_ACT_FLAGS_REPLACE
)) {
157 nparms
->net
= rtnl_dereference(ci
->parms
)->net
;
158 nparms
->zone
= parm
->zone
;
166 err
= tcf_action_check_ctrlact(parm
->action
, tp
, &goto_ch
, extack
);
170 spin_lock_bh(&ci
->tcf_lock
);
171 goto_ch
= tcf_action_set_ctrlact(*a
, parm
->action
, goto_ch
);
172 oparms
= rcu_replace_pointer(ci
->parms
, nparms
, lockdep_is_held(&ci
->tcf_lock
));
173 spin_unlock_bh(&ci
->tcf_lock
);
176 tcf_chain_put_by_act(goto_ch
);
179 kfree_rcu(oparms
, rcu
);
184 tcf_idr_release(*a
, bind
);
190 static inline int tcf_connmark_dump(struct sk_buff
*skb
, struct tc_action
*a
,
193 unsigned char *b
= skb_tail_pointer(skb
);
194 struct tcf_connmark_info
*ci
= to_connmark(a
);
195 struct tc_connmark opt
= {
196 .index
= ci
->tcf_index
,
197 .refcnt
= refcount_read(&ci
->tcf_refcnt
) - ref
,
198 .bindcnt
= atomic_read(&ci
->tcf_bindcnt
) - bind
,
200 struct tcf_connmark_parms
*parms
;
203 spin_lock_bh(&ci
->tcf_lock
);
204 parms
= rcu_dereference_protected(ci
->parms
, lockdep_is_held(&ci
->tcf_lock
));
206 opt
.action
= ci
->tcf_action
;
207 opt
.zone
= parms
->zone
;
208 if (nla_put(skb
, TCA_CONNMARK_PARMS
, sizeof(opt
), &opt
))
209 goto nla_put_failure
;
211 tcf_tm_dump(&t
, &ci
->tcf_tm
);
212 if (nla_put_64bit(skb
, TCA_CONNMARK_TM
, sizeof(t
), &t
,
214 goto nla_put_failure
;
215 spin_unlock_bh(&ci
->tcf_lock
);
220 spin_unlock_bh(&ci
->tcf_lock
);
225 static void tcf_connmark_cleanup(struct tc_action
*a
)
227 struct tcf_connmark_info
*ci
= to_connmark(a
);
228 struct tcf_connmark_parms
*parms
;
230 parms
= rcu_dereference_protected(ci
->parms
, 1);
232 kfree_rcu(parms
, rcu
);
235 static struct tc_action_ops act_connmark_ops
= {
237 .id
= TCA_ID_CONNMARK
,
238 .owner
= THIS_MODULE
,
239 .act
= tcf_connmark_act
,
240 .dump
= tcf_connmark_dump
,
241 .init
= tcf_connmark_init
,
242 .cleanup
= tcf_connmark_cleanup
,
243 .size
= sizeof(struct tcf_connmark_info
),
245 MODULE_ALIAS_NET_ACT("connmark");
247 static __net_init
int connmark_init_net(struct net
*net
)
249 struct tc_action_net
*tn
= net_generic(net
, act_connmark_ops
.net_id
);
251 return tc_action_net_init(net
, tn
, &act_connmark_ops
);
254 static void __net_exit
connmark_exit_net(struct list_head
*net_list
)
256 tc_action_net_exit(net_list
, act_connmark_ops
.net_id
);
259 static struct pernet_operations connmark_net_ops
= {
260 .init
= connmark_init_net
,
261 .exit_batch
= connmark_exit_net
,
262 .id
= &act_connmark_ops
.net_id
,
263 .size
= sizeof(struct tc_action_net
),
266 static int __init
connmark_init_module(void)
268 return tcf_register_action(&act_connmark_ops
, &connmark_net_ops
);
271 static void __exit
connmark_cleanup_module(void)
273 tcf_unregister_action(&act_connmark_ops
, &connmark_net_ops
);
276 module_init(connmark_init_module
);
277 module_exit(connmark_cleanup_module
);
278 MODULE_AUTHOR("Felix Fietkau <nbd@openwrt.org>");
279 MODULE_DESCRIPTION("Connection tracking mark restoring");
280 MODULE_LICENSE("GPL");