| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  |  * Copyright (c) 2010, 2011 Nicira Networks. | 
					
						
							| 
									
										
										
										
											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>
 | 
					
						
							|  |  |  |  | #include <linux/rtnetlink.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"
 | 
					
						
							| 
									
										
										
										
											2011-03-11 15:18:30 -08:00
										 |  |  |  | #include "daemon.h"
 | 
					
						
							|  |  |  |  | #include "dirs.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | #include "dpif-linux.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"
 | 
					
						
							| 
									
										
										
										
											2011-04-28 11:13:53 -07:00
										 |  |  |  | #include "netdev-linux.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | #include "netdev-provider.h"
 | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | #include "netlink.h"
 | 
					
						
							|  |  |  |  | #include "netlink-socket.h"
 | 
					
						
							|  |  |  |  | #include "ofpbuf.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | #include "openvswitch/datapath-protocol.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | #include "openvswitch/tunnel.h"
 | 
					
						
							|  |  |  |  | #include "packets.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  | #include "route-table.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | #include "rtnetlink.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | #include "shash.h"
 | 
					
						
							|  |  |  |  | #include "socket-util.h"
 | 
					
						
							|  |  |  |  | #include "vlog.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-19 14:47:01 -07:00
										 |  |  |  | VLOG_DEFINE_THIS_MODULE(netdev_vport); | 
					
						
							| 
									
										
										
										
											2010-07-16 11:02:49 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | struct netdev_vport_notifier { | 
					
						
							|  |  |  |  |     struct netdev_notifier notifier; | 
					
						
							|  |  |  |  |     struct list list_node; | 
					
						
							| 
									
										
										
										
											2010-08-30 00:24:53 -07:00
										 |  |  |  |     struct shash_node *shash_node; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | struct netdev_dev_vport { | 
					
						
							|  |  |  |  |     struct netdev_dev netdev_dev; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct ofpbuf *options; | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  |     int dp_ifindex;             /* -1 if unknown. */ | 
					
						
							|  |  |  |  |     uint32_t port_no;           /* UINT32_MAX if unknown. */ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct netdev_vport { | 
					
						
							|  |  |  |  |     struct netdev netdev; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct vport_class { | 
					
						
							| 
									
										
										
										
											2011-01-23 20:01:30 -08:00
										 |  |  |  |     enum odp_vport_type type; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     struct netdev_class netdev_class; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     int (*parse_config)(const char *name, const char *type, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                         const struct shash *args, struct ofpbuf *options); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     int (*unparse_config)(const char *name, const char *type, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                           const struct nlattr *options, size_t options_len, | 
					
						
							|  |  |  |  |                           struct shash *args); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | static struct shash netdev_vport_notifiers = | 
					
						
							|  |  |  |  |                                     SHASH_INITIALIZER(&netdev_vport_notifiers); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vlog_rate_limit rl = VLOG_RATE_LIMIT_INIT(5, 20); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int netdev_vport_create(const struct netdev_class *, const char *, | 
					
						
							|  |  |  |  |                                const struct shash *, struct netdev_dev **); | 
					
						
							|  |  |  |  | static void netdev_vport_poll_notify(const struct netdev *); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | static int tnl_port_config_from_nlattr(const struct nlattr *options, | 
					
						
							|  |  |  |  |                                        size_t options_len, | 
					
						
							|  |  |  |  |                                        struct nlattr *a[ODP_TUNNEL_ATTR_MAX + 1]); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  | static const char *netdev_vport_get_tnl_iface(const struct netdev *netdev); | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     return class->create == netdev_vport_create; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static const struct vport_class * | 
					
						
							|  |  |  |  | vport_class_cast(const struct netdev_class *class) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     assert(is_vport_class(class)); | 
					
						
							|  |  |  |  |     return CONTAINER_OF(class, struct vport_class, netdev_class); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct netdev_dev_vport * | 
					
						
							|  |  |  |  | netdev_dev_vport_cast(const struct netdev_dev *netdev_dev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     assert(is_vport_class(netdev_dev_get_class(netdev_dev))); | 
					
						
							|  |  |  |  |     return CONTAINER_OF(netdev_dev, struct netdev_dev_vport, netdev_dev); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct netdev_vport * | 
					
						
							|  |  |  |  | netdev_vport_cast(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dev *netdev_dev = netdev_get_dev(netdev); | 
					
						
							|  |  |  |  |     assert(is_vport_class(netdev_dev_get_class(netdev_dev))); | 
					
						
							|  |  |  |  |     return CONTAINER_OF(netdev, struct netdev_vport, netdev); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | /* If 'netdev' is a vport netdev, returns an ofpbuf that contains Netlink
 | 
					
						
							|  |  |  |  |  * options to include in ODP_VPORT_ATTR_OPTIONS for configuring that vport. | 
					
						
							|  |  |  |  |  * Otherwise returns NULL. */ | 
					
						
							|  |  |  |  | const struct ofpbuf * | 
					
						
							|  |  |  |  | netdev_vport_get_options(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const struct netdev_dev *dev = netdev_get_dev(netdev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return (is_vport_class(netdev_dev_get_class(dev)) | 
					
						
							|  |  |  |  |             ? netdev_dev_vport_cast(dev)->options | 
					
						
							|  |  |  |  |             : NULL); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | enum odp_vport_type | 
					
						
							|  |  |  |  | netdev_vport_get_vport_type(const struct netdev *netdev) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     const struct netdev_dev *dev = netdev_get_dev(netdev); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     const struct netdev_class *class = netdev_dev_get_class(dev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return (is_vport_class(class) ? vport_class_cast(class)->type | 
					
						
							|  |  |  |  |             : class == &netdev_internal_class ? ODP_VPORT_TYPE_INTERNAL | 
					
						
							|  |  |  |  |             : class == &netdev_linux_class ? ODP_VPORT_TYPE_NETDEV | 
					
						
							|  |  |  |  |             : ODP_VPORT_TYPE_UNSPEC); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | const char * | 
					
						
							|  |  |  |  | netdev_vport_get_netdev_type(const struct dpif_linux_vport *vport) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct nlattr *a[ODP_TUNNEL_ATTR_MAX + 1]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (vport->type) { | 
					
						
							|  |  |  |  |     case ODP_VPORT_TYPE_UNSPEC: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ODP_VPORT_TYPE_NETDEV: | 
					
						
							|  |  |  |  |         return "system"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ODP_VPORT_TYPE_INTERNAL: | 
					
						
							|  |  |  |  |         return "internal"; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     case ODP_VPORT_TYPE_PATCH: | 
					
						
							|  |  |  |  |         return "patch"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ODP_VPORT_TYPE_GRE: | 
					
						
							|  |  |  |  |         if (tnl_port_config_from_nlattr(vport->options, vport->options_len, | 
					
						
							|  |  |  |  |                                         a)) { | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return (nl_attr_get_u32(a[ODP_TUNNEL_ATTR_FLAGS]) & TNL_F_IPSEC | 
					
						
							|  |  |  |  |                 ? "ipsec_gre" : "gre"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ODP_VPORT_TYPE_CAPWAP: | 
					
						
							|  |  |  |  |         return "capwap"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case __ODP_VPORT_TYPE_MAX: | 
					
						
							|  |  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     VLOG_WARN_RL(&rl, "dp%d: port `%s' has unsupported type %u", | 
					
						
							| 
									
										
										
										
											2011-01-21 17:01:56 -08:00
										 |  |  |  |                  vport->dp_ifindex, vport->name, (unsigned int) vport->type); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     return "unknown"; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  | netdev_vport_create(const struct netdev_class *netdev_class, const char *name, | 
					
						
							|  |  |  |  |                     const struct shash *args, | 
					
						
							|  |  |  |  |                     struct netdev_dev **netdev_devp) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     const struct vport_class *vport_class = vport_class_cast(netdev_class); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct ofpbuf *options = NULL; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     struct shash fetched_args; | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  |     int dp_ifindex; | 
					
						
							|  |  |  |  |     uint32_t port_no; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     int error; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     shash_init(&fetched_args); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  |     dp_ifindex = -1; | 
					
						
							|  |  |  |  |     port_no = UINT32_MAX; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     if (!shash_is_empty(args)) { | 
					
						
							|  |  |  |  |         /* Parse the provided configuration. */ | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         options = ofpbuf_new(64); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         error = vport_class->parse_config(name, netdev_class->type, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                                           args, options); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         /* Fetch an existing configuration from the kernel.
 | 
					
						
							|  |  |  |  |          * | 
					
						
							|  |  |  |  |          * This case could be ambiguous with initializing a new vport with an | 
					
						
							|  |  |  |  |          * empty configuration, but none of the existing vport classes accept | 
					
						
							|  |  |  |  |          * an empty configuration. */ | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         struct dpif_linux_vport reply; | 
					
						
							|  |  |  |  |         struct ofpbuf *buf; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         error = dpif_linux_vport_get(name, &reply, &buf); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         if (!error) { | 
					
						
							|  |  |  |  |             /* XXX verify correct type */ | 
					
						
							|  |  |  |  |             error = vport_class->unparse_config(name, netdev_class->type, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                                                 reply.options, | 
					
						
							|  |  |  |  |                                                 reply.options_len, | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |                                                 &fetched_args); | 
					
						
							|  |  |  |  |             if (error) { | 
					
						
							|  |  |  |  |                 VLOG_ERR_RL(&rl, "%s: failed to parse kernel config (%s)", | 
					
						
							|  |  |  |  |                             name, strerror(error)); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |             } else { | 
					
						
							|  |  |  |  |                 options = ofpbuf_clone_data(reply.options, reply.options_len); | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  |                 dp_ifindex = reply.dp_ifindex; | 
					
						
							|  |  |  |  |                 port_no = reply.port_no; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |             ofpbuf_delete(buf); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             VLOG_ERR_RL(&rl, "%s: vport query failed (%s)", | 
					
						
							|  |  |  |  |                         name, strerror(error)); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         struct netdev_dev_vport *dev; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         dev = xmalloc(sizeof *dev); | 
					
						
							|  |  |  |  |         netdev_dev_init(&dev->netdev_dev, name, | 
					
						
							|  |  |  |  |                         shash_is_empty(&fetched_args) ? args : &fetched_args, | 
					
						
							|  |  |  |  |                         netdev_class); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         dev->options = options; | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  |         dev->dp_ifindex = dp_ifindex; | 
					
						
							|  |  |  |  |         dev->port_no = port_no; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         *netdev_devp = &dev->netdev_dev; | 
					
						
							| 
									
										
										
										
											2011-01-31 12:02:54 -08:00
										 |  |  |  |         route_table_register(); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         ofpbuf_delete(options); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     shash_destroy(&fetched_args); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_destroy(struct netdev_dev *netdev_dev_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dev_vport *netdev_dev = netdev_dev_vport_cast(netdev_dev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  |     route_table_unregister(); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     free(netdev_dev); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_vport_open(struct netdev_dev *netdev_dev_, int ethertype OVS_UNUSED, | 
					
						
							|  |  |  |  |                 struct netdev **netdevp) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *netdev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     netdev = xmalloc(sizeof *netdev); | 
					
						
							|  |  |  |  |     netdev_init(&netdev->netdev, netdev_dev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *netdevp = &netdev->netdev; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_close(struct netdev *netdev_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport *netdev = netdev_vport_cast(netdev_); | 
					
						
							|  |  |  |  |     free(netdev); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | netdev_vport_set_config(struct netdev_dev *dev_, const struct shash *args) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     const struct netdev_class *netdev_class = netdev_dev_get_class(dev_); | 
					
						
							|  |  |  |  |     const struct vport_class *vport_class = vport_class_cast(netdev_class); | 
					
						
							|  |  |  |  |     struct netdev_dev_vport *dev = netdev_dev_vport_cast(dev_); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     const char *name = netdev_dev_get_name(dev_); | 
					
						
							|  |  |  |  |     struct ofpbuf *options; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     options = ofpbuf_new(64); | 
					
						
							|  |  |  |  |     error = vport_class->parse_config(name, netdev_dev_get_type(dev_), | 
					
						
							|  |  |  |  |                                       args, options); | 
					
						
							|  |  |  |  |     if (!error | 
					
						
							|  |  |  |  |         && (options->size != dev->options->size | 
					
						
							|  |  |  |  |             || memcmp(options->data, dev->options->data, options->size))) { | 
					
						
							|  |  |  |  |         struct dpif_linux_vport vport; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         dpif_linux_vport_init(&vport); | 
					
						
							| 
									
										
										
										
											2011-01-28 13:59:03 -08:00
										 |  |  |  |         vport.cmd = ODP_VPORT_CMD_SET; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         vport.name = name; | 
					
						
							|  |  |  |  |         vport.options = options->data; | 
					
						
							|  |  |  |  |         vport.options_len = options->size; | 
					
						
							|  |  |  |  |         error = dpif_linux_vport_transact(&vport, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |         if (!error || error == ENODEV) { | 
					
						
							|  |  |  |  |             /* Either reconfiguration succeeded or this vport is not installed
 | 
					
						
							|  |  |  |  |              * in the kernel (e.g. it hasn't been added to a dpif yet with | 
					
						
							|  |  |  |  |              * dpif_port_add()). */ | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |             ofpbuf_delete(dev->options); | 
					
						
							|  |  |  |  |             dev->options = options; | 
					
						
							|  |  |  |  |             options = NULL; | 
					
						
							|  |  |  |  |             error = 0; | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     ofpbuf_delete(options); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_vport_send(struct netdev *netdev, const void *data, size_t size) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dev *dev_ = netdev_get_dev(netdev); | 
					
						
							|  |  |  |  |     struct netdev_dev_vport *dev = netdev_dev_vport_cast(dev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (dev->dp_ifindex == -1) { | 
					
						
							|  |  |  |  |         const char *name = netdev_get_name(netdev); | 
					
						
							|  |  |  |  |         struct dpif_linux_vport reply; | 
					
						
							|  |  |  |  |         struct ofpbuf *buf; | 
					
						
							|  |  |  |  |         int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = dpif_linux_vport_get(name, &reply, &buf); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             VLOG_ERR_RL(&rl, "%s: failed to query vport for send (%s)", | 
					
						
							|  |  |  |  |                         name, strerror(error)); | 
					
						
							|  |  |  |  |             return error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         dev->dp_ifindex = reply.dp_ifindex; | 
					
						
							|  |  |  |  |         dev->port_no = reply.port_no; | 
					
						
							|  |  |  |  |         ofpbuf_delete(buf); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return dpif_linux_vport_send(dev->dp_ifindex, dev->port_no, data, size); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | netdev_vport_set_etheraddr(struct netdev *netdev, | 
					
						
							|  |  |  |  |                            const uint8_t mac[ETH_ADDR_LEN]) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct dpif_linux_vport vport; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     dpif_linux_vport_init(&vport); | 
					
						
							| 
									
										
										
										
											2011-01-28 13:59:03 -08:00
										 |  |  |  |     vport.cmd = ODP_VPORT_CMD_SET; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     vport.name = netdev_get_name(netdev); | 
					
						
							|  |  |  |  |     vport.address = mac; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     error = dpif_linux_vport_transact(&vport, NULL, NULL); | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         netdev_vport_poll_notify(netdev); | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | netdev_vport_get_etheraddr(const struct netdev *netdev, | 
					
						
							|  |  |  |  |                            uint8_t mac[ETH_ADDR_LEN]) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct dpif_linux_vport reply; | 
					
						
							|  |  |  |  |     struct ofpbuf *buf; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     error = dpif_linux_vport_get(netdev_get_name(netdev), &reply, &buf); | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         if (reply.address) { | 
					
						
							|  |  |  |  |             memcpy(mac, reply.address, ETH_ADDR_LEN); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             error = EOPNOTSUPP; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         ofpbuf_delete(buf); | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | netdev_vport_get_mtu(const struct netdev *netdev, int *mtup) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct dpif_linux_vport reply; | 
					
						
							|  |  |  |  |     struct ofpbuf *buf; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     error = dpif_linux_vport_get(netdev_get_name(netdev), &reply, &buf); | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         *mtup = reply.mtu; | 
					
						
							|  |  |  |  |         ofpbuf_delete(buf); | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int | 
					
						
							|  |  |  |  | netdev_vport_get_stats(const struct netdev *netdev, struct netdev_stats *stats) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct dpif_linux_vport reply; | 
					
						
							|  |  |  |  |     struct ofpbuf *buf; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     error = dpif_linux_vport_get(netdev_get_name(netdev), &reply, &buf); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } else if (!reply.stats) { | 
					
						
							|  |  |  |  |         ofpbuf_delete(buf); | 
					
						
							|  |  |  |  |         return EOPNOTSUPP; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-28 11:13:53 -07:00
										 |  |  |  |     netdev_stats_from_rtnl_link_stats64(stats, reply.stats); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ofpbuf_delete(buf); | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-09 12:54:34 -07:00
										 |  |  |  | int | 
					
						
							|  |  |  |  | netdev_vport_set_stats(struct netdev *netdev, const struct netdev_stats *stats) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct rtnl_link_stats64 rtnl_stats; | 
					
						
							|  |  |  |  |     struct dpif_linux_vport vport; | 
					
						
							| 
									
										
										
										
											2010-06-09 12:54:34 -07:00
										 |  |  |  |     int err; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-28 11:13:53 -07:00
										 |  |  |  |     netdev_stats_to_rtnl_link_stats64(&rtnl_stats, stats); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     dpif_linux_vport_init(&vport); | 
					
						
							| 
									
										
										
										
											2011-01-28 13:59:03 -08:00
										 |  |  |  |     vport.cmd = ODP_VPORT_CMD_SET; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     vport.name = netdev_get_name(netdev); | 
					
						
							|  |  |  |  |     vport.stats = &rtnl_stats; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     err = dpif_linux_vport_transact(&vport, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2010-06-09 12:54:34 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* If the vport layer doesn't know about the device, that doesn't mean it
 | 
					
						
							|  |  |  |  |      * doesn't exist (after all were able to open it when netdev_open() was | 
					
						
							|  |  |  |  |      * called), it just means that it isn't attached and we'll be getting | 
					
						
							|  |  |  |  |      * stats a different way. */ | 
					
						
							|  |  |  |  |     if (err == ENODEV) { | 
					
						
							|  |  |  |  |         err = EOPNOTSUPP; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return err; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_vport_get_status(const struct netdev *netdev, struct shash *sh) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *iface = netdev_vport_get_tnl_iface(netdev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (iface) { | 
					
						
							| 
									
										
										
										
											2011-01-19 14:50:01 -08:00
										 |  |  |  |         struct netdev *egress_netdev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  |         shash_add(sh, "tunnel_egress_iface", xstrdup(iface)); | 
					
						
							| 
									
										
										
										
											2011-01-19 14:50:01 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!netdev_open_default(iface, &egress_netdev)) { | 
					
						
							|  |  |  |  |             shash_add(sh, "tunnel_egress_iface_carrier", | 
					
						
							|  |  |  |  |                       xstrdup(netdev_get_carrier(egress_netdev) | 
					
						
							|  |  |  |  |                               ? "up" : "down")); | 
					
						
							|  |  |  |  |             netdev_close(egress_netdev); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -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) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (off & (NETDEV_UP | NETDEV_PROMISC)) { | 
					
						
							|  |  |  |  |         return EOPNOTSUPP; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *old_flagsp = NETDEV_UP | NETDEV_PROMISC; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static char * | 
					
						
							|  |  |  |  | make_poll_name(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return xasprintf("%s:%s", netdev_get_type(netdev), netdev_get_name(netdev)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | netdev_vport_poll_add(struct netdev *netdev, | 
					
						
							|  |  |  |  |                       void (*cb)(struct netdev_notifier *), void *aux, | 
					
						
							|  |  |  |  |                       struct netdev_notifier **notifierp) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     char *poll_name = make_poll_name(netdev); | 
					
						
							|  |  |  |  |     struct netdev_vport_notifier *notifier; | 
					
						
							|  |  |  |  |     struct list *list; | 
					
						
							|  |  |  |  |     struct shash_node *shash_node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-04 13:44:35 -08:00
										 |  |  |  |     shash_node = shash_find(&netdev_vport_notifiers, poll_name); | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |     if (!shash_node) { | 
					
						
							|  |  |  |  |         list = xmalloc(sizeof *list); | 
					
						
							|  |  |  |  |         list_init(list); | 
					
						
							| 
									
										
										
										
											2010-11-15 12:48:31 -08:00
										 |  |  |  |         shash_node = shash_add(&netdev_vport_notifiers, poll_name, list); | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         list = shash_node->data; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     notifier = xmalloc(sizeof *notifier); | 
					
						
							|  |  |  |  |     netdev_notifier_init(¬ifier->notifier, netdev, cb, aux); | 
					
						
							|  |  |  |  |     list_push_back(list, ¬ifier->list_node); | 
					
						
							|  |  |  |  |     notifier->shash_node = shash_node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *notifierp = ¬ifier->notifier; | 
					
						
							|  |  |  |  |     free(poll_name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | netdev_vport_poll_remove(struct netdev_notifier *notifier_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_vport_notifier *notifier = | 
					
						
							|  |  |  |  |                 CONTAINER_OF(notifier_, struct netdev_vport_notifier, notifier); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     struct list *list; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     list = list_remove(¬ifier->list_node); | 
					
						
							|  |  |  |  |     if (list_is_empty(list)) { | 
					
						
							|  |  |  |  |         shash_delete(&netdev_vport_notifiers, notifier->shash_node); | 
					
						
							|  |  |  |  |         free(list); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     free(notifier); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_run(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  |     route_table_run(); | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_vport_wait(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-12 14:55:18 -08:00
										 |  |  |  |     route_table_wait(); | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  |  | 
					
						
							|  |  |  |  | /* get_tnl_iface() implementation. */ | 
					
						
							|  |  |  |  | static const char * | 
					
						
							|  |  |  |  | netdev_vport_get_tnl_iface(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct nlattr *a[ODP_TUNNEL_ATTR_MAX + 1]; | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  |     uint32_t route; | 
					
						
							|  |  |  |  |     struct netdev_dev_vport *ndv; | 
					
						
							| 
									
										
										
										
											2011-01-30 17:52:19 -08:00
										 |  |  |  |     static char name[IFNAMSIZ]; | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ndv = netdev_dev_vport_cast(netdev_get_dev(netdev)); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (tnl_port_config_from_nlattr(ndv->options->data, ndv->options->size, | 
					
						
							|  |  |  |  |                                     a)) { | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     route = nl_attr_get_be32(a[ODP_TUNNEL_ATTR_DST_IPV4]); | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-30 17:52:19 -08:00
										 |  |  |  |     if (route_table_get_name(route, name)) { | 
					
						
							|  |  |  |  |         return name; | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | /* Helper functions. */ | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  | netdev_vport_poll_notify(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     char *poll_name = make_poll_name(netdev); | 
					
						
							|  |  |  |  |     struct list *list = shash_find_data(&netdev_vport_notifiers, | 
					
						
							|  |  |  |  |                                         poll_name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (list) { | 
					
						
							|  |  |  |  |         struct netdev_vport_notifier *notifier; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-17 10:33:10 -07:00
										 |  |  |  |         LIST_FOR_EACH (notifier, list_node, list) { | 
					
						
							| 
									
										
										
										
											2010-05-17 15:04:10 -07:00
										 |  |  |  |             struct netdev_notifier *n = ¬ifier->notifier; | 
					
						
							|  |  |  |  |             n->cb(n); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     free(poll_name); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | /* Code specific to individual vport types. */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | set_key(const struct shash *args, const char *name, uint16_t type, | 
					
						
							|  |  |  |  |         struct ofpbuf *options) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *s; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     s = shash_find_data(args, name); | 
					
						
							|  |  |  |  |     if (!s) { | 
					
						
							|  |  |  |  |         s = shash_find_data(args, "key"); | 
					
						
							|  |  |  |  |         if (!s) { | 
					
						
							|  |  |  |  |             s = "0"; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!strcmp(s, "flow")) { | 
					
						
							|  |  |  |  |         /* This is the default if no attribute is present. */ | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         nl_msg_put_be64(options, type, htonll(strtoull(s, NULL, 0))); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | parse_tunnel_config(const char *name, const char *type, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                     const struct shash *args, struct ofpbuf *options) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-01 17:23:33 -08:00
										 |  |  |  |     bool is_gre = false; | 
					
						
							|  |  |  |  |     bool is_ipsec = false; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     struct shash_node *node; | 
					
						
							|  |  |  |  |     bool ipsec_mech_set = false; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     ovs_be32 daddr = htonl(0); | 
					
						
							|  |  |  |  |     uint32_t flags; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-29 17:05:58 -07:00
										 |  |  |  |     flags = TNL_F_DF_DEFAULT | TNL_F_PMTUD | TNL_F_HDR_CACHE; | 
					
						
							| 
									
										
										
										
											2010-12-01 17:23:33 -08:00
										 |  |  |  |     if (!strcmp(type, "gre")) { | 
					
						
							|  |  |  |  |         is_gre = true; | 
					
						
							|  |  |  |  |     } else if (!strcmp(type, "ipsec_gre")) { | 
					
						
							|  |  |  |  |         is_gre = true; | 
					
						
							|  |  |  |  |         is_ipsec = true; | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         flags |= TNL_F_IPSEC; | 
					
						
							|  |  |  |  |         flags &= ~TNL_F_HDR_CACHE; | 
					
						
							| 
									
										
										
										
											2010-12-01 17:23:33 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     SHASH_FOR_EACH (node, args) { | 
					
						
							|  |  |  |  |         if (!strcmp(node->name, "remote_ip")) { | 
					
						
							|  |  |  |  |             struct in_addr in_addr; | 
					
						
							|  |  |  |  |             if (lookup_ip(node->data, &in_addr)) { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |                 VLOG_WARN("%s: bad %s 'remote_ip'", name, type); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 daddr = in_addr.s_addr; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "local_ip")) { | 
					
						
							|  |  |  |  |             struct in_addr in_addr; | 
					
						
							|  |  |  |  |             if (lookup_ip(node->data, &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 { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 nl_msg_put_be32(options, ODP_TUNNEL_ATTR_SRC_IPV4, | 
					
						
							|  |  |  |  |                                 in_addr.s_addr); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "tos")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "inherit")) { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 flags |= TNL_F_TOS_INHERIT; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 nl_msg_put_u8(options, ODP_TUNNEL_ATTR_TOS, atoi(node->data)); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "ttl")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "inherit")) { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 flags |= TNL_F_TTL_INHERIT; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 nl_msg_put_u8(options, ODP_TUNNEL_ATTR_TTL, atoi(node->data)); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "csum") && is_gre) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "true")) { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 flags |= TNL_F_CSUM; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-04-29 17:05:58 -07:00
										 |  |  |  |         } else if (!strcmp(node->name, "df_inherit")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "true")) { | 
					
						
							|  |  |  |  |                 flags |= TNL_F_DF_INHERIT; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "df_default")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "false")) { | 
					
						
							|  |  |  |  |                 flags &= ~TNL_F_DF_DEFAULT; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         } else if (!strcmp(node->name, "pmtud")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "false")) { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 flags &= ~TNL_F_PMTUD; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "header_cache")) { | 
					
						
							|  |  |  |  |             if (!strcmp(node->data, "false")) { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                 flags &= ~TNL_F_HDR_CACHE; | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |         } else if (!strcmp(node->name, "peer_cert") && is_ipsec) { | 
					
						
							|  |  |  |  |             if (shash_find(args, "certificate")) { | 
					
						
							|  |  |  |  |                 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. | 
					
						
							|  |  |  |  |                  */ | 
					
						
							|  |  |  |  |                 use_ssl_cert = shash_find_data(args, "use_ssl_cert"); | 
					
						
							|  |  |  |  |                 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); | 
					
						
							| 
									
										
										
										
											2010-12-27 17:44:33 -08:00
										 |  |  |  |                     return EINVAL; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |                 ipsec_mech_set = true; | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (!strcmp(node->name, "psk") && is_ipsec) { | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |             ipsec_mech_set = true; | 
					
						
							| 
									
										
										
										
											2010-12-21 16:26:21 -08:00
										 |  |  |  |         } else if (is_ipsec | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |                 && (!strcmp(node->name, "certificate") | 
					
						
							| 
									
										
										
										
											2010-12-27 17:44:33 -08:00
										 |  |  |  |                     || !strcmp(node->name, "private_key") | 
					
						
							|  |  |  |  |                     || !strcmp(node->name, "use_ssl_cert"))) { | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |             /* Ignore options not used by the netdev. */ | 
					
						
							| 
									
										
										
										
											2011-03-10 13:47:31 -08:00
										 |  |  |  |         } else if (is_gre && (!strcmp(node->name, "key") || | 
					
						
							|  |  |  |  |                               !strcmp(node->name, "in_key") || | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                               !strcmp(node->name, "out_key"))) { | 
					
						
							|  |  |  |  |             /* Handled separately below. */ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |             VLOG_WARN("%s: unknown %s argument '%s'", name, type, node->name); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |     if (is_ipsec) { | 
					
						
							| 
									
										
										
										
											2011-03-11 15:18:30 -08:00
										 |  |  |  |         char *file_name = xasprintf("%s/%s", ovs_rundir(), | 
					
						
							|  |  |  |  |                 "ovs-monitor-ipsec.pid"); | 
					
						
							| 
									
										
										
										
											2011-03-14 13:15:25 -07:00
										 |  |  |  |         pid_t pid = read_pidfile(file_name); | 
					
						
							| 
									
										
										
										
											2011-03-11 15:18:30 -08:00
										 |  |  |  |         free(file_name); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2011-03-14 13:15:25 -07:00
										 |  |  |  |             return EINVAL; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-03-11 15:18:30 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |         if (shash_find(args, "peer_cert") && shash_find(args, "psk")) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |             VLOG_ERR("%s: cannot define both 'peer_cert' and 'psk'", name); | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |             return EINVAL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!ipsec_mech_set) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |             VLOG_ERR("%s: IPsec requires an 'peer_cert' or psk' argument", | 
					
						
							|  |  |  |  |                      name); | 
					
						
							| 
									
										
										
										
											2010-12-15 23:44:41 -08:00
										 |  |  |  |             return EINVAL; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (is_gre) { | 
					
						
							|  |  |  |  |         set_key(args, "in_key", ODP_TUNNEL_ATTR_IN_KEY, options); | 
					
						
							|  |  |  |  |         set_key(args, "out_key", ODP_TUNNEL_ATTR_OUT_KEY, options); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!daddr) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |         VLOG_ERR("%s: %s type requires valid 'remote_ip' argument", | 
					
						
							|  |  |  |  |                  name, type); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     nl_msg_put_be32(options, ODP_TUNNEL_ATTR_DST_IPV4, daddr); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     nl_msg_put_u32(options, ODP_TUNNEL_ATTR_FLAGS, flags); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | tnl_port_config_from_nlattr(const struct nlattr *options, size_t options_len, | 
					
						
							|  |  |  |  |                             struct nlattr *a[ODP_TUNNEL_ATTR_MAX + 1]) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static const struct nl_policy odp_tunnel_policy[] = { | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_FLAGS] = { .type = NL_A_U32 }, | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_DST_IPV4] = { .type = NL_A_BE32 }, | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_SRC_IPV4] = { .type = NL_A_BE32, .optional = true }, | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_IN_KEY] = { .type = NL_A_BE64, .optional = true }, | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_OUT_KEY] = { .type = NL_A_BE64, .optional = true }, | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_TOS] = { .type = NL_A_U8, .optional = true }, | 
					
						
							|  |  |  |  |         [ODP_TUNNEL_ATTR_TTL] = { .type = NL_A_U8, .optional = true }, | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  |     struct ofpbuf buf; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ofpbuf_use_const(&buf, options, options_len); | 
					
						
							|  |  |  |  |     if (!nl_policy_parse(&buf, 0, odp_tunnel_policy, | 
					
						
							|  |  |  |  |                          a, ARRAY_SIZE(odp_tunnel_policy))) { | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static uint64_t | 
					
						
							|  |  |  |  | get_be64_or_zero(const struct nlattr *a) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return a ? ntohll(nl_attr_get_be64(a)) : 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | unparse_tunnel_config(const char *name OVS_UNUSED, const char *type OVS_UNUSED, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                       const struct nlattr *options, size_t options_len, | 
					
						
							|  |  |  |  |                       struct shash *args) | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     struct nlattr *a[ODP_TUNNEL_ATTR_MAX + 1]; | 
					
						
							|  |  |  |  |     ovs_be32 daddr; | 
					
						
							|  |  |  |  |     uint32_t flags; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     error = tnl_port_config_from_nlattr(options, options_len, a); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     flags = nl_attr_get_u32(a[ODP_TUNNEL_ATTR_FLAGS]); | 
					
						
							|  |  |  |  |     if (!(flags & TNL_F_HDR_CACHE) == !(flags & TNL_F_IPSEC)) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "header_cache", | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                  flags & TNL_F_HDR_CACHE ? "true" : "false"); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     daddr = nl_attr_get_be32(a[ODP_TUNNEL_ATTR_DST_IPV4]); | 
					
						
							|  |  |  |  |     shash_add(args, "remote_ip", xasprintf(IP_FMT, IP_ARGS(&daddr))); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (a[ODP_TUNNEL_ATTR_SRC_IPV4]) { | 
					
						
							|  |  |  |  |         ovs_be32 saddr = nl_attr_get_be32(a[ODP_TUNNEL_ATTR_SRC_IPV4]); | 
					
						
							|  |  |  |  |         shash_add(args, "local_ip", xasprintf(IP_FMT, IP_ARGS(&saddr))); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (!a[ODP_TUNNEL_ATTR_IN_KEY] && !a[ODP_TUNNEL_ATTR_OUT_KEY]) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "key", "flow"); | 
					
						
							|  |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |         uint64_t in_key = get_be64_or_zero(a[ODP_TUNNEL_ATTR_IN_KEY]); | 
					
						
							|  |  |  |  |         uint64_t out_key = get_be64_or_zero(a[ODP_TUNNEL_ATTR_OUT_KEY]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (in_key && in_key == out_key) { | 
					
						
							|  |  |  |  |             shash_add(args, "key", xasprintf("%"PRIu64, in_key)); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             if (!a[ODP_TUNNEL_ATTR_IN_KEY]) { | 
					
						
							|  |  |  |  |                 smap_add(args, "in_key", "flow"); | 
					
						
							|  |  |  |  |             } else if (in_key) { | 
					
						
							|  |  |  |  |                 shash_add(args, "in_key", xasprintf("%"PRIu64, in_key)); | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |             if (!a[ODP_TUNNEL_ATTR_OUT_KEY]) { | 
					
						
							|  |  |  |  |                 smap_add(args, "out_key", "flow"); | 
					
						
							|  |  |  |  |             } else if (out_key) { | 
					
						
							|  |  |  |  |                 shash_add(args, "out_key", xasprintf("%"PRIu64, out_key)); | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (flags & TNL_F_TTL_INHERIT) { | 
					
						
							|  |  |  |  |         smap_add(args, "tos", "inherit"); | 
					
						
							|  |  |  |  |     } else if (a[ODP_TUNNEL_ATTR_TTL]) { | 
					
						
							|  |  |  |  |         int ttl = nl_attr_get_u8(a[ODP_TUNNEL_ATTR_TTL]); | 
					
						
							|  |  |  |  |         shash_add(args, "tos", xasprintf("%d", ttl)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (flags & TNL_F_TOS_INHERIT) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "tos", "inherit"); | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     } else if (a[ODP_TUNNEL_ATTR_TOS]) { | 
					
						
							|  |  |  |  |         int tos = nl_attr_get_u8(a[ODP_TUNNEL_ATTR_TOS]); | 
					
						
							|  |  |  |  |         shash_add(args, "tos", xasprintf("%d", tos)); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (flags & TNL_F_CSUM) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "csum", "true"); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-04-29 17:05:58 -07:00
										 |  |  |  |     if (flags & TNL_F_DF_INHERIT) { | 
					
						
							|  |  |  |  |         smap_add(args, "df_inherit", "true"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (!(flags & TNL_F_DF_DEFAULT)) { | 
					
						
							|  |  |  |  |         smap_add(args, "df_default", "false"); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (!(flags & TNL_F_PMTUD)) { | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |         smap_add(args, "pmtud", "false"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | parse_patch_config(const char *name, const char *type OVS_UNUSED, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                    const struct shash *args, struct ofpbuf *options) | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     const char *peer; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     peer = shash_find_data(args, "peer"); | 
					
						
							|  |  |  |  |     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; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (shash_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; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     if (strlen(peer) >= IFNAMSIZ) { | 
					
						
							| 
									
										
										
										
											2011-03-14 13:45:43 -07:00
										 |  |  |  |         VLOG_ERR("%s: patch 'peer' arg too long", 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; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     nl_msg_put_string(options, ODP_PATCH_ATTR_PEER, peer); | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | unparse_patch_config(const char *name OVS_UNUSED, const char *type OVS_UNUSED, | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |                      const struct nlattr *options, size_t options_len, | 
					
						
							|  |  |  |  |                      struct shash *args) | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     static const struct nl_policy odp_patch_policy[] = { | 
					
						
							|  |  |  |  |         [ODP_PATCH_ATTR_PEER] = { .type = NL_A_STRING, | 
					
						
							|  |  |  |  |                                .max_len = IFNAMSIZ, | 
					
						
							|  |  |  |  |                                .optional = false } | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     struct nlattr *a[ARRAY_SIZE(odp_patch_policy)]; | 
					
						
							|  |  |  |  |     struct ofpbuf buf; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ofpbuf_use_const(&buf, options, options_len); | 
					
						
							|  |  |  |  |     if (!nl_policy_parse(&buf, 0, odp_patch_policy, | 
					
						
							|  |  |  |  |                          a, ARRAY_SIZE(odp_patch_policy))) { | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-26 12:28:59 -08:00
										 |  |  |  |     smap_add(args, "peer", nl_attr_get_string(a[ODP_PATCH_ATTR_PEER])); | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |  | 
					
						
							| 
									
										
										
										
											2011-01-05 11:51:15 -08:00
										 |  |  |  | #define VPORT_FUNCTIONS(GET_STATUS)                         \
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     netdev_vport_create,                                    \ | 
					
						
							|  |  |  |  |     netdev_vport_destroy,                                   \ | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |     netdev_vport_set_config,                                \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     netdev_vport_open,                                      \ | 
					
						
							|  |  |  |  |     netdev_vport_close,                                     \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     NULL,                       /* enumerate */             \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     NULL,                       /* recv */                  \ | 
					
						
							|  |  |  |  |     NULL,                       /* recv_wait */             \ | 
					
						
							|  |  |  |  |     NULL,                       /* drain */                 \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							| 
									
										
										
										
											2011-04-01 10:17:52 -07:00
										 |  |  |  |     netdev_vport_send,          /* send */                  \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     NULL,                       /* send_wait */             \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     netdev_vport_set_etheraddr,                             \ | 
					
						
							|  |  |  |  |     netdev_vport_get_etheraddr,                             \ | 
					
						
							|  |  |  |  |     netdev_vport_get_mtu,                                   \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_ifindex */           \ | 
					
						
							| 
									
										
										
										
											2010-10-27 15:29:16 -07:00
										 |  |  |  |     NULL,                       /* get_carrier */           \ | 
					
						
							| 
									
										
										
										
											2011-01-07 16:22:34 -08:00
										 |  |  |  |     NULL,                       /* get_miimon */            \ | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  |     netdev_vport_get_stats,                                 \ | 
					
						
							|  |  |  |  |     netdev_vport_set_stats,                                 \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_features */          \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_advertisements */    \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_vlan_vid */          \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     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 */       \ | 
					
						
							|  |  |  |  |     NULL,                       /* dump_queues */           \ | 
					
						
							|  |  |  |  |     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,                              \ | 
					
						
							|  |  |  |  |                                                             \ | 
					
						
							|  |  |  |  |     netdev_vport_poll_add,                                  \ | 
					
						
							|  |  |  |  |     netdev_vport_poll_remove, | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | netdev_vport_register(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     static const struct vport_class vport_classes[] = { | 
					
						
							| 
									
										
										
										
											2011-01-23 20:01:30 -08:00
										 |  |  |  |         { ODP_VPORT_TYPE_GRE, | 
					
						
							|  |  |  |  |           { "gre", VPORT_FUNCTIONS(netdev_vport_get_status) }, | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |           parse_tunnel_config, unparse_tunnel_config }, | 
					
						
							| 
									
										
										
										
											2011-01-23 20:01:30 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         { ODP_VPORT_TYPE_GRE, | 
					
						
							|  |  |  |  |           { "ipsec_gre", VPORT_FUNCTIONS(netdev_vport_get_status) }, | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |           parse_tunnel_config, unparse_tunnel_config }, | 
					
						
							| 
									
										
										
										
											2011-01-23 20:01:30 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         { ODP_VPORT_TYPE_CAPWAP, | 
					
						
							|  |  |  |  |           { "capwap", VPORT_FUNCTIONS(netdev_vport_get_status) }, | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |           parse_tunnel_config, unparse_tunnel_config }, | 
					
						
							| 
									
										
										
										
											2011-01-23 20:01:30 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         { ODP_VPORT_TYPE_PATCH, | 
					
						
							|  |  |  |  |           { "patch", VPORT_FUNCTIONS(NULL) }, | 
					
						
							| 
									
										
										
										
											2010-12-29 16:02:22 -08:00
										 |  |  |  |           parse_patch_config, unparse_patch_config } | 
					
						
							| 
									
										
										
										
											2010-12-03 14:41:38 -08:00
										 |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < ARRAY_SIZE(vport_classes); i++) { | 
					
						
							|  |  |  |  |         netdev_register_provider(&vport_classes[i].netdev_class); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-10-06 15:35:53 -07:00
										 |  |  |  | } |