2012-10-01 19:32:35 +07:00
|
|
|
/*
|
2012-11-13 20:29:15 +07:00
|
|
|
* VXLAN: Virtual eXtensible Local Area Network
|
2012-10-01 19:32:35 +07:00
|
|
|
*
|
2013-04-27 18:31:52 +07:00
|
|
|
* Copyright (c) 2012-2013 Vyatta Inc.
|
2012-10-01 19:32:35 +07:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/skbuff.h>
|
|
|
|
#include <linux/rculist.h>
|
|
|
|
#include <linux/netdevice.h>
|
|
|
|
#include <linux/in.h>
|
|
|
|
#include <linux/ip.h>
|
|
|
|
#include <linux/udp.h>
|
|
|
|
#include <linux/igmp.h>
|
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/if_ether.h>
|
2013-08-20 01:23:29 +07:00
|
|
|
#include <linux/if_vlan.h>
|
2012-10-01 19:32:35 +07:00
|
|
|
#include <linux/hash.h>
|
2013-01-30 06:43:07 +07:00
|
|
|
#include <linux/ethtool.h>
|
2012-11-20 09:50:14 +07:00
|
|
|
#include <net/arp.h>
|
|
|
|
#include <net/ndisc.h>
|
2012-10-01 19:32:35 +07:00
|
|
|
#include <net/ip.h>
|
2013-03-25 21:49:35 +07:00
|
|
|
#include <net/ip_tunnels.h>
|
2012-10-01 19:32:35 +07:00
|
|
|
#include <net/icmp.h>
|
|
|
|
#include <net/udp.h>
|
2014-07-14 09:49:42 +07:00
|
|
|
#include <net/udp_tunnel.h>
|
2012-10-01 19:32:35 +07:00
|
|
|
#include <net/rtnetlink.h>
|
|
|
|
#include <net/route.h>
|
|
|
|
#include <net/dsfield.h>
|
|
|
|
#include <net/inet_ecn.h>
|
|
|
|
#include <net/net_namespace.h>
|
|
|
|
#include <net/netns/generic.h>
|
2013-08-20 01:23:07 +07:00
|
|
|
#include <net/vxlan.h>
|
2014-01-20 18:59:21 +07:00
|
|
|
#include <net/protocol.h>
|
2014-09-17 07:31:18 +07:00
|
|
|
#include <net/udp_tunnel.h>
|
2013-08-31 12:44:33 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
#include <net/ipv6.h>
|
|
|
|
#include <net/addrconf.h>
|
|
|
|
#include <net/ip6_tunnel.h>
|
2013-09-02 09:06:52 +07:00
|
|
|
#include <net/ip6_checksum.h>
|
2013-08-31 12:44:33 +07:00
|
|
|
#endif
|
2015-07-21 15:43:58 +07:00
|
|
|
#include <net/dst_metadata.h>
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
#define VXLAN_VERSION "0.1"
|
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
#define PORT_HASH_BITS 8
|
|
|
|
#define PORT_HASH_SIZE (1<<PORT_HASH_BITS)
|
2012-10-01 19:32:35 +07:00
|
|
|
#define FDB_AGE_DEFAULT 300 /* 5 min */
|
|
|
|
#define FDB_AGE_INTERVAL (10 * HZ) /* rescan interval */
|
|
|
|
|
2013-04-27 18:31:53 +07:00
|
|
|
/* UDP port for VXLAN traffic.
|
|
|
|
* The IANA assigned port is 4789, but the Linux default is 8472
|
2013-06-18 04:16:41 +07:00
|
|
|
* for compatibility with early adopters.
|
2013-04-27 18:31:53 +07:00
|
|
|
*/
|
2013-06-18 04:16:12 +07:00
|
|
|
static unsigned short vxlan_port __read_mostly = 8472;
|
|
|
|
module_param_named(udp_port, vxlan_port, ushort, 0444);
|
2012-10-01 19:32:35 +07:00
|
|
|
MODULE_PARM_DESC(udp_port, "Destination UDP port");
|
|
|
|
|
|
|
|
static bool log_ecn_error = true;
|
|
|
|
module_param(log_ecn_error, bool, 0644);
|
|
|
|
MODULE_PARM_DESC(log_ecn_error, "Log packets received with corrupted ECN");
|
|
|
|
|
2013-06-20 14:26:31 +07:00
|
|
|
static int vxlan_net_id;
|
2015-07-21 15:44:02 +07:00
|
|
|
static struct rtnl_link_ops vxlan_link_ops;
|
2013-05-16 18:35:20 +07:00
|
|
|
|
2013-06-25 20:01:51 +07:00
|
|
|
static const u8 all_zeros_mac[ETH_ALEN];
|
|
|
|
|
2015-09-24 18:50:01 +07:00
|
|
|
static int vxlan_sock_add(struct vxlan_dev *vxlan);
|
2015-07-21 15:44:06 +07:00
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
/* per-network namespace private data for this module */
|
|
|
|
struct vxlan_net {
|
|
|
|
struct list_head vxlan_list;
|
|
|
|
struct hlist_head sock_list[PORT_HASH_SIZE];
|
2013-06-18 04:16:11 +07:00
|
|
|
spinlock_t sock_lock;
|
2013-05-16 18:35:20 +07:00
|
|
|
};
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Forwarding table entry */
|
|
|
|
struct vxlan_fdb {
|
|
|
|
struct hlist_node hlist; /* linked list of entries */
|
|
|
|
struct rcu_head rcu;
|
|
|
|
unsigned long updated; /* jiffies */
|
|
|
|
unsigned long used;
|
2013-06-18 04:16:12 +07:00
|
|
|
struct list_head remotes;
|
2015-07-20 14:54:50 +07:00
|
|
|
u8 eth_addr[ETH_ALEN];
|
2012-10-01 19:32:35 +07:00
|
|
|
u16 state; /* see ndm_state */
|
2013-04-19 07:36:26 +07:00
|
|
|
u8 flags; /* see ndm_flags */
|
2012-10-01 19:32:35 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
/* salt for hash table */
|
|
|
|
static u32 vxlan_salt __read_mostly;
|
2013-06-18 04:16:09 +07:00
|
|
|
static struct workqueue_struct *vxlan_wq;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:43:58 +07:00
|
|
|
static inline bool vxlan_collect_metadata(struct vxlan_sock *vs)
|
|
|
|
{
|
2015-07-21 15:44:01 +07:00
|
|
|
return vs->flags & VXLAN_F_COLLECT_METADATA ||
|
|
|
|
ip_tunnel_collect_metadata();
|
2015-07-21 15:43:58 +07:00
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
static inline
|
|
|
|
bool vxlan_addr_equal(const union vxlan_addr *a, const union vxlan_addr *b)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
if (a->sa.sa_family != b->sa.sa_family)
|
|
|
|
return false;
|
|
|
|
if (a->sa.sa_family == AF_INET6)
|
|
|
|
return ipv6_addr_equal(&a->sin6.sin6_addr, &b->sin6.sin6_addr);
|
|
|
|
else
|
|
|
|
return a->sin.sin_addr.s_addr == b->sin.sin_addr.s_addr;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool vxlan_addr_any(const union vxlan_addr *ipa)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
if (ipa->sa.sa_family == AF_INET6)
|
|
|
|
return ipv6_addr_any(&ipa->sin6.sin6_addr);
|
|
|
|
else
|
|
|
|
return ipa->sin.sin_addr.s_addr == htonl(INADDR_ANY);
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool vxlan_addr_multicast(const union vxlan_addr *ipa)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
if (ipa->sa.sa_family == AF_INET6)
|
|
|
|
return ipv6_addr_is_multicast(&ipa->sin6.sin6_addr);
|
|
|
|
else
|
|
|
|
return IN_MULTICAST(ntohl(ipa->sin.sin_addr.s_addr));
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_nla_get_addr(union vxlan_addr *ip, struct nlattr *nla)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
if (nla_len(nla) >= sizeof(struct in6_addr)) {
|
2015-03-29 21:59:26 +07:00
|
|
|
ip->sin6.sin6_addr = nla_get_in6_addr(nla);
|
2015-03-29 21:17:37 +07:00
|
|
|
ip->sa.sa_family = AF_INET6;
|
|
|
|
return 0;
|
|
|
|
} else if (nla_len(nla) >= sizeof(__be32)) {
|
2015-03-29 21:59:26 +07:00
|
|
|
ip->sin.sin_addr.s_addr = nla_get_in_addr(nla);
|
2015-03-29 21:17:37 +07:00
|
|
|
ip->sa.sa_family = AF_INET;
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return -EAFNOSUPPORT;
|
|
|
|
}
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_nla_put_addr(struct sk_buff *skb, int attr,
|
2015-03-29 21:17:37 +07:00
|
|
|
const union vxlan_addr *ip)
|
2013-08-31 12:44:33 +07:00
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
if (ip->sa.sa_family == AF_INET6)
|
2015-03-29 21:59:25 +07:00
|
|
|
return nla_put_in6_addr(skb, attr, &ip->sin6.sin6_addr);
|
2015-03-29 21:17:37 +07:00
|
|
|
else
|
2015-03-29 21:59:25 +07:00
|
|
|
return nla_put_in_addr(skb, attr, ip->sin.sin_addr.s_addr);
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
#else /* !CONFIG_IPV6 */
|
|
|
|
|
|
|
|
static inline
|
|
|
|
bool vxlan_addr_equal(const union vxlan_addr *a, const union vxlan_addr *b)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
return a->sin.sin_addr.s_addr == b->sin.sin_addr.s_addr;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool vxlan_addr_any(const union vxlan_addr *ipa)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
return ipa->sin.sin_addr.s_addr == htonl(INADDR_ANY);
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool vxlan_addr_multicast(const union vxlan_addr *ipa)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
return IN_MULTICAST(ntohl(ipa->sin.sin_addr.s_addr));
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_nla_get_addr(union vxlan_addr *ip, struct nlattr *nla)
|
|
|
|
{
|
2015-03-29 21:17:37 +07:00
|
|
|
if (nla_len(nla) >= sizeof(struct in6_addr)) {
|
|
|
|
return -EAFNOSUPPORT;
|
|
|
|
} else if (nla_len(nla) >= sizeof(__be32)) {
|
2015-03-29 21:59:26 +07:00
|
|
|
ip->sin.sin_addr.s_addr = nla_get_in_addr(nla);
|
2015-03-29 21:17:37 +07:00
|
|
|
ip->sa.sa_family = AF_INET;
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return -EAFNOSUPPORT;
|
|
|
|
}
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_nla_put_addr(struct sk_buff *skb, int attr,
|
2015-03-29 21:17:37 +07:00
|
|
|
const union vxlan_addr *ip)
|
2013-08-31 12:44:33 +07:00
|
|
|
{
|
2015-03-29 21:59:25 +07:00
|
|
|
return nla_put_in_addr(skb, attr, ip->sin.sin_addr.s_addr);
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
/* Virtual Network hash table head */
|
|
|
|
static inline struct hlist_head *vni_head(struct vxlan_sock *vs, u32 id)
|
|
|
|
{
|
|
|
|
return &vs->vni_list[hash_32(id, VNI_HASH_BITS)];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Socket hash table head */
|
|
|
|
static inline struct hlist_head *vs_head(struct net *net, __be16 port)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
struct vxlan_net *vn = net_generic(net, vxlan_net_id);
|
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
return &vn->sock_list[hash_32(ntohs(port), PORT_HASH_BITS)];
|
|
|
|
}
|
|
|
|
|
2013-06-18 04:16:12 +07:00
|
|
|
/* First remote destination for a forwarding entry.
|
|
|
|
* Guaranteed to be non-NULL because remotes are never deleted.
|
|
|
|
*/
|
2013-08-05 07:17:39 +07:00
|
|
|
static inline struct vxlan_rdst *first_remote_rcu(struct vxlan_fdb *fdb)
|
2013-06-18 04:16:12 +07:00
|
|
|
{
|
2013-08-05 07:17:39 +07:00
|
|
|
return list_entry_rcu(fdb->remotes.next, struct vxlan_rdst, list);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct vxlan_rdst *first_remote_rtnl(struct vxlan_fdb *fdb)
|
|
|
|
{
|
|
|
|
return list_first_entry(&fdb->remotes, struct vxlan_rdst, list);
|
2013-06-18 04:16:12 +07:00
|
|
|
}
|
|
|
|
|
2015-01-15 09:53:56 +07:00
|
|
|
/* Find VXLAN socket based on network namespace, address family and UDP port
|
|
|
|
* and enabled unshareable flags.
|
|
|
|
*/
|
|
|
|
static struct vxlan_sock *vxlan_find_sock(struct net *net, sa_family_t family,
|
|
|
|
__be16 port, u32 flags)
|
2013-05-16 18:35:20 +07:00
|
|
|
{
|
|
|
|
struct vxlan_sock *vs;
|
2015-01-21 02:23:05 +07:00
|
|
|
|
|
|
|
flags &= VXLAN_F_RCV_FLAGS;
|
2013-05-16 18:35:20 +07:00
|
|
|
|
|
|
|
hlist_for_each_entry_rcu(vs, vs_head(net, port), hlist) {
|
2014-11-13 23:43:08 +07:00
|
|
|
if (inet_sk(vs->sock->sk)->inet_sport == port &&
|
2015-08-20 18:56:28 +07:00
|
|
|
vxlan_get_sk_family(vs) == family &&
|
2015-01-21 02:23:05 +07:00
|
|
|
vs->flags == flags)
|
2013-05-16 18:35:20 +07:00
|
|
|
return vs;
|
|
|
|
}
|
|
|
|
return NULL;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
2013-08-20 01:23:02 +07:00
|
|
|
static struct vxlan_dev *vxlan_vs_find_vni(struct vxlan_sock *vs, u32 id)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan;
|
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
hlist_for_each_entry_rcu(vxlan, vni_head(vs, id), hlist) {
|
2013-04-16 09:50:52 +07:00
|
|
|
if (vxlan->default_dst.remote_vni == id)
|
2012-10-01 19:32:35 +07:00
|
|
|
return vxlan;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-08-20 01:23:02 +07:00
|
|
|
/* Look up VNI in a per net namespace table */
|
2014-11-13 23:43:08 +07:00
|
|
|
static struct vxlan_dev *vxlan_find_vni(struct net *net, u32 id,
|
2015-01-15 09:53:56 +07:00
|
|
|
sa_family_t family, __be16 port,
|
|
|
|
u32 flags)
|
2013-08-20 01:23:02 +07:00
|
|
|
{
|
|
|
|
struct vxlan_sock *vs;
|
|
|
|
|
2015-01-15 09:53:56 +07:00
|
|
|
vs = vxlan_find_sock(net, family, port, flags);
|
2013-08-20 01:23:02 +07:00
|
|
|
if (!vs)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return vxlan_vs_find_vni(vs, id);
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Fill in neighbour message in skbuff. */
|
|
|
|
static int vxlan_fdb_info(struct sk_buff *skb, struct vxlan_dev *vxlan,
|
2013-06-18 04:16:41 +07:00
|
|
|
const struct vxlan_fdb *fdb,
|
|
|
|
u32 portid, u32 seq, int type, unsigned int flags,
|
|
|
|
const struct vxlan_rdst *rdst)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
unsigned long now = jiffies;
|
|
|
|
struct nda_cacheinfo ci;
|
|
|
|
struct nlmsghdr *nlh;
|
|
|
|
struct ndmsg *ndm;
|
2012-11-20 09:50:14 +07:00
|
|
|
bool send_ip, send_eth;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
nlh = nlmsg_put(skb, portid, seq, type, sizeof(*ndm), flags);
|
|
|
|
if (nlh == NULL)
|
|
|
|
return -EMSGSIZE;
|
|
|
|
|
|
|
|
ndm = nlmsg_data(nlh);
|
|
|
|
memset(ndm, 0, sizeof(*ndm));
|
2012-11-20 09:50:14 +07:00
|
|
|
|
|
|
|
send_eth = send_ip = true;
|
|
|
|
|
|
|
|
if (type == RTM_GETNEIGH) {
|
|
|
|
ndm->ndm_family = AF_INET;
|
2013-08-31 12:44:33 +07:00
|
|
|
send_ip = !vxlan_addr_any(&rdst->remote_ip);
|
2012-11-20 09:50:14 +07:00
|
|
|
send_eth = !is_zero_ether_addr(fdb->eth_addr);
|
|
|
|
} else
|
|
|
|
ndm->ndm_family = AF_BRIDGE;
|
2012-10-01 19:32:35 +07:00
|
|
|
ndm->ndm_state = fdb->state;
|
|
|
|
ndm->ndm_ifindex = vxlan->dev->ifindex;
|
2013-04-19 07:36:26 +07:00
|
|
|
ndm->ndm_flags = fdb->flags;
|
2014-07-25 23:38:59 +07:00
|
|
|
ndm->ndm_type = RTN_UNICAST;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-01-20 21:15:47 +07:00
|
|
|
if (!net_eq(dev_net(vxlan->dev), vxlan->net) &&
|
2015-01-26 20:10:53 +07:00
|
|
|
nla_put_s32(skb, NDA_LINK_NETNSID,
|
2015-05-07 16:02:49 +07:00
|
|
|
peernet2id_alloc(dev_net(vxlan->dev), vxlan->net)))
|
2015-01-20 21:15:47 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
if (send_eth && nla_put(skb, NDA_LLADDR, ETH_ALEN, &fdb->eth_addr))
|
2012-10-01 19:32:35 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (send_ip && vxlan_nla_put_addr(skb, NDA_DST, &rdst->remote_ip))
|
2013-03-15 11:35:51 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (rdst->remote_port && rdst->remote_port != vxlan->cfg.dst_port &&
|
2013-03-15 11:35:51 +07:00
|
|
|
nla_put_be16(skb, NDA_PORT, rdst->remote_port))
|
|
|
|
goto nla_put_failure;
|
2013-04-16 09:50:52 +07:00
|
|
|
if (rdst->remote_vni != vxlan->default_dst.remote_vni &&
|
2013-06-20 14:26:31 +07:00
|
|
|
nla_put_u32(skb, NDA_VNI, rdst->remote_vni))
|
2013-03-15 11:35:51 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
if (rdst->remote_ifindex &&
|
|
|
|
nla_put_u32(skb, NDA_IFINDEX, rdst->remote_ifindex))
|
2012-10-01 19:32:35 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
ci.ndm_used = jiffies_to_clock_t(now - fdb->used);
|
|
|
|
ci.ndm_confirmed = 0;
|
|
|
|
ci.ndm_updated = jiffies_to_clock_t(now - fdb->updated);
|
|
|
|
ci.ndm_refcnt = 0;
|
|
|
|
|
|
|
|
if (nla_put(skb, NDA_CACHEINFO, sizeof(ci), &ci))
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2015-01-17 04:09:00 +07:00
|
|
|
nlmsg_end(skb, nlh);
|
|
|
|
return 0;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
nlmsg_cancel(skb, nlh);
|
|
|
|
return -EMSGSIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline size_t vxlan_nlmsg_size(void)
|
|
|
|
{
|
|
|
|
return NLMSG_ALIGN(sizeof(struct ndmsg))
|
|
|
|
+ nla_total_size(ETH_ALEN) /* NDA_LLADDR */
|
2013-08-31 12:44:33 +07:00
|
|
|
+ nla_total_size(sizeof(struct in6_addr)) /* NDA_DST */
|
2013-04-27 18:31:54 +07:00
|
|
|
+ nla_total_size(sizeof(__be16)) /* NDA_PORT */
|
2013-03-15 11:35:51 +07:00
|
|
|
+ nla_total_size(sizeof(__be32)) /* NDA_VNI */
|
|
|
|
+ nla_total_size(sizeof(__u32)) /* NDA_IFINDEX */
|
2015-01-26 20:10:53 +07:00
|
|
|
+ nla_total_size(sizeof(__s32)) /* NDA_LINK_NETNSID */
|
2012-10-01 19:32:35 +07:00
|
|
|
+ nla_total_size(sizeof(struct nda_cacheinfo));
|
|
|
|
}
|
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
static void vxlan_fdb_notify(struct vxlan_dev *vxlan, struct vxlan_fdb *fdb,
|
|
|
|
struct vxlan_rdst *rd, int type)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
struct net *net = dev_net(vxlan->dev);
|
|
|
|
struct sk_buff *skb;
|
|
|
|
int err = -ENOBUFS;
|
|
|
|
|
|
|
|
skb = nlmsg_new(vxlan_nlmsg_size(), GFP_ATOMIC);
|
|
|
|
if (skb == NULL)
|
|
|
|
goto errout;
|
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
err = vxlan_fdb_info(skb, vxlan, fdb, 0, 0, type, 0, rd);
|
2012-10-01 19:32:35 +07:00
|
|
|
if (err < 0) {
|
|
|
|
/* -EMSGSIZE implies BUG in vxlan_nlmsg_size() */
|
|
|
|
WARN_ON(err == -EMSGSIZE);
|
|
|
|
kfree_skb(skb);
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
rtnl_notify(skb, net, 0, RTNLGRP_NEIGH, NULL, GFP_ATOMIC);
|
|
|
|
return;
|
|
|
|
errout:
|
|
|
|
if (err < 0)
|
|
|
|
rtnl_set_sk_err(net, RTNLGRP_NEIGH, err);
|
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
static void vxlan_ip_miss(struct net_device *dev, union vxlan_addr *ipa)
|
2012-11-20 09:50:14 +07:00
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2013-06-18 04:16:40 +07:00
|
|
|
struct vxlan_fdb f = {
|
|
|
|
.state = NUD_STALE,
|
|
|
|
};
|
|
|
|
struct vxlan_rdst remote = {
|
2013-08-31 12:44:33 +07:00
|
|
|
.remote_ip = *ipa, /* goes to NDA_DST */
|
2013-06-18 04:16:40 +07:00
|
|
|
.remote_vni = VXLAN_N_VID,
|
|
|
|
};
|
2013-06-18 04:16:12 +07:00
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
vxlan_fdb_notify(vxlan, &f, &remote, RTM_GETNEIGH);
|
2012-11-20 09:50:14 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void vxlan_fdb_miss(struct vxlan_dev *vxlan, const u8 eth_addr[ETH_ALEN])
|
|
|
|
{
|
2013-06-18 04:16:40 +07:00
|
|
|
struct vxlan_fdb f = {
|
|
|
|
.state = NUD_STALE,
|
|
|
|
};
|
2014-04-22 20:01:30 +07:00
|
|
|
struct vxlan_rdst remote = { };
|
2012-11-20 09:50:14 +07:00
|
|
|
|
|
|
|
memcpy(f.eth_addr, eth_addr, ETH_ALEN);
|
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
vxlan_fdb_notify(vxlan, &f, &remote, RTM_GETNEIGH);
|
2012-11-20 09:50:14 +07:00
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Hash Ethernet address */
|
|
|
|
static u32 eth_hash(const unsigned char *addr)
|
|
|
|
{
|
|
|
|
u64 value = get_unaligned((u64 *)addr);
|
|
|
|
|
|
|
|
/* only want 6 bytes */
|
|
|
|
#ifdef __BIG_ENDIAN
|
|
|
|
value >>= 16;
|
2012-10-10 03:35:47 +07:00
|
|
|
#else
|
|
|
|
value <<= 16;
|
2012-10-01 19:32:35 +07:00
|
|
|
#endif
|
|
|
|
return hash_64(value, FDB_HASH_BITS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Hash chain to use given mac address */
|
|
|
|
static inline struct hlist_head *vxlan_fdb_head(struct vxlan_dev *vxlan,
|
|
|
|
const u8 *mac)
|
|
|
|
{
|
|
|
|
return &vxlan->fdb_head[eth_hash(mac)];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Look up Ethernet address in forwarding table */
|
2013-05-17 13:39:07 +07:00
|
|
|
static struct vxlan_fdb *__vxlan_find_mac(struct vxlan_dev *vxlan,
|
2012-10-01 19:32:35 +07:00
|
|
|
const u8 *mac)
|
|
|
|
{
|
|
|
|
struct hlist_head *head = vxlan_fdb_head(vxlan, mac);
|
|
|
|
struct vxlan_fdb *f;
|
|
|
|
|
hlist: drop the node parameter from iterators
I'm not sure why, but the hlist for each entry iterators were conceived
list_for_each_entry(pos, head, member)
The hlist ones were greedy and wanted an extra parameter:
hlist_for_each_entry(tpos, pos, head, member)
Why did they need an extra pos parameter? I'm not quite sure. Not only
they don't really need it, it also prevents the iterator from looking
exactly like the list iterator, which is unfortunate.
Besides the semantic patch, there was some manual work required:
- Fix up the actual hlist iterators in linux/list.h
- Fix up the declaration of other iterators based on the hlist ones.
- A very small amount of places were using the 'node' parameter, this
was modified to use 'obj->member' instead.
- Coccinelle didn't handle the hlist_for_each_entry_safe iterator
properly, so those had to be fixed up manually.
The semantic patch which is mostly the work of Peter Senna Tschudin is here:
@@
iterator name hlist_for_each_entry, hlist_for_each_entry_continue, hlist_for_each_entry_from, hlist_for_each_entry_rcu, hlist_for_each_entry_rcu_bh, hlist_for_each_entry_continue_rcu_bh, for_each_busy_worker, ax25_uid_for_each, ax25_for_each, inet_bind_bucket_for_each, sctp_for_each_hentry, sk_for_each, sk_for_each_rcu, sk_for_each_from, sk_for_each_safe, sk_for_each_bound, hlist_for_each_entry_safe, hlist_for_each_entry_continue_rcu, nr_neigh_for_each, nr_neigh_for_each_safe, nr_node_for_each, nr_node_for_each_safe, for_each_gfn_indirect_valid_sp, for_each_gfn_sp, for_each_host;
type T;
expression a,c,d,e;
identifier b;
statement S;
@@
-T b;
<+... when != b
(
hlist_for_each_entry(a,
- b,
c, d) S
|
hlist_for_each_entry_continue(a,
- b,
c) S
|
hlist_for_each_entry_from(a,
- b,
c) S
|
hlist_for_each_entry_rcu(a,
- b,
c, d) S
|
hlist_for_each_entry_rcu_bh(a,
- b,
c, d) S
|
hlist_for_each_entry_continue_rcu_bh(a,
- b,
c) S
|
for_each_busy_worker(a, c,
- b,
d) S
|
ax25_uid_for_each(a,
- b,
c) S
|
ax25_for_each(a,
- b,
c) S
|
inet_bind_bucket_for_each(a,
- b,
c) S
|
sctp_for_each_hentry(a,
- b,
c) S
|
sk_for_each(a,
- b,
c) S
|
sk_for_each_rcu(a,
- b,
c) S
|
sk_for_each_from
-(a, b)
+(a)
S
+ sk_for_each_from(a) S
|
sk_for_each_safe(a,
- b,
c, d) S
|
sk_for_each_bound(a,
- b,
c) S
|
hlist_for_each_entry_safe(a,
- b,
c, d, e) S
|
hlist_for_each_entry_continue_rcu(a,
- b,
c) S
|
nr_neigh_for_each(a,
- b,
c) S
|
nr_neigh_for_each_safe(a,
- b,
c, d) S
|
nr_node_for_each(a,
- b,
c) S
|
nr_node_for_each_safe(a,
- b,
c, d) S
|
- for_each_gfn_sp(a, c, d, b) S
+ for_each_gfn_sp(a, c, d) S
|
- for_each_gfn_indirect_valid_sp(a, c, d, b) S
+ for_each_gfn_indirect_valid_sp(a, c, d) S
|
for_each_host(a,
- b,
c) S
|
for_each_host_safe(a,
- b,
c, d) S
|
for_each_mesh_entry(a,
- b,
c, d) S
)
...+>
[akpm@linux-foundation.org: drop bogus change from net/ipv4/raw.c]
[akpm@linux-foundation.org: drop bogus hunk from net/ipv6/raw.c]
[akpm@linux-foundation.org: checkpatch fixes]
[akpm@linux-foundation.org: fix warnings]
[akpm@linux-foudnation.org: redo intrusive kvm changes]
Tested-by: Peter Senna Tschudin <peter.senna@gmail.com>
Acked-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Signed-off-by: Sasha Levin <sasha.levin@oracle.com>
Cc: Wu Fengguang <fengguang.wu@intel.com>
Cc: Marcelo Tosatti <mtosatti@redhat.com>
Cc: Gleb Natapov <gleb@redhat.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2013-02-28 08:06:00 +07:00
|
|
|
hlist_for_each_entry_rcu(f, head, hlist) {
|
drivers/net: Convert uses of compare_ether_addr to ether_addr_equal
Use the new bool function ether_addr_equal to add
some clarity and reduce the likelihood for misuse
of compare_ether_addr for sorting.
Done via cocci script: (and a little typing)
$ cat compare_ether_addr.cocci
@@
expression a,b;
@@
- !compare_ether_addr(a, b)
+ ether_addr_equal(a, b)
@@
expression a,b;
@@
- compare_ether_addr(a, b)
+ !ether_addr_equal(a, b)
@@
expression a,b;
@@
- !ether_addr_equal(a, b) == 0
+ ether_addr_equal(a, b)
@@
expression a,b;
@@
- !ether_addr_equal(a, b) != 0
+ !ether_addr_equal(a, b)
@@
expression a,b;
@@
- ether_addr_equal(a, b) == 0
+ !ether_addr_equal(a, b)
@@
expression a,b;
@@
- ether_addr_equal(a, b) != 0
+ ether_addr_equal(a, b)
@@
expression a,b;
@@
- !!ether_addr_equal(a, b)
+ ether_addr_equal(a, b)
Signed-off-by: Joe Perches <joe@perches.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2013-09-02 01:51:23 +07:00
|
|
|
if (ether_addr_equal(mac, f->eth_addr))
|
2012-10-01 19:32:35 +07:00
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-05-17 13:39:07 +07:00
|
|
|
static struct vxlan_fdb *vxlan_find_mac(struct vxlan_dev *vxlan,
|
|
|
|
const u8 *mac)
|
|
|
|
{
|
|
|
|
struct vxlan_fdb *f;
|
|
|
|
|
|
|
|
f = __vxlan_find_mac(vxlan, mac);
|
|
|
|
if (f)
|
|
|
|
f->used = jiffies;
|
|
|
|
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
2013-06-25 20:01:52 +07:00
|
|
|
/* caller should hold vxlan->hash_lock */
|
|
|
|
static struct vxlan_rdst *vxlan_fdb_find_rdst(struct vxlan_fdb *f,
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *ip, __be16 port,
|
2013-06-25 20:01:52 +07:00
|
|
|
__u32 vni, __u32 ifindex)
|
2013-03-15 11:35:51 +07:00
|
|
|
{
|
2013-06-18 04:16:12 +07:00
|
|
|
struct vxlan_rdst *rd;
|
2013-03-15 11:35:51 +07:00
|
|
|
|
2013-06-18 04:16:12 +07:00
|
|
|
list_for_each_entry(rd, &f->remotes, list) {
|
2013-08-31 12:44:33 +07:00
|
|
|
if (vxlan_addr_equal(&rd->remote_ip, ip) &&
|
2013-03-15 11:35:51 +07:00
|
|
|
rd->remote_port == port &&
|
|
|
|
rd->remote_vni == vni &&
|
|
|
|
rd->remote_ifindex == ifindex)
|
2013-06-25 20:01:52 +07:00
|
|
|
return rd;
|
2013-03-15 11:35:51 +07:00
|
|
|
}
|
2013-06-18 04:16:12 +07:00
|
|
|
|
2013-06-25 20:01:52 +07:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-07-19 22:20:07 +07:00
|
|
|
/* Replace destination of unicast mac */
|
|
|
|
static int vxlan_fdb_replace(struct vxlan_fdb *f,
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *ip, __be16 port, __u32 vni, __u32 ifindex)
|
2013-07-19 22:20:07 +07:00
|
|
|
{
|
|
|
|
struct vxlan_rdst *rd;
|
|
|
|
|
|
|
|
rd = vxlan_fdb_find_rdst(f, ip, port, vni, ifindex);
|
|
|
|
if (rd)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
rd = list_first_entry_or_null(&f->remotes, struct vxlan_rdst, list);
|
|
|
|
if (!rd)
|
|
|
|
return 0;
|
2013-08-31 12:44:33 +07:00
|
|
|
rd->remote_ip = *ip;
|
2013-07-19 22:20:07 +07:00
|
|
|
rd->remote_port = port;
|
|
|
|
rd->remote_vni = vni;
|
|
|
|
rd->remote_ifindex = ifindex;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-06-25 20:01:52 +07:00
|
|
|
/* Add/update destinations for multicast */
|
|
|
|
static int vxlan_fdb_append(struct vxlan_fdb *f,
|
2014-04-22 20:01:30 +07:00
|
|
|
union vxlan_addr *ip, __be16 port, __u32 vni,
|
|
|
|
__u32 ifindex, struct vxlan_rdst **rdp)
|
2013-06-25 20:01:52 +07:00
|
|
|
{
|
|
|
|
struct vxlan_rdst *rd;
|
|
|
|
|
|
|
|
rd = vxlan_fdb_find_rdst(f, ip, port, vni, ifindex);
|
|
|
|
if (rd)
|
|
|
|
return 0;
|
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
rd = kmalloc(sizeof(*rd), GFP_ATOMIC);
|
|
|
|
if (rd == NULL)
|
|
|
|
return -ENOBUFS;
|
2013-08-31 12:44:33 +07:00
|
|
|
rd->remote_ip = *ip;
|
2013-03-15 11:35:51 +07:00
|
|
|
rd->remote_port = port;
|
|
|
|
rd->remote_vni = vni;
|
|
|
|
rd->remote_ifindex = ifindex;
|
2013-06-18 04:16:12 +07:00
|
|
|
|
|
|
|
list_add_tail_rcu(&rd->list, &f->remotes);
|
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
*rdp = rd;
|
2013-03-15 11:35:51 +07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
static struct vxlanhdr *vxlan_gro_remcsum(struct sk_buff *skb,
|
|
|
|
unsigned int off,
|
|
|
|
struct vxlanhdr *vh, size_t hdrlen,
|
2015-02-11 07:30:32 +07:00
|
|
|
u32 data, struct gro_remcsum *grc,
|
|
|
|
bool nopartial)
|
2015-01-13 08:00:38 +07:00
|
|
|
{
|
2015-08-20 07:07:32 +07:00
|
|
|
size_t start, offset;
|
2015-01-13 08:00:38 +07:00
|
|
|
|
|
|
|
if (skb->remcsum_offload)
|
2015-08-20 07:07:32 +07:00
|
|
|
return vh;
|
2015-01-13 08:00:38 +07:00
|
|
|
|
|
|
|
if (!NAPI_GRO_CB(skb)->csum_valid)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
start = (data & VXLAN_RCO_MASK) << VXLAN_RCO_SHIFT;
|
|
|
|
offset = start + ((data & VXLAN_RCO_UDP) ?
|
|
|
|
offsetof(struct udphdr, check) :
|
|
|
|
offsetof(struct tcphdr, check));
|
|
|
|
|
2015-08-20 07:07:32 +07:00
|
|
|
vh = skb_gro_remcsum_process(skb, (void *)vh, off, hdrlen,
|
|
|
|
start, offset, grc, nopartial);
|
2015-01-13 08:00:38 +07:00
|
|
|
|
|
|
|
skb->remcsum_offload = 1;
|
|
|
|
|
|
|
|
return vh;
|
|
|
|
}
|
|
|
|
|
2015-01-13 08:00:37 +07:00
|
|
|
static struct sk_buff **vxlan_gro_receive(struct sk_buff **head,
|
|
|
|
struct sk_buff *skb,
|
|
|
|
struct udp_offload *uoff)
|
2014-01-20 18:59:21 +07:00
|
|
|
{
|
|
|
|
struct sk_buff *p, **pp = NULL;
|
|
|
|
struct vxlanhdr *vh, *vh2;
|
2014-12-31 10:10:15 +07:00
|
|
|
unsigned int hlen, off_vx;
|
2014-01-20 18:59:21 +07:00
|
|
|
int flush = 1;
|
2015-01-13 08:00:38 +07:00
|
|
|
struct vxlan_sock *vs = container_of(uoff, struct vxlan_sock,
|
|
|
|
udp_offloads);
|
|
|
|
u32 flags;
|
2015-02-11 07:30:27 +07:00
|
|
|
struct gro_remcsum grc;
|
|
|
|
|
|
|
|
skb_gro_remcsum_init(&grc);
|
2014-01-20 18:59:21 +07:00
|
|
|
|
|
|
|
off_vx = skb_gro_offset(skb);
|
|
|
|
hlen = off_vx + sizeof(*vh);
|
|
|
|
vh = skb_gro_header_fast(skb, off_vx);
|
|
|
|
if (skb_gro_header_hard(skb, hlen)) {
|
|
|
|
vh = skb_gro_header_slow(skb, hlen, off_vx);
|
|
|
|
if (unlikely(!vh))
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
skb_gro_postpull_rcsum(skb, vh, sizeof(struct vxlanhdr));
|
|
|
|
|
|
|
|
flags = ntohl(vh->vx_flags);
|
|
|
|
|
|
|
|
if ((flags & VXLAN_HF_RCO) && (vs->flags & VXLAN_F_REMCSUM_RX)) {
|
|
|
|
vh = vxlan_gro_remcsum(skb, off_vx, vh, sizeof(struct vxlanhdr),
|
2015-02-11 07:30:32 +07:00
|
|
|
ntohl(vh->vx_vni), &grc,
|
|
|
|
!!(vs->flags &
|
|
|
|
VXLAN_F_REMCSUM_NOPARTIAL));
|
2015-01-13 08:00:38 +07:00
|
|
|
|
|
|
|
if (!vh)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2015-08-20 07:07:32 +07:00
|
|
|
skb_gro_pull(skb, sizeof(struct vxlanhdr)); /* pull vxlan header */
|
|
|
|
|
2014-01-20 18:59:21 +07:00
|
|
|
flush = 0;
|
|
|
|
|
|
|
|
for (p = *head; p; p = p->next) {
|
|
|
|
if (!NAPI_GRO_CB(p)->same_flow)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
vh2 = (struct vxlanhdr *)(p->data + off_vx);
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
if (vh->vx_flags != vh2->vx_flags ||
|
|
|
|
vh->vx_vni != vh2->vx_vni) {
|
2014-01-20 18:59:21 +07:00
|
|
|
NAPI_GRO_CB(p)->same_flow = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-31 10:10:15 +07:00
|
|
|
pp = eth_gro_receive(head, skb);
|
2014-01-20 18:59:21 +07:00
|
|
|
|
|
|
|
out:
|
2015-02-11 07:30:27 +07:00
|
|
|
skb_gro_remcsum_cleanup(skb, &grc);
|
2014-01-20 18:59:21 +07:00
|
|
|
NAPI_GRO_CB(skb)->flush |= flush;
|
|
|
|
|
|
|
|
return pp;
|
|
|
|
}
|
|
|
|
|
2015-01-13 08:00:37 +07:00
|
|
|
static int vxlan_gro_complete(struct sk_buff *skb, int nhoff,
|
|
|
|
struct udp_offload *uoff)
|
2014-01-20 18:59:21 +07:00
|
|
|
{
|
2014-11-11 02:45:13 +07:00
|
|
|
udp_tunnel_gro_complete(skb, nhoff);
|
|
|
|
|
2014-12-31 10:10:15 +07:00
|
|
|
return eth_gro_complete(skb, nhoff + sizeof(struct vxlanhdr));
|
2014-01-20 18:59:21 +07:00
|
|
|
}
|
|
|
|
|
2013-09-04 16:13:38 +07:00
|
|
|
/* Notify netdevs that UDP port started listening */
|
2014-01-20 18:59:21 +07:00
|
|
|
static void vxlan_notify_add_rx_port(struct vxlan_sock *vs)
|
2013-09-04 16:13:38 +07:00
|
|
|
{
|
|
|
|
struct net_device *dev;
|
2014-01-20 18:59:21 +07:00
|
|
|
struct sock *sk = vs->sock->sk;
|
2013-09-04 16:13:38 +07:00
|
|
|
struct net *net = sock_net(sk);
|
2015-08-20 18:56:28 +07:00
|
|
|
sa_family_t sa_family = vxlan_get_sk_family(vs);
|
2013-09-13 21:34:13 +07:00
|
|
|
__be16 port = inet_sk(sk)->inet_sport;
|
2014-01-20 18:59:21 +07:00
|
|
|
int err;
|
|
|
|
|
|
|
|
if (sa_family == AF_INET) {
|
|
|
|
err = udp_add_offload(&vs->udp_offloads);
|
|
|
|
if (err)
|
|
|
|
pr_warn("vxlan: udp_add_offload failed with status %d\n", err);
|
|
|
|
}
|
2013-09-04 16:13:38 +07:00
|
|
|
|
|
|
|
rcu_read_lock();
|
|
|
|
for_each_netdev_rcu(net, dev) {
|
|
|
|
if (dev->netdev_ops->ndo_add_vxlan_port)
|
|
|
|
dev->netdev_ops->ndo_add_vxlan_port(dev, sa_family,
|
|
|
|
port);
|
|
|
|
}
|
|
|
|
rcu_read_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Notify netdevs that UDP port is no more listening */
|
2014-01-20 18:59:21 +07:00
|
|
|
static void vxlan_notify_del_rx_port(struct vxlan_sock *vs)
|
2013-09-04 16:13:38 +07:00
|
|
|
{
|
|
|
|
struct net_device *dev;
|
2014-01-20 18:59:21 +07:00
|
|
|
struct sock *sk = vs->sock->sk;
|
2013-09-04 16:13:38 +07:00
|
|
|
struct net *net = sock_net(sk);
|
2015-08-20 18:56:28 +07:00
|
|
|
sa_family_t sa_family = vxlan_get_sk_family(vs);
|
2013-09-13 21:34:13 +07:00
|
|
|
__be16 port = inet_sk(sk)->inet_sport;
|
2013-09-04 16:13:38 +07:00
|
|
|
|
|
|
|
rcu_read_lock();
|
|
|
|
for_each_netdev_rcu(net, dev) {
|
|
|
|
if (dev->netdev_ops->ndo_del_vxlan_port)
|
|
|
|
dev->netdev_ops->ndo_del_vxlan_port(dev, sa_family,
|
|
|
|
port);
|
|
|
|
}
|
|
|
|
rcu_read_unlock();
|
2014-01-20 18:59:21 +07:00
|
|
|
|
|
|
|
if (sa_family == AF_INET)
|
|
|
|
udp_del_offload(&vs->udp_offloads);
|
2013-09-04 16:13:38 +07:00
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Add new entry to forwarding table -- assumes lock held */
|
|
|
|
static int vxlan_fdb_create(struct vxlan_dev *vxlan,
|
2013-08-31 12:44:33 +07:00
|
|
|
const u8 *mac, union vxlan_addr *ip,
|
2013-03-15 11:35:51 +07:00
|
|
|
__u16 state, __u16 flags,
|
2013-04-27 18:31:54 +07:00
|
|
|
__be16 port, __u32 vni, __u32 ifindex,
|
2013-04-19 07:36:26 +07:00
|
|
|
__u8 ndm_flags)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
2014-04-22 20:01:30 +07:00
|
|
|
struct vxlan_rdst *rd = NULL;
|
2012-10-01 19:32:35 +07:00
|
|
|
struct vxlan_fdb *f;
|
|
|
|
int notify = 0;
|
|
|
|
|
2013-05-17 13:39:07 +07:00
|
|
|
f = __vxlan_find_mac(vxlan, mac);
|
2012-10-01 19:32:35 +07:00
|
|
|
if (f) {
|
|
|
|
if (flags & NLM_F_EXCL) {
|
|
|
|
netdev_dbg(vxlan->dev,
|
|
|
|
"lost race to create %pM\n", mac);
|
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
if (f->state != state) {
|
|
|
|
f->state = state;
|
|
|
|
f->updated = jiffies;
|
|
|
|
notify = 1;
|
|
|
|
}
|
2013-04-19 07:36:26 +07:00
|
|
|
if (f->flags != ndm_flags) {
|
|
|
|
f->flags = ndm_flags;
|
|
|
|
f->updated = jiffies;
|
|
|
|
notify = 1;
|
|
|
|
}
|
2013-07-19 22:20:07 +07:00
|
|
|
if ((flags & NLM_F_REPLACE)) {
|
|
|
|
/* Only change unicasts */
|
|
|
|
if (!(is_multicast_ether_addr(f->eth_addr) ||
|
|
|
|
is_zero_ether_addr(f->eth_addr))) {
|
2015-04-22 14:49:10 +07:00
|
|
|
notify |= vxlan_fdb_replace(f, ip, port, vni,
|
2013-07-19 22:20:07 +07:00
|
|
|
ifindex);
|
|
|
|
} else
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2013-03-15 11:35:51 +07:00
|
|
|
if ((flags & NLM_F_APPEND) &&
|
2013-06-25 20:01:56 +07:00
|
|
|
(is_multicast_ether_addr(f->eth_addr) ||
|
|
|
|
is_zero_ether_addr(f->eth_addr))) {
|
2014-04-22 20:01:30 +07:00
|
|
|
int rc = vxlan_fdb_append(f, ip, port, vni, ifindex,
|
|
|
|
&rd);
|
2013-03-15 11:35:51 +07:00
|
|
|
|
|
|
|
if (rc < 0)
|
|
|
|
return rc;
|
|
|
|
notify |= rc;
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
} else {
|
|
|
|
if (!(flags & NLM_F_CREATE))
|
|
|
|
return -ENOENT;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (vxlan->cfg.addrmax &&
|
|
|
|
vxlan->addrcnt >= vxlan->cfg.addrmax)
|
2012-10-01 19:32:35 +07:00
|
|
|
return -ENOSPC;
|
|
|
|
|
2013-07-19 22:20:07 +07:00
|
|
|
/* Disallow replace to add a multicast entry */
|
|
|
|
if ((flags & NLM_F_REPLACE) &&
|
|
|
|
(is_multicast_ether_addr(mac) || is_zero_ether_addr(mac)))
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
netdev_dbg(vxlan->dev, "add %pM -> %pIS\n", mac, ip);
|
2012-10-01 19:32:35 +07:00
|
|
|
f = kmalloc(sizeof(*f), GFP_ATOMIC);
|
|
|
|
if (!f)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
notify = 1;
|
|
|
|
f->state = state;
|
2013-04-19 07:36:26 +07:00
|
|
|
f->flags = ndm_flags;
|
2012-10-01 19:32:35 +07:00
|
|
|
f->updated = f->used = jiffies;
|
2013-06-18 04:16:12 +07:00
|
|
|
INIT_LIST_HEAD(&f->remotes);
|
2012-10-01 19:32:35 +07:00
|
|
|
memcpy(f->eth_addr, mac, ETH_ALEN);
|
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
vxlan_fdb_append(f, ip, port, vni, ifindex, &rd);
|
2013-06-18 04:16:12 +07:00
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
++vxlan->addrcnt;
|
|
|
|
hlist_add_head_rcu(&f->hlist,
|
|
|
|
vxlan_fdb_head(vxlan, mac));
|
|
|
|
}
|
|
|
|
|
2014-04-22 20:01:30 +07:00
|
|
|
if (notify) {
|
|
|
|
if (rd == NULL)
|
|
|
|
rd = first_remote_rtnl(f);
|
|
|
|
vxlan_fdb_notify(vxlan, f, rd, RTM_NEWNEIGH);
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-04-12 02:00:35 +07:00
|
|
|
static void vxlan_fdb_free(struct rcu_head *head)
|
2013-03-15 11:35:51 +07:00
|
|
|
{
|
|
|
|
struct vxlan_fdb *f = container_of(head, struct vxlan_fdb, rcu);
|
2013-06-18 04:16:12 +07:00
|
|
|
struct vxlan_rdst *rd, *nd;
|
2013-03-15 11:35:51 +07:00
|
|
|
|
2013-06-18 04:16:12 +07:00
|
|
|
list_for_each_entry_safe(rd, nd, &f->remotes, list)
|
2013-03-15 11:35:51 +07:00
|
|
|
kfree(rd);
|
|
|
|
kfree(f);
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
static void vxlan_fdb_destroy(struct vxlan_dev *vxlan, struct vxlan_fdb *f)
|
|
|
|
{
|
|
|
|
netdev_dbg(vxlan->dev,
|
|
|
|
"delete %pM\n", f->eth_addr);
|
|
|
|
|
|
|
|
--vxlan->addrcnt;
|
2014-04-22 20:01:30 +07:00
|
|
|
vxlan_fdb_notify(vxlan, f, first_remote_rtnl(f), RTM_DELNEIGH);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
hlist_del_rcu(&f->hlist);
|
2013-03-15 11:35:51 +07:00
|
|
|
call_rcu(&f->rcu, vxlan_fdb_free);
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
2013-06-25 20:01:53 +07:00
|
|
|
static int vxlan_fdb_parse(struct nlattr *tb[], struct vxlan_dev *vxlan,
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *ip, __be16 *port, u32 *vni, u32 *ifindex)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
2013-03-15 11:35:51 +07:00
|
|
|
struct net *net = dev_net(vxlan->dev);
|
2013-08-31 12:44:33 +07:00
|
|
|
int err;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-06-25 20:01:53 +07:00
|
|
|
if (tb[NDA_DST]) {
|
2013-08-31 12:44:33 +07:00
|
|
|
err = vxlan_nla_get_addr(ip, tb[NDA_DST]);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2013-06-25 20:01:53 +07:00
|
|
|
} else {
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *remote = &vxlan->default_dst.remote_ip;
|
|
|
|
if (remote->sa.sa_family == AF_INET) {
|
|
|
|
ip->sin.sin_addr.s_addr = htonl(INADDR_ANY);
|
|
|
|
ip->sa.sa_family = AF_INET;
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
|
|
|
ip->sin6.sin6_addr = in6addr_any;
|
|
|
|
ip->sa.sa_family = AF_INET6;
|
|
|
|
#endif
|
|
|
|
}
|
2013-06-25 20:01:53 +07:00
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
if (tb[NDA_PORT]) {
|
2013-04-27 18:31:54 +07:00
|
|
|
if (nla_len(tb[NDA_PORT]) != sizeof(__be16))
|
2013-03-15 11:35:51 +07:00
|
|
|
return -EINVAL;
|
2013-06-25 20:01:53 +07:00
|
|
|
*port = nla_get_be16(tb[NDA_PORT]);
|
|
|
|
} else {
|
2015-07-21 15:44:02 +07:00
|
|
|
*port = vxlan->cfg.dst_port;
|
2013-06-25 20:01:53 +07:00
|
|
|
}
|
2013-03-15 11:35:51 +07:00
|
|
|
|
|
|
|
if (tb[NDA_VNI]) {
|
|
|
|
if (nla_len(tb[NDA_VNI]) != sizeof(u32))
|
|
|
|
return -EINVAL;
|
2013-06-25 20:01:53 +07:00
|
|
|
*vni = nla_get_u32(tb[NDA_VNI]);
|
|
|
|
} else {
|
|
|
|
*vni = vxlan->default_dst.remote_vni;
|
|
|
|
}
|
2013-03-15 11:35:51 +07:00
|
|
|
|
|
|
|
if (tb[NDA_IFINDEX]) {
|
2013-03-26 15:29:30 +07:00
|
|
|
struct net_device *tdev;
|
2013-03-15 11:35:51 +07:00
|
|
|
|
|
|
|
if (nla_len(tb[NDA_IFINDEX]) != sizeof(u32))
|
|
|
|
return -EINVAL;
|
2013-06-25 20:01:53 +07:00
|
|
|
*ifindex = nla_get_u32(tb[NDA_IFINDEX]);
|
2014-01-15 09:23:41 +07:00
|
|
|
tdev = __dev_get_by_index(net, *ifindex);
|
2013-03-26 15:29:30 +07:00
|
|
|
if (!tdev)
|
2013-03-15 11:35:51 +07:00
|
|
|
return -EADDRNOTAVAIL;
|
2013-06-25 20:01:53 +07:00
|
|
|
} else {
|
|
|
|
*ifindex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add static entry (via netlink) */
|
|
|
|
static int vxlan_fdb_add(struct ndmsg *ndm, struct nlattr *tb[],
|
|
|
|
struct net_device *dev,
|
2014-11-28 20:34:15 +07:00
|
|
|
const unsigned char *addr, u16 vid, u16 flags)
|
2013-06-25 20:01:53 +07:00
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
/* struct net *net = dev_net(vxlan->dev); */
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr ip;
|
2013-06-25 20:01:53 +07:00
|
|
|
__be16 port;
|
|
|
|
u32 vni, ifindex;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (!(ndm->ndm_state & (NUD_PERMANENT|NUD_REACHABLE))) {
|
|
|
|
pr_info("RTM_NEWNEIGH with invalid state %#x\n",
|
|
|
|
ndm->ndm_state);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tb[NDA_DST] == NULL)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
err = vxlan_fdb_parse(tb, vxlan, &ip, &port, &vni, &ifindex);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2013-03-15 11:35:51 +07:00
|
|
|
|
2014-04-01 13:23:01 +07:00
|
|
|
if (vxlan->default_dst.remote_ip.sa.sa_family != ip.sa.sa_family)
|
|
|
|
return -EAFNOSUPPORT;
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
spin_lock_bh(&vxlan->hash_lock);
|
2013-08-31 12:44:33 +07:00
|
|
|
err = vxlan_fdb_create(vxlan, addr, &ip, ndm->ndm_state, flags,
|
2013-04-27 18:31:54 +07:00
|
|
|
port, vni, ifindex, ndm->ndm_flags);
|
2012-10-01 19:32:35 +07:00
|
|
|
spin_unlock_bh(&vxlan->hash_lock);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Delete entry (via netlink) */
|
2013-02-13 19:00:18 +07:00
|
|
|
static int vxlan_fdb_delete(struct ndmsg *ndm, struct nlattr *tb[],
|
|
|
|
struct net_device *dev,
|
2014-11-28 20:34:15 +07:00
|
|
|
const unsigned char *addr, u16 vid)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct vxlan_fdb *f;
|
2013-06-25 20:01:54 +07:00
|
|
|
struct vxlan_rdst *rd = NULL;
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr ip;
|
2013-06-25 20:01:54 +07:00
|
|
|
__be16 port;
|
|
|
|
u32 vni, ifindex;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = vxlan_fdb_parse(tb, vxlan, &ip, &port, &vni, &ifindex);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = -ENOENT;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
spin_lock_bh(&vxlan->hash_lock);
|
|
|
|
f = vxlan_find_mac(vxlan, addr);
|
2013-06-25 20:01:54 +07:00
|
|
|
if (!f)
|
|
|
|
goto out;
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!vxlan_addr_any(&ip)) {
|
|
|
|
rd = vxlan_fdb_find_rdst(f, &ip, port, vni, ifindex);
|
2013-06-25 20:01:54 +07:00
|
|
|
if (!rd)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = 0;
|
|
|
|
|
|
|
|
/* remove a destination if it's not the only one on the list,
|
|
|
|
* otherwise destroy the fdb entry
|
|
|
|
*/
|
|
|
|
if (rd && !list_is_singular(&f->remotes)) {
|
|
|
|
list_del_rcu(&rd->list);
|
2014-04-22 20:01:30 +07:00
|
|
|
vxlan_fdb_notify(vxlan, f, rd, RTM_DELNEIGH);
|
2013-08-17 06:32:09 +07:00
|
|
|
kfree_rcu(rd, rcu);
|
2013-06-25 20:01:54 +07:00
|
|
|
goto out;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
2013-06-25 20:01:54 +07:00
|
|
|
|
|
|
|
vxlan_fdb_destroy(vxlan, f);
|
|
|
|
|
|
|
|
out:
|
2012-10-01 19:32:35 +07:00
|
|
|
spin_unlock_bh(&vxlan->hash_lock);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Dump forwarding table */
|
|
|
|
static int vxlan_fdb_dump(struct sk_buff *skb, struct netlink_callback *cb,
|
2014-07-10 18:01:58 +07:00
|
|
|
struct net_device *dev,
|
|
|
|
struct net_device *filter_dev, int idx)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
unsigned int h;
|
|
|
|
|
|
|
|
for (h = 0; h < FDB_HASH_SIZE; ++h) {
|
|
|
|
struct vxlan_fdb *f;
|
|
|
|
int err;
|
|
|
|
|
hlist: drop the node parameter from iterators
I'm not sure why, but the hlist for each entry iterators were conceived
list_for_each_entry(pos, head, member)
The hlist ones were greedy and wanted an extra parameter:
hlist_for_each_entry(tpos, pos, head, member)
Why did they need an extra pos parameter? I'm not quite sure. Not only
they don't really need it, it also prevents the iterator from looking
exactly like the list iterator, which is unfortunate.
Besides the semantic patch, there was some manual work required:
- Fix up the actual hlist iterators in linux/list.h
- Fix up the declaration of other iterators based on the hlist ones.
- A very small amount of places were using the 'node' parameter, this
was modified to use 'obj->member' instead.
- Coccinelle didn't handle the hlist_for_each_entry_safe iterator
properly, so those had to be fixed up manually.
The semantic patch which is mostly the work of Peter Senna Tschudin is here:
@@
iterator name hlist_for_each_entry, hlist_for_each_entry_continue, hlist_for_each_entry_from, hlist_for_each_entry_rcu, hlist_for_each_entry_rcu_bh, hlist_for_each_entry_continue_rcu_bh, for_each_busy_worker, ax25_uid_for_each, ax25_for_each, inet_bind_bucket_for_each, sctp_for_each_hentry, sk_for_each, sk_for_each_rcu, sk_for_each_from, sk_for_each_safe, sk_for_each_bound, hlist_for_each_entry_safe, hlist_for_each_entry_continue_rcu, nr_neigh_for_each, nr_neigh_for_each_safe, nr_node_for_each, nr_node_for_each_safe, for_each_gfn_indirect_valid_sp, for_each_gfn_sp, for_each_host;
type T;
expression a,c,d,e;
identifier b;
statement S;
@@
-T b;
<+... when != b
(
hlist_for_each_entry(a,
- b,
c, d) S
|
hlist_for_each_entry_continue(a,
- b,
c) S
|
hlist_for_each_entry_from(a,
- b,
c) S
|
hlist_for_each_entry_rcu(a,
- b,
c, d) S
|
hlist_for_each_entry_rcu_bh(a,
- b,
c, d) S
|
hlist_for_each_entry_continue_rcu_bh(a,
- b,
c) S
|
for_each_busy_worker(a, c,
- b,
d) S
|
ax25_uid_for_each(a,
- b,
c) S
|
ax25_for_each(a,
- b,
c) S
|
inet_bind_bucket_for_each(a,
- b,
c) S
|
sctp_for_each_hentry(a,
- b,
c) S
|
sk_for_each(a,
- b,
c) S
|
sk_for_each_rcu(a,
- b,
c) S
|
sk_for_each_from
-(a, b)
+(a)
S
+ sk_for_each_from(a) S
|
sk_for_each_safe(a,
- b,
c, d) S
|
sk_for_each_bound(a,
- b,
c) S
|
hlist_for_each_entry_safe(a,
- b,
c, d, e) S
|
hlist_for_each_entry_continue_rcu(a,
- b,
c) S
|
nr_neigh_for_each(a,
- b,
c) S
|
nr_neigh_for_each_safe(a,
- b,
c, d) S
|
nr_node_for_each(a,
- b,
c) S
|
nr_node_for_each_safe(a,
- b,
c, d) S
|
- for_each_gfn_sp(a, c, d, b) S
+ for_each_gfn_sp(a, c, d) S
|
- for_each_gfn_indirect_valid_sp(a, c, d, b) S
+ for_each_gfn_indirect_valid_sp(a, c, d) S
|
for_each_host(a,
- b,
c) S
|
for_each_host_safe(a,
- b,
c, d) S
|
for_each_mesh_entry(a,
- b,
c, d) S
)
...+>
[akpm@linux-foundation.org: drop bogus change from net/ipv4/raw.c]
[akpm@linux-foundation.org: drop bogus hunk from net/ipv6/raw.c]
[akpm@linux-foundation.org: checkpatch fixes]
[akpm@linux-foundation.org: fix warnings]
[akpm@linux-foudnation.org: redo intrusive kvm changes]
Tested-by: Peter Senna Tschudin <peter.senna@gmail.com>
Acked-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Signed-off-by: Sasha Levin <sasha.levin@oracle.com>
Cc: Wu Fengguang <fengguang.wu@intel.com>
Cc: Marcelo Tosatti <mtosatti@redhat.com>
Cc: Gleb Natapov <gleb@redhat.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2013-02-28 08:06:00 +07:00
|
|
|
hlist_for_each_entry_rcu(f, &vxlan->fdb_head[h], hlist) {
|
2013-03-15 11:35:51 +07:00
|
|
|
struct vxlan_rdst *rd;
|
|
|
|
|
2013-06-18 04:16:12 +07:00
|
|
|
list_for_each_entry_rcu(rd, &f->remotes, list) {
|
2015-08-10 21:39:09 +07:00
|
|
|
if (idx < cb->args[0])
|
|
|
|
goto skip;
|
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
err = vxlan_fdb_info(skb, vxlan, f,
|
|
|
|
NETLINK_CB(cb->skb).portid,
|
|
|
|
cb->nlh->nlmsg_seq,
|
|
|
|
RTM_NEWNEIGH,
|
|
|
|
NLM_F_MULTI, rd);
|
|
|
|
if (err < 0)
|
2013-06-18 04:16:12 +07:00
|
|
|
goto out;
|
|
|
|
skip:
|
2015-08-10 21:39:09 +07:00
|
|
|
++idx;
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
}
|
2013-06-18 04:16:12 +07:00
|
|
|
out:
|
2012-10-01 19:32:35 +07:00
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Watch incoming packets to learn mapping between Ethernet address
|
|
|
|
* and Tunnel endpoint.
|
2015-04-02 09:17:58 +07:00
|
|
|
* Return true if packet is bogus and should be dropped.
|
2012-10-01 19:32:35 +07:00
|
|
|
*/
|
2013-06-18 02:09:58 +07:00
|
|
|
static bool vxlan_snoop(struct net_device *dev,
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *src_ip, const u8 *src_mac)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct vxlan_fdb *f;
|
|
|
|
|
|
|
|
f = vxlan_find_mac(vxlan, src_mac);
|
|
|
|
if (likely(f)) {
|
2013-08-05 07:17:39 +07:00
|
|
|
struct vxlan_rdst *rdst = first_remote_rcu(f);
|
2013-06-18 04:16:12 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (likely(vxlan_addr_equal(&rdst->remote_ip, src_ip)))
|
2013-06-18 02:09:58 +07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Don't migrate static entries, drop packets */
|
2013-06-19 04:27:01 +07:00
|
|
|
if (f->state & NUD_NOARP)
|
2013-06-18 02:09:58 +07:00
|
|
|
return true;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
if (net_ratelimit())
|
|
|
|
netdev_info(dev,
|
2013-08-31 12:44:33 +07:00
|
|
|
"%pM migrated from %pIS to %pIS\n",
|
2015-02-07 09:17:31 +07:00
|
|
|
src_mac, &rdst->remote_ip.sa, &src_ip->sa);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
rdst->remote_ip = *src_ip;
|
2012-10-01 19:32:35 +07:00
|
|
|
f->updated = jiffies;
|
2014-04-22 20:01:30 +07:00
|
|
|
vxlan_fdb_notify(vxlan, f, rdst, RTM_NEWNEIGH);
|
2012-10-01 19:32:35 +07:00
|
|
|
} else {
|
|
|
|
/* learned new entry */
|
|
|
|
spin_lock(&vxlan->hash_lock);
|
2013-06-18 02:09:57 +07:00
|
|
|
|
|
|
|
/* close off race between vxlan_flush and incoming packets */
|
|
|
|
if (netif_running(dev))
|
|
|
|
vxlan_fdb_create(vxlan, src_mac, src_ip,
|
|
|
|
NUD_REACHABLE,
|
|
|
|
NLM_F_EXCL|NLM_F_CREATE,
|
2015-07-21 15:44:02 +07:00
|
|
|
vxlan->cfg.dst_port,
|
2013-06-18 02:09:57 +07:00
|
|
|
vxlan->default_dst.remote_vni,
|
|
|
|
0, NTF_SELF);
|
2012-10-01 19:32:35 +07:00
|
|
|
spin_unlock(&vxlan->hash_lock);
|
|
|
|
}
|
2013-06-18 02:09:58 +07:00
|
|
|
|
|
|
|
return false;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* See if multicast group is already in use by other ID */
|
2013-12-10 15:37:33 +07:00
|
|
|
static bool vxlan_group_used(struct vxlan_net *vn, struct vxlan_dev *dev)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
2013-05-16 18:35:20 +07:00
|
|
|
struct vxlan_dev *vxlan;
|
2015-09-24 18:50:02 +07:00
|
|
|
unsigned short family = dev->default_dst.remote_ip.sa.sa_family;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-12-10 15:37:33 +07:00
|
|
|
/* The vxlan_sock is only used by dev, leaving group has
|
|
|
|
* no effect on other vxlan devices.
|
|
|
|
*/
|
2015-09-24 18:50:02 +07:00
|
|
|
if (family == AF_INET && dev->vn4_sock &&
|
|
|
|
atomic_read(&dev->vn4_sock->refcnt) == 1)
|
2013-12-10 15:37:33 +07:00
|
|
|
return false;
|
2015-09-24 18:50:02 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
if (family == AF_INET6 && dev->vn6_sock &&
|
|
|
|
atomic_read(&dev->vn6_sock->refcnt) == 1)
|
|
|
|
return false;
|
|
|
|
#endif
|
2013-12-10 15:37:33 +07:00
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
list_for_each_entry(vxlan, &vn->vxlan_list, next) {
|
2013-12-10 15:37:33 +07:00
|
|
|
if (!netif_running(vxlan->dev) || vxlan == dev)
|
2013-05-16 18:35:20 +07:00
|
|
|
continue;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
if (family == AF_INET && vxlan->vn4_sock != dev->vn4_sock)
|
2013-12-10 15:37:33 +07:00
|
|
|
continue;
|
2015-09-24 18:50:02 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
if (family == AF_INET6 && vxlan->vn6_sock != dev->vn6_sock)
|
|
|
|
continue;
|
|
|
|
#endif
|
2013-12-10 15:37:33 +07:00
|
|
|
|
|
|
|
if (!vxlan_addr_equal(&vxlan->default_dst.remote_ip,
|
|
|
|
&dev->default_dst.remote_ip))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (vxlan->default_dst.remote_ifindex !=
|
|
|
|
dev->default_dst.remote_ifindex)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return true;
|
2013-05-16 18:35:20 +07:00
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
static void __vxlan_sock_release(struct vxlan_sock *vs)
|
2013-06-18 04:16:10 +07:00
|
|
|
{
|
2015-09-24 18:50:02 +07:00
|
|
|
struct vxlan_net *vn;
|
2013-08-20 01:23:07 +07:00
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
if (!vs)
|
|
|
|
return;
|
2013-06-18 04:16:10 +07:00
|
|
|
if (!atomic_dec_and_test(&vs->refcnt))
|
|
|
|
return;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
vn = net_generic(sock_net(vs->sock->sk), vxlan_net_id);
|
2013-06-18 04:16:11 +07:00
|
|
|
spin_lock(&vn->sock_lock);
|
2013-06-18 04:16:10 +07:00
|
|
|
hlist_del_rcu(&vs->hlist);
|
2014-01-20 18:59:21 +07:00
|
|
|
vxlan_notify_del_rx_port(vs);
|
2013-06-18 04:16:11 +07:00
|
|
|
spin_unlock(&vn->sock_lock);
|
|
|
|
|
2013-06-18 04:16:10 +07:00
|
|
|
queue_work(vxlan_wq, &vs->del_work);
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
static void vxlan_sock_release(struct vxlan_dev *vxlan)
|
|
|
|
{
|
|
|
|
__vxlan_sock_release(vxlan->vn4_sock);
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
__vxlan_sock_release(vxlan->vn6_sock);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-03-19 00:50:44 +07:00
|
|
|
/* Update multicast group membership when first VNI on
|
2015-04-02 09:17:58 +07:00
|
|
|
* multicast address is brought up
|
2013-06-18 04:16:10 +07:00
|
|
|
*/
|
2015-03-19 00:50:44 +07:00
|
|
|
static int vxlan_igmp_join(struct vxlan_dev *vxlan)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
2015-09-24 18:50:02 +07:00
|
|
|
struct sock *sk;
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *ip = &vxlan->default_dst.remote_ip;
|
|
|
|
int ifindex = vxlan->default_dst.remote_ifindex;
|
2015-03-20 20:26:21 +07:00
|
|
|
int ret = -EINVAL;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (ip->sa.sa_family == AF_INET) {
|
|
|
|
struct ip_mreqn mreq = {
|
|
|
|
.imr_multiaddr.s_addr = ip->sin.sin_addr.s_addr,
|
|
|
|
.imr_ifindex = ifindex,
|
|
|
|
};
|
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
sk = vxlan->vn4_sock->sock->sk;
|
|
|
|
lock_sock(sk);
|
2015-03-19 00:50:44 +07:00
|
|
|
ret = ip_mc_join_group(sk, &mreq);
|
2015-09-24 18:50:02 +07:00
|
|
|
release_sock(sk);
|
2013-08-31 12:44:33 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
2015-09-24 18:50:02 +07:00
|
|
|
sk = vxlan->vn6_sock->sock->sk;
|
|
|
|
lock_sock(sk);
|
2015-03-19 00:50:44 +07:00
|
|
|
ret = ipv6_stub->ipv6_sock_mc_join(sk, ifindex,
|
|
|
|
&ip->sin6.sin6_addr);
|
2015-09-24 18:50:02 +07:00
|
|
|
release_sock(sk);
|
2013-08-31 12:44:33 +07:00
|
|
|
#endif
|
|
|
|
}
|
2013-07-18 22:40:15 +07:00
|
|
|
|
2015-03-19 00:50:44 +07:00
|
|
|
return ret;
|
2013-07-18 22:40:15 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Inverse of vxlan_igmp_join when last VNI is brought down */
|
2015-03-19 00:50:44 +07:00
|
|
|
static int vxlan_igmp_leave(struct vxlan_dev *vxlan)
|
2013-07-18 22:40:15 +07:00
|
|
|
{
|
2015-09-24 18:50:02 +07:00
|
|
|
struct sock *sk;
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *ip = &vxlan->default_dst.remote_ip;
|
|
|
|
int ifindex = vxlan->default_dst.remote_ifindex;
|
2015-03-20 20:26:21 +07:00
|
|
|
int ret = -EINVAL;
|
2013-07-18 22:40:15 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (ip->sa.sa_family == AF_INET) {
|
|
|
|
struct ip_mreqn mreq = {
|
|
|
|
.imr_multiaddr.s_addr = ip->sin.sin_addr.s_addr,
|
|
|
|
.imr_ifindex = ifindex,
|
|
|
|
};
|
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
sk = vxlan->vn4_sock->sock->sk;
|
|
|
|
lock_sock(sk);
|
2015-03-19 00:50:44 +07:00
|
|
|
ret = ip_mc_leave_group(sk, &mreq);
|
2015-09-24 18:50:02 +07:00
|
|
|
release_sock(sk);
|
2013-08-31 12:44:33 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
2015-09-24 18:50:02 +07:00
|
|
|
sk = vxlan->vn6_sock->sock->sk;
|
|
|
|
lock_sock(sk);
|
2015-03-19 00:50:44 +07:00
|
|
|
ret = ipv6_stub->ipv6_sock_mc_drop(sk, ifindex,
|
|
|
|
&ip->sin6.sin6_addr);
|
2015-09-24 18:50:02 +07:00
|
|
|
release_sock(sk);
|
2013-08-31 12:44:33 +07:00
|
|
|
#endif
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-03-19 00:50:44 +07:00
|
|
|
return ret;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
static struct vxlanhdr *vxlan_remcsum(struct sk_buff *skb, struct vxlanhdr *vh,
|
2015-02-11 07:30:32 +07:00
|
|
|
size_t hdrlen, u32 data, bool nopartial)
|
2015-01-13 08:00:38 +07:00
|
|
|
{
|
|
|
|
size_t start, offset, plen;
|
|
|
|
|
2015-08-20 07:07:32 +07:00
|
|
|
if (skb->remcsum_offload)
|
|
|
|
return vh;
|
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
start = (data & VXLAN_RCO_MASK) << VXLAN_RCO_SHIFT;
|
|
|
|
offset = start + ((data & VXLAN_RCO_UDP) ?
|
|
|
|
offsetof(struct udphdr, check) :
|
|
|
|
offsetof(struct tcphdr, check));
|
|
|
|
|
|
|
|
plen = hdrlen + offset + sizeof(u16);
|
|
|
|
|
|
|
|
if (!pskb_may_pull(skb, plen))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
vh = (struct vxlanhdr *)(udp_hdr(skb) + 1);
|
|
|
|
|
2015-02-11 07:30:32 +07:00
|
|
|
skb_remcsum_process(skb, (void *)vh + hdrlen, start, offset,
|
|
|
|
nopartial);
|
2015-01-13 08:00:38 +07:00
|
|
|
|
|
|
|
return vh;
|
|
|
|
}
|
|
|
|
|
2015-07-21 15:44:06 +07:00
|
|
|
static void vxlan_rcv(struct vxlan_sock *vs, struct sk_buff *skb,
|
|
|
|
struct vxlan_metadata *md, u32 vni,
|
|
|
|
struct metadata_dst *tun_dst)
|
|
|
|
{
|
|
|
|
struct iphdr *oip = NULL;
|
|
|
|
struct ipv6hdr *oip6 = NULL;
|
|
|
|
struct vxlan_dev *vxlan;
|
|
|
|
struct pcpu_sw_netstats *stats;
|
|
|
|
union vxlan_addr saddr;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
/* For flow based devices, map all packets to VNI 0 */
|
2015-08-05 12:51:07 +07:00
|
|
|
if (vs->flags & VXLAN_F_COLLECT_METADATA)
|
2015-07-21 15:44:06 +07:00
|
|
|
vni = 0;
|
|
|
|
|
|
|
|
/* Is this VNI defined? */
|
|
|
|
vxlan = vxlan_vs_find_vni(vs, vni);
|
|
|
|
if (!vxlan)
|
|
|
|
goto drop;
|
|
|
|
|
|
|
|
skb_reset_mac_header(skb);
|
|
|
|
skb_scrub_packet(skb, !net_eq(vxlan->net, dev_net(vxlan->dev)));
|
|
|
|
skb->protocol = eth_type_trans(skb, vxlan->dev);
|
|
|
|
skb_postpull_rcsum(skb, eth_hdr(skb), ETH_HLEN);
|
|
|
|
|
|
|
|
/* Ignore packet loops (and multicast echo) */
|
|
|
|
if (ether_addr_equal(eth_hdr(skb)->h_source, vxlan->dev->dev_addr))
|
|
|
|
goto drop;
|
|
|
|
|
2015-12-07 22:29:08 +07:00
|
|
|
/* Get data from the outer IP header */
|
|
|
|
if (vxlan_get_sk_family(vs) == AF_INET) {
|
2015-07-21 15:44:06 +07:00
|
|
|
oip = ip_hdr(skb);
|
|
|
|
saddr.sin.sin_addr.s_addr = oip->saddr;
|
|
|
|
saddr.sa.sa_family = AF_INET;
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
|
|
|
oip6 = ipv6_hdr(skb);
|
|
|
|
saddr.sin6.sin6_addr = oip6->saddr;
|
|
|
|
saddr.sa.sa_family = AF_INET6;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tun_dst) {
|
|
|
|
skb_dst_set(skb, (struct dst_entry *)tun_dst);
|
|
|
|
tun_dst = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((vxlan->flags & VXLAN_F_LEARN) &&
|
|
|
|
vxlan_snoop(skb->dev, &saddr, eth_hdr(skb)->h_source))
|
|
|
|
goto drop;
|
|
|
|
|
|
|
|
skb_reset_network_header(skb);
|
|
|
|
/* In flow-based mode, GBP is carried in dst_metadata */
|
2015-08-05 12:51:07 +07:00
|
|
|
if (!(vs->flags & VXLAN_F_COLLECT_METADATA))
|
2015-07-21 15:44:06 +07:00
|
|
|
skb->mark = md->gbp;
|
|
|
|
|
|
|
|
if (oip6)
|
|
|
|
err = IP6_ECN_decapsulate(oip6, skb);
|
|
|
|
if (oip)
|
|
|
|
err = IP_ECN_decapsulate(oip, skb);
|
|
|
|
|
|
|
|
if (unlikely(err)) {
|
|
|
|
if (log_ecn_error) {
|
|
|
|
if (oip6)
|
|
|
|
net_info_ratelimited("non-ECT from %pI6\n",
|
|
|
|
&oip6->saddr);
|
|
|
|
if (oip)
|
|
|
|
net_info_ratelimited("non-ECT from %pI4 with TOS=%#x\n",
|
|
|
|
&oip->saddr, oip->tos);
|
|
|
|
}
|
|
|
|
if (err > 1) {
|
|
|
|
++vxlan->dev->stats.rx_frame_errors;
|
|
|
|
++vxlan->dev->stats.rx_errors;
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stats = this_cpu_ptr(vxlan->dev->tstats);
|
|
|
|
u64_stats_update_begin(&stats->syncp);
|
|
|
|
stats->rx_packets++;
|
|
|
|
stats->rx_bytes += skb->len;
|
|
|
|
u64_stats_update_end(&stats->syncp);
|
|
|
|
|
2015-08-20 07:07:33 +07:00
|
|
|
gro_cells_receive(&vxlan->gro_cells, skb);
|
2015-07-21 15:44:06 +07:00
|
|
|
|
|
|
|
return;
|
|
|
|
drop:
|
|
|
|
if (tun_dst)
|
|
|
|
dst_release((struct dst_entry *)tun_dst);
|
|
|
|
|
|
|
|
/* Consume bad packet */
|
|
|
|
kfree_skb(skb);
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Callback from net/ipv4/udp.c to receive packets */
|
|
|
|
static int vxlan_udp_encap_recv(struct sock *sk, struct sk_buff *skb)
|
|
|
|
{
|
2015-07-21 15:43:58 +07:00
|
|
|
struct metadata_dst *tun_dst = NULL;
|
2013-08-20 01:23:02 +07:00
|
|
|
struct vxlan_sock *vs;
|
2012-10-01 19:32:35 +07:00
|
|
|
struct vxlanhdr *vxh;
|
2015-01-09 03:31:18 +07:00
|
|
|
u32 flags, vni;
|
2015-07-21 15:43:58 +07:00
|
|
|
struct vxlan_metadata _md;
|
|
|
|
struct vxlan_metadata *md = &_md;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
/* Need Vxlan and inner Ethernet header to be present */
|
2013-08-20 01:22:54 +07:00
|
|
|
if (!pskb_may_pull(skb, VXLAN_HLEN))
|
2012-10-01 19:32:35 +07:00
|
|
|
goto error;
|
|
|
|
|
2013-08-20 01:22:54 +07:00
|
|
|
vxh = (struct vxlanhdr *)(udp_hdr(skb) + 1);
|
2015-01-09 03:31:18 +07:00
|
|
|
flags = ntohl(vxh->vx_flags);
|
|
|
|
vni = ntohl(vxh->vx_vni);
|
|
|
|
|
|
|
|
if (flags & VXLAN_HF_VNI) {
|
|
|
|
flags &= ~VXLAN_HF_VNI;
|
|
|
|
} else {
|
|
|
|
/* VNI flag always required to be set */
|
|
|
|
goto bad_flags;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
2013-08-20 01:23:02 +07:00
|
|
|
if (iptunnel_pull_header(skb, VXLAN_HLEN, htons(ETH_P_TEB)))
|
|
|
|
goto drop;
|
2015-01-13 08:00:38 +07:00
|
|
|
vxh = (struct vxlanhdr *)(udp_hdr(skb) + 1);
|
2013-08-20 01:23:02 +07:00
|
|
|
|
2013-09-25 00:25:40 +07:00
|
|
|
vs = rcu_dereference_sk_user_data(sk);
|
2013-08-20 01:23:02 +07:00
|
|
|
if (!vs)
|
2012-10-01 19:32:35 +07:00
|
|
|
goto drop;
|
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
if ((flags & VXLAN_HF_RCO) && (vs->flags & VXLAN_F_REMCSUM_RX)) {
|
2015-02-11 07:30:32 +07:00
|
|
|
vxh = vxlan_remcsum(skb, vxh, sizeof(struct vxlanhdr), vni,
|
|
|
|
!!(vs->flags & VXLAN_F_REMCSUM_NOPARTIAL));
|
2015-01-13 08:00:38 +07:00
|
|
|
if (!vxh)
|
|
|
|
goto drop;
|
|
|
|
|
|
|
|
flags &= ~VXLAN_HF_RCO;
|
2015-03-13 23:13:53 +07:00
|
|
|
vni &= VXLAN_VNI_MASK;
|
2015-01-13 08:00:38 +07:00
|
|
|
}
|
|
|
|
|
2015-07-21 15:43:58 +07:00
|
|
|
if (vxlan_collect_metadata(vs)) {
|
2015-08-27 13:46:50 +07:00
|
|
|
tun_dst = udp_tun_rx_dst(skb, vxlan_get_sk_family(vs), TUNNEL_KEY,
|
|
|
|
cpu_to_be64(vni >> 8), sizeof(*md));
|
|
|
|
|
2015-07-21 15:43:58 +07:00
|
|
|
if (!tun_dst)
|
|
|
|
goto drop;
|
|
|
|
|
2015-09-04 17:49:32 +07:00
|
|
|
md = ip_tunnel_info_opts(&tun_dst->u.tun_info);
|
2015-07-21 15:43:58 +07:00
|
|
|
} else {
|
|
|
|
memset(md, 0, sizeof(*md));
|
|
|
|
}
|
|
|
|
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
/* For backwards compatibility, only allow reserved fields to be
|
|
|
|
* used by VXLAN extensions if explicitly requested.
|
|
|
|
*/
|
|
|
|
if ((flags & VXLAN_HF_GBP) && (vs->flags & VXLAN_F_GBP)) {
|
|
|
|
struct vxlanhdr_gbp *gbp;
|
|
|
|
|
|
|
|
gbp = (struct vxlanhdr_gbp *)vxh;
|
2015-07-21 15:43:58 +07:00
|
|
|
md->gbp = ntohs(gbp->policy_id);
|
|
|
|
|
|
|
|
if (tun_dst)
|
2015-09-04 17:49:32 +07:00
|
|
|
tun_dst->u.tun_info.key.tun_flags |= TUNNEL_VXLAN_OPT;
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
|
|
|
|
if (gbp->dont_learn)
|
2015-07-21 15:43:58 +07:00
|
|
|
md->gbp |= VXLAN_GBP_DONT_LEARN;
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
|
|
|
|
if (gbp->policy_applied)
|
2015-07-21 15:43:58 +07:00
|
|
|
md->gbp |= VXLAN_GBP_POLICY_APPLIED;
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
|
|
|
|
flags &= ~VXLAN_GBP_USED_BITS;
|
|
|
|
}
|
|
|
|
|
2015-03-13 23:13:53 +07:00
|
|
|
if (flags || vni & ~VXLAN_VNI_MASK) {
|
2015-01-09 03:31:18 +07:00
|
|
|
/* If there are any unprocessed flags remaining treat
|
|
|
|
* this as a malformed packet. This behavior diverges from
|
|
|
|
* VXLAN RFC (RFC7348) which stipulates that bits in reserved
|
|
|
|
* in reserved fields are to be ignored. The approach here
|
2015-04-02 09:17:58 +07:00
|
|
|
* maintains compatibility with previous stack code, and also
|
2015-01-09 03:31:18 +07:00
|
|
|
* is more robust and provides a little more security in
|
|
|
|
* adding extensions to VXLAN.
|
|
|
|
*/
|
|
|
|
|
|
|
|
goto bad_flags;
|
|
|
|
}
|
|
|
|
|
2015-07-21 15:44:06 +07:00
|
|
|
vxlan_rcv(vs, skb, md, vni >> 8, tun_dst);
|
2013-08-20 01:23:02 +07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
drop:
|
|
|
|
/* Consume bad packet */
|
|
|
|
kfree_skb(skb);
|
|
|
|
return 0;
|
|
|
|
|
2015-01-09 03:31:18 +07:00
|
|
|
bad_flags:
|
|
|
|
netdev_dbg(skb->dev, "invalid vxlan flags=%#x vni=%#x\n",
|
|
|
|
ntohl(vxh->vx_flags), ntohl(vxh->vx_vni));
|
|
|
|
|
2013-08-20 01:23:02 +07:00
|
|
|
error:
|
2015-07-21 15:43:58 +07:00
|
|
|
if (tun_dst)
|
|
|
|
dst_release((struct dst_entry *)tun_dst);
|
|
|
|
|
2013-08-20 01:23:02 +07:00
|
|
|
/* Return non vxlan pkt */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
static int arp_reduce(struct net_device *dev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct arphdr *parp;
|
|
|
|
u8 *arpptr, *sha;
|
|
|
|
__be32 sip, tip;
|
|
|
|
struct neighbour *n;
|
|
|
|
|
|
|
|
if (dev->flags & IFF_NOARP)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (!pskb_may_pull(skb, arp_hdr_len(dev))) {
|
|
|
|
dev->stats.tx_dropped++;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
parp = arp_hdr(skb);
|
|
|
|
|
|
|
|
if ((parp->ar_hrd != htons(ARPHRD_ETHER) &&
|
|
|
|
parp->ar_hrd != htons(ARPHRD_IEEE802)) ||
|
|
|
|
parp->ar_pro != htons(ETH_P_IP) ||
|
|
|
|
parp->ar_op != htons(ARPOP_REQUEST) ||
|
|
|
|
parp->ar_hln != dev->addr_len ||
|
|
|
|
parp->ar_pln != 4)
|
|
|
|
goto out;
|
|
|
|
arpptr = (u8 *)parp + sizeof(struct arphdr);
|
|
|
|
sha = arpptr;
|
|
|
|
arpptr += dev->addr_len; /* sha */
|
|
|
|
memcpy(&sip, arpptr, sizeof(sip));
|
|
|
|
arpptr += sizeof(sip);
|
|
|
|
arpptr += dev->addr_len; /* tha */
|
|
|
|
memcpy(&tip, arpptr, sizeof(tip));
|
|
|
|
|
|
|
|
if (ipv4_is_loopback(tip) ||
|
|
|
|
ipv4_is_multicast(tip))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
n = neigh_lookup(&arp_tbl, &tip, dev);
|
|
|
|
|
|
|
|
if (n) {
|
|
|
|
struct vxlan_fdb *f;
|
|
|
|
struct sk_buff *reply;
|
|
|
|
|
|
|
|
if (!(n->nud_state & NUD_CONNECTED)) {
|
|
|
|
neigh_release(n);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
f = vxlan_find_mac(vxlan, n->ha);
|
2013-08-31 12:44:33 +07:00
|
|
|
if (f && vxlan_addr_any(&(first_remote_rcu(f)->remote_ip))) {
|
2012-11-20 09:50:14 +07:00
|
|
|
/* bridge-local neighbor */
|
|
|
|
neigh_release(n);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
reply = arp_create(ARPOP_REPLY, ETH_P_ARP, sip, dev, tip, sha,
|
|
|
|
n->ha, sha);
|
|
|
|
|
|
|
|
neigh_release(n);
|
|
|
|
|
2014-03-18 23:32:29 +07:00
|
|
|
if (reply == NULL)
|
|
|
|
goto out;
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
skb_reset_mac_header(reply);
|
|
|
|
__skb_pull(reply, skb_network_offset(reply));
|
|
|
|
reply->ip_summed = CHECKSUM_UNNECESSARY;
|
|
|
|
reply->pkt_type = PACKET_HOST;
|
|
|
|
|
|
|
|
if (netif_rx_ni(reply) == NET_RX_DROP)
|
|
|
|
dev->stats.rx_dropped++;
|
2013-08-31 12:44:33 +07:00
|
|
|
} else if (vxlan->flags & VXLAN_F_L3MISS) {
|
|
|
|
union vxlan_addr ipa = {
|
|
|
|
.sin.sin_addr.s_addr = tip,
|
2014-08-23 02:34:16 +07:00
|
|
|
.sin.sin_family = AF_INET,
|
2013-08-31 12:44:33 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
vxlan_ip_miss(dev, &ipa);
|
|
|
|
}
|
2012-11-20 09:50:14 +07:00
|
|
|
out:
|
|
|
|
consume_skb(skb);
|
|
|
|
return NETDEV_TX_OK;
|
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:36 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
static struct sk_buff *vxlan_na_create(struct sk_buff *request,
|
|
|
|
struct neighbour *n, bool isrouter)
|
|
|
|
{
|
|
|
|
struct net_device *dev = request->dev;
|
|
|
|
struct sk_buff *reply;
|
|
|
|
struct nd_msg *ns, *na;
|
|
|
|
struct ipv6hdr *pip6;
|
|
|
|
u8 *daddr;
|
|
|
|
int na_olen = 8; /* opt hdr + ETH_ALEN for target */
|
|
|
|
int ns_olen;
|
|
|
|
int i, len;
|
|
|
|
|
|
|
|
if (dev == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
len = LL_RESERVED_SPACE(dev) + sizeof(struct ipv6hdr) +
|
|
|
|
sizeof(*na) + na_olen + dev->needed_tailroom;
|
|
|
|
reply = alloc_skb(len, GFP_ATOMIC);
|
|
|
|
if (reply == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
reply->protocol = htons(ETH_P_IPV6);
|
|
|
|
reply->dev = dev;
|
|
|
|
skb_reserve(reply, LL_RESERVED_SPACE(request->dev));
|
|
|
|
skb_push(reply, sizeof(struct ethhdr));
|
|
|
|
skb_set_mac_header(reply, 0);
|
|
|
|
|
|
|
|
ns = (struct nd_msg *)skb_transport_header(request);
|
|
|
|
|
|
|
|
daddr = eth_hdr(request)->h_source;
|
|
|
|
ns_olen = request->len - skb_transport_offset(request) - sizeof(*ns);
|
|
|
|
for (i = 0; i < ns_olen-1; i += (ns->opt[i+1]<<3)) {
|
|
|
|
if (ns->opt[i] == ND_OPT_SOURCE_LL_ADDR) {
|
|
|
|
daddr = ns->opt + i + sizeof(struct nd_opt_hdr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ethernet header */
|
|
|
|
ether_addr_copy(eth_hdr(reply)->h_dest, daddr);
|
|
|
|
ether_addr_copy(eth_hdr(reply)->h_source, n->ha);
|
|
|
|
eth_hdr(reply)->h_proto = htons(ETH_P_IPV6);
|
|
|
|
reply->protocol = htons(ETH_P_IPV6);
|
|
|
|
|
|
|
|
skb_pull(reply, sizeof(struct ethhdr));
|
|
|
|
skb_set_network_header(reply, 0);
|
|
|
|
skb_put(reply, sizeof(struct ipv6hdr));
|
|
|
|
|
|
|
|
/* IPv6 header */
|
|
|
|
|
|
|
|
pip6 = ipv6_hdr(reply);
|
|
|
|
memset(pip6, 0, sizeof(struct ipv6hdr));
|
|
|
|
pip6->version = 6;
|
|
|
|
pip6->priority = ipv6_hdr(request)->priority;
|
|
|
|
pip6->nexthdr = IPPROTO_ICMPV6;
|
|
|
|
pip6->hop_limit = 255;
|
|
|
|
pip6->daddr = ipv6_hdr(request)->saddr;
|
|
|
|
pip6->saddr = *(struct in6_addr *)n->primary_key;
|
|
|
|
|
|
|
|
skb_pull(reply, sizeof(struct ipv6hdr));
|
|
|
|
skb_set_transport_header(reply, 0);
|
|
|
|
|
|
|
|
na = (struct nd_msg *)skb_put(reply, sizeof(*na) + na_olen);
|
|
|
|
|
|
|
|
/* Neighbor Advertisement */
|
|
|
|
memset(na, 0, sizeof(*na)+na_olen);
|
|
|
|
na->icmph.icmp6_type = NDISC_NEIGHBOUR_ADVERTISEMENT;
|
|
|
|
na->icmph.icmp6_router = isrouter;
|
|
|
|
na->icmph.icmp6_override = 1;
|
|
|
|
na->icmph.icmp6_solicited = 1;
|
|
|
|
na->target = ns->target;
|
|
|
|
ether_addr_copy(&na->opt[2], n->ha);
|
|
|
|
na->opt[0] = ND_OPT_TARGET_LL_ADDR;
|
|
|
|
na->opt[1] = na_olen >> 3;
|
|
|
|
|
|
|
|
na->icmph.icmp6_cksum = csum_ipv6_magic(&pip6->saddr,
|
|
|
|
&pip6->daddr, sizeof(*na)+na_olen, IPPROTO_ICMPV6,
|
|
|
|
csum_partial(na, sizeof(*na)+na_olen, 0));
|
|
|
|
|
|
|
|
pip6->payload_len = htons(sizeof(*na)+na_olen);
|
|
|
|
|
|
|
|
skb_push(reply, sizeof(struct ipv6hdr));
|
|
|
|
|
|
|
|
reply->ip_summed = CHECKSUM_UNNECESSARY;
|
|
|
|
|
|
|
|
return reply;
|
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:36 +07:00
|
|
|
static int neigh_reduce(struct net_device *dev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
struct nd_msg *msg;
|
2013-08-31 12:44:36 +07:00
|
|
|
const struct ipv6hdr *iphdr;
|
|
|
|
const struct in6_addr *saddr, *daddr;
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
struct neighbour *n;
|
|
|
|
struct inet6_dev *in6_dev;
|
2013-08-31 12:44:36 +07:00
|
|
|
|
|
|
|
in6_dev = __in6_dev_get(dev);
|
|
|
|
if (!in6_dev)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
iphdr = ipv6_hdr(skb);
|
|
|
|
saddr = &iphdr->saddr;
|
|
|
|
daddr = &iphdr->daddr;
|
|
|
|
|
|
|
|
msg = (struct nd_msg *)skb_transport_header(skb);
|
|
|
|
if (msg->icmph.icmp6_code != 0 ||
|
|
|
|
msg->icmph.icmp6_type != NDISC_NEIGHBOUR_SOLICITATION)
|
|
|
|
goto out;
|
|
|
|
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
if (ipv6_addr_loopback(daddr) ||
|
|
|
|
ipv6_addr_is_multicast(&msg->target))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
n = neigh_lookup(ipv6_stub->nd_tbl, &msg->target, dev);
|
2013-08-31 12:44:36 +07:00
|
|
|
|
|
|
|
if (n) {
|
|
|
|
struct vxlan_fdb *f;
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
struct sk_buff *reply;
|
2013-08-31 12:44:36 +07:00
|
|
|
|
|
|
|
if (!(n->nud_state & NUD_CONNECTED)) {
|
|
|
|
neigh_release(n);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
f = vxlan_find_mac(vxlan, n->ha);
|
|
|
|
if (f && vxlan_addr_any(&(first_remote_rcu(f)->remote_ip))) {
|
|
|
|
/* bridge-local neighbor */
|
|
|
|
neigh_release(n);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
reply = vxlan_na_create(skb, n,
|
|
|
|
!!(f ? f->flags & NTF_ROUTER : 0));
|
|
|
|
|
2013-08-31 12:44:36 +07:00
|
|
|
neigh_release(n);
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
|
|
|
|
if (reply == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (netif_rx_ni(reply) == NET_RX_DROP)
|
|
|
|
dev->stats.rx_dropped++;
|
|
|
|
|
2013-08-31 12:44:36 +07:00
|
|
|
} else if (vxlan->flags & VXLAN_F_L3MISS) {
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
union vxlan_addr ipa = {
|
|
|
|
.sin6.sin6_addr = msg->target,
|
2014-08-23 02:34:16 +07:00
|
|
|
.sin6.sin6_family = AF_INET6,
|
vxlan: fix nonfunctional neigh_reduce()
The VXLAN neigh_reduce() code is completely non-functional since
check-in. Specific errors:
1) The original code drops all packets with a multicast destination address,
even though neighbor solicitations are sent to the solicited-node
address, a multicast address. The code after this check was never run.
2) The neighbor table lookup used the IPv6 header destination, which is the
solicited node address, rather than the target address from the
neighbor solicitation. So neighbor lookups would always fail if it
got this far. Also for L3MISSes.
3) The code calls ndisc_send_na(), which does a send on the tunnel device.
The context for neigh_reduce() is the transmit path, vxlan_xmit(),
where the host or a bridge-attached neighbor is trying to transmit
a neighbor solicitation. To respond to it, the tunnel endpoint needs
to do a *receive* of the appropriate neighbor advertisement. Doing a
send, would only try to send the advertisement, encapsulated, to the
remote destinations in the fdb -- hosts that definitely did not do the
corresponding solicitation.
4) The code uses the tunnel endpoint IPv6 forwarding flag to determine the
isrouter flag in the advertisement. This has nothing to do with whether
or not the target is a router, and generally won't be set since the
tunnel endpoint is bridging, not routing, traffic.
The patch below creates a proxy neighbor advertisement to respond to
neighbor solicitions as intended, providing proper IPv6 support for neighbor
reduction.
Signed-off-by: David L Stevens <dlstevens@us.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-03-24 21:39:58 +07:00
|
|
|
};
|
|
|
|
|
2013-08-31 12:44:36 +07:00
|
|
|
vxlan_ip_miss(dev, &ipa);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
consume_skb(skb);
|
|
|
|
return NETDEV_TX_OK;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
static bool route_shortcircuit(struct net_device *dev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct neighbour *n;
|
|
|
|
|
|
|
|
if (is_multicast_ether_addr(eth_hdr(skb)->h_dest))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
n = NULL;
|
|
|
|
switch (ntohs(eth_hdr(skb)->h_proto)) {
|
|
|
|
case ETH_P_IP:
|
2013-08-31 12:44:34 +07:00
|
|
|
{
|
|
|
|
struct iphdr *pip;
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
if (!pskb_may_pull(skb, sizeof(struct iphdr)))
|
|
|
|
return false;
|
|
|
|
pip = ip_hdr(skb);
|
|
|
|
n = neigh_lookup(&arp_tbl, &pip->daddr, dev);
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!n && (vxlan->flags & VXLAN_F_L3MISS)) {
|
|
|
|
union vxlan_addr ipa = {
|
|
|
|
.sin.sin_addr.s_addr = pip->daddr,
|
2014-08-23 02:34:16 +07:00
|
|
|
.sin.sin_family = AF_INET,
|
2013-08-31 12:44:33 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
vxlan_ip_miss(dev, &ipa);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
break;
|
2013-08-31 12:44:34 +07:00
|
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
case ETH_P_IPV6:
|
|
|
|
{
|
|
|
|
struct ipv6hdr *pip6;
|
|
|
|
|
|
|
|
if (!pskb_may_pull(skb, sizeof(struct ipv6hdr)))
|
|
|
|
return false;
|
|
|
|
pip6 = ipv6_hdr(skb);
|
|
|
|
n = neigh_lookup(ipv6_stub->nd_tbl, &pip6->daddr, dev);
|
|
|
|
if (!n && (vxlan->flags & VXLAN_F_L3MISS)) {
|
|
|
|
union vxlan_addr ipa = {
|
|
|
|
.sin6.sin6_addr = pip6->daddr,
|
2014-08-23 02:34:16 +07:00
|
|
|
.sin6.sin6_family = AF_INET6,
|
2013-08-31 12:44:34 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
vxlan_ip_miss(dev, &ipa);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
2012-11-20 09:50:14 +07:00
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (n) {
|
|
|
|
bool diff;
|
|
|
|
|
drivers/net: Convert uses of compare_ether_addr to ether_addr_equal
Use the new bool function ether_addr_equal to add
some clarity and reduce the likelihood for misuse
of compare_ether_addr for sorting.
Done via cocci script: (and a little typing)
$ cat compare_ether_addr.cocci
@@
expression a,b;
@@
- !compare_ether_addr(a, b)
+ ether_addr_equal(a, b)
@@
expression a,b;
@@
- compare_ether_addr(a, b)
+ !ether_addr_equal(a, b)
@@
expression a,b;
@@
- !ether_addr_equal(a, b) == 0
+ ether_addr_equal(a, b)
@@
expression a,b;
@@
- !ether_addr_equal(a, b) != 0
+ !ether_addr_equal(a, b)
@@
expression a,b;
@@
- ether_addr_equal(a, b) == 0
+ !ether_addr_equal(a, b)
@@
expression a,b;
@@
- ether_addr_equal(a, b) != 0
+ ether_addr_equal(a, b)
@@
expression a,b;
@@
- !!ether_addr_equal(a, b)
+ ether_addr_equal(a, b)
Signed-off-by: Joe Perches <joe@perches.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2013-09-02 01:51:23 +07:00
|
|
|
diff = !ether_addr_equal(eth_hdr(skb)->h_dest, n->ha);
|
2012-11-20 09:50:14 +07:00
|
|
|
if (diff) {
|
|
|
|
memcpy(eth_hdr(skb)->h_source, eth_hdr(skb)->h_dest,
|
|
|
|
dev->addr_len);
|
|
|
|
memcpy(eth_hdr(skb)->h_dest, n->ha, dev->addr_len);
|
|
|
|
}
|
|
|
|
neigh_release(n);
|
|
|
|
return diff;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-01-21 02:23:05 +07:00
|
|
|
static void vxlan_build_gbp_hdr(struct vxlanhdr *vxh, u32 vxflags,
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
struct vxlan_metadata *md)
|
|
|
|
{
|
|
|
|
struct vxlanhdr_gbp *gbp;
|
|
|
|
|
2015-02-04 23:00:04 +07:00
|
|
|
if (!md->gbp)
|
|
|
|
return;
|
|
|
|
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
gbp = (struct vxlanhdr_gbp *)vxh;
|
|
|
|
vxh->vx_flags |= htonl(VXLAN_HF_GBP);
|
|
|
|
|
|
|
|
if (md->gbp & VXLAN_GBP_DONT_LEARN)
|
|
|
|
gbp->dont_learn = 1;
|
|
|
|
|
|
|
|
if (md->gbp & VXLAN_GBP_POLICY_APPLIED)
|
|
|
|
gbp->policy_applied = 1;
|
|
|
|
|
|
|
|
gbp->policy_id = htons(md->gbp & VXLAN_GBP_ID_MASK);
|
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
2015-04-06 09:19:09 +07:00
|
|
|
static int vxlan6_xmit_skb(struct dst_entry *dst, struct sock *sk,
|
|
|
|
struct sk_buff *skb,
|
2013-08-31 12:44:33 +07:00
|
|
|
struct net_device *dev, struct in6_addr *saddr,
|
|
|
|
struct in6_addr *daddr, __u8 prio, __u8 ttl,
|
2015-07-22 22:08:42 +07:00
|
|
|
__be16 src_port, __be16 dst_port, __be32 vni,
|
2015-01-21 02:23:05 +07:00
|
|
|
struct vxlan_metadata *md, bool xnet, u32 vxflags)
|
2013-08-31 12:44:33 +07:00
|
|
|
{
|
|
|
|
struct vxlanhdr *vxh;
|
|
|
|
int min_headroom;
|
|
|
|
int err;
|
2015-01-21 02:23:05 +07:00
|
|
|
bool udp_sum = !(vxflags & VXLAN_F_UDP_ZERO_CSUM6_TX);
|
2015-01-13 08:00:38 +07:00
|
|
|
int type = udp_sum ? SKB_GSO_UDP_TUNNEL_CSUM : SKB_GSO_UDP_TUNNEL;
|
|
|
|
u16 hdrlen = sizeof(struct vxlanhdr);
|
|
|
|
|
2015-01-21 02:23:05 +07:00
|
|
|
if ((vxflags & VXLAN_F_REMCSUM_TX) &&
|
2015-01-13 08:00:38 +07:00
|
|
|
skb->ip_summed == CHECKSUM_PARTIAL) {
|
|
|
|
int csum_start = skb_checksum_start_offset(skb);
|
|
|
|
|
|
|
|
if (csum_start <= VXLAN_MAX_REMCSUM_START &&
|
|
|
|
!(csum_start & VXLAN_RCO_SHIFT_MASK) &&
|
|
|
|
(skb->csum_offset == offsetof(struct udphdr, check) ||
|
|
|
|
skb->csum_offset == offsetof(struct tcphdr, check))) {
|
|
|
|
udp_sum = false;
|
|
|
|
type |= SKB_GSO_TUNNEL_REMCSUM;
|
|
|
|
}
|
|
|
|
}
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2014-04-24 15:02:49 +07:00
|
|
|
skb_scrub_packet(skb, xnet);
|
2013-09-02 20:34:57 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
min_headroom = LL_RESERVED_SPACE(dst->dev) + dst->header_len
|
|
|
|
+ VXLAN_HLEN + sizeof(struct ipv6hdr)
|
2015-01-13 23:13:44 +07:00
|
|
|
+ (skb_vlan_tag_present(skb) ? VLAN_HLEN : 0);
|
2013-08-31 12:44:33 +07:00
|
|
|
|
|
|
|
/* Need space for new headers (invalidates iph ptr) */
|
|
|
|
err = skb_cow_head(skb, min_headroom);
|
2014-12-24 07:20:36 +07:00
|
|
|
if (unlikely(err)) {
|
|
|
|
kfree_skb(skb);
|
|
|
|
goto err;
|
|
|
|
}
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2014-11-19 20:04:59 +07:00
|
|
|
skb = vlan_hwaccel_push_inside(skb);
|
2014-12-24 07:20:36 +07:00
|
|
|
if (WARN_ON(!skb)) {
|
|
|
|
err = -ENOMEM;
|
|
|
|
goto err;
|
|
|
|
}
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2015-04-10 01:19:14 +07:00
|
|
|
skb = iptunnel_handle_offloads(skb, udp_sum, type);
|
|
|
|
if (IS_ERR(skb)) {
|
|
|
|
err = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
vxh = (struct vxlanhdr *) __skb_push(skb, sizeof(*vxh));
|
2015-01-09 03:31:18 +07:00
|
|
|
vxh->vx_flags = htonl(VXLAN_HF_VNI);
|
2015-07-21 15:44:06 +07:00
|
|
|
vxh->vx_vni = vni;
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
if (type & SKB_GSO_TUNNEL_REMCSUM) {
|
|
|
|
u32 data = (skb_checksum_start_offset(skb) - hdrlen) >>
|
|
|
|
VXLAN_RCO_SHIFT;
|
|
|
|
|
|
|
|
if (skb->csum_offset == offsetof(struct udphdr, check))
|
|
|
|
data |= VXLAN_RCO_UDP;
|
|
|
|
|
|
|
|
vxh->vx_vni |= htonl(data);
|
|
|
|
vxh->vx_flags |= htonl(VXLAN_HF_RCO);
|
|
|
|
|
|
|
|
if (!skb_is_gso(skb)) {
|
|
|
|
skb->ip_summed = CHECKSUM_NONE;
|
|
|
|
skb->encapsulation = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-21 02:23:05 +07:00
|
|
|
if (vxflags & VXLAN_F_GBP)
|
|
|
|
vxlan_build_gbp_hdr(vxh, vxflags, md);
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
|
2014-09-30 10:22:33 +07:00
|
|
|
skb_set_inner_protocol(skb, htons(ETH_P_TEB));
|
|
|
|
|
2015-04-06 09:19:09 +07:00
|
|
|
udp_tunnel6_xmit_skb(dst, sk, skb, dev, saddr, daddr, prio,
|
2015-01-21 02:23:04 +07:00
|
|
|
ttl, src_port, dst_port,
|
2015-01-21 02:23:05 +07:00
|
|
|
!!(vxflags & VXLAN_F_UDP_ZERO_CSUM6_TX));
|
2013-08-31 12:44:33 +07:00
|
|
|
return 0;
|
2014-12-24 07:20:36 +07:00
|
|
|
err:
|
|
|
|
dst_release(dst);
|
|
|
|
return err;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-07-21 15:44:06 +07:00
|
|
|
static int vxlan_xmit_skb(struct rtable *rt, struct sock *sk, struct sk_buff *skb,
|
|
|
|
__be32 src, __be32 dst, __u8 tos, __u8 ttl, __be16 df,
|
2015-07-22 22:08:42 +07:00
|
|
|
__be16 src_port, __be16 dst_port, __be32 vni,
|
2015-07-21 15:44:06 +07:00
|
|
|
struct vxlan_metadata *md, bool xnet, u32 vxflags)
|
2013-08-20 01:23:17 +07:00
|
|
|
{
|
|
|
|
struct vxlanhdr *vxh;
|
2013-08-20 01:23:22 +07:00
|
|
|
int min_headroom;
|
2013-08-20 01:23:17 +07:00
|
|
|
int err;
|
2015-01-21 02:23:05 +07:00
|
|
|
bool udp_sum = !!(vxflags & VXLAN_F_UDP_CSUM);
|
2015-01-13 08:00:38 +07:00
|
|
|
int type = udp_sum ? SKB_GSO_UDP_TUNNEL_CSUM : SKB_GSO_UDP_TUNNEL;
|
|
|
|
u16 hdrlen = sizeof(struct vxlanhdr);
|
|
|
|
|
2015-01-21 02:23:05 +07:00
|
|
|
if ((vxflags & VXLAN_F_REMCSUM_TX) &&
|
2015-01-13 08:00:38 +07:00
|
|
|
skb->ip_summed == CHECKSUM_PARTIAL) {
|
|
|
|
int csum_start = skb_checksum_start_offset(skb);
|
|
|
|
|
|
|
|
if (csum_start <= VXLAN_MAX_REMCSUM_START &&
|
|
|
|
!(csum_start & VXLAN_RCO_SHIFT_MASK) &&
|
|
|
|
(skb->csum_offset == offsetof(struct udphdr, check) ||
|
|
|
|
skb->csum_offset == offsetof(struct tcphdr, check))) {
|
|
|
|
udp_sum = false;
|
|
|
|
type |= SKB_GSO_TUNNEL_REMCSUM;
|
|
|
|
}
|
|
|
|
}
|
2013-08-20 01:23:17 +07:00
|
|
|
|
2013-08-20 01:23:22 +07:00
|
|
|
min_headroom = LL_RESERVED_SPACE(rt->dst.dev) + rt->dst.header_len
|
2013-08-20 01:23:29 +07:00
|
|
|
+ VXLAN_HLEN + sizeof(struct iphdr)
|
2015-01-13 23:13:44 +07:00
|
|
|
+ (skb_vlan_tag_present(skb) ? VLAN_HLEN : 0);
|
2013-08-20 01:23:22 +07:00
|
|
|
|
|
|
|
/* Need space for new headers (invalidates iph ptr) */
|
|
|
|
err = skb_cow_head(skb, min_headroom);
|
2014-12-24 07:20:36 +07:00
|
|
|
if (unlikely(err)) {
|
|
|
|
kfree_skb(skb);
|
2013-08-20 01:23:22 +07:00
|
|
|
return err;
|
2014-12-24 07:20:36 +07:00
|
|
|
}
|
2013-08-20 01:23:22 +07:00
|
|
|
|
2014-11-19 20:04:59 +07:00
|
|
|
skb = vlan_hwaccel_push_inside(skb);
|
|
|
|
if (WARN_ON(!skb))
|
|
|
|
return -ENOMEM;
|
2013-08-20 01:23:29 +07:00
|
|
|
|
2015-04-10 01:19:14 +07:00
|
|
|
skb = iptunnel_handle_offloads(skb, udp_sum, type);
|
|
|
|
if (IS_ERR(skb))
|
|
|
|
return PTR_ERR(skb);
|
|
|
|
|
2013-08-20 01:23:17 +07:00
|
|
|
vxh = (struct vxlanhdr *) __skb_push(skb, sizeof(*vxh));
|
2015-01-09 03:31:18 +07:00
|
|
|
vxh->vx_flags = htonl(VXLAN_HF_VNI);
|
2015-07-21 15:44:06 +07:00
|
|
|
vxh->vx_vni = vni;
|
2013-08-20 01:23:17 +07:00
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
if (type & SKB_GSO_TUNNEL_REMCSUM) {
|
|
|
|
u32 data = (skb_checksum_start_offset(skb) - hdrlen) >>
|
|
|
|
VXLAN_RCO_SHIFT;
|
|
|
|
|
|
|
|
if (skb->csum_offset == offsetof(struct udphdr, check))
|
|
|
|
data |= VXLAN_RCO_UDP;
|
|
|
|
|
|
|
|
vxh->vx_vni |= htonl(data);
|
|
|
|
vxh->vx_flags |= htonl(VXLAN_HF_RCO);
|
|
|
|
|
|
|
|
if (!skb_is_gso(skb)) {
|
|
|
|
skb->ip_summed = CHECKSUM_NONE;
|
|
|
|
skb->encapsulation = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-21 02:23:05 +07:00
|
|
|
if (vxflags & VXLAN_F_GBP)
|
|
|
|
vxlan_build_gbp_hdr(vxh, vxflags, md);
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
|
2014-09-30 10:22:33 +07:00
|
|
|
skb_set_inner_protocol(skb, htons(ETH_P_TEB));
|
|
|
|
|
2015-04-06 09:19:09 +07:00
|
|
|
return udp_tunnel_xmit_skb(rt, sk, skb, src, dst, tos,
|
2015-01-21 02:23:04 +07:00
|
|
|
ttl, df, src_port, dst_port, xnet,
|
2015-01-21 02:23:05 +07:00
|
|
|
!(vxflags & VXLAN_F_UDP_CSUM));
|
2013-08-20 01:23:17 +07:00
|
|
|
}
|
|
|
|
|
2015-12-07 19:04:30 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
static struct dst_entry *vxlan6_get_route(struct vxlan_dev *vxlan,
|
|
|
|
struct sk_buff *skb, int oif,
|
|
|
|
const struct in6_addr *daddr,
|
|
|
|
struct in6_addr *saddr)
|
|
|
|
{
|
|
|
|
struct dst_entry *ndst;
|
|
|
|
struct flowi6 fl6;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
memset(&fl6, 0, sizeof(fl6));
|
|
|
|
fl6.flowi6_oif = oif;
|
|
|
|
fl6.daddr = *daddr;
|
|
|
|
fl6.saddr = vxlan->cfg.saddr.sin6.sin6_addr;
|
|
|
|
fl6.flowi6_mark = skb->mark;
|
|
|
|
fl6.flowi6_proto = IPPROTO_UDP;
|
|
|
|
|
|
|
|
err = ipv6_stub->ipv6_dst_lookup(vxlan->net,
|
|
|
|
vxlan->vn6_sock->sock->sk,
|
|
|
|
&ndst, &fl6);
|
|
|
|
if (err < 0)
|
|
|
|
return ERR_PTR(err);
|
|
|
|
|
|
|
|
*saddr = fl6.saddr;
|
|
|
|
return ndst;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-04-02 19:31:52 +07:00
|
|
|
/* Bypass encapsulation if the destination is local */
|
|
|
|
static void vxlan_encap_bypass(struct sk_buff *skb, struct vxlan_dev *src_vxlan,
|
|
|
|
struct vxlan_dev *dst_vxlan)
|
|
|
|
{
|
2014-01-04 12:57:59 +07:00
|
|
|
struct pcpu_sw_netstats *tx_stats, *rx_stats;
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr loopback;
|
|
|
|
union vxlan_addr *remote_ip = &dst_vxlan->default_dst.remote_ip;
|
2014-10-16 07:49:41 +07:00
|
|
|
struct net_device *dev = skb->dev;
|
|
|
|
int len = skb->len;
|
2013-04-02 19:31:52 +07:00
|
|
|
|
2014-01-04 12:57:59 +07:00
|
|
|
tx_stats = this_cpu_ptr(src_vxlan->dev->tstats);
|
|
|
|
rx_stats = this_cpu_ptr(dst_vxlan->dev->tstats);
|
2013-04-02 19:31:52 +07:00
|
|
|
skb->pkt_type = PACKET_HOST;
|
|
|
|
skb->encapsulation = 0;
|
|
|
|
skb->dev = dst_vxlan->dev;
|
|
|
|
__skb_pull(skb, skb_network_offset(skb));
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (remote_ip->sa.sa_family == AF_INET) {
|
|
|
|
loopback.sin.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
|
|
|
|
loopback.sa.sa_family = AF_INET;
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
|
|
|
loopback.sin6.sin6_addr = in6addr_loopback;
|
|
|
|
loopback.sa.sa_family = AF_INET6;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-04-02 19:31:52 +07:00
|
|
|
if (dst_vxlan->flags & VXLAN_F_LEARN)
|
2013-08-31 12:44:33 +07:00
|
|
|
vxlan_snoop(skb->dev, &loopback, eth_hdr(skb)->h_source);
|
2013-04-02 19:31:52 +07:00
|
|
|
|
|
|
|
u64_stats_update_begin(&tx_stats->syncp);
|
|
|
|
tx_stats->tx_packets++;
|
2014-10-16 07:49:41 +07:00
|
|
|
tx_stats->tx_bytes += len;
|
2013-04-02 19:31:52 +07:00
|
|
|
u64_stats_update_end(&tx_stats->syncp);
|
|
|
|
|
|
|
|
if (netif_rx(skb) == NET_RX_SUCCESS) {
|
|
|
|
u64_stats_update_begin(&rx_stats->syncp);
|
|
|
|
rx_stats->rx_packets++;
|
2014-10-16 07:49:41 +07:00
|
|
|
rx_stats->rx_bytes += len;
|
2013-04-02 19:31:52 +07:00
|
|
|
u64_stats_update_end(&rx_stats->syncp);
|
|
|
|
} else {
|
2014-10-16 07:49:41 +07:00
|
|
|
dev->stats.rx_dropped++;
|
2013-04-02 19:31:52 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-18 04:16:11 +07:00
|
|
|
static void vxlan_xmit_one(struct sk_buff *skb, struct net_device *dev,
|
|
|
|
struct vxlan_rdst *rdst, bool did_rsc)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
2015-07-21 15:44:00 +07:00
|
|
|
struct ip_tunnel_info *info;
|
2012-10-01 19:32:35 +07:00
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2015-09-24 18:50:02 +07:00
|
|
|
struct sock *sk;
|
2013-08-31 12:44:33 +07:00
|
|
|
struct rtable *rt = NULL;
|
2012-10-01 19:32:35 +07:00
|
|
|
const struct iphdr *old_iph;
|
|
|
|
struct flowi4 fl4;
|
2013-08-31 12:44:33 +07:00
|
|
|
union vxlan_addr *dst;
|
2015-07-21 15:43:58 +07:00
|
|
|
union vxlan_addr remote_ip;
|
|
|
|
struct vxlan_metadata _md;
|
|
|
|
struct vxlan_metadata *md = &_md;
|
2013-08-31 12:44:33 +07:00
|
|
|
__be16 src_port = 0, dst_port;
|
2013-06-18 04:16:41 +07:00
|
|
|
u32 vni;
|
2012-10-01 19:32:35 +07:00
|
|
|
__be16 df = 0;
|
|
|
|
__u8 tos, ttl;
|
2013-06-18 07:49:56 +07:00
|
|
|
int err;
|
2015-07-21 15:43:58 +07:00
|
|
|
u32 flags = vxlan->flags;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-08-20 18:56:25 +07:00
|
|
|
info = skb_tunnel_info(skb);
|
2015-07-21 15:44:00 +07:00
|
|
|
|
2015-07-21 15:43:58 +07:00
|
|
|
if (rdst) {
|
2015-07-21 15:44:02 +07:00
|
|
|
dst_port = rdst->remote_port ? rdst->remote_port : vxlan->cfg.dst_port;
|
2015-07-21 15:43:58 +07:00
|
|
|
vni = rdst->remote_vni;
|
|
|
|
dst = &rdst->remote_ip;
|
|
|
|
} else {
|
|
|
|
if (!info) {
|
|
|
|
WARN_ONCE(1, "%s: Missing encapsulation instructions\n",
|
|
|
|
dev->name);
|
|
|
|
goto drop;
|
|
|
|
}
|
2015-07-21 15:44:02 +07:00
|
|
|
dst_port = info->key.tp_dst ? : vxlan->cfg.dst_port;
|
2015-07-21 15:43:58 +07:00
|
|
|
vni = be64_to_cpu(info->key.tun_id);
|
2015-09-24 18:50:02 +07:00
|
|
|
remote_ip.sa.sa_family = ip_tunnel_info_af(info);
|
|
|
|
if (remote_ip.sa.sa_family == AF_INET)
|
2015-08-20 18:56:30 +07:00
|
|
|
remote_ip.sin.sin_addr.s_addr = info->key.u.ipv4.dst;
|
|
|
|
else
|
|
|
|
remote_ip.sin6.sin6_addr = info->key.u.ipv6.dst;
|
2015-07-21 15:43:58 +07:00
|
|
|
dst = &remote_ip;
|
|
|
|
}
|
2012-11-20 09:50:14 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (vxlan_addr_any(dst)) {
|
2012-11-20 09:50:14 +07:00
|
|
|
if (did_rsc) {
|
|
|
|
/* short-circuited back to local bridge */
|
2013-04-02 19:31:52 +07:00
|
|
|
vxlan_encap_bypass(skb, vxlan, vxlan);
|
2013-06-18 04:16:11 +07:00
|
|
|
return;
|
2012-11-20 09:50:14 +07:00
|
|
|
}
|
2012-10-10 03:35:46 +07:00
|
|
|
goto drop;
|
2012-11-20 09:50:14 +07:00
|
|
|
}
|
2012-10-10 03:35:46 +07:00
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
old_iph = ip_hdr(skb);
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
ttl = vxlan->cfg.ttl;
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!ttl && vxlan_addr_multicast(dst))
|
2012-10-01 19:32:35 +07:00
|
|
|
ttl = 1;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
tos = vxlan->cfg.tos;
|
2012-10-01 19:32:35 +07:00
|
|
|
if (tos == 1)
|
2013-03-25 21:49:53 +07:00
|
|
|
tos = ip_tunnel_get_dsfield(old_iph, skb);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
src_port = udp_flow_src_port(dev_net(dev), skb, vxlan->cfg.port_min,
|
|
|
|
vxlan->cfg.port_max, true);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-08-20 18:56:30 +07:00
|
|
|
if (info) {
|
|
|
|
if (info->key.tun_flags & TUNNEL_CSUM)
|
|
|
|
flags |= VXLAN_F_UDP_CSUM;
|
|
|
|
else
|
|
|
|
flags &= ~VXLAN_F_UDP_CSUM;
|
|
|
|
|
|
|
|
ttl = info->key.ttl;
|
|
|
|
tos = info->key.tos;
|
|
|
|
|
|
|
|
if (info->options_len)
|
2015-08-31 08:09:38 +07:00
|
|
|
md = ip_tunnel_info_opts(info);
|
2015-08-20 18:56:30 +07:00
|
|
|
} else {
|
|
|
|
md->gbp = skb->mark;
|
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (dst->sa.sa_family == AF_INET) {
|
2015-09-24 18:50:02 +07:00
|
|
|
if (!vxlan->vn4_sock)
|
|
|
|
goto drop;
|
|
|
|
sk = vxlan->vn4_sock->sock->sk;
|
|
|
|
|
2015-08-20 18:56:30 +07:00
|
|
|
if (info && (info->key.tun_flags & TUNNEL_DONT_FRAGMENT))
|
|
|
|
df = htons(IP_DF);
|
2015-07-21 15:43:58 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
memset(&fl4, 0, sizeof(fl4));
|
2015-07-21 15:43:58 +07:00
|
|
|
fl4.flowi4_oif = rdst ? rdst->remote_ifindex : 0;
|
2013-08-31 12:44:33 +07:00
|
|
|
fl4.flowi4_tos = RT_TOS(tos);
|
2015-05-05 20:09:21 +07:00
|
|
|
fl4.flowi4_mark = skb->mark;
|
|
|
|
fl4.flowi4_proto = IPPROTO_UDP;
|
2013-08-31 12:44:33 +07:00
|
|
|
fl4.daddr = dst->sin.sin_addr.s_addr;
|
2015-07-21 15:44:02 +07:00
|
|
|
fl4.saddr = vxlan->cfg.saddr.sin.sin_addr.s_addr;
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2014-04-24 15:02:49 +07:00
|
|
|
rt = ip_route_output_key(vxlan->net, &fl4);
|
2013-08-31 12:44:33 +07:00
|
|
|
if (IS_ERR(rt)) {
|
|
|
|
netdev_dbg(dev, "no route to %pI4\n",
|
|
|
|
&dst->sin.sin_addr.s_addr);
|
|
|
|
dev->stats.tx_carrier_errors++;
|
|
|
|
goto tx_error;
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (rt->dst.dev == dev) {
|
|
|
|
netdev_dbg(dev, "circular route to %pI4\n",
|
|
|
|
&dst->sin.sin_addr.s_addr);
|
|
|
|
dev->stats.collisions++;
|
2013-12-09 09:33:53 +07:00
|
|
|
goto rt_tx_error;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Bypass encapsulation if the destination is local */
|
|
|
|
if (rt->rt_flags & RTCF_LOCAL &&
|
|
|
|
!(rt->rt_flags & (RTCF_BROADCAST | RTCF_MULTICAST))) {
|
|
|
|
struct vxlan_dev *dst_vxlan;
|
|
|
|
|
|
|
|
ip_rt_put(rt);
|
2014-11-13 23:43:08 +07:00
|
|
|
dst_vxlan = vxlan_find_vni(vxlan->net, vni,
|
2015-01-15 09:53:56 +07:00
|
|
|
dst->sa.sa_family, dst_port,
|
|
|
|
vxlan->flags);
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!dst_vxlan)
|
|
|
|
goto tx_error;
|
|
|
|
vxlan_encap_bypass(skb, vxlan, dst_vxlan);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
tos = ip_tunnel_ecn_encap(tos, old_iph, skb);
|
|
|
|
ttl = ttl ? : ip4_dst_hoplimit(&rt->dst);
|
2015-04-06 09:19:09 +07:00
|
|
|
err = vxlan_xmit_skb(rt, sk, skb, fl4.saddr,
|
2015-01-21 02:23:05 +07:00
|
|
|
dst->sin.sin_addr.s_addr, tos, ttl, df,
|
2015-07-21 15:44:06 +07:00
|
|
|
src_port, dst_port, htonl(vni << 8), md,
|
2015-01-21 02:23:05 +07:00
|
|
|
!net_eq(vxlan->net, dev_net(vxlan->dev)),
|
2015-07-21 15:43:58 +07:00
|
|
|
flags);
|
2014-12-24 07:20:36 +07:00
|
|
|
if (err < 0) {
|
|
|
|
/* skb is already freed. */
|
|
|
|
skb = NULL;
|
2013-08-31 12:44:33 +07:00
|
|
|
goto rt_tx_error;
|
2014-12-24 07:20:36 +07:00
|
|
|
}
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
iptunnel_xmit_stats(err, &dev->stats, dev->tstats);
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
|
|
|
struct dst_entry *ndst;
|
2015-12-07 19:04:30 +07:00
|
|
|
struct in6_addr saddr;
|
2015-08-20 18:56:29 +07:00
|
|
|
u32 rt6i_flags;
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
if (!vxlan->vn6_sock)
|
|
|
|
goto drop;
|
|
|
|
sk = vxlan->vn6_sock->sock->sk;
|
|
|
|
|
2015-12-07 19:04:30 +07:00
|
|
|
ndst = vxlan6_get_route(vxlan, skb,
|
|
|
|
rdst ? rdst->remote_ifindex : 0,
|
|
|
|
&dst->sin6.sin6_addr, &saddr);
|
|
|
|
if (IS_ERR(ndst)) {
|
2013-08-31 12:44:33 +07:00
|
|
|
netdev_dbg(dev, "no route to %pI6\n",
|
|
|
|
&dst->sin6.sin6_addr);
|
|
|
|
dev->stats.tx_carrier_errors++;
|
2013-04-02 19:31:52 +07:00
|
|
|
goto tx_error;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (ndst->dev == dev) {
|
|
|
|
netdev_dbg(dev, "circular route to %pI6\n",
|
|
|
|
&dst->sin6.sin6_addr);
|
|
|
|
dst_release(ndst);
|
|
|
|
dev->stats.collisions++;
|
|
|
|
goto tx_error;
|
|
|
|
}
|
2013-06-18 07:49:56 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
/* Bypass encapsulation if the destination is local */
|
2015-08-20 18:56:29 +07:00
|
|
|
rt6i_flags = ((struct rt6_info *)ndst)->rt6i_flags;
|
|
|
|
if (rt6i_flags & RTF_LOCAL &&
|
|
|
|
!(rt6i_flags & (RTCF_BROADCAST | RTCF_MULTICAST))) {
|
2013-08-31 12:44:33 +07:00
|
|
|
struct vxlan_dev *dst_vxlan;
|
|
|
|
|
|
|
|
dst_release(ndst);
|
2014-11-13 23:43:08 +07:00
|
|
|
dst_vxlan = vxlan_find_vni(vxlan->net, vni,
|
2015-01-15 09:53:56 +07:00
|
|
|
dst->sa.sa_family, dst_port,
|
|
|
|
vxlan->flags);
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!dst_vxlan)
|
|
|
|
goto tx_error;
|
|
|
|
vxlan_encap_bypass(skb, vxlan, dst_vxlan);
|
|
|
|
return;
|
|
|
|
}
|
2013-08-20 01:23:17 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
ttl = ttl ? : ip6_dst_hoplimit(ndst);
|
2015-12-07 19:04:30 +07:00
|
|
|
err = vxlan6_xmit_skb(ndst, sk, skb, dev, &saddr, &dst->sin6.sin6_addr,
|
2015-07-21 15:44:06 +07:00
|
|
|
0, ttl, src_port, dst_port, htonl(vni << 8), md,
|
2015-01-21 02:23:05 +07:00
|
|
|
!net_eq(vxlan->net, dev_net(vxlan->dev)),
|
2015-08-20 18:56:30 +07:00
|
|
|
flags);
|
2013-08-31 12:44:33 +07:00
|
|
|
#endif
|
|
|
|
}
|
2013-06-18 07:49:56 +07:00
|
|
|
|
2013-06-18 04:16:11 +07:00
|
|
|
return;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
drop:
|
|
|
|
dev->stats.tx_dropped++;
|
|
|
|
goto tx_free;
|
|
|
|
|
2013-08-20 01:23:17 +07:00
|
|
|
rt_tx_error:
|
|
|
|
ip_rt_put(rt);
|
2012-10-01 19:32:35 +07:00
|
|
|
tx_error:
|
|
|
|
dev->stats.tx_errors++;
|
|
|
|
tx_free:
|
|
|
|
dev_kfree_skb(skb);
|
|
|
|
}
|
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
/* Transmit local packets over Vxlan
|
|
|
|
*
|
|
|
|
* Outer IP header inherits ECN and DF from inner header.
|
|
|
|
* Outer UDP destination is the VXLAN assigned port.
|
|
|
|
* source port is based on hash of flow
|
|
|
|
*/
|
|
|
|
static netdev_tx_t vxlan_xmit(struct sk_buff *skb, struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2015-07-21 15:44:00 +07:00
|
|
|
const struct ip_tunnel_info *info;
|
2013-03-15 11:35:51 +07:00
|
|
|
struct ethhdr *eth;
|
|
|
|
bool did_rsc = false;
|
2014-01-07 00:54:31 +07:00
|
|
|
struct vxlan_rdst *rdst, *fdst = NULL;
|
2013-03-15 11:35:51 +07:00
|
|
|
struct vxlan_fdb *f;
|
|
|
|
|
2015-08-20 18:56:25 +07:00
|
|
|
info = skb_tunnel_info(skb);
|
2015-07-21 15:44:00 +07:00
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
skb_reset_mac_header(skb);
|
|
|
|
eth = eth_hdr(skb);
|
|
|
|
|
2013-08-31 12:44:36 +07:00
|
|
|
if ((vxlan->flags & VXLAN_F_PROXY)) {
|
|
|
|
if (ntohs(eth->h_proto) == ETH_P_ARP)
|
|
|
|
return arp_reduce(dev, skb);
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
else if (ntohs(eth->h_proto) == ETH_P_IPV6 &&
|
2014-10-16 08:17:18 +07:00
|
|
|
pskb_may_pull(skb, sizeof(struct ipv6hdr)
|
|
|
|
+ sizeof(struct nd_msg)) &&
|
2013-08-31 12:44:36 +07:00
|
|
|
ipv6_hdr(skb)->nexthdr == IPPROTO_ICMPV6) {
|
|
|
|
struct nd_msg *msg;
|
|
|
|
|
|
|
|
msg = (struct nd_msg *)skb_transport_header(skb);
|
|
|
|
if (msg->icmph.icmp6_code == 0 &&
|
|
|
|
msg->icmph.icmp6_type == NDISC_NEIGHBOUR_SOLICITATION)
|
|
|
|
return neigh_reduce(dev, skb);
|
|
|
|
}
|
2014-10-17 13:06:16 +07:00
|
|
|
eth = eth_hdr(skb);
|
2013-08-31 12:44:36 +07:00
|
|
|
#endif
|
|
|
|
}
|
2013-03-15 11:35:51 +07:00
|
|
|
|
2015-08-05 12:51:07 +07:00
|
|
|
if (vxlan->flags & VXLAN_F_COLLECT_METADATA &&
|
2015-08-29 01:48:19 +07:00
|
|
|
info && info->mode & IP_TUNNEL_INFO_TX) {
|
2015-07-21 15:43:58 +07:00
|
|
|
vxlan_xmit_one(skb, dev, NULL, false);
|
|
|
|
return NETDEV_TX_OK;
|
|
|
|
}
|
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
f = vxlan_find_mac(vxlan, eth->h_dest);
|
2013-04-19 07:36:26 +07:00
|
|
|
did_rsc = false;
|
|
|
|
|
|
|
|
if (f && (f->flags & NTF_ROUTER) && (vxlan->flags & VXLAN_F_RSC) &&
|
2013-08-31 12:44:34 +07:00
|
|
|
(ntohs(eth->h_proto) == ETH_P_IP ||
|
|
|
|
ntohs(eth->h_proto) == ETH_P_IPV6)) {
|
2013-04-19 07:36:26 +07:00
|
|
|
did_rsc = route_shortcircuit(dev, skb);
|
|
|
|
if (did_rsc)
|
|
|
|
f = vxlan_find_mac(vxlan, eth->h_dest);
|
|
|
|
}
|
|
|
|
|
2013-03-15 11:35:51 +07:00
|
|
|
if (f == NULL) {
|
2013-06-25 20:01:51 +07:00
|
|
|
f = vxlan_find_mac(vxlan, all_zeros_mac);
|
|
|
|
if (f == NULL) {
|
|
|
|
if ((vxlan->flags & VXLAN_F_L2MISS) &&
|
|
|
|
!is_multicast_ether_addr(eth->h_dest))
|
|
|
|
vxlan_fdb_miss(vxlan, eth->h_dest);
|
|
|
|
|
|
|
|
dev->stats.tx_dropped++;
|
2014-01-07 00:54:31 +07:00
|
|
|
kfree_skb(skb);
|
2013-06-25 20:01:51 +07:00
|
|
|
return NETDEV_TX_OK;
|
|
|
|
}
|
|
|
|
}
|
2013-03-15 11:35:51 +07:00
|
|
|
|
2013-06-25 20:01:51 +07:00
|
|
|
list_for_each_entry_rcu(rdst, &f->remotes, list) {
|
|
|
|
struct sk_buff *skb1;
|
2013-03-15 11:35:51 +07:00
|
|
|
|
2014-01-07 00:54:31 +07:00
|
|
|
if (!fdst) {
|
|
|
|
fdst = rdst;
|
|
|
|
continue;
|
|
|
|
}
|
2013-06-25 20:01:51 +07:00
|
|
|
skb1 = skb_clone(skb, GFP_ATOMIC);
|
|
|
|
if (skb1)
|
|
|
|
vxlan_xmit_one(skb1, dev, rdst, did_rsc);
|
2013-03-15 11:35:51 +07:00
|
|
|
}
|
|
|
|
|
2014-01-07 00:54:31 +07:00
|
|
|
if (fdst)
|
|
|
|
vxlan_xmit_one(skb, dev, fdst, did_rsc);
|
|
|
|
else
|
|
|
|
kfree_skb(skb);
|
2013-06-18 04:16:11 +07:00
|
|
|
return NETDEV_TX_OK;
|
2013-03-15 11:35:51 +07:00
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Walk the forwarding table and purge stale entries */
|
|
|
|
static void vxlan_cleanup(unsigned long arg)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = (struct vxlan_dev *) arg;
|
|
|
|
unsigned long next_timer = jiffies + FDB_AGE_INTERVAL;
|
|
|
|
unsigned int h;
|
|
|
|
|
|
|
|
if (!netif_running(vxlan->dev))
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (h = 0; h < FDB_HASH_SIZE; ++h) {
|
|
|
|
struct hlist_node *p, *n;
|
2015-05-26 14:42:04 +07:00
|
|
|
|
|
|
|
spin_lock_bh(&vxlan->hash_lock);
|
2012-10-01 19:32:35 +07:00
|
|
|
hlist_for_each_safe(p, n, &vxlan->fdb_head[h]) {
|
|
|
|
struct vxlan_fdb *f
|
|
|
|
= container_of(p, struct vxlan_fdb, hlist);
|
|
|
|
unsigned long timeout;
|
|
|
|
|
2012-10-26 13:24:34 +07:00
|
|
|
if (f->state & NUD_PERMANENT)
|
2012-10-01 19:32:35 +07:00
|
|
|
continue;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
timeout = f->used + vxlan->cfg.age_interval * HZ;
|
2012-10-01 19:32:35 +07:00
|
|
|
if (time_before_eq(timeout, jiffies)) {
|
|
|
|
netdev_dbg(vxlan->dev,
|
|
|
|
"garbage collect %pM\n",
|
|
|
|
f->eth_addr);
|
|
|
|
f->state = NUD_STALE;
|
|
|
|
vxlan_fdb_destroy(vxlan, f);
|
|
|
|
} else if (time_before(timeout, next_timer))
|
|
|
|
next_timer = timeout;
|
|
|
|
}
|
2015-05-26 14:42:04 +07:00
|
|
|
spin_unlock_bh(&vxlan->hash_lock);
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
mod_timer(&vxlan->age_timer, next_timer);
|
|
|
|
}
|
|
|
|
|
2013-08-20 01:22:48 +07:00
|
|
|
static void vxlan_vs_add_dev(struct vxlan_sock *vs, struct vxlan_dev *vxlan)
|
|
|
|
{
|
2015-03-19 00:50:44 +07:00
|
|
|
struct vxlan_net *vn = net_generic(vxlan->net, vxlan_net_id);
|
2013-08-20 01:22:48 +07:00
|
|
|
__u32 vni = vxlan->default_dst.remote_vni;
|
|
|
|
|
2015-03-19 00:50:44 +07:00
|
|
|
spin_lock(&vn->sock_lock);
|
2013-08-20 01:22:48 +07:00
|
|
|
hlist_add_head_rcu(&vxlan->hlist, vni_head(vs, vni));
|
2015-03-19 00:50:44 +07:00
|
|
|
spin_unlock(&vn->sock_lock);
|
2013-08-20 01:22:48 +07:00
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Setup stats when device is created */
|
|
|
|
static int vxlan_init(struct net_device *dev)
|
|
|
|
{
|
2014-02-14 02:46:28 +07:00
|
|
|
dev->tstats = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats);
|
2013-03-25 21:49:46 +07:00
|
|
|
if (!dev->tstats)
|
2012-10-01 19:32:35 +07:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-06-26 07:06:01 +07:00
|
|
|
static void vxlan_fdb_delete_default(struct vxlan_dev *vxlan)
|
2013-06-25 20:01:51 +07:00
|
|
|
{
|
|
|
|
struct vxlan_fdb *f;
|
|
|
|
|
|
|
|
spin_lock_bh(&vxlan->hash_lock);
|
|
|
|
f = __vxlan_find_mac(vxlan, all_zeros_mac);
|
|
|
|
if (f)
|
|
|
|
vxlan_fdb_destroy(vxlan, f);
|
|
|
|
spin_unlock_bh(&vxlan->hash_lock);
|
|
|
|
}
|
|
|
|
|
2013-06-18 04:16:11 +07:00
|
|
|
static void vxlan_uninit(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
|
2013-06-26 07:06:01 +07:00
|
|
|
vxlan_fdb_delete_default(vxlan);
|
2013-06-25 20:01:51 +07:00
|
|
|
|
2013-06-18 04:16:11 +07:00
|
|
|
free_percpu(dev->tstats);
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Start ageing timer and join group when device is brought up */
|
|
|
|
static int vxlan_open(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2015-09-24 18:50:01 +07:00
|
|
|
int ret;
|
2015-03-19 00:50:44 +07:00
|
|
|
|
2015-09-24 18:50:01 +07:00
|
|
|
ret = vxlan_sock_add(vxlan);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-12-10 15:37:32 +07:00
|
|
|
if (vxlan_addr_multicast(&vxlan->default_dst.remote_ip)) {
|
2015-03-19 00:50:44 +07:00
|
|
|
ret = vxlan_igmp_join(vxlan);
|
2015-08-26 06:22:35 +07:00
|
|
|
if (ret == -EADDRINUSE)
|
|
|
|
ret = 0;
|
2015-03-19 00:50:44 +07:00
|
|
|
if (ret) {
|
2015-09-24 18:50:01 +07:00
|
|
|
vxlan_sock_release(vxlan);
|
2015-03-19 00:50:44 +07:00
|
|
|
return ret;
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (vxlan->cfg.age_interval)
|
2012-10-01 19:32:35 +07:00
|
|
|
mod_timer(&vxlan->age_timer, jiffies + FDB_AGE_INTERVAL);
|
|
|
|
|
2015-03-19 00:50:44 +07:00
|
|
|
return ret;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Purge the forwarding table */
|
|
|
|
static void vxlan_flush(struct vxlan_dev *vxlan)
|
|
|
|
{
|
2013-05-28 05:35:52 +07:00
|
|
|
unsigned int h;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
spin_lock_bh(&vxlan->hash_lock);
|
|
|
|
for (h = 0; h < FDB_HASH_SIZE; ++h) {
|
|
|
|
struct hlist_node *p, *n;
|
|
|
|
hlist_for_each_safe(p, n, &vxlan->fdb_head[h]) {
|
|
|
|
struct vxlan_fdb *f
|
|
|
|
= container_of(p, struct vxlan_fdb, hlist);
|
2013-06-25 20:01:51 +07:00
|
|
|
/* the all_zeros_mac entry is deleted at vxlan_uninit */
|
|
|
|
if (!is_zero_ether_addr(f->eth_addr))
|
|
|
|
vxlan_fdb_destroy(vxlan, f);
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock_bh(&vxlan->hash_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Cleanup timer and forwarding table on shutdown */
|
|
|
|
static int vxlan_stop(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2014-04-24 15:02:49 +07:00
|
|
|
struct vxlan_net *vn = net_generic(vxlan->net, vxlan_net_id);
|
2015-03-19 00:50:44 +07:00
|
|
|
int ret = 0;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-03-24 02:23:12 +07:00
|
|
|
if (vxlan_addr_multicast(&vxlan->default_dst.remote_ip) &&
|
2015-04-09 04:48:30 +07:00
|
|
|
!vxlan_group_used(vn, vxlan))
|
2015-03-19 00:50:44 +07:00
|
|
|
ret = vxlan_igmp_leave(vxlan);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
del_timer_sync(&vxlan->age_timer);
|
|
|
|
|
|
|
|
vxlan_flush(vxlan);
|
2015-09-24 18:50:01 +07:00
|
|
|
vxlan_sock_release(vxlan);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-03-19 00:50:44 +07:00
|
|
|
return ret;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Stub, nothing needs to be done. */
|
|
|
|
static void vxlan_set_multicast_list(struct net_device *dev)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-12-18 06:21:08 +07:00
|
|
|
static int vxlan_change_mtu(struct net_device *dev, int new_mtu)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct vxlan_rdst *dst = &vxlan->default_dst;
|
|
|
|
struct net_device *lowerdev;
|
|
|
|
int max_mtu;
|
|
|
|
|
2014-04-24 15:02:49 +07:00
|
|
|
lowerdev = __dev_get_by_index(vxlan->net, dst->remote_ifindex);
|
2013-12-18 06:21:08 +07:00
|
|
|
if (lowerdev == NULL)
|
|
|
|
return eth_change_mtu(dev, new_mtu);
|
|
|
|
|
|
|
|
if (dst->remote_ip.sa.sa_family == AF_INET6)
|
|
|
|
max_mtu = lowerdev->mtu - VXLAN6_HEADROOM;
|
|
|
|
else
|
|
|
|
max_mtu = lowerdev->mtu - VXLAN_HEADROOM;
|
|
|
|
|
|
|
|
if (new_mtu < 68 || new_mtu > max_mtu)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
dev->mtu = new_mtu;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-10-23 08:17:16 +07:00
|
|
|
static int egress_ipv4_tun_info(struct net_device *dev, struct sk_buff *skb,
|
|
|
|
struct ip_tunnel_info *info,
|
|
|
|
__be16 sport, __be16 dport)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct rtable *rt;
|
|
|
|
struct flowi4 fl4;
|
|
|
|
|
|
|
|
memset(&fl4, 0, sizeof(fl4));
|
|
|
|
fl4.flowi4_tos = RT_TOS(info->key.tos);
|
|
|
|
fl4.flowi4_mark = skb->mark;
|
|
|
|
fl4.flowi4_proto = IPPROTO_UDP;
|
|
|
|
fl4.daddr = info->key.u.ipv4.dst;
|
|
|
|
|
|
|
|
rt = ip_route_output_key(vxlan->net, &fl4);
|
|
|
|
if (IS_ERR(rt))
|
|
|
|
return PTR_ERR(rt);
|
|
|
|
ip_rt_put(rt);
|
|
|
|
|
|
|
|
info->key.u.ipv4.src = fl4.saddr;
|
|
|
|
info->key.tp_src = sport;
|
|
|
|
info->key.tp_dst = dport;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_fill_metadata_dst(struct net_device *dev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
struct ip_tunnel_info *info = skb_tunnel_info(skb);
|
|
|
|
__be16 sport, dport;
|
|
|
|
|
|
|
|
sport = udp_flow_src_port(dev_net(dev), skb, vxlan->cfg.port_min,
|
|
|
|
vxlan->cfg.port_max, true);
|
|
|
|
dport = info->key.tp_dst ? : vxlan->cfg.dst_port;
|
|
|
|
|
2015-12-07 19:04:31 +07:00
|
|
|
if (ip_tunnel_info_af(info) == AF_INET) {
|
|
|
|
if (!vxlan->vn4_sock)
|
|
|
|
return -EINVAL;
|
2015-10-23 08:17:16 +07:00
|
|
|
return egress_ipv4_tun_info(dev, skb, info, sport, dport);
|
2015-12-07 19:04:31 +07:00
|
|
|
} else {
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
struct dst_entry *ndst;
|
|
|
|
|
|
|
|
if (!vxlan->vn6_sock)
|
|
|
|
return -EINVAL;
|
|
|
|
ndst = vxlan6_get_route(vxlan, skb, 0,
|
|
|
|
&info->key.u.ipv6.dst,
|
|
|
|
&info->key.u.ipv6.src);
|
|
|
|
if (IS_ERR(ndst))
|
|
|
|
return PTR_ERR(ndst);
|
|
|
|
dst_release(ndst);
|
|
|
|
|
|
|
|
info->key.tp_src = sport;
|
|
|
|
info->key.tp_dst = dport;
|
|
|
|
#else /* !CONFIG_IPV6 */
|
|
|
|
return -EPFNOSUPPORT;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return 0;
|
2015-10-23 08:17:16 +07:00
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
static const struct net_device_ops vxlan_netdev_ops = {
|
|
|
|
.ndo_init = vxlan_init,
|
2013-06-18 04:16:11 +07:00
|
|
|
.ndo_uninit = vxlan_uninit,
|
2012-10-01 19:32:35 +07:00
|
|
|
.ndo_open = vxlan_open,
|
|
|
|
.ndo_stop = vxlan_stop,
|
|
|
|
.ndo_start_xmit = vxlan_xmit,
|
2013-03-25 21:49:46 +07:00
|
|
|
.ndo_get_stats64 = ip_tunnel_get_stats64,
|
2012-10-01 19:32:35 +07:00
|
|
|
.ndo_set_rx_mode = vxlan_set_multicast_list,
|
2013-12-18 06:21:08 +07:00
|
|
|
.ndo_change_mtu = vxlan_change_mtu,
|
2012-10-01 19:32:35 +07:00
|
|
|
.ndo_validate_addr = eth_validate_addr,
|
|
|
|
.ndo_set_mac_address = eth_mac_addr,
|
|
|
|
.ndo_fdb_add = vxlan_fdb_add,
|
|
|
|
.ndo_fdb_del = vxlan_fdb_delete,
|
|
|
|
.ndo_fdb_dump = vxlan_fdb_dump,
|
2015-10-23 08:17:16 +07:00
|
|
|
.ndo_fill_metadata_dst = vxlan_fill_metadata_dst,
|
2012-10-01 19:32:35 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Info for udev, that this is a virtual tunnel endpoint */
|
|
|
|
static struct device_type vxlan_type = {
|
|
|
|
.name = "vxlan",
|
|
|
|
};
|
|
|
|
|
2013-09-04 16:13:38 +07:00
|
|
|
/* Calls the ndo_add_vxlan_port of the caller in order to
|
2013-09-13 21:34:13 +07:00
|
|
|
* supply the listening VXLAN udp ports. Callers are expected
|
|
|
|
* to implement the ndo_add_vxlan_port.
|
2013-09-04 16:13:38 +07:00
|
|
|
*/
|
|
|
|
void vxlan_get_rx_port(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_sock *vs;
|
|
|
|
struct net *net = dev_net(dev);
|
|
|
|
struct vxlan_net *vn = net_generic(net, vxlan_net_id);
|
|
|
|
sa_family_t sa_family;
|
2013-09-13 21:34:13 +07:00
|
|
|
__be16 port;
|
|
|
|
unsigned int i;
|
2013-09-04 16:13:38 +07:00
|
|
|
|
|
|
|
spin_lock(&vn->sock_lock);
|
|
|
|
for (i = 0; i < PORT_HASH_SIZE; ++i) {
|
2013-09-13 21:34:13 +07:00
|
|
|
hlist_for_each_entry_rcu(vs, &vn->sock_list[i], hlist) {
|
|
|
|
port = inet_sk(vs->sock->sk)->inet_sport;
|
2015-08-20 18:56:28 +07:00
|
|
|
sa_family = vxlan_get_sk_family(vs);
|
2013-09-04 16:13:38 +07:00
|
|
|
dev->netdev_ops->ndo_add_vxlan_port(dev, sa_family,
|
|
|
|
port);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock(&vn->sock_lock);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(vxlan_get_rx_port);
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
/* Initialize the device structure. */
|
|
|
|
static void vxlan_setup(struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2013-05-28 05:35:52 +07:00
|
|
|
unsigned int h;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
eth_hw_addr_random(dev);
|
|
|
|
ether_setup(dev);
|
|
|
|
|
|
|
|
dev->netdev_ops = &vxlan_netdev_ops;
|
2013-06-18 04:16:11 +07:00
|
|
|
dev->destructor = free_netdev;
|
2012-10-01 19:32:35 +07:00
|
|
|
SET_NETDEV_DEVTYPE(dev, &vxlan_type);
|
|
|
|
|
|
|
|
dev->features |= NETIF_F_LLTX;
|
2012-12-07 21:14:16 +07:00
|
|
|
dev->features |= NETIF_F_SG | NETIF_F_HW_CSUM;
|
2012-12-07 21:14:18 +07:00
|
|
|
dev->features |= NETIF_F_RXCSUM;
|
2013-03-07 20:22:36 +07:00
|
|
|
dev->features |= NETIF_F_GSO_SOFTWARE;
|
2012-12-07 21:14:18 +07:00
|
|
|
|
2013-08-20 01:23:29 +07:00
|
|
|
dev->vlan_features = dev->features;
|
|
|
|
dev->features |= NETIF_F_HW_VLAN_CTAG_TX | NETIF_F_HW_VLAN_STAG_TX;
|
2012-12-07 21:14:18 +07:00
|
|
|
dev->hw_features |= NETIF_F_SG | NETIF_F_HW_CSUM | NETIF_F_RXCSUM;
|
2013-03-07 20:22:36 +07:00
|
|
|
dev->hw_features |= NETIF_F_GSO_SOFTWARE;
|
2013-08-20 01:23:29 +07:00
|
|
|
dev->hw_features |= NETIF_F_HW_VLAN_CTAG_TX | NETIF_F_HW_VLAN_STAG_TX;
|
2014-10-06 08:38:35 +07:00
|
|
|
netif_keep_dst(dev);
|
2015-08-18 15:30:35 +07:00
|
|
|
dev->priv_flags |= IFF_LIVE_ADDR_CHANGE | IFF_NO_QUEUE;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
INIT_LIST_HEAD(&vxlan->next);
|
2012-10-01 19:32:35 +07:00
|
|
|
spin_lock_init(&vxlan->hash_lock);
|
|
|
|
|
|
|
|
init_timer_deferrable(&vxlan->age_timer);
|
|
|
|
vxlan->age_timer.function = vxlan_cleanup;
|
|
|
|
vxlan->age_timer.data = (unsigned long) vxlan;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
vxlan->cfg.dst_port = htons(vxlan_port);
|
2012-10-10 03:35:50 +07:00
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
vxlan->dev = dev;
|
|
|
|
|
2015-08-20 07:07:33 +07:00
|
|
|
gro_cells_init(&vxlan->gro_cells, dev);
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
for (h = 0; h < FDB_HASH_SIZE; ++h)
|
|
|
|
INIT_HLIST_HEAD(&vxlan->fdb_head[h]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct nla_policy vxlan_policy[IFLA_VXLAN_MAX + 1] = {
|
|
|
|
[IFLA_VXLAN_ID] = { .type = NLA_U32 },
|
2013-04-27 18:31:55 +07:00
|
|
|
[IFLA_VXLAN_GROUP] = { .len = FIELD_SIZEOF(struct iphdr, daddr) },
|
2013-08-31 12:44:33 +07:00
|
|
|
[IFLA_VXLAN_GROUP6] = { .len = sizeof(struct in6_addr) },
|
2012-10-01 19:32:35 +07:00
|
|
|
[IFLA_VXLAN_LINK] = { .type = NLA_U32 },
|
|
|
|
[IFLA_VXLAN_LOCAL] = { .len = FIELD_SIZEOF(struct iphdr, saddr) },
|
2013-08-31 12:44:33 +07:00
|
|
|
[IFLA_VXLAN_LOCAL6] = { .len = sizeof(struct in6_addr) },
|
2012-10-01 19:32:35 +07:00
|
|
|
[IFLA_VXLAN_TOS] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_TTL] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_LEARNING] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_AGEING] = { .type = NLA_U32 },
|
|
|
|
[IFLA_VXLAN_LIMIT] = { .type = NLA_U32 },
|
2012-10-10 03:35:50 +07:00
|
|
|
[IFLA_VXLAN_PORT_RANGE] = { .len = sizeof(struct ifla_vxlan_port_range) },
|
2012-11-20 09:50:14 +07:00
|
|
|
[IFLA_VXLAN_PROXY] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_RSC] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_L2MISS] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_L3MISS] = { .type = NLA_U8 },
|
2015-07-31 10:10:22 +07:00
|
|
|
[IFLA_VXLAN_COLLECT_METADATA] = { .type = NLA_U8 },
|
2013-04-27 18:31:57 +07:00
|
|
|
[IFLA_VXLAN_PORT] = { .type = NLA_U16 },
|
2014-11-07 09:06:01 +07:00
|
|
|
[IFLA_VXLAN_UDP_CSUM] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_UDP_ZERO_CSUM6_TX] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_UDP_ZERO_CSUM6_RX] = { .type = NLA_U8 },
|
2015-01-13 08:00:38 +07:00
|
|
|
[IFLA_VXLAN_REMCSUM_TX] = { .type = NLA_U8 },
|
|
|
|
[IFLA_VXLAN_REMCSUM_RX] = { .type = NLA_U8 },
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
[IFLA_VXLAN_GBP] = { .type = NLA_FLAG, },
|
2015-02-11 07:30:32 +07:00
|
|
|
[IFLA_VXLAN_REMCSUM_NOPARTIAL] = { .type = NLA_FLAG },
|
2012-10-01 19:32:35 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
static int vxlan_validate(struct nlattr *tb[], struct nlattr *data[])
|
|
|
|
{
|
|
|
|
if (tb[IFLA_ADDRESS]) {
|
|
|
|
if (nla_len(tb[IFLA_ADDRESS]) != ETH_ALEN) {
|
|
|
|
pr_debug("invalid link address (not ethernet)\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_valid_ether_addr(nla_data(tb[IFLA_ADDRESS]))) {
|
|
|
|
pr_debug("invalid all zero ethernet address\n");
|
|
|
|
return -EADDRNOTAVAIL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_ID]) {
|
|
|
|
__u32 id = nla_get_u32(data[IFLA_VXLAN_ID]);
|
|
|
|
if (id >= VXLAN_VID_MASK)
|
|
|
|
return -ERANGE;
|
|
|
|
}
|
|
|
|
|
2012-10-10 03:35:50 +07:00
|
|
|
if (data[IFLA_VXLAN_PORT_RANGE]) {
|
|
|
|
const struct ifla_vxlan_port_range *p
|
|
|
|
= nla_data(data[IFLA_VXLAN_PORT_RANGE]);
|
|
|
|
|
|
|
|
if (ntohs(p->high) < ntohs(p->low)) {
|
|
|
|
pr_debug("port range %u .. %u not valid\n",
|
|
|
|
ntohs(p->low), ntohs(p->high));
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-01-30 06:43:07 +07:00
|
|
|
static void vxlan_get_drvinfo(struct net_device *netdev,
|
|
|
|
struct ethtool_drvinfo *drvinfo)
|
|
|
|
{
|
|
|
|
strlcpy(drvinfo->version, VXLAN_VERSION, sizeof(drvinfo->version));
|
|
|
|
strlcpy(drvinfo->driver, "vxlan", sizeof(drvinfo->driver));
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct ethtool_ops vxlan_ethtool_ops = {
|
|
|
|
.get_drvinfo = vxlan_get_drvinfo,
|
|
|
|
.get_link = ethtool_op_get_link,
|
|
|
|
};
|
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
static void vxlan_del_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct vxlan_sock *vs = container_of(work, struct vxlan_sock, del_work);
|
2014-09-17 07:31:18 +07:00
|
|
|
udp_tunnel_sock_release(vs->sock);
|
2013-05-16 18:35:20 +07:00
|
|
|
kfree_rcu(vs, rcu);
|
|
|
|
}
|
|
|
|
|
2014-07-14 09:49:42 +07:00
|
|
|
static struct socket *vxlan_create_sock(struct net *net, bool ipv6,
|
|
|
|
__be16 port, u32 flags)
|
2013-05-16 18:35:20 +07:00
|
|
|
{
|
2013-08-31 12:44:33 +07:00
|
|
|
struct socket *sock;
|
2014-07-14 09:49:42 +07:00
|
|
|
struct udp_port_cfg udp_conf;
|
|
|
|
int err;
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2014-07-14 09:49:42 +07:00
|
|
|
memset(&udp_conf, 0, sizeof(udp_conf));
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2014-07-14 09:49:42 +07:00
|
|
|
if (ipv6) {
|
|
|
|
udp_conf.family = AF_INET6;
|
|
|
|
udp_conf.use_udp6_rx_checksums =
|
2014-11-25 11:08:38 +07:00
|
|
|
!(flags & VXLAN_F_UDP_ZERO_CSUM6_RX);
|
2015-08-29 01:48:22 +07:00
|
|
|
udp_conf.ipv6_v6only = 1;
|
2014-07-14 09:49:42 +07:00
|
|
|
} else {
|
|
|
|
udp_conf.family = AF_INET;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
2014-07-14 09:49:42 +07:00
|
|
|
udp_conf.local_udp_port = port;
|
2013-05-16 18:35:20 +07:00
|
|
|
|
2014-07-14 09:49:42 +07:00
|
|
|
/* Open UDP socket */
|
|
|
|
err = udp_sock_create(net, &udp_conf, &sock);
|
|
|
|
if (err < 0)
|
|
|
|
return ERR_PTR(err);
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2013-10-28 13:01:48 +07:00
|
|
|
return sock;
|
2013-08-31 12:44:33 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Create new listen socket if needed */
|
2015-09-24 18:50:02 +07:00
|
|
|
static struct vxlan_sock *vxlan_socket_create(struct net *net, bool ipv6,
|
|
|
|
__be16 port, u32 flags)
|
2013-08-31 12:44:33 +07:00
|
|
|
{
|
|
|
|
struct vxlan_net *vn = net_generic(net, vxlan_net_id);
|
|
|
|
struct vxlan_sock *vs;
|
|
|
|
struct socket *sock;
|
|
|
|
unsigned int h;
|
2014-09-17 07:31:18 +07:00
|
|
|
struct udp_tunnel_sock_cfg tunnel_cfg;
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2014-01-20 18:59:21 +07:00
|
|
|
vs = kzalloc(sizeof(*vs), GFP_KERNEL);
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!vs)
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
|
|
|
for (h = 0; h < VNI_HASH_SIZE; ++h)
|
|
|
|
INIT_HLIST_HEAD(&vs->vni_list[h]);
|
|
|
|
|
|
|
|
INIT_WORK(&vs->del_work, vxlan_del_work);
|
|
|
|
|
2014-07-14 09:49:42 +07:00
|
|
|
sock = vxlan_create_sock(net, ipv6, port, flags);
|
2013-10-28 13:01:48 +07:00
|
|
|
if (IS_ERR(sock)) {
|
2015-03-19 00:50:44 +07:00
|
|
|
pr_info("Cannot bind port %d, err=%ld\n", ntohs(port),
|
|
|
|
PTR_ERR(sock));
|
2013-05-16 18:35:20 +07:00
|
|
|
kfree(vs);
|
2013-11-01 12:09:43 +07:00
|
|
|
return ERR_CAST(sock);
|
2013-05-16 18:35:20 +07:00
|
|
|
}
|
2013-08-31 12:44:33 +07:00
|
|
|
|
|
|
|
vs->sock = sock;
|
2013-08-20 01:22:48 +07:00
|
|
|
atomic_set(&vs->refcnt, 1);
|
2015-01-21 02:23:05 +07:00
|
|
|
vs->flags = (flags & VXLAN_F_RCV_FLAGS);
|
2013-05-16 18:35:20 +07:00
|
|
|
|
2014-01-20 18:59:21 +07:00
|
|
|
/* Initialize the vxlan udp offloads structure */
|
|
|
|
vs->udp_offloads.port = port;
|
|
|
|
vs->udp_offloads.callbacks.gro_receive = vxlan_gro_receive;
|
|
|
|
vs->udp_offloads.callbacks.gro_complete = vxlan_gro_complete;
|
|
|
|
|
2013-08-20 01:22:48 +07:00
|
|
|
spin_lock(&vn->sock_lock);
|
|
|
|
hlist_add_head_rcu(&vs->hlist, vs_head(net, port));
|
2014-01-20 18:59:21 +07:00
|
|
|
vxlan_notify_add_rx_port(vs);
|
2013-08-20 01:22:48 +07:00
|
|
|
spin_unlock(&vn->sock_lock);
|
2013-05-16 18:35:20 +07:00
|
|
|
|
|
|
|
/* Mark socket as an encapsulation socket. */
|
2014-09-17 07:31:18 +07:00
|
|
|
tunnel_cfg.sk_user_data = vs;
|
|
|
|
tunnel_cfg.encap_type = 1;
|
|
|
|
tunnel_cfg.encap_rcv = vxlan_udp_encap_recv;
|
|
|
|
tunnel_cfg.encap_destroy = NULL;
|
|
|
|
|
|
|
|
setup_udp_tunnel_sock(net, sock, &tunnel_cfg);
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2013-08-20 01:22:48 +07:00
|
|
|
return vs;
|
|
|
|
}
|
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
static int __vxlan_sock_add(struct vxlan_dev *vxlan, bool ipv6)
|
2013-08-20 01:22:48 +07:00
|
|
|
{
|
2015-09-24 18:50:01 +07:00
|
|
|
struct vxlan_net *vn = net_generic(vxlan->net, vxlan_net_id);
|
|
|
|
struct vxlan_sock *vs = NULL;
|
2013-08-20 01:22:48 +07:00
|
|
|
|
2015-09-24 18:50:01 +07:00
|
|
|
if (!vxlan->cfg.no_share) {
|
2015-03-19 00:50:44 +07:00
|
|
|
spin_lock(&vn->sock_lock);
|
2015-09-24 18:50:01 +07:00
|
|
|
vs = vxlan_find_sock(vxlan->net, ipv6 ? AF_INET6 : AF_INET,
|
|
|
|
vxlan->cfg.dst_port, vxlan->flags);
|
|
|
|
if (vs && !atomic_add_unless(&vs->refcnt, 1, 0)) {
|
2015-03-19 00:50:44 +07:00
|
|
|
spin_unlock(&vn->sock_lock);
|
2015-09-24 18:50:01 +07:00
|
|
|
return -EBUSY;
|
2015-03-19 00:50:44 +07:00
|
|
|
}
|
|
|
|
spin_unlock(&vn->sock_lock);
|
|
|
|
}
|
2015-09-24 18:50:01 +07:00
|
|
|
if (!vs)
|
2015-09-24 18:50:02 +07:00
|
|
|
vs = vxlan_socket_create(vxlan->net, ipv6,
|
|
|
|
vxlan->cfg.dst_port, vxlan->flags);
|
2015-09-24 18:50:01 +07:00
|
|
|
if (IS_ERR(vs))
|
|
|
|
return PTR_ERR(vs);
|
2015-09-24 18:50:02 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
if (ipv6)
|
|
|
|
vxlan->vn6_sock = vs;
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
vxlan->vn4_sock = vs;
|
2015-09-24 18:50:01 +07:00
|
|
|
vxlan_vs_add_dev(vs, vxlan);
|
|
|
|
return 0;
|
2013-05-16 18:35:20 +07:00
|
|
|
}
|
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
static int vxlan_sock_add(struct vxlan_dev *vxlan)
|
|
|
|
{
|
|
|
|
bool ipv6 = vxlan->flags & VXLAN_F_IPV6;
|
|
|
|
bool metadata = vxlan->flags & VXLAN_F_COLLECT_METADATA;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
vxlan->vn4_sock = NULL;
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
vxlan->vn6_sock = NULL;
|
|
|
|
if (ipv6 || metadata)
|
|
|
|
ret = __vxlan_sock_add(vxlan, true);
|
|
|
|
#endif
|
|
|
|
if (!ret && (!ipv6 || metadata))
|
|
|
|
ret = __vxlan_sock_add(vxlan, false);
|
|
|
|
if (ret < 0)
|
|
|
|
vxlan_sock_release(vxlan);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
static int vxlan_dev_configure(struct net *src_net, struct net_device *dev,
|
|
|
|
struct vxlan_config *conf)
|
2012-10-01 19:32:35 +07:00
|
|
|
{
|
2015-01-27 04:28:14 +07:00
|
|
|
struct vxlan_net *vn = net_generic(src_net, vxlan_net_id);
|
2012-10-01 19:32:35 +07:00
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2013-04-16 09:50:52 +07:00
|
|
|
struct vxlan_rdst *dst = &vxlan->default_dst;
|
2015-09-24 18:50:02 +07:00
|
|
|
unsigned short needed_headroom = ETH_HLEN;
|
2012-10-01 19:32:35 +07:00
|
|
|
int err;
|
2013-08-31 12:44:33 +07:00
|
|
|
bool use_ipv6 = false;
|
2015-07-21 15:44:02 +07:00
|
|
|
__be16 default_port = vxlan->cfg.dst_port;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-01-27 04:28:14 +07:00
|
|
|
vxlan->net = src_net;
|
2014-04-24 15:02:49 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
dst->remote_vni = conf->vni;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
memcpy(&dst->remote_ip, &conf->remote_ip, sizeof(conf->remote_ip));
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
/* Unless IPv6 is explicitly requested, assume IPv4 */
|
|
|
|
if (!dst->remote_ip.sa.sa_family)
|
|
|
|
dst->remote_ip.sa.sa_family = AF_INET;
|
2013-08-31 12:44:33 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (dst->remote_ip.sa.sa_family == AF_INET6 ||
|
2015-09-17 21:11:11 +07:00
|
|
|
vxlan->cfg.saddr.sa.sa_family == AF_INET6) {
|
|
|
|
if (!IS_ENABLED(CONFIG_IPV6))
|
|
|
|
return -EPFNOSUPPORT;
|
2013-08-31 12:44:33 +07:00
|
|
|
use_ipv6 = true;
|
2015-09-24 18:50:02 +07:00
|
|
|
vxlan->flags |= VXLAN_F_IPV6;
|
2015-09-17 21:11:11 +07:00
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (conf->remote_ifindex) {
|
2012-10-10 03:35:53 +07:00
|
|
|
struct net_device *lowerdev
|
2015-07-21 15:44:02 +07:00
|
|
|
= __dev_get_by_index(src_net, conf->remote_ifindex);
|
|
|
|
|
|
|
|
dst->remote_ifindex = conf->remote_ifindex;
|
2012-10-10 03:35:53 +07:00
|
|
|
|
|
|
|
if (!lowerdev) {
|
2013-04-16 09:50:52 +07:00
|
|
|
pr_info("ifindex %d does not exist\n", dst->remote_ifindex);
|
2012-10-10 03:35:53 +07:00
|
|
|
return -ENODEV;
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
if (use_ipv6) {
|
|
|
|
struct inet6_dev *idev = __in6_dev_get(lowerdev);
|
|
|
|
if (idev && idev->cnf.disable_ipv6) {
|
|
|
|
pr_info("IPv6 is disabled via sysctl\n");
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (!conf->mtu)
|
2013-08-31 12:44:33 +07:00
|
|
|
dev->mtu = lowerdev->mtu - (use_ipv6 ? VXLAN6_HEADROOM : VXLAN_HEADROOM);
|
2012-11-13 20:10:59 +07:00
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
needed_headroom = lowerdev->hard_header_len;
|
2015-09-17 21:11:10 +07:00
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-09-24 18:50:02 +07:00
|
|
|
if (use_ipv6 || conf->flags & VXLAN_F_COLLECT_METADATA)
|
|
|
|
needed_headroom += VXLAN6_HEADROOM;
|
|
|
|
else
|
|
|
|
needed_headroom += VXLAN_HEADROOM;
|
|
|
|
dev->needed_headroom = needed_headroom;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
memcpy(&vxlan->cfg, conf, sizeof(*conf));
|
|
|
|
if (!vxlan->cfg.dst_port)
|
|
|
|
vxlan->cfg.dst_port = default_port;
|
|
|
|
vxlan->flags |= conf->flags;
|
|
|
|
|
|
|
|
if (!vxlan->cfg.age_interval)
|
|
|
|
vxlan->cfg.age_interval = FDB_AGE_DEFAULT;
|
|
|
|
|
|
|
|
if (vxlan_find_vni(src_net, conf->vni, use_ipv6 ? AF_INET6 : AF_INET,
|
|
|
|
vxlan->cfg.dst_port, vxlan->flags))
|
|
|
|
return -EEXIST;
|
|
|
|
|
|
|
|
dev->ethtool_ops = &vxlan_ethtool_ops;
|
|
|
|
|
|
|
|
/* create an fdb entry for a valid default destination */
|
|
|
|
if (!vxlan_addr_any(&vxlan->default_dst.remote_ip)) {
|
|
|
|
err = vxlan_fdb_create(vxlan, all_zeros_mac,
|
|
|
|
&vxlan->default_dst.remote_ip,
|
|
|
|
NUD_REACHABLE|NUD_PERMANENT,
|
|
|
|
NLM_F_EXCL|NLM_F_CREATE,
|
|
|
|
vxlan->cfg.dst_port,
|
|
|
|
vxlan->default_dst.remote_vni,
|
|
|
|
vxlan->default_dst.remote_ifindex,
|
|
|
|
NTF_SELF);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = register_netdevice(dev);
|
|
|
|
if (err) {
|
|
|
|
vxlan_fdb_delete_default(vxlan);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
list_add(&vxlan->next, &vn->vxlan_list);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct net_device *vxlan_dev_create(struct net *net, const char *name,
|
|
|
|
u8 name_assign_type, struct vxlan_config *conf)
|
|
|
|
{
|
|
|
|
struct nlattr *tb[IFLA_MAX+1];
|
|
|
|
struct net_device *dev;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
memset(&tb, 0, sizeof(tb));
|
|
|
|
|
|
|
|
dev = rtnl_create_link(net, name, name_assign_type,
|
|
|
|
&vxlan_link_ops, tb);
|
|
|
|
if (IS_ERR(dev))
|
|
|
|
return dev;
|
|
|
|
|
|
|
|
err = vxlan_dev_configure(net, dev, conf);
|
|
|
|
if (err < 0) {
|
|
|
|
free_netdev(dev);
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return dev;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(vxlan_dev_create);
|
|
|
|
|
|
|
|
static int vxlan_newlink(struct net *src_net, struct net_device *dev,
|
|
|
|
struct nlattr *tb[], struct nlattr *data[])
|
|
|
|
{
|
|
|
|
struct vxlan_config conf;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
memset(&conf, 0, sizeof(conf));
|
2015-10-17 06:36:00 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_ID])
|
|
|
|
conf.vni = nla_get_u32(data[IFLA_VXLAN_ID]);
|
2015-07-21 15:44:02 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_GROUP]) {
|
|
|
|
conf.remote_ip.sin.sin_addr.s_addr = nla_get_in_addr(data[IFLA_VXLAN_GROUP]);
|
|
|
|
} else if (data[IFLA_VXLAN_GROUP6]) {
|
|
|
|
if (!IS_ENABLED(CONFIG_IPV6))
|
|
|
|
return -EPFNOSUPPORT;
|
|
|
|
|
|
|
|
conf.remote_ip.sin6.sin6_addr = nla_get_in6_addr(data[IFLA_VXLAN_GROUP6]);
|
|
|
|
conf.remote_ip.sa.sa_family = AF_INET6;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_LOCAL]) {
|
|
|
|
conf.saddr.sin.sin_addr.s_addr = nla_get_in_addr(data[IFLA_VXLAN_LOCAL]);
|
|
|
|
conf.saddr.sa.sa_family = AF_INET;
|
|
|
|
} else if (data[IFLA_VXLAN_LOCAL6]) {
|
|
|
|
if (!IS_ENABLED(CONFIG_IPV6))
|
|
|
|
return -EPFNOSUPPORT;
|
|
|
|
|
|
|
|
/* TODO: respect scope id */
|
|
|
|
conf.saddr.sin6.sin6_addr = nla_get_in6_addr(data[IFLA_VXLAN_LOCAL6]);
|
|
|
|
conf.saddr.sa.sa_family = AF_INET6;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_LINK])
|
|
|
|
conf.remote_ifindex = nla_get_u32(data[IFLA_VXLAN_LINK]);
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
if (data[IFLA_VXLAN_TOS])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.tos = nla_get_u8(data[IFLA_VXLAN_TOS]);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2012-10-30 17:27:16 +07:00
|
|
|
if (data[IFLA_VXLAN_TTL])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.ttl = nla_get_u8(data[IFLA_VXLAN_TTL]);
|
2012-10-30 17:27:16 +07:00
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
if (!data[IFLA_VXLAN_LEARNING] || nla_get_u8(data[IFLA_VXLAN_LEARNING]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_LEARN;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_AGEING])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.age_interval = nla_get_u32(data[IFLA_VXLAN_AGEING]);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2012-11-20 09:50:14 +07:00
|
|
|
if (data[IFLA_VXLAN_PROXY] && nla_get_u8(data[IFLA_VXLAN_PROXY]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_PROXY;
|
2012-11-20 09:50:14 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_RSC] && nla_get_u8(data[IFLA_VXLAN_RSC]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_RSC;
|
2012-11-20 09:50:14 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_L2MISS] && nla_get_u8(data[IFLA_VXLAN_L2MISS]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_L2MISS;
|
2012-11-20 09:50:14 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_L3MISS] && nla_get_u8(data[IFLA_VXLAN_L3MISS]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_L3MISS;
|
2012-11-20 09:50:14 +07:00
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
if (data[IFLA_VXLAN_LIMIT])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.addrmax = nla_get_u32(data[IFLA_VXLAN_LIMIT]);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-31 10:10:22 +07:00
|
|
|
if (data[IFLA_VXLAN_COLLECT_METADATA] &&
|
|
|
|
nla_get_u8(data[IFLA_VXLAN_COLLECT_METADATA]))
|
|
|
|
conf.flags |= VXLAN_F_COLLECT_METADATA;
|
|
|
|
|
2012-10-10 03:35:50 +07:00
|
|
|
if (data[IFLA_VXLAN_PORT_RANGE]) {
|
|
|
|
const struct ifla_vxlan_port_range *p
|
|
|
|
= nla_data(data[IFLA_VXLAN_PORT_RANGE]);
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.port_min = ntohs(p->low);
|
|
|
|
conf.port_max = ntohs(p->high);
|
2012-10-10 03:35:50 +07:00
|
|
|
}
|
|
|
|
|
2013-04-27 18:31:57 +07:00
|
|
|
if (data[IFLA_VXLAN_PORT])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.dst_port = nla_get_be16(data[IFLA_VXLAN_PORT]);
|
2013-04-27 18:31:57 +07:00
|
|
|
|
2014-06-05 07:20:29 +07:00
|
|
|
if (data[IFLA_VXLAN_UDP_CSUM] && nla_get_u8(data[IFLA_VXLAN_UDP_CSUM]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_UDP_CSUM;
|
2014-06-05 07:20:29 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_UDP_ZERO_CSUM6_TX] &&
|
|
|
|
nla_get_u8(data[IFLA_VXLAN_UDP_ZERO_CSUM6_TX]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_UDP_ZERO_CSUM6_TX;
|
2014-06-05 07:20:29 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_UDP_ZERO_CSUM6_RX] &&
|
|
|
|
nla_get_u8(data[IFLA_VXLAN_UDP_ZERO_CSUM6_RX]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_UDP_ZERO_CSUM6_RX;
|
2014-06-05 07:20:29 +07:00
|
|
|
|
2015-01-13 08:00:38 +07:00
|
|
|
if (data[IFLA_VXLAN_REMCSUM_TX] &&
|
|
|
|
nla_get_u8(data[IFLA_VXLAN_REMCSUM_TX]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_REMCSUM_TX;
|
2015-01-13 08:00:38 +07:00
|
|
|
|
|
|
|
if (data[IFLA_VXLAN_REMCSUM_RX] &&
|
|
|
|
nla_get_u8(data[IFLA_VXLAN_REMCSUM_RX]))
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_REMCSUM_RX;
|
2015-01-13 08:00:38 +07:00
|
|
|
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
if (data[IFLA_VXLAN_GBP])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_GBP;
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
|
2015-02-11 07:30:32 +07:00
|
|
|
if (data[IFLA_VXLAN_REMCSUM_NOPARTIAL])
|
2015-07-21 15:44:02 +07:00
|
|
|
conf.flags |= VXLAN_F_REMCSUM_NOPARTIAL;
|
2015-02-11 07:30:32 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
err = vxlan_dev_configure(src_net, dev, &conf);
|
|
|
|
switch (err) {
|
|
|
|
case -ENODEV:
|
|
|
|
pr_info("ifindex %d does not exist\n", conf.remote_ifindex);
|
|
|
|
break;
|
2013-01-30 06:43:07 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
case -EPERM:
|
|
|
|
pr_info("IPv6 is disabled via sysctl\n");
|
|
|
|
break;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
case -EEXIST:
|
|
|
|
pr_info("duplicate VNI %u\n", conf.vni);
|
|
|
|
break;
|
2013-06-25 20:01:51 +07:00
|
|
|
}
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
return err;
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void vxlan_dellink(struct net_device *dev, struct list_head *head)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
2014-04-24 15:02:49 +07:00
|
|
|
struct vxlan_net *vn = net_generic(vxlan->net, vxlan_net_id);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-07-14 00:18:18 +07:00
|
|
|
spin_lock(&vn->sock_lock);
|
2013-08-20 01:22:48 +07:00
|
|
|
if (!hlist_unhashed(&vxlan->hlist))
|
|
|
|
hlist_del_rcu(&vxlan->hlist);
|
2013-07-14 00:18:18 +07:00
|
|
|
spin_unlock(&vn->sock_lock);
|
|
|
|
|
2015-08-20 07:07:33 +07:00
|
|
|
gro_cells_destroy(&vxlan->gro_cells);
|
2013-05-16 18:35:20 +07:00
|
|
|
list_del(&vxlan->next);
|
2012-10-01 19:32:35 +07:00
|
|
|
unregister_netdevice_queue(dev, head);
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t vxlan_get_size(const struct net_device *dev)
|
|
|
|
{
|
|
|
|
|
|
|
|
return nla_total_size(sizeof(__u32)) + /* IFLA_VXLAN_ID */
|
2013-08-31 12:44:33 +07:00
|
|
|
nla_total_size(sizeof(struct in6_addr)) + /* IFLA_VXLAN_GROUP{6} */
|
2012-10-01 19:32:35 +07:00
|
|
|
nla_total_size(sizeof(__u32)) + /* IFLA_VXLAN_LINK */
|
2013-08-31 12:44:33 +07:00
|
|
|
nla_total_size(sizeof(struct in6_addr)) + /* IFLA_VXLAN_LOCAL{6} */
|
2012-10-01 19:32:35 +07:00
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_TTL */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_TOS */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_LEARNING */
|
2012-11-20 09:50:14 +07:00
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_PROXY */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_RSC */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_L2MISS */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_L3MISS */
|
2015-08-05 12:51:07 +07:00
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_COLLECT_METADATA */
|
2012-10-01 19:32:35 +07:00
|
|
|
nla_total_size(sizeof(__u32)) + /* IFLA_VXLAN_AGEING */
|
|
|
|
nla_total_size(sizeof(__u32)) + /* IFLA_VXLAN_LIMIT */
|
2012-10-10 03:35:50 +07:00
|
|
|
nla_total_size(sizeof(struct ifla_vxlan_port_range)) +
|
2014-06-05 07:20:29 +07:00
|
|
|
nla_total_size(sizeof(__be16)) + /* IFLA_VXLAN_PORT */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_UDP_CSUM */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_UDP_ZERO_CSUM6_TX */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_UDP_ZERO_CSUM6_RX */
|
2015-01-13 08:00:38 +07:00
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_REMCSUM_TX */
|
|
|
|
nla_total_size(sizeof(__u8)) + /* IFLA_VXLAN_REMCSUM_RX */
|
2012-10-01 19:32:35 +07:00
|
|
|
0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_fill_info(struct sk_buff *skb, const struct net_device *dev)
|
|
|
|
{
|
|
|
|
const struct vxlan_dev *vxlan = netdev_priv(dev);
|
2013-04-16 09:50:52 +07:00
|
|
|
const struct vxlan_rdst *dst = &vxlan->default_dst;
|
2012-10-10 03:35:50 +07:00
|
|
|
struct ifla_vxlan_port_range ports = {
|
2015-07-21 15:44:02 +07:00
|
|
|
.low = htons(vxlan->cfg.port_min),
|
|
|
|
.high = htons(vxlan->cfg.port_max),
|
2012-10-10 03:35:50 +07:00
|
|
|
};
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-04-16 09:50:52 +07:00
|
|
|
if (nla_put_u32(skb, IFLA_VXLAN_ID, dst->remote_vni))
|
2012-10-01 19:32:35 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
2013-08-31 12:44:33 +07:00
|
|
|
if (!vxlan_addr_any(&dst->remote_ip)) {
|
|
|
|
if (dst->remote_ip.sa.sa_family == AF_INET) {
|
2015-03-29 21:59:25 +07:00
|
|
|
if (nla_put_in_addr(skb, IFLA_VXLAN_GROUP,
|
|
|
|
dst->remote_ip.sin.sin_addr.s_addr))
|
2013-08-31 12:44:33 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
2015-03-29 21:59:25 +07:00
|
|
|
if (nla_put_in6_addr(skb, IFLA_VXLAN_GROUP6,
|
|
|
|
&dst->remote_ip.sin6.sin6_addr))
|
2013-08-31 12:44:33 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-04-16 09:50:52 +07:00
|
|
|
if (dst->remote_ifindex && nla_put_u32(skb, IFLA_VXLAN_LINK, dst->remote_ifindex))
|
2012-10-01 19:32:35 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (!vxlan_addr_any(&vxlan->cfg.saddr)) {
|
|
|
|
if (vxlan->cfg.saddr.sa.sa_family == AF_INET) {
|
2015-03-29 21:59:25 +07:00
|
|
|
if (nla_put_in_addr(skb, IFLA_VXLAN_LOCAL,
|
2015-07-21 15:44:02 +07:00
|
|
|
vxlan->cfg.saddr.sin.sin_addr.s_addr))
|
2013-08-31 12:44:33 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
#if IS_ENABLED(CONFIG_IPV6)
|
|
|
|
} else {
|
2015-03-29 21:59:25 +07:00
|
|
|
if (nla_put_in6_addr(skb, IFLA_VXLAN_LOCAL6,
|
2015-07-21 15:44:02 +07:00
|
|
|
&vxlan->cfg.saddr.sin6.sin6_addr))
|
2013-08-31 12:44:33 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2015-07-21 15:44:02 +07:00
|
|
|
if (nla_put_u8(skb, IFLA_VXLAN_TTL, vxlan->cfg.ttl) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_TOS, vxlan->cfg.tos) ||
|
2012-11-20 09:50:14 +07:00
|
|
|
nla_put_u8(skb, IFLA_VXLAN_LEARNING,
|
|
|
|
!!(vxlan->flags & VXLAN_F_LEARN)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_PROXY,
|
|
|
|
!!(vxlan->flags & VXLAN_F_PROXY)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_RSC, !!(vxlan->flags & VXLAN_F_RSC)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_L2MISS,
|
|
|
|
!!(vxlan->flags & VXLAN_F_L2MISS)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_L3MISS,
|
|
|
|
!!(vxlan->flags & VXLAN_F_L3MISS)) ||
|
2015-08-05 12:51:07 +07:00
|
|
|
nla_put_u8(skb, IFLA_VXLAN_COLLECT_METADATA,
|
|
|
|
!!(vxlan->flags & VXLAN_F_COLLECT_METADATA)) ||
|
2015-07-21 15:44:02 +07:00
|
|
|
nla_put_u32(skb, IFLA_VXLAN_AGEING, vxlan->cfg.age_interval) ||
|
|
|
|
nla_put_u32(skb, IFLA_VXLAN_LIMIT, vxlan->cfg.addrmax) ||
|
|
|
|
nla_put_be16(skb, IFLA_VXLAN_PORT, vxlan->cfg.dst_port) ||
|
2014-06-05 07:20:29 +07:00
|
|
|
nla_put_u8(skb, IFLA_VXLAN_UDP_CSUM,
|
|
|
|
!!(vxlan->flags & VXLAN_F_UDP_CSUM)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_UDP_ZERO_CSUM6_TX,
|
|
|
|
!!(vxlan->flags & VXLAN_F_UDP_ZERO_CSUM6_TX)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_UDP_ZERO_CSUM6_RX,
|
2015-01-13 08:00:38 +07:00
|
|
|
!!(vxlan->flags & VXLAN_F_UDP_ZERO_CSUM6_RX)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_REMCSUM_TX,
|
|
|
|
!!(vxlan->flags & VXLAN_F_REMCSUM_TX)) ||
|
|
|
|
nla_put_u8(skb, IFLA_VXLAN_REMCSUM_RX,
|
|
|
|
!!(vxlan->flags & VXLAN_F_REMCSUM_RX)))
|
2012-10-01 19:32:35 +07:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
2012-10-10 03:35:50 +07:00
|
|
|
if (nla_put(skb, IFLA_VXLAN_PORT_RANGE, sizeof(ports), &ports))
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
vxlan: Group Policy extension
Implements supports for the Group Policy VXLAN extension [0] to provide
a lightweight and simple security label mechanism across network peers
based on VXLAN. The security context and associated metadata is mapped
to/from skb->mark. This allows further mapping to a SELinux context
using SECMARK, to implement ACLs directly with nftables, iptables, OVS,
tc, etc.
The group membership is defined by the lower 16 bits of skb->mark, the
upper 16 bits are used for flags.
SELinux allows to manage label to secure local resources. However,
distributed applications require ACLs to implemented across hosts. This
is typically achieved by matching on L2-L4 fields to identify the
original sending host and process on the receiver. On top of that,
netlabel and specifically CIPSO [1] allow to map security contexts to
universal labels. However, netlabel and CIPSO are relatively complex.
This patch provides a lightweight alternative for overlay network
environments with a trusted underlay. No additional control protocol
is required.
Host 1: Host 2:
Group A Group B Group B Group A
+-----+ +-------------+ +-------+ +-----+
| lxc | | SELinux CTX | | httpd | | VM |
+--+--+ +--+----------+ +---+---+ +--+--+
\---+---/ \----+---/
| |
+---+---+ +---+---+
| vxlan | | vxlan |
+---+---+ +---+---+
+------------------------------+
Backwards compatibility:
A VXLAN-GBP socket can receive standard VXLAN frames and will assign
the default group 0x0000 to such frames. A Linux VXLAN socket will
drop VXLAN-GBP frames. The extension is therefore disabled by default
and needs to be specifically enabled:
ip link add [...] type vxlan [...] gbp
In a mixed environment with VXLAN and VXLAN-GBP sockets, the GBP socket
must run on a separate port number.
Examples:
iptables:
host1# iptables -I OUTPUT -m owner --uid-owner 101 -j MARK --set-mark 0x200
host2# iptables -I INPUT -m mark --mark 0x200 -j DROP
OVS:
# ovs-ofctl add-flow br0 'in_port=1,actions=load:0x200->NXM_NX_TUN_GBP_ID[],NORMAL'
# ovs-ofctl add-flow br0 'in_port=2,tun_gbp_id=0x200,actions=drop'
[0] https://tools.ietf.org/html/draft-smith-vxlan-group-policy
[1] http://lwn.net/Articles/204905/
Signed-off-by: Thomas Graf <tgraf@suug.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>
2015-01-15 09:53:55 +07:00
|
|
|
if (vxlan->flags & VXLAN_F_GBP &&
|
|
|
|
nla_put_flag(skb, IFLA_VXLAN_GBP))
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2015-02-11 07:30:32 +07:00
|
|
|
if (vxlan->flags & VXLAN_F_REMCSUM_NOPARTIAL &&
|
|
|
|
nla_put_flag(skb, IFLA_VXLAN_REMCSUM_NOPARTIAL))
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
return -EMSGSIZE;
|
|
|
|
}
|
|
|
|
|
2015-01-15 21:11:17 +07:00
|
|
|
static struct net *vxlan_get_link_net(const struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan = netdev_priv(dev);
|
|
|
|
|
|
|
|
return vxlan->net;
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
static struct rtnl_link_ops vxlan_link_ops __read_mostly = {
|
|
|
|
.kind = "vxlan",
|
|
|
|
.maxtype = IFLA_VXLAN_MAX,
|
|
|
|
.policy = vxlan_policy,
|
|
|
|
.priv_size = sizeof(struct vxlan_dev),
|
|
|
|
.setup = vxlan_setup,
|
|
|
|
.validate = vxlan_validate,
|
|
|
|
.newlink = vxlan_newlink,
|
|
|
|
.dellink = vxlan_dellink,
|
|
|
|
.get_size = vxlan_get_size,
|
|
|
|
.fill_info = vxlan_fill_info,
|
2015-01-15 21:11:17 +07:00
|
|
|
.get_link_net = vxlan_get_link_net,
|
2012-10-01 19:32:35 +07:00
|
|
|
};
|
|
|
|
|
2014-01-14 00:41:19 +07:00
|
|
|
static void vxlan_handle_lowerdev_unregister(struct vxlan_net *vn,
|
|
|
|
struct net_device *dev)
|
|
|
|
{
|
|
|
|
struct vxlan_dev *vxlan, *next;
|
|
|
|
LIST_HEAD(list_kill);
|
|
|
|
|
|
|
|
list_for_each_entry_safe(vxlan, next, &vn->vxlan_list, next) {
|
|
|
|
struct vxlan_rdst *dst = &vxlan->default_dst;
|
|
|
|
|
|
|
|
/* In case we created vxlan device with carrier
|
|
|
|
* and we loose the carrier due to module unload
|
|
|
|
* we also need to remove vxlan device. In other
|
|
|
|
* cases, it's not necessary and remote_ifindex
|
|
|
|
* is 0 here, so no matches.
|
|
|
|
*/
|
|
|
|
if (dst->remote_ifindex == dev->ifindex)
|
|
|
|
vxlan_dellink(vxlan->dev, &list_kill);
|
|
|
|
}
|
|
|
|
|
|
|
|
unregister_netdevice_many(&list_kill);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vxlan_lowerdev_event(struct notifier_block *unused,
|
|
|
|
unsigned long event, void *ptr)
|
|
|
|
{
|
|
|
|
struct net_device *dev = netdev_notifier_info_to_dev(ptr);
|
net: vxlan: convert to act as a pernet subsystem
As per suggestion from Eric W. Biederman, vxlan should be using
{un,}register_pernet_subsys() instead of {un,}register_pernet_device()
to ensure the vxlan_net structure is initialized before and cleaned
up after all network devices in a given network namespace i.e. when
dealing with network notifiers. This is similarly handeled already in
commit 91e2ff3528ac ("net: Teach vlans to cleanup as a pernet subsystem")
and, thus, improves upon fd27e0d44a89 ("net: vxlan: do not use vxlan_net
before checking event type"). Just as in 91e2ff3528ac, we do not need
to explicitly handle deletion of vxlan devices as network namespace
exit calls dellink on all remaining virtual devices, and
rtnl_link_unregister() calls dellink on all outstanding devices in that
network namespace, so we can entirely drop the pernet exit operation
as well. Moreover, on vxlan module exit, rcu_barrier() is called by
netns since commit 3a765edadb28 ("netns: Add an explicit rcu_barrier
to unregister_pernet_{device|subsys}"), so this may be omitted. Tested
with various scenarios and works well on my side.
Suggested-by: Eric W. Biederman <ebiederm@xmission.com>
Cc: Jesse Brandeburg <jesse.brandeburg@intel.com>
Cc: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: Daniel Borkmann <dborkman@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-01-23 03:07:53 +07:00
|
|
|
struct vxlan_net *vn = net_generic(dev_net(dev), vxlan_net_id);
|
2014-01-14 00:41:19 +07:00
|
|
|
|
net: vxlan: convert to act as a pernet subsystem
As per suggestion from Eric W. Biederman, vxlan should be using
{un,}register_pernet_subsys() instead of {un,}register_pernet_device()
to ensure the vxlan_net structure is initialized before and cleaned
up after all network devices in a given network namespace i.e. when
dealing with network notifiers. This is similarly handeled already in
commit 91e2ff3528ac ("net: Teach vlans to cleanup as a pernet subsystem")
and, thus, improves upon fd27e0d44a89 ("net: vxlan: do not use vxlan_net
before checking event type"). Just as in 91e2ff3528ac, we do not need
to explicitly handle deletion of vxlan devices as network namespace
exit calls dellink on all remaining virtual devices, and
rtnl_link_unregister() calls dellink on all outstanding devices in that
network namespace, so we can entirely drop the pernet exit operation
as well. Moreover, on vxlan module exit, rcu_barrier() is called by
netns since commit 3a765edadb28 ("netns: Add an explicit rcu_barrier
to unregister_pernet_{device|subsys}"), so this may be omitted. Tested
with various scenarios and works well on my side.
Suggested-by: Eric W. Biederman <ebiederm@xmission.com>
Cc: Jesse Brandeburg <jesse.brandeburg@intel.com>
Cc: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: Daniel Borkmann <dborkman@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-01-23 03:07:53 +07:00
|
|
|
if (event == NETDEV_UNREGISTER)
|
2014-01-14 00:41:19 +07:00
|
|
|
vxlan_handle_lowerdev_unregister(vn, dev);
|
|
|
|
|
|
|
|
return NOTIFY_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct notifier_block vxlan_notifier_block __read_mostly = {
|
|
|
|
.notifier_call = vxlan_lowerdev_event,
|
|
|
|
};
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
static __net_init int vxlan_init_net(struct net *net)
|
|
|
|
{
|
|
|
|
struct vxlan_net *vn = net_generic(net, vxlan_net_id);
|
2013-05-28 05:35:52 +07:00
|
|
|
unsigned int h;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
INIT_LIST_HEAD(&vn->vxlan_list);
|
2013-06-18 04:16:11 +07:00
|
|
|
spin_lock_init(&vn->sock_lock);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2013-05-16 18:35:20 +07:00
|
|
|
for (h = 0; h < PORT_HASH_SIZE; ++h)
|
|
|
|
INIT_HLIST_HEAD(&vn->sock_list[h]);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-24 15:02:49 +07:00
|
|
|
static void __net_exit vxlan_exit_net(struct net *net)
|
|
|
|
{
|
|
|
|
struct vxlan_net *vn = net_generic(net, vxlan_net_id);
|
|
|
|
struct vxlan_dev *vxlan, *next;
|
|
|
|
struct net_device *dev, *aux;
|
|
|
|
LIST_HEAD(list);
|
|
|
|
|
|
|
|
rtnl_lock();
|
|
|
|
for_each_netdev_safe(net, dev, aux)
|
|
|
|
if (dev->rtnl_link_ops == &vxlan_link_ops)
|
|
|
|
unregister_netdevice_queue(dev, &list);
|
|
|
|
|
|
|
|
list_for_each_entry_safe(vxlan, next, &vn->vxlan_list, next) {
|
|
|
|
/* If vxlan->dev is in the same netns, it has already been added
|
|
|
|
* to the list by the previous loop.
|
|
|
|
*/
|
2015-08-20 07:07:33 +07:00
|
|
|
if (!net_eq(dev_net(vxlan->dev), net)) {
|
|
|
|
gro_cells_destroy(&vxlan->gro_cells);
|
2015-05-19 00:51:24 +07:00
|
|
|
unregister_netdevice_queue(vxlan->dev, &list);
|
2015-08-20 07:07:33 +07:00
|
|
|
}
|
2014-04-24 15:02:49 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
unregister_netdevice_many(&list);
|
|
|
|
rtnl_unlock();
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
static struct pernet_operations vxlan_net_ops = {
|
|
|
|
.init = vxlan_init_net,
|
2014-04-24 15:02:49 +07:00
|
|
|
.exit = vxlan_exit_net,
|
2012-10-01 19:32:35 +07:00
|
|
|
.id = &vxlan_net_id,
|
|
|
|
.size = sizeof(struct vxlan_net),
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init vxlan_init_module(void)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
2013-06-18 04:16:09 +07:00
|
|
|
vxlan_wq = alloc_workqueue("vxlan", 0, 0);
|
|
|
|
if (!vxlan_wq)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2012-10-01 19:32:35 +07:00
|
|
|
get_random_bytes(&vxlan_salt, sizeof(vxlan_salt));
|
|
|
|
|
net: vxlan: convert to act as a pernet subsystem
As per suggestion from Eric W. Biederman, vxlan should be using
{un,}register_pernet_subsys() instead of {un,}register_pernet_device()
to ensure the vxlan_net structure is initialized before and cleaned
up after all network devices in a given network namespace i.e. when
dealing with network notifiers. This is similarly handeled already in
commit 91e2ff3528ac ("net: Teach vlans to cleanup as a pernet subsystem")
and, thus, improves upon fd27e0d44a89 ("net: vxlan: do not use vxlan_net
before checking event type"). Just as in 91e2ff3528ac, we do not need
to explicitly handle deletion of vxlan devices as network namespace
exit calls dellink on all remaining virtual devices, and
rtnl_link_unregister() calls dellink on all outstanding devices in that
network namespace, so we can entirely drop the pernet exit operation
as well. Moreover, on vxlan module exit, rcu_barrier() is called by
netns since commit 3a765edadb28 ("netns: Add an explicit rcu_barrier
to unregister_pernet_{device|subsys}"), so this may be omitted. Tested
with various scenarios and works well on my side.
Suggested-by: Eric W. Biederman <ebiederm@xmission.com>
Cc: Jesse Brandeburg <jesse.brandeburg@intel.com>
Cc: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: Daniel Borkmann <dborkman@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-01-23 03:07:53 +07:00
|
|
|
rc = register_pernet_subsys(&vxlan_net_ops);
|
2012-10-01 19:32:35 +07:00
|
|
|
if (rc)
|
|
|
|
goto out1;
|
|
|
|
|
2014-01-14 00:41:19 +07:00
|
|
|
rc = register_netdevice_notifier(&vxlan_notifier_block);
|
2012-10-01 19:32:35 +07:00
|
|
|
if (rc)
|
|
|
|
goto out2;
|
|
|
|
|
2014-01-14 00:41:19 +07:00
|
|
|
rc = rtnl_link_register(&vxlan_link_ops);
|
|
|
|
if (rc)
|
|
|
|
goto out3;
|
2012-10-01 19:32:35 +07:00
|
|
|
|
2014-01-14 00:41:19 +07:00
|
|
|
return 0;
|
|
|
|
out3:
|
|
|
|
unregister_netdevice_notifier(&vxlan_notifier_block);
|
2012-10-01 19:32:35 +07:00
|
|
|
out2:
|
net: vxlan: convert to act as a pernet subsystem
As per suggestion from Eric W. Biederman, vxlan should be using
{un,}register_pernet_subsys() instead of {un,}register_pernet_device()
to ensure the vxlan_net structure is initialized before and cleaned
up after all network devices in a given network namespace i.e. when
dealing with network notifiers. This is similarly handeled already in
commit 91e2ff3528ac ("net: Teach vlans to cleanup as a pernet subsystem")
and, thus, improves upon fd27e0d44a89 ("net: vxlan: do not use vxlan_net
before checking event type"). Just as in 91e2ff3528ac, we do not need
to explicitly handle deletion of vxlan devices as network namespace
exit calls dellink on all remaining virtual devices, and
rtnl_link_unregister() calls dellink on all outstanding devices in that
network namespace, so we can entirely drop the pernet exit operation
as well. Moreover, on vxlan module exit, rcu_barrier() is called by
netns since commit 3a765edadb28 ("netns: Add an explicit rcu_barrier
to unregister_pernet_{device|subsys}"), so this may be omitted. Tested
with various scenarios and works well on my side.
Suggested-by: Eric W. Biederman <ebiederm@xmission.com>
Cc: Jesse Brandeburg <jesse.brandeburg@intel.com>
Cc: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: Daniel Borkmann <dborkman@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-01-23 03:07:53 +07:00
|
|
|
unregister_pernet_subsys(&vxlan_net_ops);
|
2012-10-01 19:32:35 +07:00
|
|
|
out1:
|
2013-06-18 04:16:09 +07:00
|
|
|
destroy_workqueue(vxlan_wq);
|
2012-10-01 19:32:35 +07:00
|
|
|
return rc;
|
|
|
|
}
|
2013-05-28 05:35:53 +07:00
|
|
|
late_initcall(vxlan_init_module);
|
2012-10-01 19:32:35 +07:00
|
|
|
|
|
|
|
static void __exit vxlan_cleanup_module(void)
|
|
|
|
{
|
2013-06-18 04:16:09 +07:00
|
|
|
rtnl_link_unregister(&vxlan_link_ops);
|
2014-01-14 00:41:19 +07:00
|
|
|
unregister_netdevice_notifier(&vxlan_notifier_block);
|
2013-06-18 04:16:09 +07:00
|
|
|
destroy_workqueue(vxlan_wq);
|
net: vxlan: convert to act as a pernet subsystem
As per suggestion from Eric W. Biederman, vxlan should be using
{un,}register_pernet_subsys() instead of {un,}register_pernet_device()
to ensure the vxlan_net structure is initialized before and cleaned
up after all network devices in a given network namespace i.e. when
dealing with network notifiers. This is similarly handeled already in
commit 91e2ff3528ac ("net: Teach vlans to cleanup as a pernet subsystem")
and, thus, improves upon fd27e0d44a89 ("net: vxlan: do not use vxlan_net
before checking event type"). Just as in 91e2ff3528ac, we do not need
to explicitly handle deletion of vxlan devices as network namespace
exit calls dellink on all remaining virtual devices, and
rtnl_link_unregister() calls dellink on all outstanding devices in that
network namespace, so we can entirely drop the pernet exit operation
as well. Moreover, on vxlan module exit, rcu_barrier() is called by
netns since commit 3a765edadb28 ("netns: Add an explicit rcu_barrier
to unregister_pernet_{device|subsys}"), so this may be omitted. Tested
with various scenarios and works well on my side.
Suggested-by: Eric W. Biederman <ebiederm@xmission.com>
Cc: Jesse Brandeburg <jesse.brandeburg@intel.com>
Cc: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: Daniel Borkmann <dborkman@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-01-23 03:07:53 +07:00
|
|
|
unregister_pernet_subsys(&vxlan_net_ops);
|
|
|
|
/* rcu_barrier() is called by netns */
|
2012-10-01 19:32:35 +07:00
|
|
|
}
|
|
|
|
module_exit(vxlan_cleanup_module);
|
|
|
|
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_VERSION(VXLAN_VERSION);
|
2013-04-27 18:31:52 +07:00
|
|
|
MODULE_AUTHOR("Stephen Hemminger <stephen@networkplumber.org>");
|
2014-01-18 02:00:33 +07:00
|
|
|
MODULE_DESCRIPTION("Driver for VXLAN encapsulated traffic");
|
2012-10-01 19:32:35 +07:00
|
|
|
MODULE_ALIAS_RTNL_LINK("vxlan");
|