aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/skbuff.h5
-rw-r--r--net/core/skbuff.c19
-rw-r--r--net/openvswitch/actions.c5
-rw-r--r--net/sched/act_mpls.c12
4 files changed, 25 insertions, 16 deletions
diff --git a/include/linux/skbuff.h b/include/linux/skbuff.h
index 4351577b14d7..7914fdaf4226 100644
--- a/include/linux/skbuff.h
+++ b/include/linux/skbuff.h
@@ -3510,8 +3510,9 @@ int skb_ensure_writable(struct sk_buff *skb, int write_len);
3510int __skb_vlan_pop(struct sk_buff *skb, u16 *vlan_tci); 3510int __skb_vlan_pop(struct sk_buff *skb, u16 *vlan_tci);
3511int skb_vlan_pop(struct sk_buff *skb); 3511int skb_vlan_pop(struct sk_buff *skb);
3512int skb_vlan_push(struct sk_buff *skb, __be16 vlan_proto, u16 vlan_tci); 3512int skb_vlan_push(struct sk_buff *skb, __be16 vlan_proto, u16 vlan_tci);
3513int skb_mpls_push(struct sk_buff *skb, __be32 mpls_lse, __be16 mpls_proto); 3513int skb_mpls_push(struct sk_buff *skb, __be32 mpls_lse, __be16 mpls_proto,
3514int skb_mpls_pop(struct sk_buff *skb, __be16 next_proto); 3514 int mac_len);
3515int skb_mpls_pop(struct sk_buff *skb, __be16 next_proto, int mac_len);
3515int skb_mpls_update_lse(struct sk_buff *skb, __be32 mpls_lse); 3516int skb_mpls_update_lse(struct sk_buff *skb, __be32 mpls_lse);
3516int skb_mpls_dec_ttl(struct sk_buff *skb); 3517int skb_mpls_dec_ttl(struct sk_buff *skb);
3517struct sk_buff *pskb_extract(struct sk_buff *skb, int off, int to_copy, 3518struct sk_buff *pskb_extract(struct sk_buff *skb, int off, int to_copy,
diff --git a/net/core/skbuff.c b/net/core/skbuff.c
index 03b6809ebde4..867e61df00db 100644
--- a/net/core/skbuff.c
+++ b/net/core/skbuff.c
@@ -5477,12 +5477,14 @@ static void skb_mod_eth_type(struct sk_buff *skb, struct ethhdr *hdr,
5477 * @skb: buffer 5477 * @skb: buffer
5478 * @mpls_lse: MPLS label stack entry to push 5478 * @mpls_lse: MPLS label stack entry to push
5479 * @mpls_proto: ethertype of the new MPLS header (expects 0x8847 or 0x8848) 5479 * @mpls_proto: ethertype of the new MPLS header (expects 0x8847 or 0x8848)
5480 * @mac_len: length of the MAC header
5480 * 5481 *
5481 * Expects skb->data at mac header. 5482 * Expects skb->data at mac header.
5482 * 5483 *
5483 * Returns 0 on success, -errno otherwise. 5484 * Returns 0 on success, -errno otherwise.
5484 */ 5485 */
5485int skb_mpls_push(struct sk_buff *skb, __be32 mpls_lse, __be16 mpls_proto) 5486int skb_mpls_push(struct sk_buff *skb, __be32 mpls_lse, __be16 mpls_proto,
5487 int mac_len)
5486{ 5488{
5487 struct mpls_shim_hdr *lse; 5489 struct mpls_shim_hdr *lse;
5488 int err; 5490 int err;
@@ -5499,15 +5501,15 @@ int skb_mpls_push(struct sk_buff *skb, __be32 mpls_lse, __be16 mpls_proto)
5499 return err; 5501 return err;
5500 5502
5501 if (!skb->inner_protocol) { 5503 if (!skb->inner_protocol) {
5502 skb_set_inner_network_header(skb, skb->mac_len); 5504 skb_set_inner_network_header(skb, mac_len);
5503 skb_set_inner_protocol(skb, skb->protocol); 5505 skb_set_inner_protocol(skb, skb->protocol);
5504 } 5506 }
5505 5507
5506 skb_push(skb, MPLS_HLEN); 5508 skb_push(skb, MPLS_HLEN);
5507 memmove(skb_mac_header(skb) - MPLS_HLEN, skb_mac_header(skb), 5509 memmove(skb_mac_header(skb) - MPLS_HLEN, skb_mac_header(skb),
5508 skb->mac_len); 5510 mac_len);
5509 skb_reset_mac_header(skb); 5511 skb_reset_mac_header(skb);
5510 skb_set_network_header(skb, skb->mac_len); 5512 skb_set_network_header(skb, mac_len);
5511 5513
5512 lse = mpls_hdr(skb); 5514 lse = mpls_hdr(skb);
5513 lse->label_stack_entry = mpls_lse; 5515 lse->label_stack_entry = mpls_lse;
@@ -5526,29 +5528,30 @@ EXPORT_SYMBOL_GPL(skb_mpls_push);
5526 * 5528 *
5527 * @skb: buffer 5529 * @skb: buffer
5528 * @next_proto: ethertype of header after popped MPLS header 5530 * @next_proto: ethertype of header after popped MPLS header
5531 * @mac_len: length of the MAC header
5529 * 5532 *
5530 * Expects skb->data at mac header. 5533 * Expects skb->data at mac header.
5531 * 5534 *
5532 * Returns 0 on success, -errno otherwise. 5535 * Returns 0 on success, -errno otherwise.
5533 */ 5536 */
5534int skb_mpls_pop(struct sk_buff *skb, __be16 next_proto) 5537int skb_mpls_pop(struct sk_buff *skb, __be16 next_proto, int mac_len)
5535{ 5538{
5536 int err; 5539 int err;
5537 5540
5538 if (unlikely(!eth_p_mpls(skb->protocol))) 5541 if (unlikely(!eth_p_mpls(skb->protocol)))
5539 return 0; 5542 return 0;
5540 5543
5541 err = skb_ensure_writable(skb, skb->mac_len + MPLS_HLEN); 5544 err = skb_ensure_writable(skb, mac_len + MPLS_HLEN);
5542 if (unlikely(err)) 5545 if (unlikely(err))
5543 return err; 5546 return err;
5544 5547
5545 skb_postpull_rcsum(skb, mpls_hdr(skb), MPLS_HLEN); 5548 skb_postpull_rcsum(skb, mpls_hdr(skb), MPLS_HLEN);
5546 memmove(skb_mac_header(skb) + MPLS_HLEN, skb_mac_header(skb), 5549 memmove(skb_mac_header(skb) + MPLS_HLEN, skb_mac_header(skb),
5547 skb->mac_len); 5550 mac_len);
5548 5551
5549 __skb_pull(skb, MPLS_HLEN); 5552 __skb_pull(skb, MPLS_HLEN);
5550 skb_reset_mac_header(skb); 5553 skb_reset_mac_header(skb);
5551 skb_set_network_header(skb, skb->mac_len); 5554 skb_set_network_header(skb, mac_len);
5552 5555
5553 if (skb->dev && skb->dev->type == ARPHRD_ETHER) { 5556 if (skb->dev && skb->dev->type == ARPHRD_ETHER) {
5554 struct ethhdr *hdr; 5557 struct ethhdr *hdr;
diff --git a/net/openvswitch/actions.c b/net/openvswitch/actions.c
index 3572e11b6f21..1c77f520f474 100644
--- a/net/openvswitch/actions.c
+++ b/net/openvswitch/actions.c
@@ -165,7 +165,8 @@ static int push_mpls(struct sk_buff *skb, struct sw_flow_key *key,
165{ 165{
166 int err; 166 int err;
167 167
168 err = skb_mpls_push(skb, mpls->mpls_lse, mpls->mpls_ethertype); 168 err = skb_mpls_push(skb, mpls->mpls_lse, mpls->mpls_ethertype,
169 skb->mac_len);
169 if (err) 170 if (err)
170 return err; 171 return err;
171 172
@@ -178,7 +179,7 @@ static int pop_mpls(struct sk_buff *skb, struct sw_flow_key *key,
178{ 179{
179 int err; 180 int err;
180 181
181 err = skb_mpls_pop(skb, ethertype); 182 err = skb_mpls_pop(skb, ethertype, skb->mac_len);
182 if (err) 183 if (err)
183 return err; 184 return err;
184 185
diff --git a/net/sched/act_mpls.c b/net/sched/act_mpls.c
index e168df0e008a..4cf6c553bb0b 100644
--- a/net/sched/act_mpls.c
+++ b/net/sched/act_mpls.c
@@ -55,7 +55,7 @@ static int tcf_mpls_act(struct sk_buff *skb, const struct tc_action *a,
55 struct tcf_mpls *m = to_mpls(a); 55 struct tcf_mpls *m = to_mpls(a);
56 struct tcf_mpls_params *p; 56 struct tcf_mpls_params *p;
57 __be32 new_lse; 57 __be32 new_lse;
58 int ret; 58 int ret, mac_len;
59 59
60 tcf_lastuse_update(&m->tcf_tm); 60 tcf_lastuse_update(&m->tcf_tm);
61 bstats_cpu_update(this_cpu_ptr(m->common.cpu_bstats), skb); 61 bstats_cpu_update(this_cpu_ptr(m->common.cpu_bstats), skb);
@@ -63,8 +63,12 @@ static int tcf_mpls_act(struct sk_buff *skb, const struct tc_action *a,
63 /* Ensure 'data' points at mac_header prior calling mpls manipulating 63 /* Ensure 'data' points at mac_header prior calling mpls manipulating
64 * functions. 64 * functions.
65 */ 65 */
66 if (skb_at_tc_ingress(skb)) 66 if (skb_at_tc_ingress(skb)) {
67 skb_push_rcsum(skb, skb->mac_len); 67 skb_push_rcsum(skb, skb->mac_len);
68 mac_len = skb->mac_len;
69 } else {
70 mac_len = skb_network_header(skb) - skb_mac_header(skb);
71 }
68 72
69 ret = READ_ONCE(m->tcf_action); 73 ret = READ_ONCE(m->tcf_action);
70 74
@@ -72,12 +76,12 @@ static int tcf_mpls_act(struct sk_buff *skb, const struct tc_action *a,
72 76
73 switch (p->tcfm_action) { 77 switch (p->tcfm_action) {
74 case TCA_MPLS_ACT_POP: 78 case TCA_MPLS_ACT_POP:
75 if (skb_mpls_pop(skb, p->tcfm_proto)) 79 if (skb_mpls_pop(skb, p->tcfm_proto, mac_len))
76 goto drop; 80 goto drop;
77 break; 81 break;
78 case TCA_MPLS_ACT_PUSH: 82 case TCA_MPLS_ACT_PUSH:
79 new_lse = tcf_mpls_get_lse(NULL, p, !eth_p_mpls(skb->protocol)); 83 new_lse = tcf_mpls_get_lse(NULL, p, !eth_p_mpls(skb->protocol));
80 if (skb_mpls_push(skb, new_lse, p->tcfm_proto)) 84 if (skb_mpls_push(skb, new_lse, p->tcfm_proto, mac_len))
81 goto drop; 85 goto drop;
82 break; 86 break;
83 case TCA_MPLS_ACT_MODIFY: 87 case TCA_MPLS_ACT_MODIFY: