12a2ea508SJohn Hurley // SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) 22a2ea508SJohn Hurley /* Copyright (C) 2019 Netronome Systems, Inc. */ 32a2ea508SJohn Hurley 4040b5cfbSMartin Varghese #include <linux/if_arp.h> 52a2ea508SJohn Hurley #include <linux/init.h> 62a2ea508SJohn Hurley #include <linux/kernel.h> 72a2ea508SJohn Hurley #include <linux/module.h> 82a2ea508SJohn Hurley #include <linux/mpls.h> 92a2ea508SJohn Hurley #include <linux/rtnetlink.h> 102a2ea508SJohn Hurley #include <linux/skbuff.h> 112a2ea508SJohn Hurley #include <linux/tc_act/tc_mpls.h> 122a2ea508SJohn Hurley #include <net/mpls.h> 132a2ea508SJohn Hurley #include <net/netlink.h> 142a2ea508SJohn Hurley #include <net/pkt_sched.h> 152a2ea508SJohn Hurley #include <net/pkt_cls.h> 162a2ea508SJohn Hurley #include <net/tc_act/tc_mpls.h> 17871cf386SPedro Tammela #include <net/tc_wrapper.h> 182a2ea508SJohn Hurley 192a2ea508SJohn Hurley static struct tc_action_ops act_mpls_ops; 202a2ea508SJohn Hurley 212a2ea508SJohn Hurley #define ACT_MPLS_TTL_DEFAULT 255 222a2ea508SJohn Hurley 232a2ea508SJohn Hurley static __be32 tcf_mpls_get_lse(struct mpls_shim_hdr *lse, 242a2ea508SJohn Hurley struct tcf_mpls_params *p, bool set_bos) 252a2ea508SJohn Hurley { 262a2ea508SJohn Hurley u32 new_lse = 0; 272a2ea508SJohn Hurley 282a2ea508SJohn Hurley if (lse) 292a2ea508SJohn Hurley new_lse = be32_to_cpu(lse->label_stack_entry); 302a2ea508SJohn Hurley 312a2ea508SJohn Hurley if (p->tcfm_label != ACT_MPLS_LABEL_NOT_SET) { 322a2ea508SJohn Hurley new_lse &= ~MPLS_LS_LABEL_MASK; 332a2ea508SJohn Hurley new_lse |= p->tcfm_label << MPLS_LS_LABEL_SHIFT; 342a2ea508SJohn Hurley } 352a2ea508SJohn Hurley if (p->tcfm_ttl) { 362a2ea508SJohn Hurley new_lse &= ~MPLS_LS_TTL_MASK; 372a2ea508SJohn Hurley new_lse |= p->tcfm_ttl << MPLS_LS_TTL_SHIFT; 382a2ea508SJohn Hurley } 392a2ea508SJohn Hurley if (p->tcfm_tc != ACT_MPLS_TC_NOT_SET) { 402a2ea508SJohn Hurley new_lse &= ~MPLS_LS_TC_MASK; 412a2ea508SJohn Hurley new_lse |= p->tcfm_tc << MPLS_LS_TC_SHIFT; 422a2ea508SJohn Hurley } 432a2ea508SJohn Hurley if (p->tcfm_bos != ACT_MPLS_BOS_NOT_SET) { 442a2ea508SJohn Hurley new_lse &= ~MPLS_LS_S_MASK; 452a2ea508SJohn Hurley new_lse |= p->tcfm_bos << MPLS_LS_S_SHIFT; 462a2ea508SJohn Hurley } else if (set_bos) { 472a2ea508SJohn Hurley new_lse |= 1 << MPLS_LS_S_SHIFT; 482a2ea508SJohn Hurley } 492a2ea508SJohn Hurley 502a2ea508SJohn Hurley return cpu_to_be32(new_lse); 512a2ea508SJohn Hurley } 522a2ea508SJohn Hurley 53871cf386SPedro Tammela TC_INDIRECT_SCOPE int tcf_mpls_act(struct sk_buff *skb, 54871cf386SPedro Tammela const struct tc_action *a, 552a2ea508SJohn Hurley struct tcf_result *res) 562a2ea508SJohn Hurley { 572a2ea508SJohn Hurley struct tcf_mpls *m = to_mpls(a); 582a2ea508SJohn Hurley struct tcf_mpls_params *p; 592a2ea508SJohn Hurley __be32 new_lse; 60fa4e0f88SDavide Caratti int ret, mac_len; 612a2ea508SJohn Hurley 622a2ea508SJohn Hurley tcf_lastuse_update(&m->tcf_tm); 6350dc9a85SAhmed S. Darwish bstats_update(this_cpu_ptr(m->common.cpu_bstats), skb); 642a2ea508SJohn Hurley 652a2ea508SJohn Hurley /* Ensure 'data' points at mac_header prior calling mpls manipulating 662a2ea508SJohn Hurley * functions. 672a2ea508SJohn Hurley */ 68fa4e0f88SDavide Caratti if (skb_at_tc_ingress(skb)) { 692a2ea508SJohn Hurley skb_push_rcsum(skb, skb->mac_len); 70fa4e0f88SDavide Caratti mac_len = skb->mac_len; 71fa4e0f88SDavide Caratti } else { 72*b3be9488SEric Dumazet mac_len = skb_network_offset(skb); 73fa4e0f88SDavide Caratti } 742a2ea508SJohn Hurley 752a2ea508SJohn Hurley ret = READ_ONCE(m->tcf_action); 762a2ea508SJohn Hurley 772a2ea508SJohn Hurley p = rcu_dereference_bh(m->mpls_p); 782a2ea508SJohn Hurley 792a2ea508SJohn Hurley switch (p->tcfm_action) { 802a2ea508SJohn Hurley case TCA_MPLS_ACT_POP: 81040b5cfbSMartin Varghese if (skb_mpls_pop(skb, p->tcfm_proto, mac_len, 82040b5cfbSMartin Varghese skb->dev && skb->dev->type == ARPHRD_ETHER)) 832a2ea508SJohn Hurley goto drop; 842a2ea508SJohn Hurley break; 852a2ea508SJohn Hurley case TCA_MPLS_ACT_PUSH: 86d7bf2ebeSToke Høiland-Jørgensen new_lse = tcf_mpls_get_lse(NULL, p, !eth_p_mpls(skb_protocol(skb, true))); 87d04ac224SMartin Varghese if (skb_mpls_push(skb, new_lse, p->tcfm_proto, mac_len, 88d04ac224SMartin Varghese skb->dev && skb->dev->type == ARPHRD_ETHER)) 892a2ea508SJohn Hurley goto drop; 902a2ea508SJohn Hurley break; 91a45294afSGuillaume Nault case TCA_MPLS_ACT_MAC_PUSH: 92a45294afSGuillaume Nault if (skb_vlan_tag_present(skb)) { 93a45294afSGuillaume Nault if (__vlan_insert_inner_tag(skb, skb->vlan_proto, 94a45294afSGuillaume Nault skb_vlan_tag_get(skb), 95a45294afSGuillaume Nault ETH_HLEN) < 0) 96a45294afSGuillaume Nault goto drop; 97a45294afSGuillaume Nault 98a45294afSGuillaume Nault skb->protocol = skb->vlan_proto; 99a45294afSGuillaume Nault __vlan_hwaccel_clear_tag(skb); 100a45294afSGuillaume Nault } 101a45294afSGuillaume Nault 102a45294afSGuillaume Nault new_lse = tcf_mpls_get_lse(NULL, p, mac_len || 103a45294afSGuillaume Nault !eth_p_mpls(skb->protocol)); 104a45294afSGuillaume Nault 105a45294afSGuillaume Nault if (skb_mpls_push(skb, new_lse, p->tcfm_proto, 0, false)) 106a45294afSGuillaume Nault goto drop; 107a45294afSGuillaume Nault break; 1082a2ea508SJohn Hurley case TCA_MPLS_ACT_MODIFY: 1099608fa65SDavide Caratti if (!pskb_may_pull(skb, 1109608fa65SDavide Caratti skb_network_offset(skb) + MPLS_HLEN)) 1119608fa65SDavide Caratti goto drop; 1122a2ea508SJohn Hurley new_lse = tcf_mpls_get_lse(mpls_hdr(skb), p, false); 1132a2ea508SJohn Hurley if (skb_mpls_update_lse(skb, new_lse)) 1142a2ea508SJohn Hurley goto drop; 1152a2ea508SJohn Hurley break; 1162a2ea508SJohn Hurley case TCA_MPLS_ACT_DEC_TTL: 1172a2ea508SJohn Hurley if (skb_mpls_dec_ttl(skb)) 1182a2ea508SJohn Hurley goto drop; 1192a2ea508SJohn Hurley break; 1202a2ea508SJohn Hurley } 1212a2ea508SJohn Hurley 1222a2ea508SJohn Hurley if (skb_at_tc_ingress(skb)) 1232a2ea508SJohn Hurley skb_pull_rcsum(skb, skb->mac_len); 1242a2ea508SJohn Hurley 1252a2ea508SJohn Hurley return ret; 1262a2ea508SJohn Hurley 1272a2ea508SJohn Hurley drop: 1282a2ea508SJohn Hurley qstats_drop_inc(this_cpu_ptr(m->common.cpu_qstats)); 1292a2ea508SJohn Hurley return TC_ACT_SHOT; 1302a2ea508SJohn Hurley } 1312a2ea508SJohn Hurley 1322a2ea508SJohn Hurley static int valid_label(const struct nlattr *attr, 1332a2ea508SJohn Hurley struct netlink_ext_ack *extack) 1342a2ea508SJohn Hurley { 1352a2ea508SJohn Hurley const u32 *label = nla_data(attr); 1362a2ea508SJohn Hurley 1379e17f992SIdo Schimmel if (nla_len(attr) != sizeof(*label)) { 1389e17f992SIdo Schimmel NL_SET_ERR_MSG_MOD(extack, "Invalid MPLS label length"); 1399e17f992SIdo Schimmel return -EINVAL; 1409e17f992SIdo Schimmel } 1419e17f992SIdo Schimmel 1422a2ea508SJohn Hurley if (*label & ~MPLS_LABEL_MASK || *label == MPLS_LABEL_IMPLNULL) { 1432a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "MPLS label out of range"); 1442a2ea508SJohn Hurley return -EINVAL; 1452a2ea508SJohn Hurley } 1462a2ea508SJohn Hurley 1472a2ea508SJohn Hurley return 0; 1482a2ea508SJohn Hurley } 1492a2ea508SJohn Hurley 1502a2ea508SJohn Hurley static const struct nla_policy mpls_policy[TCA_MPLS_MAX + 1] = { 1512a2ea508SJohn Hurley [TCA_MPLS_PARMS] = NLA_POLICY_EXACT_LEN(sizeof(struct tc_mpls)), 1522a2ea508SJohn Hurley [TCA_MPLS_PROTO] = { .type = NLA_U16 }, 1539e17f992SIdo Schimmel [TCA_MPLS_LABEL] = NLA_POLICY_VALIDATE_FN(NLA_BINARY, 1549e17f992SIdo Schimmel valid_label), 1552a2ea508SJohn Hurley [TCA_MPLS_TC] = NLA_POLICY_RANGE(NLA_U8, 0, 7), 1562a2ea508SJohn Hurley [TCA_MPLS_TTL] = NLA_POLICY_MIN(NLA_U8, 1), 1572a2ea508SJohn Hurley [TCA_MPLS_BOS] = NLA_POLICY_RANGE(NLA_U8, 0, 1), 1582a2ea508SJohn Hurley }; 1592a2ea508SJohn Hurley 1602a2ea508SJohn Hurley static int tcf_mpls_init(struct net *net, struct nlattr *nla, 1612a2ea508SJohn Hurley struct nlattr *est, struct tc_action **a, 162abbb0d33SVlad Buslov struct tcf_proto *tp, u32 flags, 163abbb0d33SVlad Buslov struct netlink_ext_ack *extack) 1642a2ea508SJohn Hurley { 165acd0a7abSZhengchao Shao struct tc_action_net *tn = net_generic(net, act_mpls_ops.net_id); 166695176bfSCong Wang bool bind = flags & TCA_ACT_FLAGS_BIND; 1672a2ea508SJohn Hurley struct nlattr *tb[TCA_MPLS_MAX + 1]; 1682a2ea508SJohn Hurley struct tcf_chain *goto_ch = NULL; 1692a2ea508SJohn Hurley struct tcf_mpls_params *p; 1702a2ea508SJohn Hurley struct tc_mpls *parm; 1712a2ea508SJohn Hurley bool exists = false; 1722a2ea508SJohn Hurley struct tcf_mpls *m; 1732a2ea508SJohn Hurley int ret = 0, err; 1742a2ea508SJohn Hurley u8 mpls_ttl = 0; 1757be8ef2cSDmytro Linkin u32 index; 1762a2ea508SJohn Hurley 1772a2ea508SJohn Hurley if (!nla) { 1782a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Missing netlink attributes"); 1792a2ea508SJohn Hurley return -EINVAL; 1802a2ea508SJohn Hurley } 1812a2ea508SJohn Hurley 1822a2ea508SJohn Hurley err = nla_parse_nested(tb, TCA_MPLS_MAX, nla, mpls_policy, extack); 1832a2ea508SJohn Hurley if (err < 0) 1842a2ea508SJohn Hurley return err; 1852a2ea508SJohn Hurley 1862a2ea508SJohn Hurley if (!tb[TCA_MPLS_PARMS]) { 1872a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "No MPLS params"); 1882a2ea508SJohn Hurley return -EINVAL; 1892a2ea508SJohn Hurley } 1902a2ea508SJohn Hurley parm = nla_data(tb[TCA_MPLS_PARMS]); 1917be8ef2cSDmytro Linkin index = parm->index; 1922a2ea508SJohn Hurley 193e88d78a7SPedro Tammela err = tcf_idr_check_alloc(tn, &index, a, bind); 194e88d78a7SPedro Tammela if (err < 0) 195e88d78a7SPedro Tammela return err; 196e88d78a7SPedro Tammela exists = err; 197e88d78a7SPedro Tammela if (exists && bind) 198e88d78a7SPedro Tammela return 0; 199e88d78a7SPedro Tammela 200e88d78a7SPedro Tammela if (!exists) { 201e88d78a7SPedro Tammela ret = tcf_idr_create(tn, index, est, a, &act_mpls_ops, bind, 202e88d78a7SPedro Tammela true, flags); 203e88d78a7SPedro Tammela if (ret) { 204e88d78a7SPedro Tammela tcf_idr_cleanup(tn, index); 205e88d78a7SPedro Tammela return ret; 206e88d78a7SPedro Tammela } 207e88d78a7SPedro Tammela 208e88d78a7SPedro Tammela ret = ACT_P_CREATED; 209e88d78a7SPedro Tammela } else if (!(flags & TCA_ACT_FLAGS_REPLACE)) { 210e88d78a7SPedro Tammela tcf_idr_release(*a, bind); 211e88d78a7SPedro Tammela return -EEXIST; 212e88d78a7SPedro Tammela } 213e88d78a7SPedro Tammela 2142a2ea508SJohn Hurley /* Verify parameters against action type. */ 2152a2ea508SJohn Hurley switch (parm->m_action) { 2162a2ea508SJohn Hurley case TCA_MPLS_ACT_POP: 2172a2ea508SJohn Hurley if (!tb[TCA_MPLS_PROTO]) { 2182a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Protocol must be set for MPLS pop"); 219e88d78a7SPedro Tammela err = -EINVAL; 220e88d78a7SPedro Tammela goto release_idr; 2212a2ea508SJohn Hurley } 2222a2ea508SJohn Hurley if (!eth_proto_is_802_3(nla_get_be16(tb[TCA_MPLS_PROTO]))) { 2232a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Invalid protocol type for MPLS pop"); 224e88d78a7SPedro Tammela err = -EINVAL; 225e88d78a7SPedro Tammela goto release_idr; 2262a2ea508SJohn Hurley } 2272a2ea508SJohn Hurley if (tb[TCA_MPLS_LABEL] || tb[TCA_MPLS_TTL] || tb[TCA_MPLS_TC] || 2282a2ea508SJohn Hurley tb[TCA_MPLS_BOS]) { 2292a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Label, TTL, TC or BOS cannot be used with MPLS pop"); 230e88d78a7SPedro Tammela err = -EINVAL; 231e88d78a7SPedro Tammela goto release_idr; 2322a2ea508SJohn Hurley } 2332a2ea508SJohn Hurley break; 2342a2ea508SJohn Hurley case TCA_MPLS_ACT_DEC_TTL: 2352a2ea508SJohn Hurley if (tb[TCA_MPLS_PROTO] || tb[TCA_MPLS_LABEL] || 2362a2ea508SJohn Hurley tb[TCA_MPLS_TTL] || tb[TCA_MPLS_TC] || tb[TCA_MPLS_BOS]) { 2372a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Label, TTL, TC, BOS or protocol cannot be used with MPLS dec_ttl"); 238e88d78a7SPedro Tammela err = -EINVAL; 239e88d78a7SPedro Tammela goto release_idr; 2402a2ea508SJohn Hurley } 2412a2ea508SJohn Hurley break; 2422a2ea508SJohn Hurley case TCA_MPLS_ACT_PUSH: 243a45294afSGuillaume Nault case TCA_MPLS_ACT_MAC_PUSH: 2442a2ea508SJohn Hurley if (!tb[TCA_MPLS_LABEL]) { 2452a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Label is required for MPLS push"); 246e88d78a7SPedro Tammela err = -EINVAL; 247e88d78a7SPedro Tammela goto release_idr; 2482a2ea508SJohn Hurley } 2492a2ea508SJohn Hurley if (tb[TCA_MPLS_PROTO] && 2502a2ea508SJohn Hurley !eth_p_mpls(nla_get_be16(tb[TCA_MPLS_PROTO]))) { 2512a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Protocol must be an MPLS type for MPLS push"); 252e88d78a7SPedro Tammela err = -EPROTONOSUPPORT; 253e88d78a7SPedro Tammela goto release_idr; 2542a2ea508SJohn Hurley } 2552a2ea508SJohn Hurley /* Push needs a TTL - if not specified, set a default value. */ 2562a2ea508SJohn Hurley if (!tb[TCA_MPLS_TTL]) { 2572a2ea508SJohn Hurley #if IS_ENABLED(CONFIG_MPLS) 2582a2ea508SJohn Hurley mpls_ttl = net->mpls.default_ttl ? 2592a2ea508SJohn Hurley net->mpls.default_ttl : ACT_MPLS_TTL_DEFAULT; 2602a2ea508SJohn Hurley #else 2612a2ea508SJohn Hurley mpls_ttl = ACT_MPLS_TTL_DEFAULT; 2622a2ea508SJohn Hurley #endif 2632a2ea508SJohn Hurley } 2642a2ea508SJohn Hurley break; 2652a2ea508SJohn Hurley case TCA_MPLS_ACT_MODIFY: 2662a2ea508SJohn Hurley if (tb[TCA_MPLS_PROTO]) { 2672a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Protocol cannot be used with MPLS modify"); 268e88d78a7SPedro Tammela err = -EINVAL; 269e88d78a7SPedro Tammela goto release_idr; 2702a2ea508SJohn Hurley } 2712a2ea508SJohn Hurley break; 2722a2ea508SJohn Hurley default: 2732a2ea508SJohn Hurley NL_SET_ERR_MSG_MOD(extack, "Unknown MPLS action"); 274e88d78a7SPedro Tammela err = -EINVAL; 275e88d78a7SPedro Tammela goto release_idr; 2762a2ea508SJohn Hurley } 2772a2ea508SJohn Hurley 2782a2ea508SJohn Hurley err = tcf_action_check_ctrlact(parm->action, tp, &goto_ch, extack); 2792a2ea508SJohn Hurley if (err < 0) 2802a2ea508SJohn Hurley goto release_idr; 2812a2ea508SJohn Hurley 2822a2ea508SJohn Hurley m = to_mpls(*a); 2832a2ea508SJohn Hurley 2842a2ea508SJohn Hurley p = kzalloc(sizeof(*p), GFP_KERNEL); 2852a2ea508SJohn Hurley if (!p) { 2862a2ea508SJohn Hurley err = -ENOMEM; 2872a2ea508SJohn Hurley goto put_chain; 2882a2ea508SJohn Hurley } 2892a2ea508SJohn Hurley 2902a2ea508SJohn Hurley p->tcfm_action = parm->m_action; 2912a2ea508SJohn Hurley p->tcfm_label = tb[TCA_MPLS_LABEL] ? nla_get_u32(tb[TCA_MPLS_LABEL]) : 2922a2ea508SJohn Hurley ACT_MPLS_LABEL_NOT_SET; 2932a2ea508SJohn Hurley p->tcfm_tc = tb[TCA_MPLS_TC] ? nla_get_u8(tb[TCA_MPLS_TC]) : 2942a2ea508SJohn Hurley ACT_MPLS_TC_NOT_SET; 2952a2ea508SJohn Hurley p->tcfm_ttl = tb[TCA_MPLS_TTL] ? nla_get_u8(tb[TCA_MPLS_TTL]) : 2962a2ea508SJohn Hurley mpls_ttl; 2972a2ea508SJohn Hurley p->tcfm_bos = tb[TCA_MPLS_BOS] ? nla_get_u8(tb[TCA_MPLS_BOS]) : 2982a2ea508SJohn Hurley ACT_MPLS_BOS_NOT_SET; 2992a2ea508SJohn Hurley p->tcfm_proto = tb[TCA_MPLS_PROTO] ? nla_get_be16(tb[TCA_MPLS_PROTO]) : 3002a2ea508SJohn Hurley htons(ETH_P_MPLS_UC); 3012a2ea508SJohn Hurley 3022a2ea508SJohn Hurley spin_lock_bh(&m->tcf_lock); 3032a2ea508SJohn Hurley goto_ch = tcf_action_set_ctrlact(*a, parm->action, goto_ch); 304445d3749SPaul E. McKenney p = rcu_replace_pointer(m->mpls_p, p, lockdep_is_held(&m->tcf_lock)); 3052a2ea508SJohn Hurley spin_unlock_bh(&m->tcf_lock); 3062a2ea508SJohn Hurley 3072a2ea508SJohn Hurley if (goto_ch) 3082a2ea508SJohn Hurley tcf_chain_put_by_act(goto_ch); 3092a2ea508SJohn Hurley if (p) 3102a2ea508SJohn Hurley kfree_rcu(p, rcu); 3112a2ea508SJohn Hurley 3122a2ea508SJohn Hurley return ret; 3132a2ea508SJohn Hurley put_chain: 3142a2ea508SJohn Hurley if (goto_ch) 3152a2ea508SJohn Hurley tcf_chain_put_by_act(goto_ch); 3162a2ea508SJohn Hurley release_idr: 3172a2ea508SJohn Hurley tcf_idr_release(*a, bind); 3182a2ea508SJohn Hurley return err; 3192a2ea508SJohn Hurley } 3202a2ea508SJohn Hurley 3212a2ea508SJohn Hurley static void tcf_mpls_cleanup(struct tc_action *a) 3222a2ea508SJohn Hurley { 3232a2ea508SJohn Hurley struct tcf_mpls *m = to_mpls(a); 3242a2ea508SJohn Hurley struct tcf_mpls_params *p; 3252a2ea508SJohn Hurley 3262a2ea508SJohn Hurley p = rcu_dereference_protected(m->mpls_p, 1); 3272a2ea508SJohn Hurley if (p) 3282a2ea508SJohn Hurley kfree_rcu(p, rcu); 3292a2ea508SJohn Hurley } 3302a2ea508SJohn Hurley 3312a2ea508SJohn Hurley static int tcf_mpls_dump(struct sk_buff *skb, struct tc_action *a, 3322a2ea508SJohn Hurley int bind, int ref) 3332a2ea508SJohn Hurley { 3342a2ea508SJohn Hurley unsigned char *b = skb_tail_pointer(skb); 3352a2ea508SJohn Hurley struct tcf_mpls *m = to_mpls(a); 3362a2ea508SJohn Hurley struct tcf_mpls_params *p; 3372a2ea508SJohn Hurley struct tc_mpls opt = { 3382a2ea508SJohn Hurley .index = m->tcf_index, 3392a2ea508SJohn Hurley .refcnt = refcount_read(&m->tcf_refcnt) - ref, 3402a2ea508SJohn Hurley .bindcnt = atomic_read(&m->tcf_bindcnt) - bind, 3412a2ea508SJohn Hurley }; 3422a2ea508SJohn Hurley struct tcf_t t; 3432a2ea508SJohn Hurley 3442a2ea508SJohn Hurley spin_lock_bh(&m->tcf_lock); 3452a2ea508SJohn Hurley opt.action = m->tcf_action; 3462a2ea508SJohn Hurley p = rcu_dereference_protected(m->mpls_p, lockdep_is_held(&m->tcf_lock)); 3472a2ea508SJohn Hurley opt.m_action = p->tcfm_action; 3482a2ea508SJohn Hurley 3492a2ea508SJohn Hurley if (nla_put(skb, TCA_MPLS_PARMS, sizeof(opt), &opt)) 3502a2ea508SJohn Hurley goto nla_put_failure; 3512a2ea508SJohn Hurley 3522a2ea508SJohn Hurley if (p->tcfm_label != ACT_MPLS_LABEL_NOT_SET && 3532a2ea508SJohn Hurley nla_put_u32(skb, TCA_MPLS_LABEL, p->tcfm_label)) 3542a2ea508SJohn Hurley goto nla_put_failure; 3552a2ea508SJohn Hurley 3562a2ea508SJohn Hurley if (p->tcfm_tc != ACT_MPLS_TC_NOT_SET && 3572a2ea508SJohn Hurley nla_put_u8(skb, TCA_MPLS_TC, p->tcfm_tc)) 3582a2ea508SJohn Hurley goto nla_put_failure; 3592a2ea508SJohn Hurley 3602a2ea508SJohn Hurley if (p->tcfm_ttl && nla_put_u8(skb, TCA_MPLS_TTL, p->tcfm_ttl)) 3612a2ea508SJohn Hurley goto nla_put_failure; 3622a2ea508SJohn Hurley 3632a2ea508SJohn Hurley if (p->tcfm_bos != ACT_MPLS_BOS_NOT_SET && 3642a2ea508SJohn Hurley nla_put_u8(skb, TCA_MPLS_BOS, p->tcfm_bos)) 3652a2ea508SJohn Hurley goto nla_put_failure; 3662a2ea508SJohn Hurley 3672a2ea508SJohn Hurley if (nla_put_be16(skb, TCA_MPLS_PROTO, p->tcfm_proto)) 3682a2ea508SJohn Hurley goto nla_put_failure; 3692a2ea508SJohn Hurley 3702a2ea508SJohn Hurley tcf_tm_dump(&t, &m->tcf_tm); 3712a2ea508SJohn Hurley 3722a2ea508SJohn Hurley if (nla_put_64bit(skb, TCA_MPLS_TM, sizeof(t), &t, TCA_MPLS_PAD)) 3732a2ea508SJohn Hurley goto nla_put_failure; 3742a2ea508SJohn Hurley 3752a2ea508SJohn Hurley spin_unlock_bh(&m->tcf_lock); 3762a2ea508SJohn Hurley 3772a2ea508SJohn Hurley return skb->len; 3782a2ea508SJohn Hurley 3792a2ea508SJohn Hurley nla_put_failure: 3802a2ea508SJohn Hurley spin_unlock_bh(&m->tcf_lock); 3812a2ea508SJohn Hurley nlmsg_trim(skb, b); 3822a2ea508SJohn Hurley return -EMSGSIZE; 3832a2ea508SJohn Hurley } 3842a2ea508SJohn Hurley 385c54e1d92SBaowen Zheng static int tcf_mpls_offload_act_setup(struct tc_action *act, void *entry_data, 386c2ccf84eSIdo Schimmel u32 *index_inc, bool bind, 387c2ccf84eSIdo Schimmel struct netlink_ext_ack *extack) 388c54e1d92SBaowen Zheng { 389c54e1d92SBaowen Zheng if (bind) { 390c54e1d92SBaowen Zheng struct flow_action_entry *entry = entry_data; 391c54e1d92SBaowen Zheng 392c54e1d92SBaowen Zheng switch (tcf_mpls_action(act)) { 393c54e1d92SBaowen Zheng case TCA_MPLS_ACT_PUSH: 394c54e1d92SBaowen Zheng entry->id = FLOW_ACTION_MPLS_PUSH; 395c54e1d92SBaowen Zheng entry->mpls_push.proto = tcf_mpls_proto(act); 396c54e1d92SBaowen Zheng entry->mpls_push.label = tcf_mpls_label(act); 397c54e1d92SBaowen Zheng entry->mpls_push.tc = tcf_mpls_tc(act); 398c54e1d92SBaowen Zheng entry->mpls_push.bos = tcf_mpls_bos(act); 399c54e1d92SBaowen Zheng entry->mpls_push.ttl = tcf_mpls_ttl(act); 400c54e1d92SBaowen Zheng break; 401c54e1d92SBaowen Zheng case TCA_MPLS_ACT_POP: 402c54e1d92SBaowen Zheng entry->id = FLOW_ACTION_MPLS_POP; 403c54e1d92SBaowen Zheng entry->mpls_pop.proto = tcf_mpls_proto(act); 404c54e1d92SBaowen Zheng break; 405c54e1d92SBaowen Zheng case TCA_MPLS_ACT_MODIFY: 406c54e1d92SBaowen Zheng entry->id = FLOW_ACTION_MPLS_MANGLE; 407c54e1d92SBaowen Zheng entry->mpls_mangle.label = tcf_mpls_label(act); 408c54e1d92SBaowen Zheng entry->mpls_mangle.tc = tcf_mpls_tc(act); 409c54e1d92SBaowen Zheng entry->mpls_mangle.bos = tcf_mpls_bos(act); 410c54e1d92SBaowen Zheng entry->mpls_mangle.ttl = tcf_mpls_ttl(act); 411c54e1d92SBaowen Zheng break; 412bca3821dSIdo Schimmel case TCA_MPLS_ACT_DEC_TTL: 413bca3821dSIdo Schimmel NL_SET_ERR_MSG_MOD(extack, "Offload not supported when \"dec_ttl\" option is used"); 414bca3821dSIdo Schimmel return -EOPNOTSUPP; 415bca3821dSIdo Schimmel case TCA_MPLS_ACT_MAC_PUSH: 416bca3821dSIdo Schimmel NL_SET_ERR_MSG_MOD(extack, "Offload not supported when \"mac_push\" option is used"); 417bca3821dSIdo Schimmel return -EOPNOTSUPP; 418c54e1d92SBaowen Zheng default: 419bca3821dSIdo Schimmel NL_SET_ERR_MSG_MOD(extack, "Unsupported MPLS mode offload"); 420c54e1d92SBaowen Zheng return -EOPNOTSUPP; 421c54e1d92SBaowen Zheng } 422c54e1d92SBaowen Zheng *index_inc = 1; 423c54e1d92SBaowen Zheng } else { 4248cbfe939SBaowen Zheng struct flow_offload_action *fl_action = entry_data; 4258cbfe939SBaowen Zheng 4268cbfe939SBaowen Zheng switch (tcf_mpls_action(act)) { 4278cbfe939SBaowen Zheng case TCA_MPLS_ACT_PUSH: 4288cbfe939SBaowen Zheng fl_action->id = FLOW_ACTION_MPLS_PUSH; 4298cbfe939SBaowen Zheng break; 4308cbfe939SBaowen Zheng case TCA_MPLS_ACT_POP: 4318cbfe939SBaowen Zheng fl_action->id = FLOW_ACTION_MPLS_POP; 4328cbfe939SBaowen Zheng break; 4338cbfe939SBaowen Zheng case TCA_MPLS_ACT_MODIFY: 4348cbfe939SBaowen Zheng fl_action->id = FLOW_ACTION_MPLS_MANGLE; 4358cbfe939SBaowen Zheng break; 4368cbfe939SBaowen Zheng default: 437c54e1d92SBaowen Zheng return -EOPNOTSUPP; 438c54e1d92SBaowen Zheng } 4398cbfe939SBaowen Zheng } 440c54e1d92SBaowen Zheng 441c54e1d92SBaowen Zheng return 0; 442c54e1d92SBaowen Zheng } 443c54e1d92SBaowen Zheng 4442a2ea508SJohn Hurley static struct tc_action_ops act_mpls_ops = { 4452a2ea508SJohn Hurley .kind = "mpls", 4462a2ea508SJohn Hurley .id = TCA_ID_MPLS, 4472a2ea508SJohn Hurley .owner = THIS_MODULE, 4482a2ea508SJohn Hurley .act = tcf_mpls_act, 4492a2ea508SJohn Hurley .dump = tcf_mpls_dump, 4502a2ea508SJohn Hurley .init = tcf_mpls_init, 4512a2ea508SJohn Hurley .cleanup = tcf_mpls_cleanup, 452c54e1d92SBaowen Zheng .offload_act_setup = tcf_mpls_offload_act_setup, 4532a2ea508SJohn Hurley .size = sizeof(struct tcf_mpls), 4542a2ea508SJohn Hurley }; 4552a2ea508SJohn Hurley 4562a2ea508SJohn Hurley static __net_init int mpls_init_net(struct net *net) 4572a2ea508SJohn Hurley { 458acd0a7abSZhengchao Shao struct tc_action_net *tn = net_generic(net, act_mpls_ops.net_id); 4592a2ea508SJohn Hurley 460981471bdSCong Wang return tc_action_net_init(net, tn, &act_mpls_ops); 4612a2ea508SJohn Hurley } 4622a2ea508SJohn Hurley 4632a2ea508SJohn Hurley static void __net_exit mpls_exit_net(struct list_head *net_list) 4642a2ea508SJohn Hurley { 465acd0a7abSZhengchao Shao tc_action_net_exit(net_list, act_mpls_ops.net_id); 4662a2ea508SJohn Hurley } 4672a2ea508SJohn Hurley 4682a2ea508SJohn Hurley static struct pernet_operations mpls_net_ops = { 4692a2ea508SJohn Hurley .init = mpls_init_net, 4702a2ea508SJohn Hurley .exit_batch = mpls_exit_net, 471acd0a7abSZhengchao Shao .id = &act_mpls_ops.net_id, 4722a2ea508SJohn Hurley .size = sizeof(struct tc_action_net), 4732a2ea508SJohn Hurley }; 4742a2ea508SJohn Hurley 4752a2ea508SJohn Hurley static int __init mpls_init_module(void) 4762a2ea508SJohn Hurley { 4772a2ea508SJohn Hurley return tcf_register_action(&act_mpls_ops, &mpls_net_ops); 4782a2ea508SJohn Hurley } 4792a2ea508SJohn Hurley 4802a2ea508SJohn Hurley static void __exit mpls_cleanup_module(void) 4812a2ea508SJohn Hurley { 4822a2ea508SJohn Hurley tcf_unregister_action(&act_mpls_ops, &mpls_net_ops); 4832a2ea508SJohn Hurley } 4842a2ea508SJohn Hurley 4852a2ea508SJohn Hurley module_init(mpls_init_module); 4862a2ea508SJohn Hurley module_exit(mpls_cleanup_module); 4872a2ea508SJohn Hurley 488501b72aeSGuillaume Nault MODULE_SOFTDEP("post: mpls_gso"); 4892a2ea508SJohn Hurley MODULE_AUTHOR("Netronome Systems <oss-drivers@netronome.com>"); 4902a2ea508SJohn Hurley MODULE_LICENSE("GPL"); 4912a2ea508SJohn Hurley MODULE_DESCRIPTION("MPLS manipulation actions"); 492