| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2014-06-24 13:47:33 -07:00
										 |  |  |  |  * Copyright (c) 2010, 2011, 2012, 2013, 2014 Nicira, Inc. | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  |  * You may obtain a copy of the License at: | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-09-24 10:55:57 -07:00
										 |  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  |  * limitations under the License. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <config.h>
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "netdev-vport.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | #include <sys/socket.h>
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | #include <net/if.h>
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | #include <sys/ioctl.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-10 10:42:42 -08:00
										 |  |  |  | #include "byte-order.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | #include "csum.h"
 | 
					
						
							| 
									
										
										
										
											2011-03-11 15:18:30 -08:00
										 |  |  |  | #include "daemon.h"
 | 
					
						
							|  |  |  |  | #include "dirs.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | #include "dpif.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-25 12:01:53 -08:00
										 |  |  |  | #include "dp-packet.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | #include "dynamic-string.h"
 | 
					
						
							|  |  |  |  | #include "flow.h"
 | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | #include "hash.h"
 | 
					
						
							|  |  |  |  | #include "hmap.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | #include "list.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | #include "netdev-provider.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | #include "odp-netlink.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  | #include "dp-packet.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-16 11:38:12 -07:00
										 |  |  |  | #include "ovs-router.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | #include "packets.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  | #include "poll-loop.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  | #include "route-table.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | #include "shash.h"
 | 
					
						
							|  |  |  |  | #include "socket-util.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  | #include "openvswitch/vlog.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | #include "unaligned.h"
 | 
					
						
							|  |  |  |  | #include "unixctl.h"
 | 
					
						
							|  |  |  |  | #include "util.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-19 14:47:01 -07:00
										 |  |  |  | VLOG_DEFINE_THIS_MODULE(netdev_vport); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | static struct vlog_rate_limit err_rl = VLOG_RATE_LIMIT_INIT(60, 5); | 
					
						
							| 
									
										
										
										
											2010-07-16 11:02:49 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-05 19:07:32 -07:00
										 |  |  |  | #define GENEVE_DST_PORT 6081
 | 
					
						
							| 
									
										
										
										
											2013-04-26 14:30:24 -04:00
										 |  |  |  | #define VXLAN_DST_PORT 4789
 | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  |  | #define LISP_DST_PORT 4341
 | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  | #define STT_DST_PORT 7471
 | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | #define VXLAN_HLEN   (sizeof(struct eth_header) +         \
 | 
					
						
							|  |  |  |  |                       sizeof(struct ip_header)  +         \ | 
					
						
							|  |  |  |  |                       sizeof(struct udp_header) +         \ | 
					
						
							|  |  |  |  |                       sizeof(struct vxlanhdr)) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  | #define GENEVE_BASE_HLEN   (sizeof(struct eth_header) +         \
 | 
					
						
							|  |  |  |  |                             sizeof(struct ip_header)  +         \ | 
					
						
							|  |  |  |  |                             sizeof(struct udp_header) +         \ | 
					
						
							|  |  |  |  |                             sizeof(struct genevehdr)) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | #define DEFAULT_TTL 64
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | struct netdev_vport { | 
					
						
							|  |  |  |  |     struct netdev up; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Protects all members below. */ | 
					
						
							|  |  |  |  |     struct ovs_mutex mutex; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-30 11:36:42 -08:00
										 |  |  |  |     uint8_t etheraddr[ETH_ADDR_LEN]; | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |     struct netdev_stats stats; | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Tunnels. */ | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     struct netdev_tunnel_config tnl_cfg; | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     char egress_iface[IFNAMSIZ]; | 
					
						
							|  |  |  |  |     bool carrier_status; | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Patch Ports. */ | 
					
						
							|  |  |  |  |     char *peer; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct vport_class { | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |     const char *dpif_port; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     struct netdev_class netdev_class; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  | /* Last read of the route-table's change number. */ | 
					
						
							|  |  |  |  | static uint64_t rt_change_seqno; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | static int netdev_vport_construct(struct netdev *); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | static int get_patch_config(const struct netdev *netdev, struct smap *args); | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | static int get_tunnel_config(const struct netdev *, struct smap *args); | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  | static bool tunnel_check_status_change__(struct netdev_vport *); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | static uint16_t tnl_udp_port_min = 32768; | 
					
						
							|  |  |  |  | static uint16_t tnl_udp_port_max = 61000; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static bool | 
					
						
							|  |  |  |  | is_vport_class(const struct netdev_class *class) | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     return class->construct == netdev_vport_construct; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  | bool | 
					
						
							|  |  |  |  | netdev_vport_is_vport_class(const struct netdev_class *class) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return is_vport_class(class); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static const struct vport_class * | 
					
						
							|  |  |  |  | vport_class_cast(const struct netdev_class *class) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-06 13:14:55 -08:00
										 |  |  |  |     ovs_assert(is_vport_class(class)); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     return CONTAINER_OF(class, struct vport_class, netdev_class); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | static struct netdev_vport * | 
					
						
							|  |  |  |  | netdev_vport_cast(const struct netdev *netdev) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     ovs_assert(is_vport_class(netdev_get_class(netdev))); | 
					
						
							|  |  |  |  |     return CONTAINER_OF(netdev, struct netdev_vport, up); | 
					
						
							| 
									
										
										
										
											2012-12-26 16:33:58 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | static const struct netdev_tunnel_config * | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | get_netdev_tunnel_config(const struct netdev *netdev) | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     return &netdev_vport_cast(netdev)->tnl_cfg; | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | bool | 
					
						
							|  |  |  |  | netdev_vport_is_patch(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     const struct netdev_class *class = netdev_get_class(netdev); | 
					
						
							| 
									
										
										
										
											2013-01-25 14:17:49 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-25 13:30:40 -08:00
										 |  |  |  |     return class->get_config == get_patch_config; | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-19 11:06:44 +02:00
										 |  |  |  | bool | 
					
						
							|  |  |  |  | netdev_vport_is_layer3(const struct netdev *dev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *type = netdev_get_type(dev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return (!strcmp("lisp", type)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  | static bool | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | netdev_vport_needs_dst_port(const struct netdev *dev) | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     const struct netdev_class *class = netdev_get_class(dev); | 
					
						
							|  |  |  |  |     const char *type = netdev_get_type(dev); | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  |  |     return (class->get_config == get_tunnel_config && | 
					
						
							| 
									
										
										
										
											2014-06-05 19:07:32 -07:00
										 |  |  |  |             (!strcmp("geneve", type) || !strcmp("vxlan", type) || | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  |              !strcmp("lisp", type) || !strcmp("stt", type)) ); | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-16 14:11:51 -07:00
										 |  |  |  | const char * | 
					
						
							|  |  |  |  | netdev_vport_class_get_dpif_port(const struct netdev_class *class) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return is_vport_class(class) ? vport_class_cast(class)->dpif_port : NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-16 17:08:50 -08:00
										 |  |  |  | const char * | 
					
						
							| 
									
										
										
										
											2013-05-01 11:05:28 -07:00
										 |  |  |  | netdev_vport_get_dpif_port(const struct netdev *netdev, | 
					
						
							|  |  |  |  |                            char namebuf[], size_t bufsize) | 
					
						
							| 
									
										
										
										
											2012-12-16 17:08:50 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-05-27 21:59:26 -07:00
										 |  |  |  |     const struct netdev_class *class = netdev_get_class(netdev); | 
					
						
							|  |  |  |  |     const char *dpif_port = netdev_vport_class_get_dpif_port(class); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!dpif_port) { | 
					
						
							|  |  |  |  |         return netdev_get_name(netdev); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     if (netdev_vport_needs_dst_port(netdev)) { | 
					
						
							|  |  |  |  |         const struct netdev_vport *vport = netdev_vport_cast(netdev); | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         /*
 | 
					
						
							| 
									
										
										
										
											2014-05-27 21:59:26 -07:00
										 |  |  |  |          * Note: IFNAMSIZ is 16 bytes long. Implementations should choose | 
					
						
							|  |  |  |  |          * a dpif port name that is short enough to fit including any | 
					
						
							|  |  |  |  |          * port numbers but assert just in case. | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  |          */ | 
					
						
							| 
									
										
										
										
											2013-05-01 11:05:28 -07:00
										 |  |  |  |         BUILD_ASSERT(NETDEV_VPORT_NAME_BUFSIZE >= IFNAMSIZ); | 
					
						
							| 
									
										
										
										
											2014-05-27 21:59:26 -07:00
										 |  |  |  |         ovs_assert(strlen(dpif_port) + 6 < IFNAMSIZ); | 
					
						
							|  |  |  |  |         snprintf(namebuf, bufsize, "%s_%d", dpif_port, | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  |                  ntohs(vport->tnl_cfg.dst_port)); | 
					
						
							| 
									
										
										
										
											2013-05-01 11:05:28 -07:00
										 |  |  |  |         return namebuf; | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2014-05-27 21:59:26 -07:00
										 |  |  |  |         return dpif_port; | 
					
						
							| 
									
										
										
										
											2013-02-14 09:37:25 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-01 11:05:28 -07:00
										 |  |  |  | char * | 
					
						
							|  |  |  |  | netdev_vport_get_dpif_port_strdup(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     char namebuf[NETDEV_VPORT_NAME_BUFSIZE]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return xstrdup(netdev_vport_get_dpif_port(netdev, namebuf, | 
					
						
							|  |  |  |  |                                               sizeof namebuf)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  | /* Whenever the route-table change number is incremented,
 | 
					
						
							|  |  |  |  |  * netdev_vport_route_changed() should be called to update | 
					
						
							|  |  |  |  |  * the corresponding tunnel interface status. */ | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_route_changed(void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev **vports; | 
					
						
							|  |  |  |  |     size_t i, n_vports; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     vports = netdev_get_vports(&n_vports); | 
					
						
							|  |  |  |  |     for (i = 0; i < n_vports; i++) { | 
					
						
							|  |  |  |  |         struct netdev *netdev_ = vports[i]; | 
					
						
							|  |  |  |  |         struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |         /* Finds all tunnel vports. */ | 
					
						
							|  |  |  |  |         if (netdev->tnl_cfg.ip_dst) { | 
					
						
							|  |  |  |  |             if (tunnel_check_status_change__(netdev)) { | 
					
						
							|  |  |  |  |                 netdev_change_seq_changed(netdev_); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2014-06-24 13:47:33 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         netdev_close(netdev_); | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     free(vports); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | static struct netdev * | 
					
						
							|  |  |  |  | netdev_vport_alloc(void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *netdev = xzalloc(sizeof *netdev); | 
					
						
							|  |  |  |  |     return &netdev->up; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | netdev_vport_construct(struct netdev *netdev_) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(netdev_); | 
					
						
							|  |  |  |  |     const char *type = netdev_get_type(netdev_); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     ovs_mutex_init(&dev->mutex); | 
					
						
							|  |  |  |  |     eth_addr_random(dev->etheraddr); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Add a default destination port for tunnel ports if none specified. */ | 
					
						
							|  |  |  |  |     if (!strcmp(type, "geneve")) { | 
					
						
							|  |  |  |  |         dev->tnl_cfg.dst_port = htons(GENEVE_DST_PORT); | 
					
						
							|  |  |  |  |     } else if (!strcmp(type, "vxlan")) { | 
					
						
							|  |  |  |  |         dev->tnl_cfg.dst_port = htons(VXLAN_DST_PORT); | 
					
						
							|  |  |  |  |     } else if (!strcmp(type, "lisp")) { | 
					
						
							|  |  |  |  |         dev->tnl_cfg.dst_port = htons(LISP_DST_PORT); | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  |     } else if (!strcmp(type, "stt")) { | 
					
						
							|  |  |  |  |         dev->tnl_cfg.dst_port = htons(STT_DST_PORT); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-01 11:02:02 -07:00
										 |  |  |  |     dev->tnl_cfg.dont_fragment = true; | 
					
						
							|  |  |  |  |     dev->tnl_cfg.ttl = DEFAULT_TTL; | 
					
						
							| 
									
										
											  
											
												netdev: Decouple creating and configuring network devices.
Until now, each call to netdev_open() for a particular network device
had to either specify a set of network device arguments that was either
empty or (for devices that already existed) equal to the existing device's
configuration.  Unfortunately, the definition of "equality" in the latter
case was mostly done in terms of strict equality of string-to-string maps,
which caused problems in cases where, for example, one set of arguments
specified the default value of an optional argument explicitly and the
other omitted it.
The netdev interface does have provisions for defining equality other ways,
but this had only been done in one case that was especially problematic in
practice.  One way to solve this particular problem would be to carefully
define equality in all the problematic cases.
This commit takes another approach based on the realization that there is
really no need to do any comparisons.  Instead, it removes configuration
at netdev_open() time entirely, because almost all of netdev_open()'s
callers are not interested in creating and configuring a netdev.  Most of
them just want to open a configured device and use it.  Therefore, this
commit stops providing any configuration arguments to netdev_open() and the
provider functions that it calls.  Instead, a caller that does want to
configure a device does so after it opens it, by calling
netdev_set_config().
This change allows us to simplify the netdev interface a bit.  There is no
longer any need to implement argument comparisons.  As a result, there is
also no need for "struct netdev_dev" to keep track of configuration at all.
Instead, the network devices that have configuration keep track of it in
their own internal form.
This new interface does mean that it becomes possible to accidentally
create and try to use an unconfigured netdev that requires configuration.
Bug #6677.
Reported-by: Paul Ingram <paul@nicira.com>
											
										 
											2011-08-08 12:49:17 -07:00
										 |  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | netdev_vport_destruct(struct netdev *netdev_) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     free(netdev->peer); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_destroy(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_dealloc(struct netdev *netdev_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     free(netdev); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | netdev_vport_set_etheraddr(struct netdev *netdev_, | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |                            const uint8_t mac[ETH_ADDR_LEN]) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     memcpy(netdev->etheraddr, mac, ETH_ADDR_LEN); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2014-04-03 00:17:34 -07:00
										 |  |  |  |     netdev_change_seq_changed(netdev_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-30 11:36:42 -08:00
										 |  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | netdev_vport_get_etheraddr(const struct netdev *netdev_, | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |                            uint8_t mac[ETH_ADDR_LEN]) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     memcpy(mac, netdev->etheraddr, ETH_ADDR_LEN); | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-30 11:36:42 -08:00
										 |  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  | /* Checks if the tunnel status has changed and returns a boolean.
 | 
					
						
							|  |  |  |  |  * Updates the tunnel status if it has changed. */ | 
					
						
							|  |  |  |  | static bool | 
					
						
							|  |  |  |  | tunnel_check_status_change__(struct netdev_vport *netdev) | 
					
						
							|  |  |  |  |     OVS_REQUIRES(netdev->mutex) | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-15 15:33:43 -07:00
										 |  |  |  |     char iface[IFNAMSIZ]; | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     bool status = false; | 
					
						
							| 
									
										
										
										
											2012-12-16 16:42:17 -08:00
										 |  |  |  |     ovs_be32 route; | 
					
						
							| 
									
										
										
										
											2014-10-16 11:38:12 -07:00
										 |  |  |  |     ovs_be32 gw; | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     iface[0] = '\0'; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     route = netdev->tnl_cfg.ip_dst; | 
					
						
							| 
									
										
										
										
											2014-10-16 11:38:12 -07:00
										 |  |  |  |     if (ovs_router_lookup(route, iface, &gw)) { | 
					
						
							| 
									
										
										
										
											2011-01-19 14:50:01 -08:00
										 |  |  |  |         struct netdev *egress_netdev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-05 14:18:06 -07:00
										 |  |  |  |         if (!netdev_open(iface, "system", &egress_netdev)) { | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |             status = netdev_get_carrier(egress_netdev); | 
					
						
							| 
									
										
										
										
											2011-01-19 14:50:01 -08:00
										 |  |  |  |             netdev_close(egress_netdev); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     if (strcmp(netdev->egress_iface, iface) | 
					
						
							|  |  |  |  |         || netdev->carrier_status != status) { | 
					
						
							|  |  |  |  |         ovs_strlcpy(netdev->egress_iface, iface, IFNAMSIZ); | 
					
						
							|  |  |  |  |         netdev->carrier_status = status; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return false; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | tunnel_get_status(const struct netdev *netdev_, struct smap *smap) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (netdev->egress_iface[0]) { | 
					
						
							|  |  |  |  |         smap_add(smap, "tunnel_egress_iface", netdev->egress_iface); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         smap_add(smap, "tunnel_egress_iface_carrier", | 
					
						
							|  |  |  |  |                  netdev->carrier_status ? "up" : "down"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | netdev_vport_update_flags(struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                           enum netdev_flags off, | 
					
						
							|  |  |  |  |                           enum netdev_flags on OVS_UNUSED, | 
					
						
							|  |  |  |  |                           enum netdev_flags *old_flagsp) | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     if (off & (NETDEV_UP | NETDEV_PROMISC)) { | 
					
						
							|  |  |  |  |         return EOPNOTSUPP; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *old_flagsp = NETDEV_UP | NETDEV_PROMISC; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_run(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     uint64_t seq; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  |     route_table_run(); | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     seq = route_table_get_change_seq(); | 
					
						
							|  |  |  |  |     if (rt_change_seqno != seq) { | 
					
						
							|  |  |  |  |         rt_change_seqno = seq; | 
					
						
							|  |  |  |  |         netdev_vport_route_changed(); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_wait(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     uint64_t seq; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  |     route_table_wait(); | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  |  |     seq = route_table_get_change_seq(); | 
					
						
							|  |  |  |  |     if (rt_change_seqno != seq) { | 
					
						
							|  |  |  |  |         poll_immediate_wake(); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  |  | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | /* Code specific to tunnel types. */ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | static ovs_be64 | 
					
						
							|  |  |  |  | parse_key(const struct smap *args, const char *name, | 
					
						
							|  |  |  |  |           bool *present, bool *flow) | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     const char *s; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     *present = false; | 
					
						
							|  |  |  |  |     *flow = false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |     s = smap_get(args, name); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (!s) { | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         s = smap_get(args, "key"); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         if (!s) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     *present = true; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (!strcmp(s, "flow")) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |         *flow = true; | 
					
						
							|  |  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |         return htonll(strtoull(s, NULL, 0)); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | set_tunnel_config(struct netdev *dev_, const struct smap *args) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(dev_); | 
					
						
							|  |  |  |  |     const char *name = netdev_get_name(dev_); | 
					
						
							|  |  |  |  |     const char *type = netdev_get_type(dev_); | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     bool ipsec_mech_set, needs_dst_port, has_csum; | 
					
						
							|  |  |  |  |     struct netdev_tunnel_config tnl_cfg; | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |     struct smap_node *node; | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 07:42:47 -07:00
										 |  |  |  |     has_csum = strstr(type, "gre") || strstr(type, "geneve") || | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  |                strstr(type, "stt") || strstr(type, "vxlan"); | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     ipsec_mech_set = false; | 
					
						
							|  |  |  |  |     memset(&tnl_cfg, 0, sizeof tnl_cfg); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     /* Add a default destination port for tunnel ports if none specified. */ | 
					
						
							|  |  |  |  |     if (!strcmp(type, "geneve")) { | 
					
						
							|  |  |  |  |         tnl_cfg.dst_port = htons(GENEVE_DST_PORT); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!strcmp(type, "vxlan")) { | 
					
						
							|  |  |  |  |         tnl_cfg.dst_port = htons(VXLAN_DST_PORT); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!strcmp(type, "lisp")) { | 
					
						
							|  |  |  |  |         tnl_cfg.dst_port = htons(LISP_DST_PORT); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  |     if (!strcmp(type, "stt")) { | 
					
						
							|  |  |  |  |         tnl_cfg.dst_port = htons(STT_DST_PORT); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-21 21:52:04 -08:00
										 |  |  |  |     needs_dst_port = netdev_vport_needs_dst_port(dev_); | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     tnl_cfg.ipsec = strstr(type, "ipsec"); | 
					
						
							|  |  |  |  |     tnl_cfg.dont_fragment = true; | 
					
						
							| 
									
										
										
										
											2010-12-01 17:23:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |     SMAP_FOR_EACH (node, args) { | 
					
						
							|  |  |  |  |         if (!strcmp(node->key, "remote_ip")) { | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             struct in_addr in_addr; | 
					
						
							| 
									
										
										
										
											2013-05-09 15:24:16 +03:00
										 |  |  |  |             if (!strcmp(node->value, "flow")) { | 
					
						
							|  |  |  |  |                 tnl_cfg.ip_dst_flow = true; | 
					
						
							|  |  |  |  |                 tnl_cfg.ip_dst = htonl(0); | 
					
						
							|  |  |  |  |             } else if (lookup_ip(node->value, &in_addr)) { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |                 VLOG_WARN("%s: bad %s 'remote_ip'", name, type); | 
					
						
							| 
									
										
										
										
											2013-03-04 13:00:25 -08:00
										 |  |  |  |             } else if (ip_is_multicast(in_addr.s_addr)) { | 
					
						
							|  |  |  |  |                 VLOG_WARN("%s: multicast remote_ip="IP_FMT" not allowed", | 
					
						
							|  |  |  |  |                           name, IP_ARGS(in_addr.s_addr)); | 
					
						
							|  |  |  |  |                 return EINVAL; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.ip_dst = in_addr.s_addr; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         } else if (!strcmp(node->key, "local_ip")) { | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             struct in_addr in_addr; | 
					
						
							| 
									
										
										
										
											2013-05-09 15:24:16 +03:00
										 |  |  |  |             if (!strcmp(node->value, "flow")) { | 
					
						
							|  |  |  |  |                 tnl_cfg.ip_src_flow = true; | 
					
						
							|  |  |  |  |                 tnl_cfg.ip_src = htonl(0); | 
					
						
							|  |  |  |  |             } else if (lookup_ip(node->value, &in_addr)) { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |                 VLOG_WARN("%s: bad %s 'local_ip'", name, type); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.ip_src = in_addr.s_addr; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         } else if (!strcmp(node->key, "tos")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->value, "inherit")) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.tos_inherit = true; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2012-05-21 12:18:19 -07:00
										 |  |  |  |                 char *endptr; | 
					
						
							|  |  |  |  |                 int tos; | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |                 tos = strtol(node->value, &endptr, 0); | 
					
						
							| 
									
										
										
										
											2012-07-20 10:24:50 -07:00
										 |  |  |  |                 if (*endptr == '\0' && tos == (tos & IP_DSCP_MASK)) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                     tnl_cfg.tos = tos; | 
					
						
							| 
									
										
										
										
											2012-07-20 10:24:50 -07:00
										 |  |  |  |                 } else { | 
					
						
							|  |  |  |  |                     VLOG_WARN("%s: invalid TOS %s", name, node->value); | 
					
						
							| 
									
										
										
										
											2012-05-21 12:18:19 -07:00
										 |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         } else if (!strcmp(node->key, "ttl")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->value, "inherit")) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.ttl_inherit = true; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.ttl = atoi(node->value); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-12-05 16:06:46 -05:00
										 |  |  |  |         } else if (!strcmp(node->key, "dst_port") && needs_dst_port) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |             tnl_cfg.dst_port = htons(atoi(node->value)); | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->key, "csum") && has_csum) { | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |             if (!strcmp(node->value, "true")) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.csum = true; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         } else if (!strcmp(node->key, "df_default")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->value, "false")) { | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |                 tnl_cfg.dont_fragment = false; | 
					
						
							| 
									
										
										
										
											2011-04-29 17:05:58 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |         } else if (!strcmp(node->key, "peer_cert") && tnl_cfg.ipsec) { | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |             if (smap_get(args, "certificate")) { | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |                 ipsec_mech_set = true; | 
					
						
							|  |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2010-12-27 17:44:33 -08:00
										 |  |  |  |                 const char *use_ssl_cert; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 /* If the "use_ssl_cert" is true, then "certificate" and
 | 
					
						
							|  |  |  |  |                  * "private_key" will be pulled from the SSL table.  The | 
					
						
							|  |  |  |  |                  * use of this option is strongly discouraged, since it | 
					
						
							|  |  |  |  |                  * will like be removed when multiple SSL configurations | 
					
						
							|  |  |  |  |                  * are supported by OVS. | 
					
						
							|  |  |  |  |                  */ | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |                 use_ssl_cert = smap_get(args, "use_ssl_cert"); | 
					
						
							| 
									
										
										
										
											2010-12-27 17:44:33 -08:00
										 |  |  |  |                 if (!use_ssl_cert || strcmp(use_ssl_cert, "true")) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |                     VLOG_ERR("%s: 'peer_cert' requires 'certificate' argument", | 
					
						
							|  |  |  |  |                              name); | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |                     return EINVAL; | 
					
						
							| 
									
										
										
										
											2010-12-27 17:44:33 -08:00
										 |  |  |  |                 } | 
					
						
							|  |  |  |  |                 ipsec_mech_set = true; | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |         } else if (!strcmp(node->key, "psk") && tnl_cfg.ipsec) { | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             ipsec_mech_set = true; | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |         } else if (tnl_cfg.ipsec | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |                 && (!strcmp(node->key, "certificate") | 
					
						
							|  |  |  |  |                     || !strcmp(node->key, "private_key") | 
					
						
							|  |  |  |  |                     || !strcmp(node->key, "use_ssl_cert"))) { | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |             /* Ignore options not used by the netdev. */ | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         } else if (!strcmp(node->key, "key") || | 
					
						
							|  |  |  |  |                    !strcmp(node->key, "in_key") || | 
					
						
							|  |  |  |  |                    !strcmp(node->key, "out_key")) { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |             /* Handled separately below. */ | 
					
						
							| 
									
										
										
										
											2015-02-06 21:10:45 +01:00
										 |  |  |  |         } else if (!strcmp(node->key, "exts")) { | 
					
						
							|  |  |  |  |             char *str = xstrdup(node->value); | 
					
						
							|  |  |  |  |             char *ext, *save_ptr = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             tnl_cfg.exts = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             ext = strtok_r(str, ",", &save_ptr); | 
					
						
							|  |  |  |  |             while (ext) { | 
					
						
							|  |  |  |  |                 if (!strcmp(type, "vxlan") && !strcmp(ext, "gbp")) { | 
					
						
							|  |  |  |  |                     tnl_cfg.exts |= (1 << OVS_VXLAN_EXT_GBP); | 
					
						
							|  |  |  |  |                 } else { | 
					
						
							|  |  |  |  |                     VLOG_WARN("%s: unknown extension '%s'", name, ext); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 ext = strtok_r(NULL, ",", &save_ptr); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             free(str); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |             VLOG_WARN("%s: unknown %s argument '%s'", name, type, node->key); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     if (tnl_cfg.ipsec) { | 
					
						
							| 
									
										
										
										
											2013-07-31 14:15:05 -07:00
										 |  |  |  |         static struct ovs_mutex mutex = OVS_MUTEX_INITIALIZER; | 
					
						
							| 
									
										
										
										
											2012-07-02 18:30:21 -07:00
										 |  |  |  |         static pid_t pid = 0; | 
					
						
							| 
									
										
										
										
											2013-04-26 13:20:30 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-12 10:50:42 -07:00
										 |  |  |  | #ifndef _WIN32
 | 
					
						
							| 
									
										
										
										
											2013-07-31 14:15:05 -07:00
										 |  |  |  |         ovs_mutex_lock(&mutex); | 
					
						
							| 
									
										
										
										
											2012-07-08 13:04:27 -07:00
										 |  |  |  |         if (pid <= 0) { | 
					
						
							| 
									
										
										
										
											2012-07-02 18:30:21 -07:00
										 |  |  |  |             char *file_name = xasprintf("%s/%s", ovs_rundir(), | 
					
						
							|  |  |  |  |                                         "ovs-monitor-ipsec.pid"); | 
					
						
							|  |  |  |  |             pid = read_pidfile(file_name); | 
					
						
							|  |  |  |  |             free(file_name); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-07-31 14:15:05 -07:00
										 |  |  |  |         ovs_mutex_unlock(&mutex); | 
					
						
							| 
									
										
										
										
											2014-03-12 10:50:42 -07:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-07-02 18:30:21 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-14 13:15:25 -07:00
										 |  |  |  |         if (pid < 0) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |             VLOG_ERR("%s: IPsec requires the ovs-monitor-ipsec daemon", | 
					
						
							|  |  |  |  |                      name); | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             return EINVAL; | 
					
						
							| 
									
										
										
										
											2011-03-14 13:15:25 -07:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-03-11 15:18:30 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |         if (smap_get(args, "peer_cert") && smap_get(args, "psk")) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |             VLOG_ERR("%s: cannot define both 'peer_cert' and 'psk'", name); | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             return EINVAL; | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!ipsec_mech_set) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |             VLOG_ERR("%s: IPsec requires an 'peer_cert' or psk' argument", | 
					
						
							|  |  |  |  |                      name); | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             return EINVAL; | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-09 15:24:16 +03:00
										 |  |  |  |     if (!tnl_cfg.ip_dst && !tnl_cfg.ip_dst_flow) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |         VLOG_ERR("%s: %s type requires valid 'remote_ip' argument", | 
					
						
							|  |  |  |  |                  name, type); | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-05-09 15:24:16 +03:00
										 |  |  |  |     if (tnl_cfg.ip_src_flow && !tnl_cfg.ip_dst_flow) { | 
					
						
							|  |  |  |  |         VLOG_ERR("%s: %s type requires 'remote_ip=flow' with 'local_ip=flow'", | 
					
						
							|  |  |  |  |                  name, type); | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     if (!tnl_cfg.ttl) { | 
					
						
							|  |  |  |  |         tnl_cfg.ttl = DEFAULT_TTL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     tnl_cfg.in_key = parse_key(args, "in_key", | 
					
						
							|  |  |  |  |                                &tnl_cfg.in_key_present, | 
					
						
							|  |  |  |  |                                &tnl_cfg.in_key_flow); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     tnl_cfg.out_key = parse_key(args, "out_key", | 
					
						
							|  |  |  |  |                                &tnl_cfg.out_key_present, | 
					
						
							|  |  |  |  |                                &tnl_cfg.out_key_flow); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2015-03-26 17:35:32 -07:00
										 |  |  |  |     if (memcmp(&dev->tnl_cfg, &tnl_cfg, sizeof tnl_cfg)) { | 
					
						
							|  |  |  |  |         dev->tnl_cfg = tnl_cfg; | 
					
						
							|  |  |  |  |         tunnel_check_status_change__(dev); | 
					
						
							|  |  |  |  |         netdev_change_seq_changed(dev_); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | get_tunnel_config(const struct netdev *dev, struct smap *args) | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(dev); | 
					
						
							|  |  |  |  |     struct netdev_tunnel_config tnl_cfg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     tnl_cfg = netdev->tnl_cfg; | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.ip_dst) { | 
					
						
							|  |  |  |  |         smap_add_format(args, "remote_ip", IP_FMT, IP_ARGS(tnl_cfg.ip_dst)); | 
					
						
							|  |  |  |  |     } else if (tnl_cfg.ip_dst_flow) { | 
					
						
							| 
									
										
										
										
											2013-05-09 15:24:16 +03:00
										 |  |  |  |         smap_add(args, "remote_ip", "flow"); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.ip_src) { | 
					
						
							|  |  |  |  |         smap_add_format(args, "local_ip", IP_FMT, IP_ARGS(tnl_cfg.ip_src)); | 
					
						
							|  |  |  |  |     } else if (tnl_cfg.ip_src_flow) { | 
					
						
							| 
									
										
										
										
											2013-05-09 15:24:16 +03:00
										 |  |  |  |         smap_add(args, "local_ip", "flow"); | 
					
						
							| 
									
										
										
										
											2012-12-29 08:58:40 +02:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.in_key_flow && tnl_cfg.out_key_flow) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "key", "flow"); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     } else if (tnl_cfg.in_key_present && tnl_cfg.out_key_present | 
					
						
							|  |  |  |  |                && tnl_cfg.in_key == tnl_cfg.out_key) { | 
					
						
							|  |  |  |  |         smap_add_format(args, "key", "%"PRIu64, ntohll(tnl_cfg.in_key)); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         if (tnl_cfg.in_key_flow) { | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             smap_add(args, "in_key", "flow"); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         } else if (tnl_cfg.in_key_present) { | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             smap_add_format(args, "in_key", "%"PRIu64, | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |                             ntohll(tnl_cfg.in_key)); | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         if (tnl_cfg.out_key_flow) { | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             smap_add(args, "out_key", "flow"); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         } else if (tnl_cfg.out_key_present) { | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |             smap_add_format(args, "out_key", "%"PRIu64, | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |                             ntohll(tnl_cfg.out_key)); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.ttl_inherit) { | 
					
						
							| 
									
										
										
										
											2012-09-25 13:22:08 -07:00
										 |  |  |  |         smap_add(args, "ttl", "inherit"); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     } else if (tnl_cfg.ttl != DEFAULT_TTL) { | 
					
						
							|  |  |  |  |         smap_add_format(args, "ttl", "%"PRIu8, tnl_cfg.ttl); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.tos_inherit) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "tos", "inherit"); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     } else if (tnl_cfg.tos) { | 
					
						
							|  |  |  |  |         smap_add_format(args, "tos", "0x%x", tnl_cfg.tos); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.dst_port) { | 
					
						
							|  |  |  |  |         uint16_t dst_port = ntohs(tnl_cfg.dst_port); | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |         const char *type = netdev_get_type(dev); | 
					
						
							| 
									
										
										
										
											2013-02-28 09:30:16 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-05 19:07:32 -07:00
										 |  |  |  |         if ((!strcmp("geneve", type) && dst_port != GENEVE_DST_PORT) || | 
					
						
							|  |  |  |  |             (!strcmp("vxlan", type) && dst_port != VXLAN_DST_PORT) || | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  |             (!strcmp("lisp", type) && dst_port != LISP_DST_PORT) || | 
					
						
							|  |  |  |  |             (!strcmp("stt", type) && dst_port != STT_DST_PORT)) { | 
					
						
							| 
									
										
										
										
											2012-12-05 16:06:46 -05:00
										 |  |  |  |             smap_add_format(args, "dst_port", "%d", dst_port); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (tnl_cfg.csum) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "csum", "true"); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-01-25 12:38:45 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (!tnl_cfg.dont_fragment) { | 
					
						
							| 
									
										
										
										
											2011-04-29 17:05:58 -07:00
										 |  |  |  |         smap_add(args, "df_default", "false"); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | /* Code specific to patch ports. */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-31 14:09:30 -07:00
										 |  |  |  | /* If 'netdev' is a patch port, returns the name of its peer as a malloc()'d
 | 
					
						
							|  |  |  |  |  * string that the caller must free. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If 'netdev' is not a patch port, returns NULL. */ | 
					
						
							|  |  |  |  | char * | 
					
						
							|  |  |  |  | netdev_vport_patch_peer(const struct netdev *netdev_) | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-31 14:09:30 -07:00
										 |  |  |  |     char *peer = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (netdev_vport_is_patch(netdev_)) { | 
					
						
							|  |  |  |  |         struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-07-31 14:09:30 -07:00
										 |  |  |  |         if (netdev->peer) { | 
					
						
							|  |  |  |  |             peer = xstrdup(netdev->peer); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-07-31 14:09:30 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return peer; | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  | netdev_vport_inc_rx(const struct netdev *netdev, | 
					
						
							| 
									
										
										
										
											2013-09-16 17:19:56 -07:00
										 |  |  |  |                     const struct dpif_flow_stats *stats) | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     if (is_vport_class(netdev_get_class(netdev))) { | 
					
						
							|  |  |  |  |         struct netdev_vport *dev = netdev_vport_cast(netdev); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |         dev->stats.rx_packets += stats->n_packets; | 
					
						
							|  |  |  |  |         dev->stats.rx_bytes += stats->n_bytes; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  | netdev_vport_inc_tx(const struct netdev *netdev, | 
					
						
							|  |  |  |  |                     const struct dpif_flow_stats *stats) | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     if (is_vport_class(netdev_get_class(netdev))) { | 
					
						
							|  |  |  |  |         struct netdev_vport *dev = netdev_vport_cast(netdev); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |         dev->stats.tx_packets += stats->n_packets; | 
					
						
							|  |  |  |  |         dev->stats.tx_bytes += stats->n_bytes; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | get_patch_config(const struct netdev *dev_, struct smap *args) | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(dev_); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |     if (dev->peer) { | 
					
						
							|  |  |  |  |         smap_add(args, "peer", dev->peer); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | set_patch_config(struct netdev *dev_, const struct smap *args) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(dev_); | 
					
						
							|  |  |  |  |     const char *name = netdev_get_name(dev_); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     const char *peer; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |     peer = smap_get(args, "peer"); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     if (!peer) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |         VLOG_ERR("%s: patch type requires valid 'peer' argument", name); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-22 03:47:36 -07:00
										 |  |  |  |     if (smap_count(args) > 1) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |         VLOG_ERR("%s: patch type takes only a 'peer' argument", name); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!strcmp(name, peer)) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |         VLOG_ERR("%s: patch peer must not be self", name); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2015-03-26 17:35:32 -07:00
										 |  |  |  |     if (!dev->peer || strcmp(dev->peer, peer)) { | 
					
						
							|  |  |  |  |         free(dev->peer); | 
					
						
							|  |  |  |  |         dev->peer = xstrdup(peer); | 
					
						
							|  |  |  |  |         netdev_change_seq_changed(dev_); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  | get_stats(const struct netdev *netdev, struct netdev_stats *stats) | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(netdev); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |     *stats = dev->stats; | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |  | 
					
						
							|  |  |  |  | /* Tunnel push pop ops. */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ip_header * | 
					
						
							|  |  |  |  | ip_hdr(void *eth) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return (void *)((char *)eth + sizeof (struct eth_header)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct gre_base_hdr * | 
					
						
							|  |  |  |  | gre_hdr(struct ip_header *ip) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |      return (void *)((char *)ip + sizeof (struct ip_header)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void * | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  | ip_extract_tnl_md(struct dp_packet *packet, struct flow_tnl *tnl) | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ip_header *nh; | 
					
						
							|  |  |  |  |     void *l4; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     nh = dp_packet_l3(packet); | 
					
						
							|  |  |  |  |     l4 = dp_packet_l4(packet); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!nh || !l4) { | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     tnl->ip_src = get_16aligned_be32(&nh->ip_src); | 
					
						
							|  |  |  |  |     tnl->ip_dst = get_16aligned_be32(&nh->ip_dst); | 
					
						
							|  |  |  |  |     tnl->ip_tos = nh->ip_tos; | 
					
						
							| 
									
										
										
										
											2015-03-27 07:39:18 -07:00
										 |  |  |  |     tnl->ip_ttl = nh->ip_ttl; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return l4; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Pushes the 'size' bytes of 'header' into the headroom of 'packet',
 | 
					
						
							|  |  |  |  |  * reallocating the packet if necessary.  'header' should contain an Ethernet | 
					
						
							|  |  |  |  |  * header, followed by an IPv4 header (without options), and an L4 header. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function sets the IP header's ip_tot_len field (which should be zeroed | 
					
						
							|  |  |  |  |  * as part of 'header') and puts its value into '*ip_tot_size' as well.  Also | 
					
						
							|  |  |  |  |  * updates IP header checksum. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Return pointer to the L4 header added to 'packet'. */ | 
					
						
							|  |  |  |  | static void * | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  | push_ip_header(struct dp_packet *packet, | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |                const void *header, int size, int *ip_tot_size) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct eth_header *eth; | 
					
						
							|  |  |  |  |     struct ip_header *ip; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     eth = dp_packet_push_uninit(packet, size); | 
					
						
							|  |  |  |  |     *ip_tot_size = dp_packet_size(packet) - sizeof (struct eth_header); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     memcpy(eth, header, size); | 
					
						
							|  |  |  |  |     ip = ip_hdr(eth); | 
					
						
							|  |  |  |  |     ip->ip_tot_len = htons(*ip_tot_size); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ip->ip_csum = recalc_csum16(ip->ip_csum, 0, ip->ip_tot_len); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ip + 1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  | static void * | 
					
						
							|  |  |  |  | udp_extract_tnl_md(struct dp_packet *packet, struct flow_tnl *tnl) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct udp_header *udp; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     udp = ip_extract_tnl_md(packet, tnl); | 
					
						
							|  |  |  |  |     if (!udp) { | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 14:27:19 -07:00
										 |  |  |  |     if (udp->udp_csum) { | 
					
						
							|  |  |  |  |         uint32_t csum = packet_csum_pseudoheader(dp_packet_l3(packet)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         csum = csum_continue(csum, udp, dp_packet_size(packet) - | 
					
						
							|  |  |  |  |                              ((const unsigned char *)udp - | 
					
						
							|  |  |  |  |                               (const unsigned char *)dp_packet_l2(packet))); | 
					
						
							|  |  |  |  |         if (csum_finish(csum)) { | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         tnl->flags |= FLOW_TNL_F_CSUM; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  |     tnl->tp_src = udp->udp_src; | 
					
						
							|  |  |  |  |     tnl->tp_dst = udp->udp_dst; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return udp + 1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static ovs_be16 | 
					
						
							|  |  |  |  | get_src_port(struct dp_packet *packet) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     uint32_t hash; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-15 19:11:48 +01:00
										 |  |  |  |     hash = dp_packet_get_rss_hash(packet); | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return htons((((uint64_t) hash * (tnl_udp_port_max - tnl_udp_port_min)) >> 32) + | 
					
						
							|  |  |  |  |                  tnl_udp_port_min); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | push_udp_header(struct dp_packet *packet, | 
					
						
							|  |  |  |  |                 const struct ovs_action_push_tnl *data) | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct udp_header *udp; | 
					
						
							|  |  |  |  |     int ip_tot_size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |     udp = push_ip_header(packet, data->header, data->header_len, &ip_tot_size); | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* set udp src port */ | 
					
						
							|  |  |  |  |     udp->udp_src = get_src_port(packet); | 
					
						
							|  |  |  |  |     udp->udp_len = htons(ip_tot_size - sizeof (struct ip_header)); | 
					
						
							| 
									
										
										
										
											2015-03-26 14:27:19 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (udp->udp_csum) { | 
					
						
							|  |  |  |  |         uint32_t csum = packet_csum_pseudoheader(ip_hdr(dp_packet_data(packet))); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         csum = csum_continue(csum, udp, | 
					
						
							|  |  |  |  |                              ip_tot_size - sizeof (struct ip_header)); | 
					
						
							|  |  |  |  |         udp->udp_csum = csum_finish(csum); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!udp->udp_csum) { | 
					
						
							|  |  |  |  |             udp->udp_csum = htons(0xffff); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void * | 
					
						
							|  |  |  |  | udp_build_header(struct netdev_tunnel_config *tnl_cfg, | 
					
						
							| 
									
										
										
										
											2015-03-26 14:27:19 -07:00
										 |  |  |  |                  const struct flow *tnl_flow, | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  |                  struct ovs_action_push_tnl *data) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ip_header *ip; | 
					
						
							|  |  |  |  |     struct udp_header *udp; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ip = ip_hdr(data->header); | 
					
						
							|  |  |  |  |     ip->ip_proto = IPPROTO_UDP; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     udp = (struct udp_header *) (ip + 1); | 
					
						
							|  |  |  |  |     udp->udp_dst = tnl_cfg->dst_port; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 14:27:19 -07:00
										 |  |  |  |     if (tnl_flow->tunnel.flags & FLOW_TNL_F_CSUM) { | 
					
						
							|  |  |  |  |         /* Write a value in now to mark that we should compute the checksum
 | 
					
						
							|  |  |  |  |          * later. 0xffff is handy because it is transparent to the | 
					
						
							|  |  |  |  |          * calculation. */ | 
					
						
							|  |  |  |  |         udp->udp_csum = htons(0xffff); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  |     return udp + 1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | gre_header_len(ovs_be16 flags) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int hlen = sizeof(struct eth_header) + | 
					
						
							|  |  |  |  |                sizeof(struct ip_header) + 4; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (flags & htons(GRE_CSUM)) { | 
					
						
							|  |  |  |  |         hlen += 4; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (flags & htons(GRE_KEY)) { | 
					
						
							|  |  |  |  |         hlen += 4; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (flags & htons(GRE_SEQ)) { | 
					
						
							|  |  |  |  |         hlen += 4; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return hlen; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  | parse_gre_header(struct dp_packet *packet, | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |                  struct flow_tnl *tnl) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const struct gre_base_hdr *greh; | 
					
						
							|  |  |  |  |     ovs_16aligned_be32 *options; | 
					
						
							|  |  |  |  |     int hlen; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     greh = ip_extract_tnl_md(packet, tnl); | 
					
						
							|  |  |  |  |     if (!greh) { | 
					
						
							|  |  |  |  |         return -EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (greh->flags & ~(htons(GRE_CSUM | GRE_KEY | GRE_SEQ))) { | 
					
						
							|  |  |  |  |         return -EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 16:53:39 -07:00
										 |  |  |  |     if (greh->protocol != htons(ETH_TYPE_TEB)) { | 
					
						
							|  |  |  |  |         return -EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     hlen = gre_header_len(greh->flags); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     if (hlen > dp_packet_size(packet)) { | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |         return -EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     options = (ovs_16aligned_be32 *)(greh + 1); | 
					
						
							|  |  |  |  |     if (greh->flags & htons(GRE_CSUM)) { | 
					
						
							|  |  |  |  |         ovs_be16 pkt_csum; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         pkt_csum = csum(greh, dp_packet_size(packet) - | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |                               ((const unsigned char *)greh - | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |                                (const unsigned char *)dp_packet_l2(packet))); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |         if (pkt_csum) { | 
					
						
							|  |  |  |  |             return -EINVAL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         tnl->flags = FLOW_TNL_F_CSUM; | 
					
						
							|  |  |  |  |         options++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (greh->flags & htons(GRE_KEY)) { | 
					
						
							|  |  |  |  |         tnl->tun_id = (OVS_FORCE ovs_be64) ((OVS_FORCE uint64_t)(get_16aligned_be32(options)) << 32); | 
					
						
							|  |  |  |  |         tnl->flags |= FLOW_TNL_F_KEY; | 
					
						
							|  |  |  |  |         options++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (greh->flags & htons(GRE_SEQ)) { | 
					
						
							|  |  |  |  |         options++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return hlen; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_gre_pop_header(struct dp_packet *packet) | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     struct pkt_metadata *md = &packet->md; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     struct flow_tnl *tnl = &md->tunnel; | 
					
						
							|  |  |  |  |     int hlen = sizeof(struct eth_header) + | 
					
						
							|  |  |  |  |                sizeof(struct ip_header) + 4; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(md, 0, sizeof *md); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     if (hlen > dp_packet_size(packet)) { | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     hlen = parse_gre_header(packet, tnl); | 
					
						
							|  |  |  |  |     if (hlen < 0) { | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return -hlen; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     dp_packet_reset_packet(packet, hlen); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  | netdev_gre_push_header(struct dp_packet *packet, | 
					
						
							|  |  |  |  |                        const struct ovs_action_push_tnl *data) | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct gre_base_hdr *greh; | 
					
						
							|  |  |  |  |     int ip_tot_size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |     greh = push_ip_header(packet, data->header, data->header_len, &ip_tot_size); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (greh->flags & htons(GRE_CSUM)) { | 
					
						
							| 
									
										
										
										
											2015-03-26 17:09:38 -07:00
										 |  |  |  |         ovs_be16 *csum_opt = (ovs_be16 *) (greh + 1); | 
					
						
							|  |  |  |  |         *csum_opt = csum(greh, ip_tot_size - sizeof (struct ip_header)); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_gre_build_header(const struct netdev *netdev, | 
					
						
							| 
									
										
										
										
											2015-03-26 06:11:28 -07:00
										 |  |  |  |                         struct ovs_action_push_tnl *data, | 
					
						
							|  |  |  |  |                         const struct flow *tnl_flow) | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(netdev); | 
					
						
							|  |  |  |  |     struct netdev_tunnel_config *tnl_cfg; | 
					
						
							|  |  |  |  |     struct ip_header *ip; | 
					
						
							|  |  |  |  |     struct gre_base_hdr *greh; | 
					
						
							|  |  |  |  |     ovs_16aligned_be32 *options; | 
					
						
							|  |  |  |  |     int hlen; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* XXX: RCUfy tnl_cfg. */ | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |     tnl_cfg = &dev->tnl_cfg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ip = ip_hdr(data->header); | 
					
						
							|  |  |  |  |     ip->ip_proto = IPPROTO_GRE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     greh = gre_hdr(ip); | 
					
						
							|  |  |  |  |     greh->protocol = htons(ETH_TYPE_TEB); | 
					
						
							|  |  |  |  |     greh->flags = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     options = (ovs_16aligned_be32 *) (greh + 1); | 
					
						
							| 
									
										
										
										
											2015-03-26 22:55:53 -07:00
										 |  |  |  |     if (tnl_flow->tunnel.flags & FLOW_TNL_F_CSUM) { | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |         greh->flags |= htons(GRE_CSUM); | 
					
						
							|  |  |  |  |         put_16aligned_be32(options, 0); | 
					
						
							|  |  |  |  |         options++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (tnl_cfg->out_key_present) { | 
					
						
							|  |  |  |  |         greh->flags |= htons(GRE_KEY); | 
					
						
							|  |  |  |  |         put_16aligned_be32(options, (OVS_FORCE ovs_be32) | 
					
						
							| 
									
										
										
										
											2015-03-26 06:11:28 -07:00
										 |  |  |  |                                     ((OVS_FORCE uint64_t) tnl_flow->tunnel.tun_id >> 32)); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |         options++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     hlen = (uint8_t *) options - (uint8_t *) greh; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     data->header_len = sizeof(struct eth_header) + | 
					
						
							|  |  |  |  |                        sizeof(struct ip_header)  + hlen; | 
					
						
							|  |  |  |  |     data->tnl_type = OVS_VPORT_TYPE_GRE; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_vxlan_pop_header(struct dp_packet *packet) | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     struct pkt_metadata *md = &packet->md; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     struct flow_tnl *tnl = &md->tunnel; | 
					
						
							|  |  |  |  |     struct vxlanhdr *vxh; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(md, 0, sizeof *md); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     if (VXLAN_HLEN > dp_packet_size(packet)) { | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 12:07:02 -07:00
										 |  |  |  |     vxh = udp_extract_tnl_md(packet, tnl); | 
					
						
							|  |  |  |  |     if (!vxh) { | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (get_16aligned_be32(&vxh->vx_flags) != htonl(VXLAN_FLAGS) || | 
					
						
							|  |  |  |  |        (get_16aligned_be32(&vxh->vx_vni) & htonl(0xff))) { | 
					
						
							|  |  |  |  |         VLOG_WARN_RL(&err_rl, "invalid vxlan flags=%#x vni=%#x\n", | 
					
						
							|  |  |  |  |                      ntohl(get_16aligned_be32(&vxh->vx_flags)), | 
					
						
							|  |  |  |  |                      ntohl(get_16aligned_be32(&vxh->vx_vni))); | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     tnl->tun_id = htonll(ntohl(get_16aligned_be32(&vxh->vx_vni)) >> 8); | 
					
						
							| 
									
										
										
										
											2015-04-07 16:40:18 -07:00
										 |  |  |  |     tnl->flags |= FLOW_TNL_F_KEY; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     dp_packet_reset_packet(packet, VXLAN_HLEN); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_vxlan_build_header(const struct netdev *netdev, | 
					
						
							| 
									
										
										
										
											2015-03-26 06:11:28 -07:00
										 |  |  |  |                           struct ovs_action_push_tnl *data, | 
					
						
							|  |  |  |  |                           const struct flow *tnl_flow) | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(netdev); | 
					
						
							|  |  |  |  |     struct netdev_tunnel_config *tnl_cfg; | 
					
						
							|  |  |  |  |     struct vxlanhdr *vxh; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* XXX: RCUfy tnl_cfg. */ | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |     tnl_cfg = &dev->tnl_cfg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 14:27:19 -07:00
										 |  |  |  |     vxh = udp_build_header(tnl_cfg, tnl_flow, data); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     put_16aligned_be32(&vxh->vx_flags, htonl(VXLAN_FLAGS)); | 
					
						
							| 
									
										
										
										
											2015-03-26 06:11:28 -07:00
										 |  |  |  |     put_16aligned_be32(&vxh->vx_vni, htonl(ntohll(tnl_flow->tunnel.tun_id) << 8)); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  |     data->header_len = VXLAN_HLEN; | 
					
						
							|  |  |  |  |     data->tnl_type = OVS_VPORT_TYPE_VXLAN; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  | netdev_geneve_pop_header(struct dp_packet *packet) | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct pkt_metadata *md = &packet->md; | 
					
						
							|  |  |  |  |     struct flow_tnl *tnl = &md->tunnel; | 
					
						
							|  |  |  |  |     struct genevehdr *gnh; | 
					
						
							|  |  |  |  |     unsigned int hlen; | 
					
						
							| 
									
										
										
										
											2015-06-22 14:23:37 -07:00
										 |  |  |  |     int err; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(md, 0, sizeof *md); | 
					
						
							|  |  |  |  |     if (GENEVE_BASE_HLEN > dp_packet_size(packet)) { | 
					
						
							|  |  |  |  |         VLOG_WARN_RL(&err_rl, "geneve packet too small: min header=%u packet size=%u\n", | 
					
						
							|  |  |  |  |                      (unsigned int)GENEVE_BASE_HLEN, dp_packet_size(packet)); | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     gnh = udp_extract_tnl_md(packet, tnl); | 
					
						
							|  |  |  |  |     if (!gnh) { | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     hlen = GENEVE_BASE_HLEN + gnh->opt_len * 4; | 
					
						
							|  |  |  |  |     if (hlen > dp_packet_size(packet)) { | 
					
						
							|  |  |  |  |         VLOG_WARN_RL(&err_rl, "geneve packet too small: header len=%u packet size=%u\n", | 
					
						
							|  |  |  |  |                      hlen, dp_packet_size(packet)); | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (gnh->ver != 0) { | 
					
						
							|  |  |  |  |         VLOG_WARN_RL(&err_rl, "unknown geneve version: %"PRIu8"\n", gnh->ver); | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (gnh->proto_type != htons(ETH_TYPE_TEB)) { | 
					
						
							|  |  |  |  |         VLOG_WARN_RL(&err_rl, "unknown geneve encapsulated protocol: %#x\n", | 
					
						
							|  |  |  |  |                      ntohs(gnh->proto_type)); | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     tnl->flags |= gnh->oam ? FLOW_TNL_F_OAM : 0; | 
					
						
							|  |  |  |  |     tnl->tun_id = htonll(ntohl(get_16aligned_be32(&gnh->vni)) >> 8); | 
					
						
							|  |  |  |  |     tnl->flags |= FLOW_TNL_F_KEY; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 14:23:37 -07:00
										 |  |  |  |     err = tun_metadata_from_geneve_header(gnh->options, gnh->opt_len * 4, | 
					
						
							|  |  |  |  |                                           &tnl->metadata); | 
					
						
							|  |  |  |  |     if (err) { | 
					
						
							|  |  |  |  |         VLOG_WARN_RL(&err_rl, "invalid geneve options"); | 
					
						
							|  |  |  |  |         return err; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     dp_packet_reset_packet(packet, hlen); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_geneve_build_header(const struct netdev *netdev, | 
					
						
							|  |  |  |  |                            struct ovs_action_push_tnl *data, | 
					
						
							|  |  |  |  |                            const struct flow *tnl_flow) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *dev = netdev_vport_cast(netdev); | 
					
						
							|  |  |  |  |     struct netdev_tunnel_config *tnl_cfg; | 
					
						
							|  |  |  |  |     struct genevehdr *gnh; | 
					
						
							| 
									
										
										
										
											2015-06-22 14:23:37 -07:00
										 |  |  |  |     int opt_len; | 
					
						
							|  |  |  |  |     bool crit_opt; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* XXX: RCUfy tnl_cfg. */ | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |     tnl_cfg = &dev->tnl_cfg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-26 14:27:19 -07:00
										 |  |  |  |     gnh = udp_build_header(tnl_cfg, tnl_flow, data); | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     put_16aligned_be32(&gnh->vni, htonl(ntohll(tnl_flow->tunnel.tun_id) << 8)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2015-06-22 14:23:37 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     opt_len = tun_metadata_to_geneve_header(&tnl_flow->tunnel.metadata, | 
					
						
							|  |  |  |  |                                             gnh->options, &crit_opt); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     gnh->opt_len = opt_len / 4; | 
					
						
							|  |  |  |  |     gnh->oam = !!(tnl_flow->tunnel.flags & FLOW_TNL_F_OAM); | 
					
						
							|  |  |  |  |     gnh->critical = crit_opt ? 1 : 0; | 
					
						
							|  |  |  |  |     gnh->proto_type = htons(ETH_TYPE_TEB); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     data->header_len = GENEVE_BASE_HLEN + opt_len; | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |     data->tnl_type = OVS_VPORT_TYPE_GENEVE; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_range(struct unixctl_conn *conn, int argc, | 
					
						
							|  |  |  |  |                    const char *argv[], void *aux OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int val1, val2; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (argc < 3) { | 
					
						
							|  |  |  |  |         struct ds ds = DS_EMPTY_INITIALIZER; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ds_put_format(&ds, "Tunnel UDP source port range: %"PRIu16"-%"PRIu16"\n", | 
					
						
							|  |  |  |  |                             tnl_udp_port_min, tnl_udp_port_max); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         unixctl_command_reply(conn, ds_cstr(&ds)); | 
					
						
							|  |  |  |  |         ds_destroy(&ds); | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (argc != 3) { | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     val1 = atoi(argv[1]); | 
					
						
							|  |  |  |  |     if (val1 <= 0 || val1 > UINT16_MAX) { | 
					
						
							|  |  |  |  |         unixctl_command_reply(conn, "Invalid min."); | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     val2 = atoi(argv[2]); | 
					
						
							|  |  |  |  |     if (val2 <= 0 || val2 > UINT16_MAX) { | 
					
						
							|  |  |  |  |         unixctl_command_reply(conn, "Invalid max."); | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (val1 > val2) { | 
					
						
							|  |  |  |  |         tnl_udp_port_min = val2; | 
					
						
							|  |  |  |  |         tnl_udp_port_max = val1; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         tnl_udp_port_min = val1; | 
					
						
							|  |  |  |  |         tnl_udp_port_max = val2; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     seq_change(tnl_conf_seq); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     unixctl_command_reply(conn, "OK"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |  | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  | #define VPORT_FUNCTIONS(GET_CONFIG, SET_CONFIG,             \
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |                         GET_TUNNEL_CONFIG, GET_STATUS,      \ | 
					
						
							|  |  |  |  |                         BUILD_HEADER,                       \ | 
					
						
							|  |  |  |  |                         PUSH_HEADER, POP_HEADER)            \ | 
					
						
							| 
									
										
										
										
											2011-01-30 17:52:19 -08:00
										 |  |  |  |     NULL,                                                   \ | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  |     netdev_vport_run,                                       \ | 
					
						
							|  |  |  |  |     netdev_vport_wait,                                      \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |                                                             \ | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     netdev_vport_alloc,                                     \ | 
					
						
							|  |  |  |  |     netdev_vport_construct,                                 \ | 
					
						
							|  |  |  |  |     netdev_vport_destruct,                                  \ | 
					
						
							|  |  |  |  |     netdev_vport_dealloc,                                   \ | 
					
						
							| 
									
										
										
										
											2012-12-20 15:32:03 -08:00
										 |  |  |  |     GET_CONFIG,                                             \ | 
					
						
							|  |  |  |  |     SET_CONFIG,                                             \ | 
					
						
							| 
									
										
										
										
											2013-01-07 16:56:04 -08:00
										 |  |  |  |     GET_TUNNEL_CONFIG,                                      \ | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     BUILD_HEADER,                                           \ | 
					
						
							|  |  |  |  |     PUSH_HEADER,                                            \ | 
					
						
							|  |  |  |  |     POP_HEADER,                                             \ | 
					
						
							| 
									
										
										
										
											2014-06-11 16:33:08 -07:00
										 |  |  |  |     NULL,                       /* get_numa_id */           \ | 
					
						
							| 
									
										
										
										
											2014-09-08 14:52:54 -07:00
										 |  |  |  |     NULL,                       /* set_multiq */            \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |                                                             \ | 
					
						
							| 
									
										
										
										
											2012-11-21 16:17:45 -08:00
										 |  |  |  |     NULL,                       /* send */                  \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     NULL,                       /* send_wait */             \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     netdev_vport_set_etheraddr,                             \ | 
					
						
							|  |  |  |  |     netdev_vport_get_etheraddr,                             \ | 
					
						
							| 
									
										
										
										
											2011-09-15 10:41:15 -07:00
										 |  |  |  |     NULL,                       /* get_mtu */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_mtu */               \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     NULL,                       /* get_ifindex */           \ | 
					
						
							| 
									
										
										
										
											2010-10-27 15:29:16 -07:00
										 |  |  |  |     NULL,                       /* get_carrier */           \ | 
					
						
							| 
									
										
										
										
											2011-10-14 12:49:57 -07:00
										 |  |  |  |     NULL,                       /* get_carrier_resets */    \ | 
					
						
							| 
									
										
										
										
											2011-01-07 16:22:34 -08:00
										 |  |  |  |     NULL,                       /* get_miimon */            \ | 
					
						
							| 
									
										
										
										
											2012-12-14 19:14:54 -08:00
										 |  |  |  |     get_stats,                                              \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_features */          \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_advertisements */    \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_policing */          \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_qos_types */         \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_qos_capabilities */  \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_qos */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_qos */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_queue */             \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_queue */             \ | 
					
						
							|  |  |  |  |     NULL,                       /* delete_queue */          \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_queue_stats */       \ | 
					
						
							| 
									
										
										
										
											2013-08-27 17:15:53 -07:00
										 |  |  |  |     NULL,                       /* queue_dump_start */      \ | 
					
						
							|  |  |  |  |     NULL,                       /* queue_dump_next */       \ | 
					
						
							|  |  |  |  |     NULL,                       /* queue_dump_done */       \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     NULL,                       /* dump_queue_stats */      \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_in4 */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_in4 */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_in6 */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* add_router */            \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_next_hop */          \ | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  |     GET_STATUS,                                             \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     NULL,                       /* arp_lookup */            \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     netdev_vport_update_flags,                              \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     NULL,                   /* rx_alloc */                  \ | 
					
						
							|  |  |  |  |     NULL,                   /* rx_construct */              \ | 
					
						
							|  |  |  |  |     NULL,                   /* rx_destruct */               \ | 
					
						
							|  |  |  |  |     NULL,                   /* rx_dealloc */                \ | 
					
						
							|  |  |  |  |     NULL,                   /* rx_recv */                   \ | 
					
						
							|  |  |  |  |     NULL,                   /* rx_wait */                   \ | 
					
						
							|  |  |  |  |     NULL,                   /* rx_drain */ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #define TUNNEL_CLASS(NAME, DPIF_PORT, BUILD_HEADER, PUSH_HEADER, POP_HEADER)   \
 | 
					
						
							|  |  |  |  |     { DPIF_PORT,                                                               \ | 
					
						
							|  |  |  |  |         { NAME, VPORT_FUNCTIONS(get_tunnel_config,                             \ | 
					
						
							|  |  |  |  |                                 set_tunnel_config,                             \ | 
					
						
							|  |  |  |  |                                 get_netdev_tunnel_config,                      \ | 
					
						
							|  |  |  |  |                                 tunnel_get_status,                             \ | 
					
						
							|  |  |  |  |                                 BUILD_HEADER, PUSH_HEADER, POP_HEADER) }} | 
					
						
							| 
									
										
										
										
											2013-01-08 20:01:24 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | void | 
					
						
							| 
									
										
										
										
											2013-01-25 13:30:40 -08:00
										 |  |  |  | netdev_vport_tunnel_register(void) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-05-27 21:59:26 -07:00
										 |  |  |  |     /* The name of the dpif_port should be short enough to accomodate adding
 | 
					
						
							|  |  |  |  |      * a port number to the end if one is necessary. */ | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     static const struct vport_class vport_classes[] = { | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |         TUNNEL_CLASS("geneve", "genev_sys", netdev_geneve_build_header, | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |                                             push_udp_header, | 
					
						
							| 
									
										
										
										
											2015-03-26 13:51:06 -07:00
										 |  |  |  |                                             netdev_geneve_pop_header), | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |         TUNNEL_CLASS("gre", "gre_sys", netdev_gre_build_header, | 
					
						
							|  |  |  |  |                                        netdev_gre_push_header, | 
					
						
							|  |  |  |  |                                        netdev_gre_pop_header), | 
					
						
							|  |  |  |  |         TUNNEL_CLASS("ipsec_gre", "gre_sys", NULL, NULL, NULL), | 
					
						
							|  |  |  |  |         TUNNEL_CLASS("gre64", "gre64_sys", NULL,  NULL, NULL), | 
					
						
							|  |  |  |  |         TUNNEL_CLASS("ipsec_gre64", "gre64_sys", NULL, NULL, NULL), | 
					
						
							|  |  |  |  |         TUNNEL_CLASS("vxlan", "vxlan_sys", netdev_vxlan_build_header, | 
					
						
							| 
									
										
										
										
											2015-04-07 17:45:40 -07:00
										 |  |  |  |                                            push_udp_header, | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |                                            netdev_vxlan_pop_header), | 
					
						
							| 
									
										
										
										
											2015-04-09 20:12:32 -07:00
										 |  |  |  |         TUNNEL_CLASS("lisp", "lisp_sys", NULL, NULL, NULL), | 
					
						
							|  |  |  |  |         TUNNEL_CLASS("stt", "stt_sys", NULL, NULL, NULL), | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     static struct ovsthread_once once = OVSTHREAD_ONCE_INITIALIZER; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     if (ovsthread_once_start(&once)) { | 
					
						
							|  |  |  |  |         int i; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-18 08:27:20 -07:00
										 |  |  |  |         for (i = 0; i < ARRAY_SIZE(vport_classes); i++) { | 
					
						
							|  |  |  |  |             netdev_register_provider(&vport_classes[i].netdev_class); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         unixctl_command_register("tnl/egress_port_range", "min max", 0, 2, | 
					
						
							|  |  |  |  |                                  netdev_vport_range, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         ovsthread_once_done(&once); | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-01-25 13:30:40 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | netdev_vport_patch_register(void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static const struct vport_class patch_class = | 
					
						
							|  |  |  |  |         { NULL, | 
					
						
							|  |  |  |  |             { "patch", VPORT_FUNCTIONS(get_patch_config, | 
					
						
							|  |  |  |  |                                        set_patch_config, | 
					
						
							|  |  |  |  |                                        NULL, | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |                                        NULL, NULL, NULL, NULL) }}; | 
					
						
							| 
									
										
										
										
											2013-01-25 13:30:40 -08:00
										 |  |  |  |     netdev_register_provider(&patch_class.netdev_class); | 
					
						
							|  |  |  |  | } |