[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20240731172332.683815-8-tom@herbertland.com>
Date: Wed, 31 Jul 2024 10:23:27 -0700
From: Tom Herbert <tom@...bertland.com>
To: davem@...emloft.net,
kuba@...nel.org,
edumazet@...gle.com,
netdev@...r.kernel.org,
felipe@...anda.io
Cc: Tom Herbert <tom@...bertland.com>
Subject: [PATCH 07/12] flow_dissector: Parse vxlan in UDP
Parse vxlan in a UDP encapsulation
Signed-off-by: Tom Herbert <tom@...bertland.com>
---
net/core/flow_dissector.c | 57 +++++++++++++++++++++++++++++++++++++++
1 file changed, 57 insertions(+)
diff --git a/net/core/flow_dissector.c b/net/core/flow_dissector.c
index 006db3b893d0..6ad45b09dda4 100644
--- a/net/core/flow_dissector.c
+++ b/net/core/flow_dissector.c
@@ -13,7 +13,9 @@
#include <net/gre.h>
#include <net/pptp.h>
#include <net/tipc.h>
+#include <net/tun_proto.h>
#include <net/udp.h>
+#include <net/vxlan.h>
#include <linux/igmp.h>
#include <linux/icmp.h>
#include <linux/sctp.h>
@@ -756,6 +758,55 @@ __skb_flow_dissect_gre(const struct sk_buff *skb,
return FLOW_DISSECT_RET_PROTO_AGAIN;
}
+static enum flow_dissect_ret
+__skb_flow_dissect_vxlan(const struct sk_buff *skb,
+ struct flow_dissector *flow_dissector,
+ void *target_container, const void *data,
+ __be16 *p_proto, int *p_nhoff, int hlen,
+ unsigned int flags)
+{
+ struct vxlanhdr *hdr, _hdr;
+ __be16 protocol;
+
+ hdr = __skb_header_pointer(skb, *p_nhoff, sizeof(_hdr), data, hlen,
+ &_hdr);
+ if (!hdr)
+ return FLOW_DISSECT_RET_OUT_BAD;
+
+ /* VNI flag always required to be set */
+ if (!(hdr->vx_flags & VXLAN_HF_VNI))
+ return FLOW_DISSECT_RET_OUT_BAD;
+
+ if (hdr->vx_flags & VXLAN_F_GPE) {
+ struct vxlanhdr_gpe *gpe = (struct vxlanhdr_gpe *)hdr;
+
+ /* Need to have Next Protocol set for interfaces in GPE mode. */
+ if (!gpe->np_applied)
+ return FLOW_DISSECT_RET_OUT_BAD;
+
+ /* The initial version is 0 */
+ if (gpe->version != 0)
+ return FLOW_DISSECT_RET_OUT_GOOD;
+
+ /* "When the O bit is set to 1, the packet is an OAM packet and
+ * OAM so ignore
+ */
+ if (gpe->oam_flag)
+ return FLOW_DISSECT_RET_OUT_GOOD;
+
+ protocol = tun_p_to_eth_p(gpe->next_protocol);
+ if (!protocol)
+ return FLOW_DISSECT_RET_OUT_GOOD;
+ } else {
+ protocol = htons(ETH_P_TEB);
+ }
+
+ *p_nhoff += sizeof(struct vxlanhdr);
+ *p_proto = protocol;
+
+ return FLOW_DISSECT_RET_PROTO_AGAIN;
+}
+
/**
* __skb_flow_dissect_batadv() - dissect batman-adv header
* @skb: sk_buff to with the batman-adv header
@@ -893,6 +944,12 @@ __skb_flow_dissect_udp(const struct sk_buff *skb, struct net *net,
ret = FLOW_DISSECT_RET_OUT_GOOD;
switch (encap_type) {
+ case UDP_ENCAP_VXLAN:
+ case UDP_ENCAP_VXLAN_GPE:
+ ret = __skb_flow_dissect_vxlan(skb, flow_dissector,
+ target_container, data,
+ p_proto, &nhoff, hlen, flags);
+ break;
default:
break;
}
--
2.34.1
Powered by blists - more mailing lists