#include <net/inet_ecn.h>
 #include <net/net_namespace.h>
 #include <net/netns/generic.h>
+#include <net/tun_proto.h>
 #include <net/vxlan.h>
 
 #if IS_ENABLED(CONFIG_IPV6)
        if (gpe->oam_flag)
                return false;
 
-       switch (gpe->next_protocol) {
-       case VXLAN_GPE_NP_IPV4:
-               *protocol = htons(ETH_P_IP);
-               break;
-       case VXLAN_GPE_NP_IPV6:
-               *protocol = htons(ETH_P_IPV6);
-               break;
-       case VXLAN_GPE_NP_ETHERNET:
-               *protocol = htons(ETH_P_TEB);
-               break;
-       default:
+       *protocol = tun_p_to_eth_p(gpe->next_protocol);
+       if (!*protocol)
                return false;
-       }
 
        unparsed->vx_flags &= ~VXLAN_GPE_USED_BITS;
        return true;
        struct vxlanhdr_gpe *gpe = (struct vxlanhdr_gpe *)vxh;
 
        gpe->np_applied = 1;
-
-       switch (protocol) {
-       case htons(ETH_P_IP):
-               gpe->next_protocol = VXLAN_GPE_NP_IPV4;
-               return 0;
-       case htons(ETH_P_IPV6):
-               gpe->next_protocol = VXLAN_GPE_NP_IPV6;
-               return 0;
-       case htons(ETH_P_TEB):
-               gpe->next_protocol = VXLAN_GPE_NP_ETHERNET;
-               return 0;
-       }
-       return -EPFNOSUPPORT;
+       gpe->next_protocol = tun_p_from_eth_p(protocol);
+       if (!gpe->next_protocol)
+               return -EPFNOSUPPORT;
+       return 0;
 }
 
 static int vxlan_build_skb(struct sk_buff *skb, struct dst_entry *dst,
 
--- /dev/null
+#ifndef __NET_TUN_PROTO_H
+#define __NET_TUN_PROTO_H
+
+#include <linux/kernel.h>
+
+/* One byte protocol values as defined by VXLAN-GPE and NSH. These will
+ * hopefully get a shared IANA registry.
+ */
+#define TUN_P_IPV4      0x01
+#define TUN_P_IPV6      0x02
+#define TUN_P_ETHERNET  0x03
+#define TUN_P_NSH       0x04
+#define TUN_P_MPLS_UC   0x05
+
+static inline __be16 tun_p_to_eth_p(u8 proto)
+{
+       switch (proto) {
+       case TUN_P_IPV4:
+               return htons(ETH_P_IP);
+       case TUN_P_IPV6:
+               return htons(ETH_P_IPV6);
+       case TUN_P_ETHERNET:
+               return htons(ETH_P_TEB);
+       case TUN_P_NSH:
+               return htons(ETH_P_NSH);
+       case TUN_P_MPLS_UC:
+               return htons(ETH_P_MPLS_UC);
+       }
+       return 0;
+}
+
+static inline u8 tun_p_from_eth_p(__be16 proto)
+{
+       switch (proto) {
+       case htons(ETH_P_IP):
+               return TUN_P_IPV4;
+       case htons(ETH_P_IPV6):
+               return TUN_P_IPV6;
+       case htons(ETH_P_TEB):
+               return TUN_P_ETHERNET;
+       case htons(ETH_P_NSH):
+               return TUN_P_NSH;
+       case htons(ETH_P_MPLS_UC):
+               return TUN_P_MPLS_UC;
+       }
+       return 0;
+}
+
+#endif
 
 #define VXLAN_GPE_USED_BITS (VXLAN_HF_VER | VXLAN_HF_NP | VXLAN_HF_OAM | \
                             cpu_to_be32(0xff))
 
-/* VXLAN-GPE header Next Protocol. */
-#define VXLAN_GPE_NP_IPV4      0x01
-#define VXLAN_GPE_NP_IPV6      0x02
-#define VXLAN_GPE_NP_ETHERNET  0x03
-#define VXLAN_GPE_NP_NSH       0x04
-
 struct vxlan_metadata {
        u32             gbp;
 };