|
|
@@ -260,7 +260,6 @@ static int erspan_rcv(struct sk_buff *skb, struct tnl_ptk_info *tpi,
|
|
|
struct net *net = dev_net(skb->dev);
|
|
|
struct metadata_dst *tun_dst = NULL;
|
|
|
struct erspan_base_hdr *ershdr;
|
|
|
- struct erspan_metadata *pkt_md;
|
|
|
struct ip_tunnel_net *itn;
|
|
|
struct ip_tunnel *tunnel;
|
|
|
const struct iphdr *iph;
|
|
|
@@ -283,9 +282,6 @@ static int erspan_rcv(struct sk_buff *skb, struct tnl_ptk_info *tpi,
|
|
|
if (unlikely(!pskb_may_pull(skb, len)))
|
|
|
return PACKET_REJECT;
|
|
|
|
|
|
- ershdr = (struct erspan_base_hdr *)(skb->data + gre_hdr_len);
|
|
|
- pkt_md = (struct erspan_metadata *)(ershdr + 1);
|
|
|
-
|
|
|
if (__iptunnel_pull_header(skb,
|
|
|
len,
|
|
|
htons(ETH_P_TEB),
|
|
|
@@ -293,8 +289,9 @@ static int erspan_rcv(struct sk_buff *skb, struct tnl_ptk_info *tpi,
|
|
|
goto drop;
|
|
|
|
|
|
if (tunnel->collect_md) {
|
|
|
+ struct erspan_metadata *pkt_md, *md;
|
|
|
struct ip_tunnel_info *info;
|
|
|
- struct erspan_metadata *md;
|
|
|
+ unsigned char *gh;
|
|
|
__be64 tun_id;
|
|
|
__be16 flags;
|
|
|
|
|
|
@@ -307,6 +304,14 @@ static int erspan_rcv(struct sk_buff *skb, struct tnl_ptk_info *tpi,
|
|
|
if (!tun_dst)
|
|
|
return PACKET_REJECT;
|
|
|
|
|
|
+ /* skb can be uncloned in __iptunnel_pull_header, so
|
|
|
+ * old pkt_md is no longer valid and we need to reset
|
|
|
+ * it
|
|
|
+ */
|
|
|
+ gh = skb_network_header(skb) +
|
|
|
+ skb_network_header_len(skb);
|
|
|
+ pkt_md = (struct erspan_metadata *)(gh + gre_hdr_len +
|
|
|
+ sizeof(*ershdr));
|
|
|
md = ip_tunnel_info_opts(&tun_dst->u.tun_info);
|
|
|
md->version = ver;
|
|
|
md2 = &md->u.md2;
|