xref: /openbmc/linux/net/sched/act_skbedit.c (revision 7ae9fb1b7ecbb5d85d07857943f677fd1a559b18)
19952f691SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2ca9b0e27SAlexander Duyck /*
3ca9b0e27SAlexander Duyck  * Copyright (c) 2008, Intel Corporation.
4ca9b0e27SAlexander Duyck  *
5ca9b0e27SAlexander Duyck  * Author: Alexander Duyck <alexander.h.duyck@intel.com>
6ca9b0e27SAlexander Duyck  */
7ca9b0e27SAlexander Duyck 
8ca9b0e27SAlexander Duyck #include <linux/module.h>
9ca9b0e27SAlexander Duyck #include <linux/init.h>
10ca9b0e27SAlexander Duyck #include <linux/kernel.h>
11ca9b0e27SAlexander Duyck #include <linux/skbuff.h>
12ca9b0e27SAlexander Duyck #include <linux/rtnetlink.h>
13ca9b0e27SAlexander Duyck #include <net/netlink.h>
14ca9b0e27SAlexander Duyck #include <net/pkt_sched.h>
15e7e3728bSQiaobin Fu #include <net/ip.h>
16e7e3728bSQiaobin Fu #include <net/ipv6.h>
17e7e3728bSQiaobin Fu #include <net/dsfield.h>
18ec7727bbSDavide Caratti #include <net/pkt_cls.h>
19*871cf386SPedro Tammela #include <net/tc_wrapper.h>
20ca9b0e27SAlexander Duyck 
21ca9b0e27SAlexander Duyck #include <linux/tc_act/tc_skbedit.h>
22ca9b0e27SAlexander Duyck #include <net/tc_act/tc_skbedit.h>
23ca9b0e27SAlexander Duyck 
24a85a970aSWANG Cong static struct tc_action_ops act_skbedit_ops;
25ddf97ccdSWANG Cong 
tcf_skbedit_hash(struct tcf_skbedit_params * params,struct sk_buff * skb)2638a6f086STonghao Zhang static u16 tcf_skbedit_hash(struct tcf_skbedit_params *params,
2738a6f086STonghao Zhang 			    struct sk_buff *skb)
2838a6f086STonghao Zhang {
2938a6f086STonghao Zhang 	u16 queue_mapping = params->queue_mapping;
3038a6f086STonghao Zhang 
3138a6f086STonghao Zhang 	if (params->flags & SKBEDIT_F_TXQ_SKBHASH) {
3238a6f086STonghao Zhang 		u32 hash = skb_get_hash(skb);
3338a6f086STonghao Zhang 
3438a6f086STonghao Zhang 		queue_mapping += hash % params->mapping_mod;
3538a6f086STonghao Zhang 	}
3638a6f086STonghao Zhang 
3738a6f086STonghao Zhang 	return netdev_cap_txqueue(skb->dev, queue_mapping);
3838a6f086STonghao Zhang }
3938a6f086STonghao Zhang 
tcf_skbedit_act(struct sk_buff * skb,const struct tc_action * a,struct tcf_result * res)40*871cf386SPedro Tammela TC_INDIRECT_SCOPE int tcf_skbedit_act(struct sk_buff *skb,
41*871cf386SPedro Tammela 				      const struct tc_action *a,
42ca9b0e27SAlexander Duyck 				      struct tcf_result *res)
43ca9b0e27SAlexander Duyck {
44a85a970aSWANG Cong 	struct tcf_skbedit *d = to_skbedit(a);
45c749cddaSDavide Caratti 	struct tcf_skbedit_params *params;
46c749cddaSDavide Caratti 	int action;
47ca9b0e27SAlexander Duyck 
489c4a4e48SJamal Hadi Salim 	tcf_lastuse_update(&d->tcf_tm);
4950dc9a85SAhmed S. Darwish 	bstats_update(this_cpu_ptr(d->common.cpu_bstats), skb);
50ca9b0e27SAlexander Duyck 
517fd4b288SPaolo Abeni 	params = rcu_dereference_bh(d->params);
52c749cddaSDavide Caratti 	action = READ_ONCE(d->tcf_action);
53c749cddaSDavide Caratti 
54c749cddaSDavide Caratti 	if (params->flags & SKBEDIT_F_PRIORITY)
55c749cddaSDavide Caratti 		skb->priority = params->priority;
56c749cddaSDavide Caratti 	if (params->flags & SKBEDIT_F_INHERITDSFIELD) {
57e7e3728bSQiaobin Fu 		int wlen = skb_network_offset(skb);
58e7e3728bSQiaobin Fu 
59d7bf2ebeSToke Høiland-Jørgensen 		switch (skb_protocol(skb, true)) {
60e7e3728bSQiaobin Fu 		case htons(ETH_P_IP):
61e7e3728bSQiaobin Fu 			wlen += sizeof(struct iphdr);
62e7e3728bSQiaobin Fu 			if (!pskb_may_pull(skb, wlen))
63e7e3728bSQiaobin Fu 				goto err;
64e7e3728bSQiaobin Fu 			skb->priority = ipv4_get_dsfield(ip_hdr(skb)) >> 2;
65e7e3728bSQiaobin Fu 			break;
66e7e3728bSQiaobin Fu 
67e7e3728bSQiaobin Fu 		case htons(ETH_P_IPV6):
68e7e3728bSQiaobin Fu 			wlen += sizeof(struct ipv6hdr);
69e7e3728bSQiaobin Fu 			if (!pskb_may_pull(skb, wlen))
70e7e3728bSQiaobin Fu 				goto err;
71e7e3728bSQiaobin Fu 			skb->priority = ipv6_get_dsfield(ipv6_hdr(skb)) >> 2;
72e7e3728bSQiaobin Fu 			break;
73e7e3728bSQiaobin Fu 		}
74e7e3728bSQiaobin Fu 	}
75c749cddaSDavide Caratti 	if (params->flags & SKBEDIT_F_QUEUE_MAPPING &&
762f1e85b1STonghao Zhang 	    skb->dev->real_num_tx_queues > params->queue_mapping) {
772f1e85b1STonghao Zhang #ifdef CONFIG_NET_EGRESS
782f1e85b1STonghao Zhang 		netdev_xmit_skip_txqueue(true);
792f1e85b1STonghao Zhang #endif
8038a6f086STonghao Zhang 		skb_set_queue_mapping(skb, tcf_skbedit_hash(params, skb));
812f1e85b1STonghao Zhang 	}
82c749cddaSDavide Caratti 	if (params->flags & SKBEDIT_F_MARK) {
83c749cddaSDavide Caratti 		skb->mark &= ~params->mask;
84c749cddaSDavide Caratti 		skb->mark |= params->mark & params->mask;
854fe77d82SAntonio Quartulli 	}
86c749cddaSDavide Caratti 	if (params->flags & SKBEDIT_F_PTYPE)
87c749cddaSDavide Caratti 		skb->pkt_type = params->ptype;
88c749cddaSDavide Caratti 	return action;
897fd4b288SPaolo Abeni 
90e7e3728bSQiaobin Fu err:
916f3dfb0dSDavide Caratti 	qstats_drop_inc(this_cpu_ptr(d->common.cpu_qstats));
927fd4b288SPaolo Abeni 	return TC_ACT_SHOT;
93ca9b0e27SAlexander Duyck }
94ca9b0e27SAlexander Duyck 
tcf_skbedit_stats_update(struct tc_action * a,u64 bytes,u64 packets,u64 drops,u64 lastuse,bool hw)95837cb17dSPetr Machata static void tcf_skbedit_stats_update(struct tc_action *a, u64 bytes,
964b61d3e8SPo Liu 				     u64 packets, u64 drops,
974b61d3e8SPo Liu 				     u64 lastuse, bool hw)
98837cb17dSPetr Machata {
99837cb17dSPetr Machata 	struct tcf_skbedit *d = to_skbedit(a);
100837cb17dSPetr Machata 	struct tcf_t *tm = &d->tcf_tm;
101837cb17dSPetr Machata 
1024b61d3e8SPo Liu 	tcf_action_update_stats(a, bytes, packets, drops, hw);
103837cb17dSPetr Machata 	tm->lastuse = max_t(u64, tm->lastuse, lastuse);
104837cb17dSPetr Machata }
105837cb17dSPetr Machata 
106ca9b0e27SAlexander Duyck static const struct nla_policy skbedit_policy[TCA_SKBEDIT_MAX + 1] = {
107ca9b0e27SAlexander Duyck 	[TCA_SKBEDIT_PARMS]		= { .len = sizeof(struct tc_skbedit) },
108ca9b0e27SAlexander Duyck 	[TCA_SKBEDIT_PRIORITY]		= { .len = sizeof(u32) },
109ca9b0e27SAlexander Duyck 	[TCA_SKBEDIT_QUEUE_MAPPING]	= { .len = sizeof(u16) },
1101c55d62eSjamal 	[TCA_SKBEDIT_MARK]		= { .len = sizeof(u32) },
111ff202ee1SJamal Hadi Salim 	[TCA_SKBEDIT_PTYPE]		= { .len = sizeof(u16) },
1124fe77d82SAntonio Quartulli 	[TCA_SKBEDIT_MASK]		= { .len = sizeof(u32) },
113e7e3728bSQiaobin Fu 	[TCA_SKBEDIT_FLAGS]		= { .len = sizeof(u64) },
11438a6f086STonghao Zhang 	[TCA_SKBEDIT_QUEUE_MAPPING_MAX]	= { .len = sizeof(u16) },
115ca9b0e27SAlexander Duyck };
116ca9b0e27SAlexander Duyck 
tcf_skbedit_init(struct net * net,struct nlattr * nla,struct nlattr * est,struct tc_action ** a,struct tcf_proto * tp,u32 act_flags,struct netlink_ext_ack * extack)117c1b52739SBenjamin LaHaise static int tcf_skbedit_init(struct net *net, struct nlattr *nla,
118a85a970aSWANG Cong 			    struct nlattr *est, struct tc_action **a,
119abbb0d33SVlad Buslov 			    struct tcf_proto *tp, u32 act_flags,
120789871bbSVlad Buslov 			    struct netlink_ext_ack *extack)
121ca9b0e27SAlexander Duyck {
122acd0a7abSZhengchao Shao 	struct tc_action_net *tn = net_generic(net, act_skbedit_ops.net_id);
123695176bfSCong Wang 	bool bind = act_flags & TCA_ACT_FLAGS_BIND;
1246d7a8df6SVlad Buslov 	struct tcf_skbedit_params *params_new;
125ca9b0e27SAlexander Duyck 	struct nlattr *tb[TCA_SKBEDIT_MAX + 1];
126ec7727bbSDavide Caratti 	struct tcf_chain *goto_ch = NULL;
127ca9b0e27SAlexander Duyck 	struct tc_skbedit *parm;
128ca9b0e27SAlexander Duyck 	struct tcf_skbedit *d;
1294fe77d82SAntonio Quartulli 	u32 flags = 0, *priority = NULL, *mark = NULL, *mask = NULL;
130ff202ee1SJamal Hadi Salim 	u16 *queue_mapping = NULL, *ptype = NULL;
13138a6f086STonghao Zhang 	u16 mapping_mod = 1;
132b2313077SWANG Cong 	bool exists = false;
133b2313077SWANG Cong 	int ret = 0, err;
1347be8ef2cSDmytro Linkin 	u32 index;
135ca9b0e27SAlexander Duyck 
136ca9b0e27SAlexander Duyck 	if (nla == NULL)
137ca9b0e27SAlexander Duyck 		return -EINVAL;
138ca9b0e27SAlexander Duyck 
1398cb08174SJohannes Berg 	err = nla_parse_nested_deprecated(tb, TCA_SKBEDIT_MAX, nla,
1408cb08174SJohannes Berg 					  skbedit_policy, NULL);
141ca9b0e27SAlexander Duyck 	if (err < 0)
142ca9b0e27SAlexander Duyck 		return err;
143ca9b0e27SAlexander Duyck 
144ca9b0e27SAlexander Duyck 	if (tb[TCA_SKBEDIT_PARMS] == NULL)
145ca9b0e27SAlexander Duyck 		return -EINVAL;
146ca9b0e27SAlexander Duyck 
147ca9b0e27SAlexander Duyck 	if (tb[TCA_SKBEDIT_PRIORITY] != NULL) {
148ca9b0e27SAlexander Duyck 		flags |= SKBEDIT_F_PRIORITY;
149ca9b0e27SAlexander Duyck 		priority = nla_data(tb[TCA_SKBEDIT_PRIORITY]);
150ca9b0e27SAlexander Duyck 	}
151ca9b0e27SAlexander Duyck 
152ca9b0e27SAlexander Duyck 	if (tb[TCA_SKBEDIT_QUEUE_MAPPING] != NULL) {
1534a6a676fSAmritha Nambiar 		if (is_tcf_skbedit_ingress(act_flags) &&
1544a6a676fSAmritha Nambiar 		    !(act_flags & TCA_ACT_FLAGS_SKIP_SW)) {
1554a6a676fSAmritha Nambiar 			NL_SET_ERR_MSG_MOD(extack, "\"queue_mapping\" option on receive side is hardware only, use skip_sw");
1564a6a676fSAmritha Nambiar 			return -EOPNOTSUPP;
1574a6a676fSAmritha Nambiar 		}
158ca9b0e27SAlexander Duyck 		flags |= SKBEDIT_F_QUEUE_MAPPING;
159ca9b0e27SAlexander Duyck 		queue_mapping = nla_data(tb[TCA_SKBEDIT_QUEUE_MAPPING]);
160ca9b0e27SAlexander Duyck 	}
1611c55d62eSjamal 
162ff202ee1SJamal Hadi Salim 	if (tb[TCA_SKBEDIT_PTYPE] != NULL) {
163ff202ee1SJamal Hadi Salim 		ptype = nla_data(tb[TCA_SKBEDIT_PTYPE]);
164ff202ee1SJamal Hadi Salim 		if (!skb_pkt_type_ok(*ptype))
165ff202ee1SJamal Hadi Salim 			return -EINVAL;
166ff202ee1SJamal Hadi Salim 		flags |= SKBEDIT_F_PTYPE;
167ff202ee1SJamal Hadi Salim 	}
168ff202ee1SJamal Hadi Salim 
1691c55d62eSjamal 	if (tb[TCA_SKBEDIT_MARK] != NULL) {
1701c55d62eSjamal 		flags |= SKBEDIT_F_MARK;
1711c55d62eSjamal 		mark = nla_data(tb[TCA_SKBEDIT_MARK]);
1721c55d62eSjamal 	}
1731c55d62eSjamal 
1744fe77d82SAntonio Quartulli 	if (tb[TCA_SKBEDIT_MASK] != NULL) {
1754fe77d82SAntonio Quartulli 		flags |= SKBEDIT_F_MASK;
1764fe77d82SAntonio Quartulli 		mask = nla_data(tb[TCA_SKBEDIT_MASK]);
1774fe77d82SAntonio Quartulli 	}
1784fe77d82SAntonio Quartulli 
179e7e3728bSQiaobin Fu 	if (tb[TCA_SKBEDIT_FLAGS] != NULL) {
180e7e3728bSQiaobin Fu 		u64 *pure_flags = nla_data(tb[TCA_SKBEDIT_FLAGS]);
181e7e3728bSQiaobin Fu 
18238a6f086STonghao Zhang 		if (*pure_flags & SKBEDIT_F_TXQ_SKBHASH) {
18338a6f086STonghao Zhang 			u16 *queue_mapping_max;
18438a6f086STonghao Zhang 
18538a6f086STonghao Zhang 			if (!tb[TCA_SKBEDIT_QUEUE_MAPPING] ||
18638a6f086STonghao Zhang 			    !tb[TCA_SKBEDIT_QUEUE_MAPPING_MAX]) {
18738a6f086STonghao Zhang 				NL_SET_ERR_MSG_MOD(extack, "Missing required range of queue_mapping.");
18838a6f086STonghao Zhang 				return -EINVAL;
18938a6f086STonghao Zhang 			}
19038a6f086STonghao Zhang 
19138a6f086STonghao Zhang 			queue_mapping_max =
19238a6f086STonghao Zhang 				nla_data(tb[TCA_SKBEDIT_QUEUE_MAPPING_MAX]);
19338a6f086STonghao Zhang 			if (*queue_mapping_max < *queue_mapping) {
19438a6f086STonghao Zhang 				NL_SET_ERR_MSG_MOD(extack, "The range of queue_mapping is invalid, max < min.");
19538a6f086STonghao Zhang 				return -EINVAL;
19638a6f086STonghao Zhang 			}
19738a6f086STonghao Zhang 
19838a6f086STonghao Zhang 			mapping_mod = *queue_mapping_max - *queue_mapping + 1;
19938a6f086STonghao Zhang 			flags |= SKBEDIT_F_TXQ_SKBHASH;
20038a6f086STonghao Zhang 		}
201e7e3728bSQiaobin Fu 		if (*pure_flags & SKBEDIT_F_INHERITDSFIELD)
202e7e3728bSQiaobin Fu 			flags |= SKBEDIT_F_INHERITDSFIELD;
203e7e3728bSQiaobin Fu 	}
204e7e3728bSQiaobin Fu 
205ca9b0e27SAlexander Duyck 	parm = nla_data(tb[TCA_SKBEDIT_PARMS]);
2067be8ef2cSDmytro Linkin 	index = parm->index;
2077be8ef2cSDmytro Linkin 	err = tcf_idr_check_alloc(tn, &index, a, bind);
2080190c1d4SVlad Buslov 	if (err < 0)
2090190c1d4SVlad Buslov 		return err;
2100190c1d4SVlad Buslov 	exists = err;
2115e1567aeSJamal Hadi Salim 	if (exists && bind)
2125e1567aeSJamal Hadi Salim 		return 0;
2135e1567aeSJamal Hadi Salim 
2145e1567aeSJamal Hadi Salim 	if (!flags) {
215af5d0184SRoman Mashak 		if (exists)
21665a206c0SChris Mi 			tcf_idr_release(*a, bind);
2170190c1d4SVlad Buslov 		else
2187be8ef2cSDmytro Linkin 			tcf_idr_cleanup(tn, index);
2195e1567aeSJamal Hadi Salim 		return -EINVAL;
2205e1567aeSJamal Hadi Salim 	}
2215e1567aeSJamal Hadi Salim 
2225e1567aeSJamal Hadi Salim 	if (!exists) {
2237be8ef2cSDmytro Linkin 		ret = tcf_idr_create(tn, index, est, a,
22440bd094dSBaowen Zheng 				     &act_skbedit_ops, bind, true, act_flags);
2250190c1d4SVlad Buslov 		if (ret) {
2267be8ef2cSDmytro Linkin 			tcf_idr_cleanup(tn, index);
22786062033SWANG Cong 			return ret;
2280190c1d4SVlad Buslov 		}
229ca9b0e27SAlexander Duyck 
230a85a970aSWANG Cong 		d = to_skbedit(*a);
231ca9b0e27SAlexander Duyck 		ret = ACT_P_CREATED;
232ca9b0e27SAlexander Duyck 	} else {
233a85a970aSWANG Cong 		d = to_skbedit(*a);
234695176bfSCong Wang 		if (!(act_flags & TCA_ACT_FLAGS_REPLACE)) {
23565a206c0SChris Mi 			tcf_idr_release(*a, bind);
236ca9b0e27SAlexander Duyck 			return -EEXIST;
237ca9b0e27SAlexander Duyck 		}
2384e8ddd7fSVlad Buslov 	}
239ec7727bbSDavide Caratti 	err = tcf_action_check_ctrlact(parm->action, tp, &goto_ch, extack);
240ec7727bbSDavide Caratti 	if (err < 0)
241ec7727bbSDavide Caratti 		goto release_idr;
242ca9b0e27SAlexander Duyck 
243c749cddaSDavide Caratti 	params_new = kzalloc(sizeof(*params_new), GFP_KERNEL);
244c749cddaSDavide Caratti 	if (unlikely(!params_new)) {
245ec7727bbSDavide Caratti 		err = -ENOMEM;
246ec7727bbSDavide Caratti 		goto put_chain;
247c749cddaSDavide Caratti 	}
248c749cddaSDavide Caratti 
249c749cddaSDavide Caratti 	params_new->flags = flags;
250ca9b0e27SAlexander Duyck 	if (flags & SKBEDIT_F_PRIORITY)
251c749cddaSDavide Caratti 		params_new->priority = *priority;
25238a6f086STonghao Zhang 	if (flags & SKBEDIT_F_QUEUE_MAPPING) {
253c749cddaSDavide Caratti 		params_new->queue_mapping = *queue_mapping;
25438a6f086STonghao Zhang 		params_new->mapping_mod = mapping_mod;
25538a6f086STonghao Zhang 	}
2561c55d62eSjamal 	if (flags & SKBEDIT_F_MARK)
257c749cddaSDavide Caratti 		params_new->mark = *mark;
258ff202ee1SJamal Hadi Salim 	if (flags & SKBEDIT_F_PTYPE)
259c749cddaSDavide Caratti 		params_new->ptype = *ptype;
2604fe77d82SAntonio Quartulli 	/* default behaviour is to use all the bits */
261c749cddaSDavide Caratti 	params_new->mask = 0xffffffff;
2624fe77d82SAntonio Quartulli 	if (flags & SKBEDIT_F_MASK)
263c749cddaSDavide Caratti 		params_new->mask = *mask;
2641c55d62eSjamal 
2656d7a8df6SVlad Buslov 	spin_lock_bh(&d->tcf_lock);
266ec7727bbSDavide Caratti 	goto_ch = tcf_action_set_ctrlact(*a, parm->action, goto_ch);
267445d3749SPaul E. McKenney 	params_new = rcu_replace_pointer(d->params, params_new,
2686d7a8df6SVlad Buslov 					 lockdep_is_held(&d->tcf_lock));
2696d7a8df6SVlad Buslov 	spin_unlock_bh(&d->tcf_lock);
2706d7a8df6SVlad Buslov 	if (params_new)
2716d7a8df6SVlad Buslov 		kfree_rcu(params_new, rcu);
272ec7727bbSDavide Caratti 	if (goto_ch)
273ec7727bbSDavide Caratti 		tcf_chain_put_by_act(goto_ch);
274ca9b0e27SAlexander Duyck 
275ca9b0e27SAlexander Duyck 	return ret;
276ec7727bbSDavide Caratti put_chain:
277ec7727bbSDavide Caratti 	if (goto_ch)
278ec7727bbSDavide Caratti 		tcf_chain_put_by_act(goto_ch);
279ec7727bbSDavide Caratti release_idr:
280ec7727bbSDavide Caratti 	tcf_idr_release(*a, bind);
281ec7727bbSDavide Caratti 	return err;
282ca9b0e27SAlexander Duyck }
283ca9b0e27SAlexander Duyck 
tcf_skbedit_dump(struct sk_buff * skb,struct tc_action * a,int bind,int ref)284cc7ec456SEric Dumazet static int tcf_skbedit_dump(struct sk_buff *skb, struct tc_action *a,
285ca9b0e27SAlexander Duyck 			    int bind, int ref)
286ca9b0e27SAlexander Duyck {
287ca9b0e27SAlexander Duyck 	unsigned char *b = skb_tail_pointer(skb);
288a85a970aSWANG Cong 	struct tcf_skbedit *d = to_skbedit(a);
289c749cddaSDavide Caratti 	struct tcf_skbedit_params *params;
2901c40be12SEric Dumazet 	struct tc_skbedit opt = {
2911c40be12SEric Dumazet 		.index   = d->tcf_index,
292036bb443SVlad Buslov 		.refcnt  = refcount_read(&d->tcf_refcnt) - ref,
293036bb443SVlad Buslov 		.bindcnt = atomic_read(&d->tcf_bindcnt) - bind,
2941c40be12SEric Dumazet 	};
295e7e3728bSQiaobin Fu 	u64 pure_flags = 0;
296c749cddaSDavide Caratti 	struct tcf_t t;
297c749cddaSDavide Caratti 
2986d7a8df6SVlad Buslov 	spin_lock_bh(&d->tcf_lock);
2996d7a8df6SVlad Buslov 	params = rcu_dereference_protected(d->params,
3006d7a8df6SVlad Buslov 					   lockdep_is_held(&d->tcf_lock));
3016d7a8df6SVlad Buslov 	opt.action = d->tcf_action;
302ca9b0e27SAlexander Duyck 
3031b34ec43SDavid S. Miller 	if (nla_put(skb, TCA_SKBEDIT_PARMS, sizeof(opt), &opt))
3041b34ec43SDavid S. Miller 		goto nla_put_failure;
305c749cddaSDavide Caratti 	if ((params->flags & SKBEDIT_F_PRIORITY) &&
306c749cddaSDavide Caratti 	    nla_put_u32(skb, TCA_SKBEDIT_PRIORITY, params->priority))
3071b34ec43SDavid S. Miller 		goto nla_put_failure;
308c749cddaSDavide Caratti 	if ((params->flags & SKBEDIT_F_QUEUE_MAPPING) &&
309c749cddaSDavide Caratti 	    nla_put_u16(skb, TCA_SKBEDIT_QUEUE_MAPPING, params->queue_mapping))
3101b34ec43SDavid S. Miller 		goto nla_put_failure;
311c749cddaSDavide Caratti 	if ((params->flags & SKBEDIT_F_MARK) &&
312c749cddaSDavide Caratti 	    nla_put_u32(skb, TCA_SKBEDIT_MARK, params->mark))
3131b34ec43SDavid S. Miller 		goto nla_put_failure;
314c749cddaSDavide Caratti 	if ((params->flags & SKBEDIT_F_PTYPE) &&
315c749cddaSDavide Caratti 	    nla_put_u16(skb, TCA_SKBEDIT_PTYPE, params->ptype))
316ff202ee1SJamal Hadi Salim 		goto nla_put_failure;
317c749cddaSDavide Caratti 	if ((params->flags & SKBEDIT_F_MASK) &&
318c749cddaSDavide Caratti 	    nla_put_u32(skb, TCA_SKBEDIT_MASK, params->mask))
3194fe77d82SAntonio Quartulli 		goto nla_put_failure;
320c749cddaSDavide Caratti 	if (params->flags & SKBEDIT_F_INHERITDSFIELD)
321e7e3728bSQiaobin Fu 		pure_flags |= SKBEDIT_F_INHERITDSFIELD;
32238a6f086STonghao Zhang 	if (params->flags & SKBEDIT_F_TXQ_SKBHASH) {
32338a6f086STonghao Zhang 		if (nla_put_u16(skb, TCA_SKBEDIT_QUEUE_MAPPING_MAX,
32438a6f086STonghao Zhang 				params->queue_mapping + params->mapping_mod - 1))
32538a6f086STonghao Zhang 			goto nla_put_failure;
32638a6f086STonghao Zhang 
32738a6f086STonghao Zhang 		pure_flags |= SKBEDIT_F_TXQ_SKBHASH;
32838a6f086STonghao Zhang 	}
329e7e3728bSQiaobin Fu 	if (pure_flags != 0 &&
330e7e3728bSQiaobin Fu 	    nla_put(skb, TCA_SKBEDIT_FLAGS, sizeof(pure_flags), &pure_flags))
331e7e3728bSQiaobin Fu 		goto nla_put_failure;
33248d8ee16SJamal Hadi Salim 
33348d8ee16SJamal Hadi Salim 	tcf_tm_dump(&t, &d->tcf_tm);
3349854518eSNicolas Dichtel 	if (nla_put_64bit(skb, TCA_SKBEDIT_TM, sizeof(t), &t, TCA_SKBEDIT_PAD))
3351b34ec43SDavid S. Miller 		goto nla_put_failure;
3366d7a8df6SVlad Buslov 	spin_unlock_bh(&d->tcf_lock);
3376d7a8df6SVlad Buslov 
338ca9b0e27SAlexander Duyck 	return skb->len;
339ca9b0e27SAlexander Duyck 
340ca9b0e27SAlexander Duyck nla_put_failure:
3416d7a8df6SVlad Buslov 	spin_unlock_bh(&d->tcf_lock);
342ca9b0e27SAlexander Duyck 	nlmsg_trim(skb, b);
343ca9b0e27SAlexander Duyck 	return -1;
344ca9b0e27SAlexander Duyck }
345ca9b0e27SAlexander Duyck 
tcf_skbedit_cleanup(struct tc_action * a)346c749cddaSDavide Caratti static void tcf_skbedit_cleanup(struct tc_action *a)
347c749cddaSDavide Caratti {
348c749cddaSDavide Caratti 	struct tcf_skbedit *d = to_skbedit(a);
349c749cddaSDavide Caratti 	struct tcf_skbedit_params *params;
350c749cddaSDavide Caratti 
351c749cddaSDavide Caratti 	params = rcu_dereference_protected(d->params, 1);
352c749cddaSDavide Caratti 	if (params)
353c749cddaSDavide Caratti 		kfree_rcu(params, rcu);
354c749cddaSDavide Caratti }
355c749cddaSDavide Caratti 
tcf_skbedit_get_fill_size(const struct tc_action * act)356e1fea322SRoman Mashak static size_t tcf_skbedit_get_fill_size(const struct tc_action *act)
357e1fea322SRoman Mashak {
358e1fea322SRoman Mashak 	return nla_total_size(sizeof(struct tc_skbedit))
359e1fea322SRoman Mashak 		+ nla_total_size(sizeof(u32)) /* TCA_SKBEDIT_PRIORITY */
360e1fea322SRoman Mashak 		+ nla_total_size(sizeof(u16)) /* TCA_SKBEDIT_QUEUE_MAPPING */
36138a6f086STonghao Zhang 		+ nla_total_size(sizeof(u16)) /* TCA_SKBEDIT_QUEUE_MAPPING_MAX */
362e1fea322SRoman Mashak 		+ nla_total_size(sizeof(u32)) /* TCA_SKBEDIT_MARK */
363e1fea322SRoman Mashak 		+ nla_total_size(sizeof(u16)) /* TCA_SKBEDIT_PTYPE */
364e1fea322SRoman Mashak 		+ nla_total_size(sizeof(u32)) /* TCA_SKBEDIT_MASK */
365e1fea322SRoman Mashak 		+ nla_total_size_64bit(sizeof(u64)); /* TCA_SKBEDIT_FLAGS */
366e1fea322SRoman Mashak }
367e1fea322SRoman Mashak 
tcf_skbedit_offload_act_setup(struct tc_action * act,void * entry_data,u32 * index_inc,bool bind,struct netlink_ext_ack * extack)368c54e1d92SBaowen Zheng static int tcf_skbedit_offload_act_setup(struct tc_action *act, void *entry_data,
369c2ccf84eSIdo Schimmel 					 u32 *index_inc, bool bind,
370c2ccf84eSIdo Schimmel 					 struct netlink_ext_ack *extack)
371c54e1d92SBaowen Zheng {
372c54e1d92SBaowen Zheng 	if (bind) {
373c54e1d92SBaowen Zheng 		struct flow_action_entry *entry = entry_data;
374c54e1d92SBaowen Zheng 
375c54e1d92SBaowen Zheng 		if (is_tcf_skbedit_mark(act)) {
376c54e1d92SBaowen Zheng 			entry->id = FLOW_ACTION_MARK;
377c54e1d92SBaowen Zheng 			entry->mark = tcf_skbedit_mark(act);
378c54e1d92SBaowen Zheng 		} else if (is_tcf_skbedit_ptype(act)) {
379c54e1d92SBaowen Zheng 			entry->id = FLOW_ACTION_PTYPE;
380c54e1d92SBaowen Zheng 			entry->ptype = tcf_skbedit_ptype(act);
381c54e1d92SBaowen Zheng 		} else if (is_tcf_skbedit_priority(act)) {
382c54e1d92SBaowen Zheng 			entry->id = FLOW_ACTION_PRIORITY;
383c54e1d92SBaowen Zheng 			entry->priority = tcf_skbedit_priority(act);
3844a6a676fSAmritha Nambiar 		} else if (is_tcf_skbedit_tx_queue_mapping(act)) {
3854a6a676fSAmritha Nambiar 			NL_SET_ERR_MSG_MOD(extack, "Offload not supported when \"queue_mapping\" option is used on transmit side");
386a9c64939SIdo Schimmel 			return -EOPNOTSUPP;
3874a6a676fSAmritha Nambiar 		} else if (is_tcf_skbedit_rx_queue_mapping(act)) {
3884a6a676fSAmritha Nambiar 			entry->id = FLOW_ACTION_RX_QUEUE_MAPPING;
3894a6a676fSAmritha Nambiar 			entry->rx_queue = tcf_skbedit_rx_queue_mapping(act);
390a9c64939SIdo Schimmel 		} else if (is_tcf_skbedit_inheritdsfield(act)) {
391a9c64939SIdo Schimmel 			NL_SET_ERR_MSG_MOD(extack, "Offload not supported when \"inheritdsfield\" option is used");
392a9c64939SIdo Schimmel 			return -EOPNOTSUPP;
393c54e1d92SBaowen Zheng 		} else {
394a9c64939SIdo Schimmel 			NL_SET_ERR_MSG_MOD(extack, "Unsupported skbedit option offload");
395c54e1d92SBaowen Zheng 			return -EOPNOTSUPP;
396c54e1d92SBaowen Zheng 		}
397c54e1d92SBaowen Zheng 		*index_inc = 1;
398c54e1d92SBaowen Zheng 	} else {
3998cbfe939SBaowen Zheng 		struct flow_offload_action *fl_action = entry_data;
4008cbfe939SBaowen Zheng 
4018cbfe939SBaowen Zheng 		if (is_tcf_skbedit_mark(act))
4028cbfe939SBaowen Zheng 			fl_action->id = FLOW_ACTION_MARK;
4038cbfe939SBaowen Zheng 		else if (is_tcf_skbedit_ptype(act))
4048cbfe939SBaowen Zheng 			fl_action->id = FLOW_ACTION_PTYPE;
4058cbfe939SBaowen Zheng 		else if (is_tcf_skbedit_priority(act))
4068cbfe939SBaowen Zheng 			fl_action->id = FLOW_ACTION_PRIORITY;
4074a6a676fSAmritha Nambiar 		else if (is_tcf_skbedit_rx_queue_mapping(act))
4084a6a676fSAmritha Nambiar 			fl_action->id = FLOW_ACTION_RX_QUEUE_MAPPING;
4098cbfe939SBaowen Zheng 		else
410c54e1d92SBaowen Zheng 			return -EOPNOTSUPP;
411c54e1d92SBaowen Zheng 	}
412c54e1d92SBaowen Zheng 
413c54e1d92SBaowen Zheng 	return 0;
414c54e1d92SBaowen Zheng }
415c54e1d92SBaowen Zheng 
416ca9b0e27SAlexander Duyck static struct tc_action_ops act_skbedit_ops = {
417ca9b0e27SAlexander Duyck 	.kind		=	"skbedit",
418eddd2cf1SEli Cohen 	.id		=	TCA_ID_SKBEDIT,
419ca9b0e27SAlexander Duyck 	.owner		=	THIS_MODULE,
42045da1dacSJamal Hadi Salim 	.act		=	tcf_skbedit_act,
421837cb17dSPetr Machata 	.stats_update	=	tcf_skbedit_stats_update,
422ca9b0e27SAlexander Duyck 	.dump		=	tcf_skbedit_dump,
423ca9b0e27SAlexander Duyck 	.init		=	tcf_skbedit_init,
424c749cddaSDavide Caratti 	.cleanup	=	tcf_skbedit_cleanup,
425e1fea322SRoman Mashak 	.get_fill_size	=	tcf_skbedit_get_fill_size,
426c54e1d92SBaowen Zheng 	.offload_act_setup =	tcf_skbedit_offload_act_setup,
427a85a970aSWANG Cong 	.size		=	sizeof(struct tcf_skbedit),
428ddf97ccdSWANG Cong };
429ddf97ccdSWANG Cong 
skbedit_init_net(struct net * net)430ddf97ccdSWANG Cong static __net_init int skbedit_init_net(struct net *net)
431ddf97ccdSWANG Cong {
432acd0a7abSZhengchao Shao 	struct tc_action_net *tn = net_generic(net, act_skbedit_ops.net_id);
433ddf97ccdSWANG Cong 
434981471bdSCong Wang 	return tc_action_net_init(net, tn, &act_skbedit_ops);
435ddf97ccdSWANG Cong }
436ddf97ccdSWANG Cong 
skbedit_exit_net(struct list_head * net_list)437039af9c6SCong Wang static void __net_exit skbedit_exit_net(struct list_head *net_list)
438ddf97ccdSWANG Cong {
439acd0a7abSZhengchao Shao 	tc_action_net_exit(net_list, act_skbedit_ops.net_id);
440ddf97ccdSWANG Cong }
441ddf97ccdSWANG Cong 
442ddf97ccdSWANG Cong static struct pernet_operations skbedit_net_ops = {
443ddf97ccdSWANG Cong 	.init = skbedit_init_net,
444039af9c6SCong Wang 	.exit_batch = skbedit_exit_net,
445acd0a7abSZhengchao Shao 	.id   = &act_skbedit_ops.net_id,
446ddf97ccdSWANG Cong 	.size = sizeof(struct tc_action_net),
447ca9b0e27SAlexander Duyck };
448ca9b0e27SAlexander Duyck 
449ca9b0e27SAlexander Duyck MODULE_AUTHOR("Alexander Duyck, <alexander.h.duyck@intel.com>");
450ca9b0e27SAlexander Duyck MODULE_DESCRIPTION("SKB Editing");
451ca9b0e27SAlexander Duyck MODULE_LICENSE("GPL");
452ca9b0e27SAlexander Duyck 
skbedit_init_module(void)453ca9b0e27SAlexander Duyck static int __init skbedit_init_module(void)
454ca9b0e27SAlexander Duyck {
455ddf97ccdSWANG Cong 	return tcf_register_action(&act_skbedit_ops, &skbedit_net_ops);
456ca9b0e27SAlexander Duyck }
457ca9b0e27SAlexander Duyck 
skbedit_cleanup_module(void)458ca9b0e27SAlexander Duyck static void __exit skbedit_cleanup_module(void)
459ca9b0e27SAlexander Duyck {
460ddf97ccdSWANG Cong 	tcf_unregister_action(&act_skbedit_ops, &skbedit_net_ops);
461ca9b0e27SAlexander Duyck }
462ca9b0e27SAlexander Duyck 
463ca9b0e27SAlexander Duyck module_init(skbedit_init_module);
464ca9b0e27SAlexander Duyck module_exit(skbedit_cleanup_module);
465