| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2010 Nicira Networks. | 
					
						
							|  |  |  |  * Distributed under the terms of the GNU GPL version 2. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Significant portions of this file may be copied from parts of the Linux | 
					
						
							|  |  |  |  * kernel, by Linus Torvalds and others. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-30 00:24:54 -07:00
										 |  |  | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | #include <linux/if.h>
 | 
					
						
							|  |  |  | #include <linux/skbuff.h>
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | #include <linux/ip.h>
 | 
					
						
							|  |  |  | #include <linux/if_tunnel.h>
 | 
					
						
							|  |  |  | #include <linux/if_vlan.h>
 | 
					
						
							|  |  |  | #include <linux/in.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <net/icmp.h>
 | 
					
						
							|  |  |  | #include <net/ip.h>
 | 
					
						
							|  |  |  | #include <net/protocol.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | #include "tunnel.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | #include "vport.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-10 17:40:22 -07:00
										 |  |  | #include "vport-generic.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * The GRE header is composed of a series of sections: a base and then a variable | 
					
						
							|  |  |  |  * number of options. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | #define GRE_HEADER_SECTION 4
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | struct gre_base_hdr { | 
					
						
							|  |  |  | 	__be16 flags; | 
					
						
							|  |  |  | 	__be16 protocol; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | static int gre_hdr_len(const struct tnl_port_config *port_config) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	int len; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	len = GRE_HEADER_SECTION; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (port_config->flags & TNL_F_CSUM) | 
					
						
							|  |  |  | 		len += GRE_HEADER_SECTION; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (port_config->out_key || | 
					
						
							|  |  |  | 	    port_config->flags & TNL_F_OUT_KEY_ACTION) | 
					
						
							|  |  |  | 		len += GRE_HEADER_SECTION; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	return len; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | static void gre_build_header(const struct vport *vport, | 
					
						
							|  |  |  | 			     const struct tnl_mutable_config *mutable, | 
					
						
							|  |  |  | 			     void *header) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	struct gre_base_hdr *greh = header; | 
					
						
							|  |  |  | 	__be32 *options = (__be32 *)(greh + 1); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | 	greh->protocol = htons(ETH_P_TEB); | 
					
						
							|  |  |  | 	greh->flags = 0; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	if (mutable->port_config.flags & TNL_F_CSUM) { | 
					
						
							|  |  |  | 		greh->flags |= GRE_CSUM; | 
					
						
							|  |  |  | 		*options = 0; | 
					
						
							|  |  |  | 		options++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	if (mutable->port_config.out_key || | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	    mutable->port_config.flags & TNL_F_OUT_KEY_ACTION) | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | 		greh->flags |= GRE_KEY; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	if (mutable->port_config.out_key) | 
					
						
							|  |  |  | 		*options = mutable->port_config.out_key; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct sk_buff *gre_update_header(const struct vport *vport, | 
					
						
							|  |  |  | 					 const struct tnl_mutable_config *mutable, | 
					
						
							|  |  |  | 					 struct dst_entry *dst, | 
					
						
							|  |  |  | 					 struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	__be32 *options = (__be32 *)(skb_network_header(skb) + mutable->tunnel_hlen | 
					
						
							|  |  |  | 					       - GRE_HEADER_SECTION); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	/* Work backwards over the options so the checksum is last. */ | 
					
						
							|  |  |  | 	if (mutable->port_config.flags & TNL_F_OUT_KEY_ACTION) { | 
					
						
							|  |  |  | 		*options = OVS_CB(skb)->tun_id; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		options--; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	if (mutable->port_config.flags & TNL_F_CSUM) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		*(__sum16 *)options = csum_fold(skb_checksum(skb, | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 						skb_transport_offset(skb), | 
					
						
							|  |  |  | 						skb->len - skb_transport_offset(skb), | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 						0)); | 
					
						
							| 
									
										
										
										
											2010-08-16 10:32:41 -04:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Allow our local IP stack to fragment the outer packet even if the | 
					
						
							|  |  |  | 	 * DF bit is set as a last resort. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	skb->local_df = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return skb; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | static int parse_header(struct iphdr *iph, __be16 *flags, __be32 *key) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	/* IP and ICMP protocol handlers check that the IHL is valid. */ | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | 	struct gre_base_hdr *greh = (struct gre_base_hdr *)((u8 *)iph + (iph->ihl << 2)); | 
					
						
							|  |  |  | 	__be32 *options = (__be32 *)(greh + 1); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	int hdr_len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | 	*flags = greh->flags; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (unlikely(greh->flags & (GRE_VERSION | GRE_ROUTING))) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (unlikely(greh->protocol != htons(ETH_P_TEB))) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	hdr_len = GRE_HEADER_SECTION; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | 	if (greh->flags & GRE_CSUM) { | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		hdr_len += GRE_HEADER_SECTION; | 
					
						
							|  |  |  | 		options++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 18:45:45 -07:00
										 |  |  | 	if (greh->flags & GRE_KEY) { | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		hdr_len += GRE_HEADER_SECTION; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		*key = *options; | 
					
						
							|  |  |  | 		options++; | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		*key = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (unlikely(greh->flags & GRE_SEQ)) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		hdr_len += GRE_HEADER_SECTION; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return hdr_len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | /* Called with rcu_read_lock and BH disabled. */ | 
					
						
							| 
									
										
										
										
											2010-07-14 19:27:18 -07:00
										 |  |  | static void gre_err(struct sk_buff *skb, u32 info) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct vport *vport; | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	const struct tnl_mutable_config *mutable; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	const int type = icmp_hdr(skb)->type; | 
					
						
							|  |  |  | 	const int code = icmp_hdr(skb)->code; | 
					
						
							|  |  |  | 	int mtu = ntohs(icmp_hdr(skb)->un.frag.mtu); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct iphdr *iph; | 
					
						
							|  |  |  | 	__be16 flags; | 
					
						
							|  |  |  | 	__be32 key; | 
					
						
							|  |  |  | 	int tunnel_hdr_len, tot_hdr_len; | 
					
						
							|  |  |  | 	unsigned int orig_mac_header; | 
					
						
							|  |  |  | 	unsigned int orig_nw_header; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (type != ICMP_DEST_UNREACH || code != ICMP_FRAG_NEEDED) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * The mimimum size packet that we would actually be able to process: | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	 * encapsulating IP header, minimum GRE header, Ethernet header, | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	 * inner IPv4 header. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	if (!pskb_may_pull(skb, sizeof(struct iphdr) + GRE_HEADER_SECTION + | 
					
						
							|  |  |  | 				ETH_HLEN + sizeof(struct iphdr))) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	iph = (struct iphdr *)skb->data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	tunnel_hdr_len = parse_header(iph, &flags, &key); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	if (tunnel_hdr_len < 0) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	vport = tnl_find_port(iph->saddr, iph->daddr, key, | 
					
						
							|  |  |  | 			      TNL_T_PROTO_GRE | TNL_T_KEY_EITHER, &mutable); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	if (!vport) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Packets received by this function were previously sent by us, so | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	 * any comparisons should be to the output values, not the input. | 
					
						
							|  |  |  | 	 * However, it's not really worth it to have a hash table based on | 
					
						
							|  |  |  | 	 * output keys (especially since ICMP error handling of tunneled packets | 
					
						
							|  |  |  | 	 * isn't that reliable anyways).  Therefore, we do a lookup based on the | 
					
						
							|  |  |  | 	 * out key as if it were the in key and then check to see if the input | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	 * and output keys are the same. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	if (mutable->port_config.in_key != mutable->port_config.out_key) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (!!(mutable->port_config.flags & TNL_F_IN_KEY_MATCH) != | 
					
						
							|  |  |  | 	    !!(mutable->port_config.flags & TNL_F_OUT_KEY_ACTION)) | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if ((mutable->port_config.flags & TNL_F_CSUM) && !(flags & GRE_CSUM)) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	tunnel_hdr_len += iph->ihl << 2; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	orig_mac_header = skb_mac_header(skb) - skb->data; | 
					
						
							|  |  |  | 	orig_nw_header = skb_network_header(skb) - skb->data; | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	skb_set_mac_header(skb, tunnel_hdr_len); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	tot_hdr_len = tunnel_hdr_len + ETH_HLEN; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	skb->protocol = eth_hdr(skb)->h_proto; | 
					
						
							|  |  |  | 	if (skb->protocol == htons(ETH_P_8021Q)) { | 
					
						
							|  |  |  | 		tot_hdr_len += VLAN_HLEN; | 
					
						
							|  |  |  | 		skb->protocol = vlan_eth_hdr(skb)->h_vlan_encapsulated_proto; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	skb_set_network_header(skb, tot_hdr_len); | 
					
						
							|  |  |  | 	mtu -= tot_hdr_len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	if (skb->protocol == htons(ETH_P_IP)) | 
					
						
							|  |  |  | 		tot_hdr_len += sizeof(struct iphdr); | 
					
						
							| 
									
										
										
										
											2010-05-18 17:30:42 -07:00
										 |  |  | #if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
 | 
					
						
							| 
									
										
										
										
											2010-04-20 16:51:59 -04:00
										 |  |  | 	else if (skb->protocol == htons(ETH_P_IPV6)) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		tot_hdr_len += sizeof(struct ipv6hdr); | 
					
						
							| 
									
										
										
										
											2010-05-18 17:30:42 -07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!pskb_may_pull(skb, tot_hdr_len)) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (skb->protocol == htons(ETH_P_IP)) { | 
					
						
							|  |  |  | 		if (mtu < IP_MIN_MTU) { | 
					
						
							|  |  |  | 			if (ntohs(ip_hdr(skb)->tot_len) >= IP_MIN_MTU) | 
					
						
							|  |  |  | 				mtu = IP_MIN_MTU; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-18 17:30:42 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | #if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
 | 
					
						
							|  |  |  | 	else if (skb->protocol == htons(ETH_P_IPV6)) { | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		if (mtu < IPV6_MIN_MTU) { | 
					
						
							|  |  |  | 			unsigned int packet_length = sizeof(struct ipv6hdr) + | 
					
						
							|  |  |  | 					      ntohs(ipv6_hdr(skb)->payload_len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (packet_length >= IPV6_MIN_MTU | 
					
						
							|  |  |  | 			    || ntohs(ipv6_hdr(skb)->payload_len) == 0) | 
					
						
							|  |  |  | 				mtu = IPV6_MIN_MTU; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-05-18 17:30:42 -07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	__skb_pull(skb, tunnel_hdr_len); | 
					
						
							|  |  |  | 	tnl_frag_needed(vport, mutable, skb, mtu, key); | 
					
						
							|  |  |  | 	__skb_push(skb, tunnel_hdr_len); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	skb_set_mac_header(skb, orig_mac_header); | 
					
						
							|  |  |  | 	skb_set_network_header(skb, orig_nw_header); | 
					
						
							|  |  |  | 	skb->protocol = htons(ETH_P_IP); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | static bool check_checksum(struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct iphdr *iph = ip_hdr(skb); | 
					
						
							|  |  |  | 	struct gre_base_hdr *greh = (struct gre_base_hdr *)(iph + 1); | 
					
						
							|  |  |  | 	__sum16 csum = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (greh->flags & GRE_CSUM) { | 
					
						
							|  |  |  | 		switch (skb->ip_summed) { | 
					
						
							|  |  |  | 		case CHECKSUM_COMPLETE: | 
					
						
							|  |  |  | 			csum = csum_fold(skb->csum); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (!csum) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			/* Fall through. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case CHECKSUM_NONE: | 
					
						
							|  |  |  | 			skb->csum = 0; | 
					
						
							|  |  |  | 			csum = __skb_checksum_complete(skb); | 
					
						
							|  |  |  | 			skb->ip_summed = CHECKSUM_COMPLETE; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (csum == 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Called with rcu_read_lock and BH disabled. */ | 
					
						
							| 
									
										
										
										
											2010-07-14 19:27:18 -07:00
										 |  |  | static int gre_rcv(struct sk_buff *skb) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct vport *vport; | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	const struct tnl_mutable_config *mutable; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	int hdr_len; | 
					
						
							|  |  |  | 	struct iphdr *iph; | 
					
						
							|  |  |  | 	__be16 flags; | 
					
						
							|  |  |  | 	__be32 key; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (unlikely(!pskb_may_pull(skb, sizeof(struct gre_base_hdr) + ETH_HLEN))) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (unlikely(!check_checksum(skb))) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	hdr_len = parse_header(ip_hdr(skb), &flags, &key); | 
					
						
							|  |  |  | 	if (unlikely(hdr_len < 0)) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (unlikely(!pskb_may_pull(skb, hdr_len + ETH_HLEN))) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	iph = ip_hdr(skb); | 
					
						
							|  |  |  | 	vport = tnl_find_port(iph->daddr, iph->saddr, key, | 
					
						
							|  |  |  | 			      TNL_T_PROTO_GRE | TNL_T_KEY_EITHER, &mutable); | 
					
						
							|  |  |  | 	if (unlikely(!vport)) { | 
					
						
							|  |  |  | 		icmp_send(skb, ICMP_DEST_UNREACH, ICMP_PORT_UNREACH, 0); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		goto error; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	if (mutable->port_config.flags & TNL_F_IN_KEY_MATCH) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 		OVS_CB(skb)->tun_id = key; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		OVS_CB(skb)->tun_id = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	__skb_pull(skb, hdr_len); | 
					
						
							|  |  |  | 	skb_postpull_rcsum(skb, skb_transport_header(skb), hdr_len + ETH_HLEN); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	tnl_rcv(vport, skb); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  | 	kfree_skb(skb); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-24 09:31:49 -08:00
										 |  |  | static const struct tnl_ops gre_tnl_ops = { | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	.tunnel_type	= TNL_T_PROTO_GRE, | 
					
						
							|  |  |  | 	.ipproto	= IPPROTO_GRE, | 
					
						
							|  |  |  | 	.hdr_len	= gre_hdr_len, | 
					
						
							|  |  |  | 	.build_header	= gre_build_header, | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	.update_header	= gre_update_header, | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-04 16:32:57 -07:00
										 |  |  | static struct vport *gre_create(const struct vport_parms *parms) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-04 16:32:57 -07:00
										 |  |  | 	return tnl_create(parms, &gre_vport_ops, &gre_tnl_ops); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-23 22:08:27 -08:00
										 |  |  | static const struct net_protocol gre_protocol_handlers = { | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	.handler	=	gre_rcv, | 
					
						
							|  |  |  | 	.err_handler	=	gre_err, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 19:27:18 -07:00
										 |  |  | static int gre_init(void) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = inet_add_protocol(&gre_protocol_handlers, IPPROTO_GRE); | 
					
						
							| 
									
										
										
										
											2010-08-27 13:55:02 -07:00
										 |  |  | 	if (err) | 
					
						
							| 
									
										
										
										
											2010-08-30 00:24:54 -07:00
										 |  |  | 		pr_warn("cannot register gre protocol handler\n"); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | static void gre_exit(void) | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	inet_del_protocol(&gre_protocol_handlers, IPPROTO_GRE); | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-23 22:08:27 -08:00
										 |  |  | const struct vport_ops gre_vport_ops = { | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | 	.type		= "gre", | 
					
						
							|  |  |  | 	.flags		= VPORT_F_GEN_STATS | VPORT_F_TUN_ID, | 
					
						
							|  |  |  | 	.init		= gre_init, | 
					
						
							|  |  |  | 	.exit		= gre_exit, | 
					
						
							|  |  |  | 	.create		= gre_create, | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	.modify		= tnl_modify, | 
					
						
							|  |  |  | 	.destroy	= tnl_destroy, | 
					
						
							|  |  |  | 	.set_mtu	= tnl_set_mtu, | 
					
						
							|  |  |  | 	.set_addr	= tnl_set_addr, | 
					
						
							|  |  |  | 	.get_name	= tnl_get_name, | 
					
						
							|  |  |  | 	.get_addr	= tnl_get_addr, | 
					
						
							| 
									
										
										
										
											2010-05-10 17:40:22 -07:00
										 |  |  | 	.get_dev_flags	= vport_gen_get_dev_flags, | 
					
						
							|  |  |  | 	.is_running	= vport_gen_is_running, | 
					
						
							|  |  |  | 	.get_operstate	= vport_gen_get_operstate, | 
					
						
							| 
									
										
										
										
											2010-08-10 20:11:48 -04:00
										 |  |  | 	.get_mtu	= tnl_get_mtu, | 
					
						
							|  |  |  | 	.send		= tnl_send, | 
					
						
							| 
									
										
										
										
											2010-04-17 15:23:31 -04:00
										 |  |  | }; |