| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2011 Nicira, Inc. | 
					
						
							|  |  |  |  * Copyright (c) 2013 Cisco Systems, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of version 2 of the GNU General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, but | 
					
						
							|  |  |  |  * WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | 
					
						
							|  |  |  |  * General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software | 
					
						
							|  |  |  |  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA | 
					
						
							|  |  |  |  * 02110-1301, USA | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/version.h>
 | 
					
						
							|  |  |  | #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,26)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/in.h>
 | 
					
						
							|  |  |  | #include <linux/ip.h>
 | 
					
						
							|  |  |  | #include <linux/net.h>
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | #include <linux/rculist.h>
 | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | #include <linux/udp.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <net/icmp.h>
 | 
					
						
							|  |  |  | #include <net/ip.h>
 | 
					
						
							|  |  |  | #include <net/udp.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "datapath.h"
 | 
					
						
							|  |  |  | #include "tunnel.h"
 | 
					
						
							|  |  |  | #include "vport.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  *  LISP encapsulation header: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | 
					
						
							|  |  |  |  *  |N|L|E|V|I|flags|            Nonce/Map-Version                  | | 
					
						
							|  |  |  |  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | 
					
						
							|  |  |  |  *  |                 Instance ID/Locator Status Bits               | | 
					
						
							|  |  |  |  *  +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * struct lisphdr - LISP header | 
					
						
							|  |  |  |  * @nonce_present: Flag indicating the presence of a 24 bit nonce value. | 
					
						
							|  |  |  |  * @locator_status_bits_present: Flag indicating the presence of Locator Status | 
					
						
							|  |  |  |  *                               Bits (LSB). | 
					
						
							|  |  |  |  * @solicit_echo_nonce: Flag indicating the use of the echo noncing mechanism. | 
					
						
							|  |  |  |  * @map_version_present: Flag indicating the use of mapping versioning. | 
					
						
							|  |  |  |  * @instance_id_present: Flag indicating the presence of a 24 bit Instance ID. | 
					
						
							|  |  |  |  * @reserved_flags: 3 bits reserved for future flags. | 
					
						
							|  |  |  |  * @nonce: 24 bit nonce value. | 
					
						
							|  |  |  |  * @map_version: 24 bit mapping version. | 
					
						
							|  |  |  |  * @locator_status_bits: Locator Status Bits: 32 bits when instance_id_present | 
					
						
							|  |  |  |  *                       is not set, 8 bits when it is. | 
					
						
							|  |  |  |  * @instance_id: 24 bit Instance ID | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct lisphdr { | 
					
						
							|  |  |  | #ifdef __LITTLE_ENDIAN_BITFIELD
 | 
					
						
							|  |  |  | 	__u8 reserved_flags:3; | 
					
						
							|  |  |  | 	__u8 instance_id_present:1; | 
					
						
							|  |  |  | 	__u8 map_version_present:1; | 
					
						
							|  |  |  | 	__u8 solicit_echo_nonce:1; | 
					
						
							|  |  |  | 	__u8 locator_status_bits_present:1; | 
					
						
							|  |  |  | 	__u8 nonce_present:1; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	__u8 nonce_present:1; | 
					
						
							|  |  |  | 	__u8 locator_status_bits_present:1; | 
					
						
							|  |  |  | 	__u8 solicit_echo_nonce:1; | 
					
						
							|  |  |  | 	__u8 map_version_present:1; | 
					
						
							|  |  |  | 	__u8 instance_id_present:1; | 
					
						
							|  |  |  | 	__u8 reserved_flags:3; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	union { | 
					
						
							|  |  |  | 		__u8 nonce[3]; | 
					
						
							|  |  |  | 		__u8 map_version[3]; | 
					
						
							|  |  |  | 	} u1; | 
					
						
							|  |  |  | 	union { | 
					
						
							|  |  |  | 		__be32 locator_status_bits; | 
					
						
							|  |  |  | 		struct { | 
					
						
							|  |  |  | 			__u8 instance_id[3]; | 
					
						
							|  |  |  | 			__u8 locator_status_bits; | 
					
						
							|  |  |  | 		} word2; | 
					
						
							|  |  |  | 	} u2; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define LISP_HLEN (sizeof(struct udphdr) + sizeof(struct lisphdr))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * struct lisp_port - Keeps track of open UDP ports | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  |  * @dst_port: lisp UDP port no. | 
					
						
							|  |  |  |  * @list: list element in @lisp_ports. | 
					
						
							|  |  |  |  * @lisp_rcv_socket: The socket created for this port number. | 
					
						
							|  |  |  |  * @name: vport name. | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  |  */ | 
					
						
							|  |  |  | struct lisp_port { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	__be16 dst_port; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	struct list_head list; | 
					
						
							|  |  |  | 	struct socket *lisp_rcv_socket; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	char name[IFNAMSIZ]; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static LIST_HEAD(lisp_ports); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | static inline struct lisp_port *lisp_vport(const struct vport *vport) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return vport_priv(vport); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | static struct lisp_port *lisp_find_port(struct net *net, __be16 port) | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct lisp_port *lisp_port; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	list_for_each_entry_rcu(lisp_port, &lisp_ports, list) { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 		if (lisp_port->dst_port == port && | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 			net_eq(sock_net(lisp_port->lisp_rcv_socket->sk), net)) | 
					
						
							|  |  |  | 			return lisp_port; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline struct lisphdr *lisp_hdr(const struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (struct lisphdr *)(udp_hdr(skb) + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Convert 64 bit tunnel ID to 24 bit Instance ID. */ | 
					
						
							|  |  |  | static void tunnel_id_to_instance_id(__be64 tun_id, __u8 *iid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef __BIG_ENDIAN
 | 
					
						
							|  |  |  | 	iid[0] = (__force __u8)(tun_id >> 16); | 
					
						
							|  |  |  | 	iid[1] = (__force __u8)(tun_id >> 8); | 
					
						
							|  |  |  | 	iid[2] = (__force __u8)tun_id; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	iid[0] = (__force __u8)((__force u64)tun_id >> 40); | 
					
						
							|  |  |  | 	iid[1] = (__force __u8)((__force u64)tun_id >> 48); | 
					
						
							|  |  |  | 	iid[2] = (__force __u8)((__force u64)tun_id >> 56); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Convert 24 bit Instance ID to 64 bit tunnel ID. */ | 
					
						
							|  |  |  | static __be64 instance_id_to_tunnel_id(__u8 *iid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef __BIG_ENDIAN
 | 
					
						
							|  |  |  | 	return (iid[0] << 16) | (iid[1] << 8) | iid[2]; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	return (__force __be64)(((__force u64)iid[0] << 40) | | 
					
						
							|  |  |  | 				((__force u64)iid[1] << 48) | | 
					
						
							|  |  |  | 				((__force u64)iid[2] << 56)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | static void lisp_build_header(const struct vport *vport, | 
					
						
							|  |  |  | 			      struct sk_buff *skb, | 
					
						
							|  |  |  | 			      int tunnel_hlen) | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	struct lisp_port *lisp_port = lisp_vport(vport); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	struct udphdr *udph = udp_hdr(skb); | 
					
						
							|  |  |  | 	struct lisphdr *lisph = (struct lisphdr *)(udph + 1); | 
					
						
							|  |  |  | 	const struct ovs_key_ipv4_tunnel *tun_key = OVS_CB(skb)->tun_key; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	udph->dest = lisp_port->dst_port; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	udph->source = htons(ovs_tnl_get_src_port(skb)); | 
					
						
							|  |  |  | 	udph->check = 0; | 
					
						
							|  |  |  | 	udph->len = htons(skb->len - skb_transport_offset(skb)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lisph->nonce_present = 0;	/* We don't support echo nonce algorithm */ | 
					
						
							|  |  |  | 	lisph->locator_status_bits_present = 1;	/* Set LSB */ | 
					
						
							|  |  |  | 	lisph->solicit_echo_nonce = 0;	/* No echo noncing */ | 
					
						
							|  |  |  | 	lisph->map_version_present = 0;	/* No mapping versioning, nonce instead */ | 
					
						
							|  |  |  | 	lisph->instance_id_present = 1;	/* Store the tun_id as Instance ID  */ | 
					
						
							|  |  |  | 	lisph->reserved_flags = 0;	/* Reserved flags, set to 0  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lisph->u1.nonce[0] = 0; | 
					
						
							|  |  |  | 	lisph->u1.nonce[1] = 0; | 
					
						
							|  |  |  | 	lisph->u1.nonce[2] = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	tunnel_id_to_instance_id(tun_key->tun_id, &lisph->u2.word2.instance_id[0]); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	lisph->u2.word2.locator_status_bits = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Called with rcu_read_lock and BH disabled. */ | 
					
						
							|  |  |  | static int lisp_rcv(struct sock *sk, struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	struct lisp_port *lisp_port; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	struct lisphdr *lisph; | 
					
						
							|  |  |  | 	struct iphdr *iph, *inner_iph; | 
					
						
							|  |  |  | 	struct ovs_key_ipv4_tunnel tun_key; | 
					
						
							|  |  |  | 	__be64 key; | 
					
						
							|  |  |  | 	struct ethhdr *ethh; | 
					
						
							|  |  |  | 	__be16 protocol; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	lisp_port = lisp_find_port(dev_net(skb->dev), udp_hdr(skb)->dest); | 
					
						
							|  |  |  | 	if (unlikely(!lisp_port)) | 
					
						
							|  |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	if (unlikely(!pskb_may_pull(skb, LISP_HLEN))) | 
					
						
							|  |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lisph = lisp_hdr(skb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	skb_pull_rcsum(skb, LISP_HLEN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (lisph->instance_id_present != 1) | 
					
						
							|  |  |  | 		key = 0; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		key = instance_id_to_tunnel_id(&lisph->u2.word2.instance_id[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Save outer tunnel values */ | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	iph = ip_hdr(skb); | 
					
						
							|  |  |  | 	tnl_tun_key_init(&tun_key, iph, key, OVS_TNL_F_KEY); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Drop non-IP inner packets */ | 
					
						
							|  |  |  | 	inner_iph = (struct iphdr *)(lisph + 1); | 
					
						
							|  |  |  | 	switch (inner_iph->version) { | 
					
						
							|  |  |  | 	case 4: | 
					
						
							|  |  |  | 		protocol = htons(ETH_P_IP); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 6: | 
					
						
							|  |  |  | 		protocol = htons(ETH_P_IPV6); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		goto error; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Add Ethernet header */ | 
					
						
							|  |  |  | 	ethh = (struct ethhdr *)skb_push(skb, ETH_HLEN); | 
					
						
							|  |  |  | 	memset(ethh, 0, ETH_HLEN); | 
					
						
							|  |  |  | 	ethh->h_dest[0] = 0x02; | 
					
						
							|  |  |  | 	ethh->h_source[0] = 0x02; | 
					
						
							|  |  |  | 	ethh->h_proto = protocol; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:34:19 -07:00
										 |  |  | 	ovs_tnl_rcv(vport_from_priv(lisp_port), skb, &tun_key); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  | 	kfree_skb(skb); | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Arbitrary value.  Irrelevant as long as it's not 0 since we set the handler. */ | 
					
						
							|  |  |  | #define UDP_ENCAP_LISP 1
 | 
					
						
							|  |  |  | static int lisp_socket_init(struct lisp_port *lisp_port, struct net *net) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct sockaddr_in sin; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	int err; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	err = sock_create_kern(AF_INET, SOCK_DGRAM, 0, | 
					
						
							|  |  |  | 			       &lisp_port->lisp_rcv_socket); | 
					
						
							|  |  |  | 	if (err) | 
					
						
							|  |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* release net ref. */ | 
					
						
							|  |  |  | 	sk_change_net(lisp_port->lisp_rcv_socket->sk, net); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	sin.sin_family = AF_INET; | 
					
						
							|  |  |  | 	sin.sin_addr.s_addr = htonl(INADDR_ANY); | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	sin.sin_port = lisp_port->dst_port; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	err = kernel_bind(lisp_port->lisp_rcv_socket, (struct sockaddr *)&sin, | 
					
						
							|  |  |  | 			  sizeof(struct sockaddr_in)); | 
					
						
							|  |  |  | 	if (err) | 
					
						
							|  |  |  | 		goto error_sock; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	udp_sk(lisp_port->lisp_rcv_socket->sk)->encap_type = UDP_ENCAP_LISP; | 
					
						
							|  |  |  | 	udp_sk(lisp_port->lisp_rcv_socket->sk)->encap_rcv = lisp_rcv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	udp_encap_enable(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | error_sock: | 
					
						
							|  |  |  | 	sk_release_kernel(lisp_port->lisp_rcv_socket->sk); | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  | 	pr_warn("cannot register lisp protocol handler: %d\n", err); | 
					
						
							|  |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | static int lisp_get_options(const struct vport *vport, struct sk_buff *skb) | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	struct lisp_port *lisp_port = lisp_vport(vport); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	if (nla_put_u16(skb, OVS_TUNNEL_ATTR_DST_PORT, ntohs(lisp_port->dst_port))) | 
					
						
							|  |  |  | 		return -EMSGSIZE; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | static void lisp_tnl_destroy(struct vport *vport) | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	struct lisp_port *lisp_port = lisp_vport(vport); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	list_del_rcu(&lisp_port->list); | 
					
						
							|  |  |  | 	/* Release socket */ | 
					
						
							|  |  |  | 	sk_release_kernel(lisp_port->lisp_rcv_socket->sk); | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ovs_vport_deferred_free(vport); | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | static struct vport *lisp_tnl_create(const struct vport_parms *parms) | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	struct net *net = ovs_dp_get_net(parms->dp); | 
					
						
							|  |  |  | 	struct nlattr *options = parms->options; | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	struct lisp_port *lisp_port; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	struct vport *vport; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	struct nlattr *a; | 
					
						
							|  |  |  | 	int err; | 
					
						
							|  |  |  | 	u16 dst_port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!options) { | 
					
						
							|  |  |  | 		err = -EINVAL; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 		goto error; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	a = nla_find_nested(options, OVS_TUNNEL_ATTR_DST_PORT); | 
					
						
							|  |  |  | 	if (a && nla_len(a) == sizeof(u16)) { | 
					
						
							|  |  |  | 		dst_port = nla_get_u16(a); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* Require destination port from userspace. */ | 
					
						
							|  |  |  | 		err = -EINVAL; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 		goto error; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Verify if we already have a socket created for this port */ | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	if (lisp_find_port(net, htons(dst_port))) { | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 		err = -EEXIST; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 		goto error; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	vport = ovs_vport_alloc(sizeof(struct lisp_port), | 
					
						
							|  |  |  | 				&ovs_lisp_vport_ops, parms); | 
					
						
							|  |  |  | 	if (IS_ERR(vport)) | 
					
						
							|  |  |  | 		return vport; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	lisp_port = lisp_vport(vport); | 
					
						
							|  |  |  | 	lisp_port->dst_port = htons(dst_port); | 
					
						
							|  |  |  | 	strncpy(lisp_port->name, parms->name, IFNAMSIZ); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	err = lisp_socket_init(lisp_port, net); | 
					
						
							|  |  |  | 	if (err) | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 		goto error_free; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	list_add_tail_rcu(&lisp_port->list, &lisp_ports); | 
					
						
							|  |  |  | 	return vport; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | error_free: | 
					
						
							|  |  |  | 	ovs_vport_free(vport); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | error: | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	return ERR_PTR(err); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | static int lisp_tnl_send(struct vport *vport, struct sk_buff *skb) | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	int tnl_len; | 
					
						
							|  |  |  | 	int network_offset = skb_network_offset(skb); | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:31:01 -07:00
										 |  |  | 	if (unlikely(!OVS_CB(skb)->tun_key)) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	/* We only encapsulate IPv4 and IPv6 packets */ | 
					
						
							|  |  |  | 	switch (skb->protocol) { | 
					
						
							|  |  |  | 	case htons(ETH_P_IP): | 
					
						
							|  |  |  | 	case htons(ETH_P_IPV6): | 
					
						
							|  |  |  | 		/* Pop off "inner" Ethernet header */ | 
					
						
							|  |  |  | 		skb_pull(skb, network_offset); | 
					
						
							|  |  |  | 		tnl_len = ovs_tnl_send(vport, skb, IPPROTO_UDP, | 
					
						
							|  |  |  | 				LISP_HLEN, lisp_build_header); | 
					
						
							|  |  |  | 		return tnl_len > 0 ? tnl_len + network_offset : tnl_len; | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2013-05-06 10:31:01 -07:00
										 |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | static const char *lisp_get_name(const struct vport *vport) | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	struct lisp_port *lisp_port = lisp_vport(vport); | 
					
						
							|  |  |  | 	return lisp_port->name; | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const struct vport_ops ovs_lisp_vport_ops = { | 
					
						
							|  |  |  | 	.type		= OVS_VPORT_TYPE_LISP, | 
					
						
							|  |  |  | 	.create		= lisp_tnl_create, | 
					
						
							|  |  |  | 	.destroy	= lisp_tnl_destroy, | 
					
						
							| 
									
										
										
										
											2013-05-06 10:29:08 -07:00
										 |  |  | 	.get_name	= lisp_get_name, | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  | 	.get_options	= lisp_get_options, | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  | 	.send		= lisp_tnl_send, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #warning LISP tunneling will not be available on kernels before 2.6.26
 | 
					
						
							|  |  |  | #endif /* Linux kernel < 2.6.26 */
 |