| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2017-01-18 02:30:26 -08:00
										 |  |  |  |  * Copyright (c) 2010, 2011, 2012, 2013, 2015, 2016, 2017 Nicira, Inc. | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08: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
 | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  |  * 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>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "dummy.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 12:01:53 -08:00
										 |  |  |  | #include "dp-packet.h"
 | 
					
						
							| 
									
										
										
										
											2014-03-20 10:54:37 -07:00
										 |  |  |  | #include "dpif-netdev.h"
 | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | #include "flow.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | #include "netdev-provider.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-25 13:30:40 -08:00
										 |  |  |  | #include "netdev-vport.h"
 | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | #include "odp-util.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-14 15:20:21 -07:00
										 |  |  |  | #include "openvswitch/dynamic-string.h"
 | 
					
						
							|  |  |  |  | #include "openvswitch/list.h"
 | 
					
						
							|  |  |  |  | #include "openvswitch/ofp-print.h"
 | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:24 -07:00
										 |  |  |  | #include "openvswitch/ofpbuf.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-14 15:20:21 -07:00
										 |  |  |  | #include "openvswitch/vlog.h"
 | 
					
						
							| 
									
										
										
										
											2015-03-29 15:49:29 -07:00
										 |  |  |  | #include "ovs-atomic.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | #include "packets.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | #include "pcap-file.h"
 | 
					
						
							| 
									
										
										
										
											2017-11-03 13:53:53 +08:00
										 |  |  |  | #include "openvswitch/poll-loop.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-12 16:37:34 -05:00
										 |  |  |  | #include "openvswitch/shash.h"
 | 
					
						
							| 
									
										
										
										
											2012-01-19 10:24:46 -08:00
										 |  |  |  | #include "sset.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | #include "stream.h"
 | 
					
						
							|  |  |  |  | #include "unaligned.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | #include "timeval.h"
 | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | #include "unixctl.h"
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | #include "reconnect.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | VLOG_DEFINE_THIS_MODULE(netdev_dummy); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  | #define C_STATS_SIZE 2
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | struct reconnect; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct dummy_packet_stream { | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     struct stream *stream; | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  |     struct ovs_list txq; | 
					
						
							| 
									
										
										
										
											2017-09-08 18:59:14 +01:00
										 |  |  |  |     struct dp_packet rxbuf; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | enum dummy_packet_conn_type { | 
					
						
							|  |  |  |  |     NONE,       /* No connection is configured. */ | 
					
						
							|  |  |  |  |     PASSIVE,    /* Listener. */ | 
					
						
							|  |  |  |  |     ACTIVE      /* Connect to listener. */ | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-05 16:01:17 -07:00
										 |  |  |  | enum dummy_netdev_conn_state { | 
					
						
							|  |  |  |  |     CONN_STATE_CONNECTED,      /* Listener connected. */ | 
					
						
							|  |  |  |  |     CONN_STATE_NOT_CONNECTED,  /* Listener not connected.  */ | 
					
						
							|  |  |  |  |     CONN_STATE_UNKNOWN,        /* No relavent information.  */ | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | struct dummy_packet_pconn { | 
					
						
							|  |  |  |  |     struct pstream *pstream; | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |     struct dummy_packet_stream **streams; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     size_t n_streams; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct dummy_packet_rconn { | 
					
						
							|  |  |  |  |     struct dummy_packet_stream *rstream; | 
					
						
							|  |  |  |  |     struct reconnect *reconnect; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct dummy_packet_conn { | 
					
						
							|  |  |  |  |     enum dummy_packet_conn_type type; | 
					
						
							|  |  |  |  |     union { | 
					
						
							|  |  |  |  |         struct dummy_packet_pconn pconn; | 
					
						
							|  |  |  |  |         struct dummy_packet_rconn rconn; | 
					
						
							|  |  |  |  |     } u; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  | struct pkt_list_node { | 
					
						
							|  |  |  |  |     struct dp_packet *pkt; | 
					
						
							|  |  |  |  |     struct ovs_list list_node; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  | /* Protects 'dummy_list'. */ | 
					
						
							|  |  |  |  | static struct ovs_mutex dummy_list_mutex = OVS_MUTEX_INITIALIZER; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Contains all 'struct dummy_dev's. */ | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  | static struct ovs_list dummy_list OVS_GUARDED_BY(dummy_list_mutex) | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  |     = OVS_LIST_INITIALIZER(&dummy_list); | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | struct netdev_dummy { | 
					
						
							|  |  |  |  |     struct netdev up; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     /* In dummy_list. */ | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  |     struct ovs_list list_node OVS_GUARDED_BY(dummy_list_mutex); | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     /* Protects all members below. */ | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     struct ovs_mutex mutex OVS_ACQ_AFTER(dummy_list_mutex); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-28 14:55:11 -07:00
										 |  |  |  |     struct eth_addr hwaddr OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  |     int mtu OVS_GUARDED; | 
					
						
							|  |  |  |  |     struct netdev_stats stats OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  |     struct netdev_custom_counter custom_stats[C_STATS_SIZE] OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  |     enum netdev_flags flags OVS_GUARDED; | 
					
						
							|  |  |  |  |     int ifindex OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |     int numa_id OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     struct dummy_packet_conn conn OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     FILE *tx_pcap, *rxq_pcap OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     struct in_addr address, netmask; | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     struct in6_addr ipv6, ipv6_mask; | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  |     struct ovs_list rxes OVS_GUARDED; /* List of child "netdev_rxq_dummy"s. */ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* The following properties are for dummy-pmd and they cannot be changed
 | 
					
						
							|  |  |  |  |      * when a device is running, so we remember the request and update them | 
					
						
							|  |  |  |  |      * next time netdev_dummy_reconfigure() is called. */ | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |     int requested_n_txq OVS_GUARDED; | 
					
						
							|  |  |  |  |     int requested_n_rxq OVS_GUARDED; | 
					
						
							|  |  |  |  |     int requested_numa_id OVS_GUARDED; | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 16:37:32 -07:00
										 |  |  |  | /* Max 'recv_queue_len' in struct netdev_dummy. */ | 
					
						
							|  |  |  |  | #define NETDEV_DUMMY_MAX_QUEUE 100
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | struct netdev_rxq_dummy { | 
					
						
							|  |  |  |  |     struct netdev_rxq up; | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  |  |     struct ovs_list node;       /* In netdev_dummy's "rxes" list. */ | 
					
						
							|  |  |  |  |     struct ovs_list recv_queue; | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     int recv_queue_len;         /* ovs_list_size(&recv_queue). */ | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     struct seq *seq;            /* Reports newly queued packets. */ | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  | static unixctl_cb_func netdev_dummy_set_admin_state; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | static int netdev_dummy_construct(struct netdev *); | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | static void netdev_dummy_queue_packet(struct netdev_dummy *, | 
					
						
							|  |  |  |  |                                       struct dp_packet *, int); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static void dummy_packet_stream_close(struct dummy_packet_stream *); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  | static void pkt_list_delete(struct ovs_list *); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | static bool | 
					
						
							|  |  |  |  | is_dummy_class(const struct netdev_class *class) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     return class->construct == netdev_dummy_construct; | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct netdev_dummy * | 
					
						
							|  |  |  |  | netdev_dummy_cast(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     ovs_assert(is_dummy_class(netdev_get_class(netdev))); | 
					
						
							| 
									
										
										
										
											2013-03-08 15:09:42 -08:00
										 |  |  |  |     return CONTAINER_OF(netdev, struct netdev_dummy, up); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | static struct netdev_rxq_dummy * | 
					
						
							|  |  |  |  | netdev_rxq_dummy_cast(const struct netdev_rxq *rx) | 
					
						
							| 
									
										
										
										
											2013-05-10 14:39:19 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     ovs_assert(is_dummy_class(netdev_get_class(rx->netdev))); | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     return CONTAINER_OF(rx, struct netdev_rxq_dummy, up); | 
					
						
							| 
									
										
										
										
											2013-05-10 14:39:19 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | dummy_packet_stream_init(struct dummy_packet_stream *s, struct stream *stream) | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     int rxbuf_size = stream ? 2048 : 0; | 
					
						
							|  |  |  |  |     s->stream = stream; | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     dp_packet_init(&s->rxbuf, rxbuf_size); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_init(&s->txq); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static struct dummy_packet_stream * | 
					
						
							|  |  |  |  | dummy_packet_stream_create(struct stream *stream) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct dummy_packet_stream *s; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     s = xzalloc(sizeof *s); | 
					
						
							|  |  |  |  |     dummy_packet_stream_init(s, stream); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     return s; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_stream_wait(struct dummy_packet_stream *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     stream_run_wait(s->stream); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     if (!ovs_list_is_empty(&s->txq)) { | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         stream_send_wait(s->stream); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     stream_recv_wait(s->stream); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_stream_send(struct dummy_packet_stream *s, const void *buffer, size_t size) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     if (ovs_list_size(&s->txq) < NETDEV_DUMMY_MAX_QUEUE) { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         struct dp_packet *b; | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |         struct pkt_list_node *node; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         b = dp_packet_clone_data_with_headroom(buffer, size, 2); | 
					
						
							|  |  |  |  |         put_unaligned_be16(dp_packet_push_uninit(b, 2), htons(size)); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         node = xmalloc(sizeof *node); | 
					
						
							|  |  |  |  |         node->pkt = b; | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |         ovs_list_push_back(&s->txq, &node->list_node); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | dummy_packet_stream_run(struct netdev_dummy *dev, struct dummy_packet_stream *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int error = 0; | 
					
						
							|  |  |  |  |     size_t n; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     stream_run(s->stream); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     if (!ovs_list_is_empty(&s->txq)) { | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |         struct pkt_list_node *txbuf_node; | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         struct dp_packet *txbuf; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         int retval; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |         ASSIGN_CONTAINER(txbuf_node, ovs_list_front(&s->txq), list_node); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |         txbuf = txbuf_node->pkt; | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         retval = stream_send(s->stream, dp_packet_data(txbuf), dp_packet_size(txbuf)); | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         if (retval > 0) { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |             dp_packet_pull(txbuf, retval); | 
					
						
							|  |  |  |  |             if (!dp_packet_size(txbuf)) { | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |                 ovs_list_remove(&txbuf_node->list_node); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |                 free(txbuf_node); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |                 dp_packet_delete(txbuf); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         } else if (retval != -EAGAIN) { | 
					
						
							|  |  |  |  |             error = -retval; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     if (!error) { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         if (dp_packet_size(&s->rxbuf) < 2) { | 
					
						
							|  |  |  |  |             n = 2 - dp_packet_size(&s->rxbuf); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             uint16_t frame_len; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |             frame_len = ntohs(get_unaligned_be16(dp_packet_data(&s->rxbuf))); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             if (frame_len < ETH_HEADER_LEN) { | 
					
						
							|  |  |  |  |                 error = EPROTO; | 
					
						
							|  |  |  |  |                 n = 0; | 
					
						
							|  |  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |                 n = (2 + frame_len) - dp_packet_size(&s->rxbuf); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         int retval; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         dp_packet_prealloc_tailroom(&s->rxbuf, n); | 
					
						
							|  |  |  |  |         retval = stream_recv(s->stream, dp_packet_tail(&s->rxbuf), n); | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         if (retval > 0) { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |             dp_packet_set_size(&s->rxbuf, dp_packet_size(&s->rxbuf) + retval); | 
					
						
							|  |  |  |  |             if (retval == n && dp_packet_size(&s->rxbuf) > 2) { | 
					
						
							|  |  |  |  |                 dp_packet_pull(&s->rxbuf, 2); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |                 netdev_dummy_queue_packet(dev, | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |                                           dp_packet_clone(&s->rxbuf), 0); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |                 dp_packet_clear(&s->rxbuf); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (retval != -EAGAIN) { | 
					
						
							|  |  |  |  |             error = (retval < 0 ? -retval | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |                      : dp_packet_size(&s->rxbuf) ? EPROTO | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |                      : EOF); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | dummy_packet_stream_close(struct dummy_packet_stream *s) | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     stream_close(s->stream); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     dp_packet_uninit(&s->rxbuf); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |     pkt_list_delete(&s->txq); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | dummy_packet_conn_init(struct dummy_packet_conn *conn) | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     memset(conn, 0, sizeof *conn); | 
					
						
							|  |  |  |  |     conn->type = NONE; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_conn_get_config(struct dummy_packet_conn *conn, struct smap *args) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     switch (conn->type) { | 
					
						
							|  |  |  |  |     case PASSIVE: | 
					
						
							|  |  |  |  |         smap_add(args, "pstream", pstream_get_name(conn->u.pconn.pstream)); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ACTIVE: | 
					
						
							|  |  |  |  |         smap_add(args, "stream", stream_get_name(conn->u.rconn.rstream->stream)); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case NONE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_conn_close(struct dummy_packet_conn *conn) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int i; | 
					
						
							|  |  |  |  |     struct dummy_packet_pconn *pconn = &conn->u.pconn; | 
					
						
							|  |  |  |  |     struct dummy_packet_rconn *rconn = &conn->u.rconn; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (conn->type) { | 
					
						
							|  |  |  |  |     case PASSIVE: | 
					
						
							|  |  |  |  |         pstream_close(pconn->pstream); | 
					
						
							|  |  |  |  |         for (i = 0; i < pconn->n_streams; i++) { | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |             dummy_packet_stream_close(pconn->streams[i]); | 
					
						
							|  |  |  |  |             free(pconn->streams[i]); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         free(pconn->streams); | 
					
						
							|  |  |  |  |         pconn->pstream = NULL; | 
					
						
							|  |  |  |  |         pconn->streams = NULL; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ACTIVE: | 
					
						
							|  |  |  |  |         dummy_packet_stream_close(rconn->rstream); | 
					
						
							|  |  |  |  |         free(rconn->rstream); | 
					
						
							|  |  |  |  |         rconn->rstream = NULL; | 
					
						
							|  |  |  |  |         reconnect_destroy(rconn->reconnect); | 
					
						
							|  |  |  |  |         rconn->reconnect = NULL; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case NONE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     conn->type = NONE; | 
					
						
							|  |  |  |  |     memset(conn, 0, sizeof *conn); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_conn_set_config(struct dummy_packet_conn *conn, | 
					
						
							|  |  |  |  |                              const struct smap *args) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *pstream = smap_get(args, "pstream"); | 
					
						
							|  |  |  |  |     const char *stream = smap_get(args, "stream"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (pstream && stream) { | 
					
						
							|  |  |  |  |          VLOG_WARN("Open failed: both %s and %s are configured", | 
					
						
							|  |  |  |  |                    pstream, stream); | 
					
						
							|  |  |  |  |          return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (conn->type) { | 
					
						
							|  |  |  |  |     case PASSIVE: | 
					
						
							| 
									
										
										
										
											2015-04-16 14:38:36 -07:00
										 |  |  |  |         if (pstream && | 
					
						
							|  |  |  |  |             !strcmp(pstream_get_name(conn->u.pconn.pstream), pstream)) { | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         dummy_packet_conn_close(conn); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     case ACTIVE: | 
					
						
							| 
									
										
										
										
											2015-04-16 14:38:36 -07:00
										 |  |  |  |         if (stream && | 
					
						
							|  |  |  |  |             !strcmp(stream_get_name(conn->u.rconn.rstream->stream), stream)) { | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         dummy_packet_conn_close(conn); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     case NONE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (pstream) { | 
					
						
							|  |  |  |  |         int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = pstream_open(pstream, &conn->u.pconn.pstream, DSCP_DEFAULT); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             VLOG_WARN("%s: open failed (%s)", pstream, ovs_strerror(error)); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             conn->type = PASSIVE; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (stream) { | 
					
						
							|  |  |  |  |         int error; | 
					
						
							|  |  |  |  |         struct stream *active_stream; | 
					
						
							| 
									
										
										
										
											2015-07-22 08:30:15 -07:00
										 |  |  |  |         struct reconnect *reconnect; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         reconnect = reconnect_create(time_msec()); | 
					
						
							|  |  |  |  |         reconnect_set_name(reconnect, stream); | 
					
						
							|  |  |  |  |         reconnect_set_passive(reconnect, false, time_msec()); | 
					
						
							|  |  |  |  |         reconnect_enable(reconnect, time_msec()); | 
					
						
							| 
									
										
										
										
											2014-02-07 17:25:42 -08:00
										 |  |  |  |         reconnect_set_backoff(reconnect, 100, INT_MAX); | 
					
						
							| 
									
										
										
										
											2014-02-07 12:50:17 -08:00
										 |  |  |  |         reconnect_set_probe_interval(reconnect, 0); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         conn->u.rconn.reconnect = reconnect; | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |         conn->type = ACTIVE; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = stream_open(stream, &active_stream, DSCP_DEFAULT); | 
					
						
							|  |  |  |  |         conn->u.rconn.rstream = dummy_packet_stream_create(active_stream); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         switch (error) { | 
					
						
							|  |  |  |  |         case 0: | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             reconnect_connected(reconnect, time_msec()); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         case EAGAIN: | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             reconnect_connecting(reconnect, time_msec()); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             reconnect_connect_failed(reconnect, time_msec(), error); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             stream_close(active_stream); | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             conn->u.rconn.rstream->stream = NULL; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_pconn_run(struct netdev_dummy *dev) | 
					
						
							|  |  |  |  |     OVS_REQUIRES(dev->mutex) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct stream *new_stream; | 
					
						
							|  |  |  |  |     struct dummy_packet_pconn *pconn = &dev->conn.u.pconn; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error = pstream_accept(pconn->pstream, &new_stream); | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							|  |  |  |  |         struct dummy_packet_stream *s; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         pconn->streams = xrealloc(pconn->streams, | 
					
						
							|  |  |  |  |                                 ((pconn->n_streams + 1) | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |                                  * sizeof s)); | 
					
						
							|  |  |  |  |         s = xmalloc(sizeof *s); | 
					
						
							|  |  |  |  |         pconn->streams[pconn->n_streams++] = s; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         dummy_packet_stream_init(s, new_stream); | 
					
						
							|  |  |  |  |     } else if (error != EAGAIN) { | 
					
						
							|  |  |  |  |         VLOG_WARN("%s: accept failed (%s)", | 
					
						
							|  |  |  |  |                   pstream_get_name(pconn->pstream), ovs_strerror(error)); | 
					
						
							|  |  |  |  |         pstream_close(pconn->pstream); | 
					
						
							|  |  |  |  |         pconn->pstream = NULL; | 
					
						
							|  |  |  |  |         dev->conn.type = NONE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |     for (i = 0; i < pconn->n_streams; ) { | 
					
						
							|  |  |  |  |         struct dummy_packet_stream *s = pconn->streams[i]; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = dummy_packet_stream_run(dev, s); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             VLOG_DBG("%s: closing connection (%s)", | 
					
						
							|  |  |  |  |                      stream_get_name(s->stream), | 
					
						
							|  |  |  |  |                      ovs_retval_to_string(error)); | 
					
						
							|  |  |  |  |             dummy_packet_stream_close(s); | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |             free(s); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             pconn->streams[i] = pconn->streams[--pconn->n_streams]; | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             i++; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_rconn_run(struct netdev_dummy *dev) | 
					
						
							|  |  |  |  | OVS_REQUIRES(dev->mutex) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct dummy_packet_rconn *rconn = &dev->conn.u.rconn; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (reconnect_run(rconn->reconnect, time_msec())) { | 
					
						
							|  |  |  |  |     case RECONNECT_CONNECT: | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (rconn->rstream->stream) { | 
					
						
							|  |  |  |  |                 error = stream_connect(rconn->rstream->stream); | 
					
						
							|  |  |  |  |             } else { | 
					
						
							|  |  |  |  |                 error = stream_open(reconnect_get_name(rconn->reconnect), | 
					
						
							|  |  |  |  |                                     &rconn->rstream->stream, DSCP_DEFAULT); | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             switch (error) { | 
					
						
							|  |  |  |  |             case 0: | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |                 reconnect_connected(rconn->reconnect, time_msec()); | 
					
						
							|  |  |  |  |                 break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             case EAGAIN: | 
					
						
							|  |  |  |  |                 reconnect_connecting(rconn->reconnect, time_msec()); | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             default: | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |                 reconnect_connect_failed(rconn->reconnect, time_msec(), error); | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |                 stream_close(rconn->rstream->stream); | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |                 rconn->rstream->stream = NULL; | 
					
						
							|  |  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case RECONNECT_DISCONNECT: | 
					
						
							|  |  |  |  |     case RECONNECT_PROBE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (reconnect_is_connected(rconn->reconnect)) { | 
					
						
							|  |  |  |  |         int err; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         err = dummy_packet_stream_run(dev, rconn->rstream); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (err) { | 
					
						
							|  |  |  |  |             reconnect_disconnected(rconn->reconnect, time_msec(), err); | 
					
						
							|  |  |  |  |             stream_close(rconn->rstream->stream); | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |             rconn->rstream->stream = NULL; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_conn_run(struct netdev_dummy *dev) | 
					
						
							|  |  |  |  |     OVS_REQUIRES(dev->mutex) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (dev->conn.type) { | 
					
						
							|  |  |  |  |     case PASSIVE: | 
					
						
							|  |  |  |  |         dummy_pconn_run(dev); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ACTIVE: | 
					
						
							|  |  |  |  |         dummy_rconn_run(dev); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case NONE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_conn_wait(struct dummy_packet_conn *conn) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int i; | 
					
						
							|  |  |  |  |     switch (conn->type) { | 
					
						
							|  |  |  |  |     case PASSIVE: | 
					
						
							|  |  |  |  |         pstream_wait(conn->u.pconn.pstream); | 
					
						
							|  |  |  |  |         for (i = 0; i < conn->u.pconn.n_streams; i++) { | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |             struct dummy_packet_stream *s = conn->u.pconn.streams[i]; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |             dummy_packet_stream_wait(s); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     case ACTIVE: | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |         if (reconnect_is_connected(conn->u.rconn.reconnect)) { | 
					
						
							|  |  |  |  |             dummy_packet_stream_wait(conn->u.rconn.rstream); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case NONE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | dummy_packet_conn_send(struct dummy_packet_conn *conn, | 
					
						
							|  |  |  |  |                        const void *buffer, size_t size) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (conn->type) { | 
					
						
							|  |  |  |  |     case PASSIVE: | 
					
						
							|  |  |  |  |         for (i = 0; i < conn->u.pconn.n_streams; i++) { | 
					
						
							| 
									
										
										
										
											2016-07-06 19:39:52 -04:00
										 |  |  |  |             struct dummy_packet_stream *s = conn->u.pconn.streams[i]; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             dummy_packet_stream_send(s, buffer, size); | 
					
						
							|  |  |  |  |             pstream_wait(conn->u.pconn.pstream); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case ACTIVE: | 
					
						
							| 
									
										
										
										
											2014-02-07 14:45:14 -08:00
										 |  |  |  |         if (reconnect_is_connected(conn->u.rconn.reconnect)) { | 
					
						
							|  |  |  |  |             dummy_packet_stream_send(conn->u.rconn.rstream, buffer, size); | 
					
						
							|  |  |  |  |             dummy_packet_stream_wait(conn->u.rconn.rstream); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case NONE: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-05 16:01:17 -07:00
										 |  |  |  | static enum dummy_netdev_conn_state | 
					
						
							|  |  |  |  | dummy_netdev_get_conn_state(struct dummy_packet_conn *conn) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     enum dummy_netdev_conn_state state; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (conn->type == ACTIVE) { | 
					
						
							|  |  |  |  |         if (reconnect_is_connected(conn->u.rconn.reconnect)) { | 
					
						
							|  |  |  |  |             state = CONN_STATE_CONNECTED; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             state = CONN_STATE_NOT_CONNECTED; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         state = CONN_STATE_UNKNOWN; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return state; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2016-02-11 13:11:10 -08:00
										 |  |  |  | netdev_dummy_run(const struct netdev_class *netdev_class) | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *dev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dummy_list_mutex); | 
					
						
							|  |  |  |  |     LIST_FOR_EACH (dev, list_node, &dummy_list) { | 
					
						
							| 
									
										
										
										
											2016-02-11 13:11:10 -08:00
										 |  |  |  |         if (netdev_get_class(&dev->up) != netdev_class) { | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |         dummy_packet_conn_run(dev); | 
					
						
							|  |  |  |  |         ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&dummy_list_mutex); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2016-02-11 13:11:10 -08:00
										 |  |  |  | netdev_dummy_wait(const struct netdev_class *netdev_class) | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *dev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dummy_list_mutex); | 
					
						
							|  |  |  |  |     LIST_FOR_EACH (dev, list_node, &dummy_list) { | 
					
						
							| 
									
										
										
										
											2016-02-11 13:11:10 -08:00
										 |  |  |  |         if (netdev_get_class(&dev->up) != netdev_class) { | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |         ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |         dummy_packet_conn_wait(&dev->conn); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dummy_list_mutex); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | static struct netdev * | 
					
						
							|  |  |  |  | netdev_dummy_alloc(void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = xzalloc(sizeof *netdev); | 
					
						
							|  |  |  |  |     return &netdev->up; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | netdev_dummy_construct(struct netdev *netdev_) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-08-29 10:34:53 -07:00
										 |  |  |  |     static atomic_count next_n = ATOMIC_COUNT_INIT(0xaa550000); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2013-04-26 12:58:14 -07:00
										 |  |  |  |     unsigned int n; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-29 10:34:53 -07:00
										 |  |  |  |     n = atomic_count_inc(&next_n); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Use "error-checking" mutexes in place of other kinds wherever possible.
We've seen a number of deadlocks in the tree since thread safety was
introduced.  So far, all of these are self-deadlocks, that is, a single
thread acquiring a lock and then attempting to re-acquire the same lock
recursively.  When this has happened, the process simply hung, and it was
somewhat difficult to find the cause.
POSIX "error-checking" mutexes check for this specific problem (and
others).  This commit switches from other types of mutexes to
error-checking mutexes everywhere that we can, that is, everywhere that
we're not using recursive mutexes.  This ought to help find problems more
quickly in the future.
There might be performance advantages to other kinds of mutexes in some
cases.  However, the existing mutex type choices were just guesses, so I'd
rather go for easy detection of errors until we know that other mutex
types actually perform better in specific cases.  Also, I did a quick
microbenchmark of glibc mutex types on my host and found that the
error checking mutexes weren't any slower than the other types, at least
when the mutex is uncontended.
Signed-off-by: Ben Pfaff <blp@nicira.com>
Acked-by: Ethan Jackson <ethan@nicira.com>
											
										 
											2013-08-20 13:40:02 -07:00
										 |  |  |  |     ovs_mutex_init(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2015-08-28 14:55:11 -07:00
										 |  |  |  |     netdev->hwaddr.ea[0] = 0xaa; | 
					
						
							|  |  |  |  |     netdev->hwaddr.ea[1] = 0x55; | 
					
						
							|  |  |  |  |     netdev->hwaddr.ea[2] = n >> 24; | 
					
						
							|  |  |  |  |     netdev->hwaddr.ea[3] = n >> 16; | 
					
						
							|  |  |  |  |     netdev->hwaddr.ea[4] = n >> 8; | 
					
						
							|  |  |  |  |     netdev->hwaddr.ea[5] = n; | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     netdev->mtu = 1500; | 
					
						
							|  |  |  |  |     netdev->flags = 0; | 
					
						
							|  |  |  |  |     netdev->ifindex = -EOPNOTSUPP; | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     netdev->requested_n_rxq = netdev_->n_rxq; | 
					
						
							|  |  |  |  |     netdev->requested_n_txq = netdev_->n_txq; | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |     netdev->numa_id = 0; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  |     memset(&netdev->custom_stats, 0, sizeof(netdev->custom_stats)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_strlcpy(netdev->custom_stats[0].name, | 
					
						
							|  |  |  |  |                 "rx_custom_packets_1", NETDEV_CUSTOM_STATS_NAME_SIZE); | 
					
						
							|  |  |  |  |     ovs_strlcpy(netdev->custom_stats[1].name, | 
					
						
							|  |  |  |  |                 "rx_custom_packets_2", NETDEV_CUSTOM_STATS_NAME_SIZE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     dummy_packet_conn_init(&netdev->conn); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_init(&netdev->rxes); | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     ovs_mutex_lock(&dummy_list_mutex); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_push_back(&dummy_list, &netdev->list_node); | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dummy_list_mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | netdev_dummy_destruct(struct netdev *netdev_) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     ovs_mutex_lock(&dummy_list_mutex); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_remove(&netdev->list_node); | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dummy_list_mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2017-08-08 16:37:15 -07:00
										 |  |  |  |     if (netdev->rxq_pcap) { | 
					
						
							|  |  |  |  |         fclose(netdev->rxq_pcap); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (netdev->tx_pcap && netdev->tx_pcap != netdev->rxq_pcap) { | 
					
						
							|  |  |  |  |         fclose(netdev->tx_pcap); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     dummy_packet_conn_close(&netdev->conn); | 
					
						
							|  |  |  |  |     netdev->conn.type = NONE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_destroy(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_dealloc(struct netdev *netdev_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     free(netdev); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | netdev_dummy_get_config(const struct netdev *dev, struct smap *args) | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(dev); | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-12 12:51:47 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     if (netdev->ifindex >= 0) { | 
					
						
							|  |  |  |  |         smap_add_format(args, "ifindex", "%d", netdev->ifindex); | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     dummy_packet_conn_get_config(&netdev->conn, args); | 
					
						
							| 
									
										
										
										
											2013-08-12 12:51:47 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     /* 'dummy-pmd' specific config. */ | 
					
						
							|  |  |  |  |     if (!netdev_is_pmd(dev)) { | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     smap_add_format(args, "requested_rx_queues", "%d", netdev->requested_n_rxq); | 
					
						
							|  |  |  |  |     smap_add_format(args, "configured_rx_queues", "%d", dev->n_rxq); | 
					
						
							|  |  |  |  |     smap_add_format(args, "requested_tx_queues", "%d", netdev->requested_n_txq); | 
					
						
							|  |  |  |  |     smap_add_format(args, "configured_tx_queues", "%d", dev->n_txq); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | exit: | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  | netdev_dummy_get_addr_list(const struct netdev *netdev_, struct in6_addr **paddr, | 
					
						
							|  |  |  |  |                            struct in6_addr **pmask, int *n_addr) | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     int cnt = 0, i = 0, err = 0; | 
					
						
							|  |  |  |  |     struct in6_addr *addr, *mask; | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     if (netdev->address.s_addr != INADDR_ANY) { | 
					
						
							|  |  |  |  |         cnt++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (ipv6_addr_is_set(&netdev->ipv6)) { | 
					
						
							|  |  |  |  |         cnt++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (!cnt) { | 
					
						
							|  |  |  |  |         err = EADDRNOTAVAIL; | 
					
						
							|  |  |  |  |         goto out; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     addr = xmalloc(sizeof *addr * cnt); | 
					
						
							|  |  |  |  |     mask = xmalloc(sizeof *mask * cnt); | 
					
						
							|  |  |  |  |     if (netdev->address.s_addr != INADDR_ANY) { | 
					
						
							|  |  |  |  |         in6_addr_set_mapped_ipv4(&addr[i], netdev->address.s_addr); | 
					
						
							|  |  |  |  |         in6_addr_set_mapped_ipv4(&mask[i], netdev->netmask.s_addr); | 
					
						
							|  |  |  |  |         i++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (ipv6_addr_is_set(&netdev->ipv6)) { | 
					
						
							|  |  |  |  |         memcpy(&addr[i], &netdev->ipv6, sizeof *addr); | 
					
						
							|  |  |  |  |         memcpy(&mask[i], &netdev->ipv6_mask, sizeof *mask); | 
					
						
							|  |  |  |  |         i++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (paddr) { | 
					
						
							|  |  |  |  |         *paddr = addr; | 
					
						
							|  |  |  |  |         *pmask = mask; | 
					
						
							|  |  |  |  |         *n_addr = cnt; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         free(addr); | 
					
						
							|  |  |  |  |         free(mask); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | out: | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     return err; | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_set_in4(struct netdev *netdev_, struct in_addr address, | 
					
						
							|  |  |  |  |                      struct in_addr netmask) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     netdev->address = address; | 
					
						
							|  |  |  |  |     netdev->netmask = netmask; | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     netdev_change_seq_changed(netdev_); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  | netdev_dummy_set_in6(struct netdev *netdev_, struct in6_addr *in6, | 
					
						
							|  |  |  |  |                      struct in6_addr *mask) | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     netdev->ipv6 = *in6; | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     netdev->ipv6_mask = *mask; | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |     netdev_change_seq_changed(netdev_); | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-09 21:56:14 -08:00
										 |  |  |  | #define DUMMY_MAX_QUEUES_PER_PORT 1024
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-12-20 17:58:14 -08:00
										 |  |  |  | netdev_dummy_set_config(struct netdev *netdev_, const struct smap *args, | 
					
						
							|  |  |  |  |                         char **errp OVS_UNUSED) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     const char *pcap; | 
					
						
							| 
									
										
										
										
											2016-07-08 16:52:38 +03:00
										 |  |  |  |     int new_n_rxq, new_n_txq, new_numa_id; | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     netdev->ifindex = smap_get_int(args, "ifindex", -EOPNOTSUPP); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  |     dummy_packet_conn_set_config(&netdev->conn, args); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     if (netdev->rxq_pcap) { | 
					
						
							|  |  |  |  |         fclose(netdev->rxq_pcap); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     if (netdev->tx_pcap && netdev->tx_pcap != netdev->rxq_pcap) { | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |         fclose(netdev->tx_pcap); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     netdev->rxq_pcap = netdev->tx_pcap = NULL; | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     pcap = smap_get(args, "pcap"); | 
					
						
							|  |  |  |  |     if (pcap) { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |         netdev->rxq_pcap = netdev->tx_pcap = ovs_pcap_open(pcap, "ab"); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |         const char *rxq_pcap = smap_get(args, "rxq_pcap"); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |         const char *tx_pcap = smap_get(args, "tx_pcap"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |         if (rxq_pcap) { | 
					
						
							|  |  |  |  |             netdev->rxq_pcap = ovs_pcap_open(rxq_pcap, "ab"); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         if (tx_pcap) { | 
					
						
							| 
									
										
										
										
											2014-01-23 17:24:03 +01:00
										 |  |  |  |             netdev->tx_pcap = ovs_pcap_open(tx_pcap, "ab"); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     netdev_change_seq_changed(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* 'dummy-pmd' specific config. */ | 
					
						
							|  |  |  |  |     if (!netdev_->netdev_class->is_pmd) { | 
					
						
							|  |  |  |  |         goto exit; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-08 17:30:21 -08:00
										 |  |  |  |     new_n_rxq = MAX(smap_get_int(args, "n_rxq", NR_QUEUE), 1); | 
					
						
							|  |  |  |  |     new_n_txq = MAX(smap_get_int(args, "n_txq", NR_QUEUE), 1); | 
					
						
							| 
									
										
										
										
											2017-01-09 21:56:14 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (new_n_rxq > DUMMY_MAX_QUEUES_PER_PORT || | 
					
						
							|  |  |  |  |         new_n_txq > DUMMY_MAX_QUEUES_PER_PORT) { | 
					
						
							|  |  |  |  |         VLOG_WARN("The one or both of interface %s queues" | 
					
						
							|  |  |  |  |                   "(rxq: %d, txq: %d) exceed %d. Sets it %d.\n", | 
					
						
							|  |  |  |  |                   netdev_get_name(netdev_), | 
					
						
							|  |  |  |  |                   new_n_rxq, | 
					
						
							|  |  |  |  |                   new_n_txq, | 
					
						
							|  |  |  |  |                   DUMMY_MAX_QUEUES_PER_PORT, | 
					
						
							|  |  |  |  |                   DUMMY_MAX_QUEUES_PER_PORT); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         new_n_rxq = MIN(DUMMY_MAX_QUEUES_PER_PORT, new_n_rxq); | 
					
						
							|  |  |  |  |         new_n_txq = MIN(DUMMY_MAX_QUEUES_PER_PORT, new_n_txq); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |     new_numa_id = smap_get_int(args, "numa_id", 0); | 
					
						
							|  |  |  |  |     if (new_n_rxq != netdev->requested_n_rxq | 
					
						
							| 
									
										
										
										
											2016-07-08 16:52:38 +03:00
										 |  |  |  |         || new_n_txq != netdev->requested_n_txq | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |         || new_numa_id != netdev->requested_numa_id) { | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |         netdev->requested_n_rxq = new_n_rxq; | 
					
						
							| 
									
										
										
										
											2016-07-08 16:52:38 +03:00
										 |  |  |  |         netdev->requested_n_txq = new_n_txq; | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |         netdev->requested_numa_id = new_numa_id; | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |         netdev_request_reconfigure(netdev_); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | exit: | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  | netdev_dummy_get_numa_id(const struct netdev *netdev_) | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     int numa_id = netdev->numa_id; | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return numa_id; | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | /* Sets the number of tx queues and rx queues for the dummy PMD interface. */ | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_reconfigure(struct netdev *netdev_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     netdev_->n_txq = netdev->requested_n_txq; | 
					
						
							|  |  |  |  |     netdev_->n_rxq = netdev->requested_n_rxq; | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |     netdev->numa_id = netdev->requested_numa_id; | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | static struct netdev_rxq * | 
					
						
							|  |  |  |  | netdev_dummy_rxq_alloc(void) | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = xzalloc(sizeof *rx); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     return &rx->up; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-05 14:15:32 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | netdev_dummy_rxq_construct(struct netdev_rxq *rxq_) | 
					
						
							| 
									
										
										
										
											2011-08-05 14:15:32 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = netdev_rxq_dummy_cast(rxq_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(rx->up.netdev); | 
					
						
							| 
									
										
										
										
											2013-05-10 14:39:19 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_push_back(&netdev->rxes, &rx->node); | 
					
						
							|  |  |  |  |     ovs_list_init(&rx->recv_queue); | 
					
						
							| 
									
										
										
										
											2012-08-16 16:37:32 -07:00
										 |  |  |  |     rx->recv_queue_len = 0; | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     rx->seq = seq_create(); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-05-10 14:39:19 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-05 14:15:32 -07:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | netdev_dummy_rxq_destruct(struct netdev_rxq *rxq_) | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = netdev_rxq_dummy_cast(rxq_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(rx->up.netdev); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_remove(&rx->node); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |     pkt_list_delete(&rx->recv_queue); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     seq_destroy(rx->seq); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | netdev_dummy_rxq_dealloc(struct netdev_rxq *rxq_) | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = netdev_rxq_dummy_cast(rxq_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:21:38 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     free(rx); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-05 14:15:32 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-07-18 17:05:35 -07:00
										 |  |  |  | netdev_dummy_rxq_recv(struct netdev_rxq *rxq_, struct dp_packet_batch *batch) | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = netdev_rxq_dummy_cast(rxq_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(rx->up.netdev); | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     struct dp_packet *packet; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     if (!ovs_list_is_empty(&rx->recv_queue)) { | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |         struct pkt_list_node *pkt_node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |         ASSIGN_CONTAINER(pkt_node, ovs_list_pop_front(&rx->recv_queue), list_node); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |         packet = pkt_node->pkt; | 
					
						
							|  |  |  |  |         free(pkt_node); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         rx->recv_queue_len--; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         packet = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!packet) { | 
					
						
							| 
									
										
										
										
											2016-06-06 17:05:49 -07:00
										 |  |  |  |         if (netdev_is_pmd(&netdev->up)) { | 
					
						
							|  |  |  |  |             /* If 'netdev' is a PMD device, this is called as part of the PMD
 | 
					
						
							|  |  |  |  |              * thread busy loop.  We yield here (without quiescing) for two | 
					
						
							|  |  |  |  |              * reasons: | 
					
						
							|  |  |  |  |              * | 
					
						
							|  |  |  |  |              * - To reduce the CPU utilization during the testsuite | 
					
						
							|  |  |  |  |              * - To give valgrind a chance to switch thread. According | 
					
						
							|  |  |  |  |              *   to the valgrind documentation, there's a big lock that | 
					
						
							|  |  |  |  |              *   prevents multiple thread from being executed at the same | 
					
						
							|  |  |  |  |              *   time.  On my system, without this sleep, the pmd threads | 
					
						
							|  |  |  |  |              *   testcases fail under valgrind, because ovs-vswitchd becomes | 
					
						
							|  |  |  |  |              *   unresponsive. */ | 
					
						
							|  |  |  |  |             sched_yield(); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-01-15 17:17:00 +09:00
										 |  |  |  |         return EAGAIN; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-20 10:54:37 -07:00
										 |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     netdev->stats.rx_packets++; | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     netdev->stats.rx_bytes += dp_packet_size(packet); | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  |     netdev->custom_stats[0].value++; | 
					
						
							|  |  |  |  |     netdev->custom_stats[1].value++; | 
					
						
							| 
									
										
										
										
											2014-03-20 10:54:37 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-18 17:05:35 -07:00
										 |  |  |  |     batch->packets[0] = packet; | 
					
						
							|  |  |  |  |     batch->count = 1; | 
					
						
							| 
									
										
										
										
											2014-03-20 10:54:37 -07:00
										 |  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | netdev_dummy_rxq_wait(struct netdev_rxq *rxq_) | 
					
						
							| 
									
										
										
										
											2013-05-10 14:39:19 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = netdev_rxq_dummy_cast(rxq_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(rx->up.netdev); | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     uint64_t seq = seq_read(rx->seq); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     if (!ovs_list_is_empty(&rx->recv_queue)) { | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |         poll_immediate_wake(); | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         seq_wait(rx->seq, seq); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  | netdev_dummy_rxq_drain(struct netdev_rxq *rxq_) | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx = netdev_rxq_dummy_cast(rxq_); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(rx->up.netdev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |     pkt_list_delete(&rx->recv_queue); | 
					
						
							| 
									
										
										
										
											2012-08-16 16:37:32 -07:00
										 |  |  |  |     rx->recv_queue_len = 0; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     seq_change(rx->seq); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2011-08-05 14:15:32 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-18 11:06:08 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2014-09-03 14:37:35 -07:00
										 |  |  |  | netdev_dummy_send(struct netdev *netdev, int qid OVS_UNUSED, | 
					
						
							| 
									
										
										
										
											2017-12-14 14:59:25 +03:00
										 |  |  |  |                   struct dp_packet_batch *batch, | 
					
						
							| 
									
										
										
										
											2016-07-27 17:44:41 +03:00
										 |  |  |  |                   bool concurrent_txq OVS_UNUSED) | 
					
						
							| 
									
										
										
										
											2012-09-18 11:06:08 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |     int error = 0; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-17 15:56:58 -08:00
										 |  |  |  |     struct dp_packet *packet; | 
					
						
							|  |  |  |  |     DP_PACKET_BATCH_FOR_EACH(packet, batch) { | 
					
						
							|  |  |  |  |         const void *buffer = dp_packet_data(packet); | 
					
						
							| 
									
										
										
										
											2017-12-14 14:59:26 +03:00
										 |  |  |  |         size_t size = dp_packet_size(packet); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-25 16:29:59 +00:00
										 |  |  |  |         if (batch->packets[i]->packet_type != htonl(PT_ETH)) { | 
					
						
							|  |  |  |  |             error = EPFNOSUPPORT; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |         if (size < ETH_HEADER_LEN) { | 
					
						
							|  |  |  |  |             error = EMSGSIZE; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             const struct eth_header *eth = buffer; | 
					
						
							|  |  |  |  |             int max_size; | 
					
						
							| 
									
										
										
										
											2013-08-16 14:04:08 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |             ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |             max_size = dev->mtu + ETH_HEADER_LEN; | 
					
						
							|  |  |  |  |             ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (eth->eth_type == htons(ETH_TYPE_VLAN)) { | 
					
						
							|  |  |  |  |                 max_size += VLAN_HEADER_LEN; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (size > max_size) { | 
					
						
							|  |  |  |  |                 error = EMSGSIZE; | 
					
						
							|  |  |  |  |                 break; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-09-18 11:06:08 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |         ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |         dev->stats.tx_packets++; | 
					
						
							|  |  |  |  |         dev->stats.tx_bytes += size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         dummy_packet_conn_send(&dev->conn, buffer, size); | 
					
						
							| 
									
										
										
										
											2012-09-18 11:06:08 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-10 00:01:59 -07:00
										 |  |  |  |         /* Reply to ARP requests for 'dev''s assigned IP address. */ | 
					
						
							|  |  |  |  |         if (dev->address.s_addr) { | 
					
						
							| 
									
										
										
										
											2017-08-02 15:03:06 -07:00
										 |  |  |  |             struct dp_packet dp; | 
					
						
							| 
									
										
										
										
											2015-05-10 00:01:59 -07:00
										 |  |  |  |             struct flow flow; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 15:03:06 -07:00
										 |  |  |  |             dp_packet_use_const(&dp, buffer, size); | 
					
						
							|  |  |  |  |             flow_extract(&dp, &flow); | 
					
						
							| 
									
										
										
										
											2015-05-10 00:01:59 -07:00
										 |  |  |  |             if (flow.dl_type == htons(ETH_TYPE_ARP) | 
					
						
							|  |  |  |  |                 && flow.nw_proto == ARP_OP_REQUEST | 
					
						
							|  |  |  |  |                 && flow.nw_dst == dev->address.s_addr) { | 
					
						
							|  |  |  |  |                 struct dp_packet *reply = dp_packet_new(0); | 
					
						
							|  |  |  |  |                 compose_arp(reply, ARP_OP_REPLY, dev->hwaddr, flow.dl_src, | 
					
						
							|  |  |  |  |                             false, flow.nw_dst, flow.nw_src); | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |                 netdev_dummy_queue_packet(dev, reply, 0); | 
					
						
							| 
									
										
										
										
											2015-05-10 00:01:59 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |         if (dev->tx_pcap) { | 
					
						
							| 
									
										
										
										
											2017-08-02 15:03:06 -07:00
										 |  |  |  |             struct dp_packet dp; | 
					
						
							| 
									
										
										
										
											2013-12-18 13:55:25 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-02 15:03:06 -07:00
										 |  |  |  |             dp_packet_use_const(&dp, buffer, size); | 
					
						
							|  |  |  |  |             ovs_pcap_write(dev->tx_pcap, &dp); | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |             fflush(dev->tx_pcap); | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |         ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-14 14:59:25 +03:00
										 |  |  |  |     dp_packet_delete_batch(batch, true); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-23 11:43:58 -07:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2012-09-18 11:06:08 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2015-08-28 14:55:11 -07:00
										 |  |  |  | netdev_dummy_set_etheraddr(struct netdev *netdev, const struct eth_addr mac) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     if (!eth_addr_equals(dev->hwaddr, mac)) { | 
					
						
							| 
									
										
										
										
											2015-08-28 14:55:11 -07:00
										 |  |  |  |         dev->hwaddr = mac; | 
					
						
							| 
									
										
										
										
											2014-04-03 00:17:34 -07:00
										 |  |  |  |         netdev_change_seq_changed(netdev); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2015-08-28 14:55:11 -07:00
										 |  |  |  | netdev_dummy_get_etheraddr(const struct netdev *netdev, struct eth_addr *mac) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2015-08-28 14:55:11 -07:00
										 |  |  |  |     *mac = dev->hwaddr; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_get_mtu(const struct netdev *netdev, int *mtup) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     *mtup = dev->mtu; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-11 20:48:56 -08:00
										 |  |  |  | #define DUMMY_MIN_MTU 68
 | 
					
						
							|  |  |  |  | #define DUMMY_MAX_MTU 65535
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-12 17:12:52 -07:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-08-09 17:01:19 +01:00
										 |  |  |  | netdev_dummy_set_mtu(struct netdev *netdev, int mtu) | 
					
						
							| 
									
										
										
										
											2011-09-12 17:12:52 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-11 20:48:56 -08:00
										 |  |  |  |     if (mtu < DUMMY_MIN_MTU || mtu > DUMMY_MAX_MTU) { | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2011-09-12 17:12:52 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2016-03-29 18:41:50 -07:00
										 |  |  |  |     if (dev->mtu != mtu) { | 
					
						
							|  |  |  |  |         dev->mtu = mtu; | 
					
						
							|  |  |  |  |         netdev_change_seq_changed(netdev); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-12 17:12:52 -07:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_get_stats(const struct netdev *netdev, struct netdev_stats *stats) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2016-05-05 09:46:01 +01:00
										 |  |  |  |     /* Passing only collected counters */ | 
					
						
							|  |  |  |  |     stats->tx_packets = dev->stats.tx_packets; | 
					
						
							|  |  |  |  |     stats->tx_bytes = dev->stats.tx_bytes; | 
					
						
							|  |  |  |  |     stats->rx_packets = dev->stats.rx_packets; | 
					
						
							|  |  |  |  |     stats->rx_bytes = dev->stats.rx_bytes; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_get_custom_stats(const struct netdev *netdev, | 
					
						
							|  |  |  |  |                              struct netdev_custom_stats *custom_stats) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     custom_stats->size = 2; | 
					
						
							|  |  |  |  |     custom_stats->counters = | 
					
						
							|  |  |  |  |             (struct netdev_custom_counter *) xcalloc(C_STATS_SIZE, | 
					
						
							|  |  |  |  |                     sizeof(struct netdev_custom_counter)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-10 15:47:20 -08:00
										 |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  |     for (i = 0 ; i < C_STATS_SIZE ; i++) { | 
					
						
							|  |  |  |  |         custom_stats->counters[i].value = dev->custom_stats[i].value; | 
					
						
							|  |  |  |  |         ovs_strlcpy(custom_stats->counters[i].name, | 
					
						
							|  |  |  |  |                     dev->custom_stats[i].name, | 
					
						
							|  |  |  |  |                     NETDEV_CUSTOM_STATS_NAME_SIZE); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-10 15:47:20 -08:00
										 |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-20 08:16:09 -08:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_get_queue(const struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                        unsigned int queue_id, struct smap *details OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (queue_id == 0) { | 
					
						
							|  |  |  |  |         return 0; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_init_queue_stats(struct netdev_queue_stats *stats) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     *stats = (struct netdev_queue_stats) { | 
					
						
							|  |  |  |  |         .tx_bytes = UINT64_MAX, | 
					
						
							|  |  |  |  |         .tx_packets = UINT64_MAX, | 
					
						
							|  |  |  |  |         .tx_errors = UINT64_MAX, | 
					
						
							|  |  |  |  |         .created = LLONG_MIN, | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_get_queue_stats(const struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                              unsigned int queue_id, | 
					
						
							|  |  |  |  |                              struct netdev_queue_stats *stats) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (queue_id == 0) { | 
					
						
							|  |  |  |  |         netdev_dummy_init_queue_stats(stats); | 
					
						
							|  |  |  |  |         return 0; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct netdev_dummy_queue_state { | 
					
						
							|  |  |  |  |     unsigned int next_queue; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_queue_dump_start(const struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                               void **statep) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy_queue_state *state = xmalloc(sizeof *state); | 
					
						
							|  |  |  |  |     state->next_queue = 0; | 
					
						
							|  |  |  |  |     *statep = state; | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_queue_dump_next(const struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                              void *state_, | 
					
						
							|  |  |  |  |                              unsigned int *queue_id, | 
					
						
							|  |  |  |  |                              struct smap *details OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy_queue_state *state = state_; | 
					
						
							|  |  |  |  |     if (state->next_queue == 0) { | 
					
						
							|  |  |  |  |         *queue_id = 0; | 
					
						
							|  |  |  |  |         state->next_queue++; | 
					
						
							|  |  |  |  |         return 0; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         return EOF; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_queue_dump_done(const struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                              void *state) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     free(state); | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_dump_queue_stats(const struct netdev *netdev OVS_UNUSED, | 
					
						
							|  |  |  |  |                               void (*cb)(unsigned int queue_id, | 
					
						
							|  |  |  |  |                                          struct netdev_queue_stats *, | 
					
						
							|  |  |  |  |                                          void *aux), | 
					
						
							|  |  |  |  |                               void *aux) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_queue_stats stats; | 
					
						
							|  |  |  |  |     netdev_dummy_init_queue_stats(&stats); | 
					
						
							|  |  |  |  |     cb(0, &stats, aux); | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_get_ifindex(const struct netdev *netdev) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     int ifindex; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&dev->mutex); | 
					
						
							|  |  |  |  |     ifindex = dev->ifindex; | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&dev->mutex); | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     return ifindex; | 
					
						
							| 
									
										
										
										
											2013-03-27 23:02:21 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | static int | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | netdev_dummy_update_flags__(struct netdev_dummy *netdev, | 
					
						
							|  |  |  |  |                             enum netdev_flags off, enum netdev_flags on, | 
					
						
							|  |  |  |  |                             enum netdev_flags *old_flagsp) | 
					
						
							|  |  |  |  |     OVS_REQUIRES(netdev->mutex) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     if ((off | on) & ~(NETDEV_UP | NETDEV_PROMISC)) { | 
					
						
							|  |  |  |  |         return EINVAL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     *old_flagsp = netdev->flags; | 
					
						
							|  |  |  |  |     netdev->flags |= on; | 
					
						
							|  |  |  |  |     netdev->flags &= ~off; | 
					
						
							|  |  |  |  |     if (*old_flagsp != netdev->flags) { | 
					
						
							| 
									
										
										
										
											2014-04-03 00:17:34 -07:00
										 |  |  |  |         netdev_change_seq_changed(&netdev->up); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | netdev_dummy_update_flags(struct netdev *netdev_, | 
					
						
							|  |  |  |  |                           enum netdev_flags off, enum netdev_flags on, | 
					
						
							|  |  |  |  |                           enum netdev_flags *old_flagsp) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev_dummy *netdev = netdev_dummy_cast(netdev_); | 
					
						
							|  |  |  |  |     int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |     error = netdev_dummy_update_flags__(netdev, off, on, old_flagsp); | 
					
						
							|  |  |  |  |     ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | /* Helper functions. */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-08 16:52:38 +03:00
										 |  |  |  | #define NETDEV_DUMMY_CLASS(NAME, PMD, RECOFIGURE)               \
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  | {                                                               \ | 
					
						
							|  |  |  |  |     NAME,                                                       \ | 
					
						
							|  |  |  |  |     PMD,                        /* is_pmd */                    \ | 
					
						
							|  |  |  |  |     NULL,                       /* init */                      \ | 
					
						
							|  |  |  |  |     netdev_dummy_run,                                           \ | 
					
						
							|  |  |  |  |     netdev_dummy_wait,                                          \ | 
					
						
							|  |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_alloc,                                         \ | 
					
						
							|  |  |  |  |     netdev_dummy_construct,                                     \ | 
					
						
							|  |  |  |  |     netdev_dummy_destruct,                                      \ | 
					
						
							|  |  |  |  |     netdev_dummy_dealloc,                                       \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_config,                                    \ | 
					
						
							|  |  |  |  |     netdev_dummy_set_config,                                    \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_tunnel_config */         \ | 
					
						
							|  |  |  |  |     NULL,                       /* build header */              \ | 
					
						
							|  |  |  |  |     NULL,                       /* push header */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* pop header */                \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_numa_id,                                   \ | 
					
						
							| 
									
										
										
										
											2016-07-08 16:52:38 +03:00
										 |  |  |  |     NULL,                       /* set_tx_multiq */             \ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_send,          /* send */                      \ | 
					
						
							|  |  |  |  |     NULL,                       /* send_wait */                 \ | 
					
						
							|  |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_set_etheraddr,                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_etheraddr,                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_mtu,                                       \ | 
					
						
							|  |  |  |  |     netdev_dummy_set_mtu,                                       \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_ifindex,                                   \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_carrier */               \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_carrier_resets */        \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_miimon */                \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_stats,                                     \ | 
					
						
							| 
									
										
										
										
											2018-01-09 07:55:37 +00:00
										 |  |  |  |     netdev_dummy_get_custom_stats,                              \ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_features */              \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_advertisements */        \ | 
					
						
							| 
									
										
										
										
											2017-06-23 16:47:59 +00:00
										 |  |  |  |     NULL,                       /* get_pt_mode */               \ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_policing */              \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_qos_types */             \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_qos_capabilities */      \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_qos */                   \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_qos */                   \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_queue,                                     \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_queue */                 \ | 
					
						
							|  |  |  |  |     NULL,                       /* delete_queue */              \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_queue_stats,                               \ | 
					
						
							|  |  |  |  |     netdev_dummy_queue_dump_start,                              \ | 
					
						
							|  |  |  |  |     netdev_dummy_queue_dump_next,                               \ | 
					
						
							|  |  |  |  |     netdev_dummy_queue_dump_done,                               \ | 
					
						
							|  |  |  |  |     netdev_dummy_dump_queue_stats,                              \ | 
					
						
							|  |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     NULL,                       /* set_in4 */                   \ | 
					
						
							|  |  |  |  |     netdev_dummy_get_addr_list,                                 \ | 
					
						
							|  |  |  |  |     NULL,                       /* add_router */                \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_next_hop */              \ | 
					
						
							|  |  |  |  |     NULL,                       /* get_status */                \ | 
					
						
							|  |  |  |  |     NULL,                       /* arp_lookup */                \ | 
					
						
							|  |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_update_flags,                                  \ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     RECOFIGURE,                                                 \ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_alloc,                                     \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_construct,                                 \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_destruct,                                  \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_dealloc,                                   \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_recv,                                      \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_wait,                                      \ | 
					
						
							|  |  |  |  |     netdev_dummy_rxq_drain,                                     \ | 
					
						
							| 
									
										
										
										
											2017-06-13 18:03:28 +03:00
										 |  |  |  |                                                                 \ | 
					
						
							|  |  |  |  |     NO_OFFLOAD_API                                              \ | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static const struct netdev_class dummy_class = | 
					
						
							| 
									
										
										
										
											2016-07-08 16:52:38 +03:00
										 |  |  |  |     NETDEV_DUMMY_CLASS("dummy", false, NULL); | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-11 13:11:10 -08:00
										 |  |  |  | static const struct netdev_class dummy_internal_class = | 
					
						
							|  |  |  |  |     NETDEV_DUMMY_CLASS("dummy-internal", false, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  | static const struct netdev_class dummy_pmd_class = | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     NETDEV_DUMMY_CLASS("dummy-pmd", true, | 
					
						
							|  |  |  |  |                        netdev_dummy_reconfigure); | 
					
						
							| 
									
										
										
										
											2013-05-10 14:39:19 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | pkt_list_delete(struct ovs_list *l) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct pkt_list_node *pkt; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     LIST_FOR_EACH_POP(pkt, list_node, l) { | 
					
						
							|  |  |  |  |         dp_packet_delete(pkt->pkt); | 
					
						
							|  |  |  |  |         free(pkt); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  | static struct dp_packet * | 
					
						
							| 
									
										
										
										
											2017-01-18 02:30:26 -08:00
										 |  |  |  | eth_from_packet(const char *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct dp_packet *packet; | 
					
						
							|  |  |  |  |     eth_from_hex(s, &packet); | 
					
						
							|  |  |  |  |     return packet; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct dp_packet * | 
					
						
							| 
									
										
										
										
											2017-07-25 16:02:01 +03:00
										 |  |  |  | eth_from_flow(const char *s, size_t packet_size) | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     enum odp_key_fitness fitness; | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     struct dp_packet *packet; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     struct ofpbuf odp_key; | 
					
						
							|  |  |  |  |     struct flow flow; | 
					
						
							|  |  |  |  |     int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Convert string to datapath key.
 | 
					
						
							|  |  |  |  |      * | 
					
						
							|  |  |  |  |      * It would actually be nicer to parse an OpenFlow-like flow key here, but | 
					
						
							|  |  |  |  |      * the code for that currently calls exit() on parse error.  We have to | 
					
						
							|  |  |  |  |      * settle for parsing a datapath key for now. | 
					
						
							|  |  |  |  |      */ | 
					
						
							|  |  |  |  |     ofpbuf_init(&odp_key, 0); | 
					
						
							| 
									
										
										
										
											2013-06-19 07:15:10 +00:00
										 |  |  |  |     error = odp_flow_from_string(s, NULL, &odp_key, NULL); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         ofpbuf_uninit(&odp_key); | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Convert odp_key to flow. */ | 
					
						
							| 
									
										
										
										
											2015-03-02 17:29:44 -08:00
										 |  |  |  |     fitness = odp_flow_key_to_flow(odp_key.data, odp_key.size, &flow); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     if (fitness == ODP_FIT_ERROR) { | 
					
						
							|  |  |  |  |         ofpbuf_uninit(&odp_key); | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |     packet = dp_packet_new(0); | 
					
						
							| 
									
										
										
										
											2017-07-25 14:26:22 -07:00
										 |  |  |  |     if (!flow_compose(packet, &flow, packet_size)) { | 
					
						
							|  |  |  |  |         dp_packet_delete(packet); | 
					
						
							|  |  |  |  |         packet = NULL; | 
					
						
							|  |  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ofpbuf_uninit(&odp_key); | 
					
						
							|  |  |  |  |     return packet; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-28 11:23:29 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  | netdev_dummy_queue_packet__(struct netdev_rxq_dummy *rx, struct dp_packet *packet) | 
					
						
							| 
									
										
										
										
											2012-08-10 15:21:58 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-05-18 10:47:45 -07:00
										 |  |  |  |     struct pkt_list_node *pkt_node = xmalloc(sizeof *pkt_node); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     pkt_node->pkt = packet; | 
					
						
							| 
									
										
										
										
											2016-03-25 14:10:22 -07:00
										 |  |  |  |     ovs_list_push_back(&rx->recv_queue, &pkt_node->list_node); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     rx->recv_queue_len++; | 
					
						
							| 
									
										
										
										
											2013-12-25 22:27:25 -08:00
										 |  |  |  |     seq_change(rx->seq); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-08-10 15:21:58 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  | netdev_dummy_queue_packet(struct netdev_dummy *dummy, struct dp_packet *packet, | 
					
						
							|  |  |  |  |                           int queue_id) | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     OVS_REQUIRES(dummy->mutex) | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     struct netdev_rxq_dummy *rx, *prev; | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 19:38:14 -07:00
										 |  |  |  |     if (dummy->rxq_pcap) { | 
					
						
							|  |  |  |  |         ovs_pcap_write(dummy->rxq_pcap, packet); | 
					
						
							|  |  |  |  |         fflush(dummy->rxq_pcap); | 
					
						
							| 
									
										
										
										
											2013-09-23 10:26:51 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     prev = NULL; | 
					
						
							| 
									
										
										
										
											2012-08-10 15:21:58 -07:00
										 |  |  |  |     LIST_FOR_EACH (rx, node, &dummy->rxes) { | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |         if (rx->up.queue_id == queue_id && | 
					
						
							|  |  |  |  |             rx->recv_queue_len < NETDEV_DUMMY_MAX_QUEUE) { | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |             if (prev) { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |                 netdev_dummy_queue_packet__(prev, dp_packet_clone(packet)); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |             prev = rx; | 
					
						
							| 
									
										
										
										
											2012-08-10 15:21:58 -07:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     if (prev) { | 
					
						
							|  |  |  |  |         netdev_dummy_queue_packet__(prev, packet); | 
					
						
							|  |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         dp_packet_delete(packet); | 
					
						
							| 
									
										
										
										
											2012-10-25 16:54:23 -07:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-08-10 15:21:58 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_receive(struct unixctl_conn *conn, | 
					
						
							|  |  |  |  |                      int argc, const char *argv[], void *aux OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |     struct netdev_dummy *dummy_dev; | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |     struct netdev *netdev; | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     int i, k = 1, rx_qid = 0; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     netdev = netdev_from_name(argv[k++]); | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |     if (!netdev || !is_dummy_class(netdev->netdev_class)) { | 
					
						
							| 
									
										
										
										
											2012-02-14 20:53:59 -08:00
										 |  |  |  |         unixctl_command_reply_error(conn, "no such dummy netdev"); | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |         goto exit_netdev; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |     dummy_dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     ovs_mutex_lock(&dummy_dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (argc > k + 1 && !strcmp(argv[k], "--qid")) { | 
					
						
							|  |  |  |  |         rx_qid = strtol(argv[k + 1], NULL, 10); | 
					
						
							|  |  |  |  |         if (rx_qid < 0 || rx_qid >= netdev->n_rxq) { | 
					
						
							|  |  |  |  |             unixctl_command_reply_error(conn, "bad rx queue id."); | 
					
						
							|  |  |  |  |             goto exit; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         k += 2; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = k; i < argc; i++) { | 
					
						
							| 
									
										
										
										
											2015-02-22 03:21:09 -08:00
										 |  |  |  |         struct dp_packet *packet; | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-18 02:30:26 -08:00
										 |  |  |  |         /* Try to parse 'argv[i]' as packet in hex. */ | 
					
						
							|  |  |  |  |         packet = eth_from_packet(argv[i]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |         if (!packet) { | 
					
						
							| 
									
										
										
										
											2017-07-25 16:02:01 +03:00
										 |  |  |  |             int packet_size = 0; | 
					
						
							|  |  |  |  |             const char *flow_str = argv[i]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             /* Parse optional --len argument immediately follows a 'flow'.  */ | 
					
						
							|  |  |  |  |             if (argc >= i + 2 && !strcmp(argv[i + 1], "--len")) { | 
					
						
							|  |  |  |  |                 packet_size = strtol(argv[i + 2], NULL, 10); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 if (packet_size < ETH_TOTAL_MIN) { | 
					
						
							|  |  |  |  |                     unixctl_command_reply_error(conn, "too small packet len"); | 
					
						
							|  |  |  |  |                     goto exit; | 
					
						
							|  |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-08-09 13:38:05 -07:00
										 |  |  |  |                 i += 2; | 
					
						
							| 
									
										
										
										
											2017-07-25 16:02:01 +03:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-01-18 02:30:26 -08:00
										 |  |  |  |             /* Try parse 'argv[i]' as odp flow. */ | 
					
						
							| 
									
										
										
										
											2017-07-25 16:02:01 +03:00
										 |  |  |  |             packet = eth_from_flow(flow_str, packet_size); | 
					
						
							| 
									
										
										
										
											2017-01-18 02:30:26 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (!packet) { | 
					
						
							|  |  |  |  |                 unixctl_command_reply_error(conn, "bad packet or flow syntax"); | 
					
						
							|  |  |  |  |                 goto exit; | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |         netdev_dummy_queue_packet(dummy_dev, packet, rx_qid); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-28 11:23:29 -07:00
										 |  |  |  |     unixctl_command_reply(conn, NULL); | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | exit: | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     ovs_mutex_unlock(&dummy_dev->mutex); | 
					
						
							|  |  |  |  | exit_netdev: | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |     netdev_close(netdev); | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  | netdev_dummy_set_admin_state__(struct netdev_dummy *dev, bool admin_state) | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |     OVS_REQUIRES(dev->mutex) | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     enum netdev_flags old_flags; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (admin_state) { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         netdev_dummy_update_flags__(dev, 0, NETDEV_UP, &old_flags); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |         netdev_dummy_update_flags__(dev, NETDEV_UP, 0, &old_flags); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_set_admin_state(struct unixctl_conn *conn, int argc, | 
					
						
							|  |  |  |  |                              const char *argv[], void *aux OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bool up; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!strcasecmp(argv[argc - 1], "up")) { | 
					
						
							|  |  |  |  |         up = true; | 
					
						
							|  |  |  |  |     } else if ( !strcasecmp(argv[argc - 1], "down")) { | 
					
						
							|  |  |  |  |         up = false; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         unixctl_command_reply_error(conn, "Invalid Admin State"); | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (argc > 2) { | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |         struct netdev *netdev = netdev_from_name(argv[1]); | 
					
						
							|  |  |  |  |         if (netdev && is_dummy_class(netdev->netdev_class)) { | 
					
						
							|  |  |  |  |             struct netdev_dummy *dummy_dev = netdev_dummy_cast(netdev); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |             ovs_mutex_lock(&dummy_dev->mutex); | 
					
						
							| 
									
										
										
										
											2013-03-15 15:54:36 -07:00
										 |  |  |  |             netdev_dummy_set_admin_state__(dummy_dev, up); | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |             ovs_mutex_unlock(&dummy_dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |             netdev_close(netdev); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             unixctl_command_reply_error(conn, "Unknown Dummy Interface"); | 
					
						
							| 
									
										
										
										
											2013-07-25 16:11:52 -07:00
										 |  |  |  |             netdev_close(netdev); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |         struct netdev_dummy *netdev; | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |         ovs_mutex_lock(&dummy_list_mutex); | 
					
						
							|  |  |  |  |         LIST_FOR_EACH (netdev, list_node, &dummy_list) { | 
					
						
							| 
									
										
										
										
											2013-08-09 21:34:02 -07:00
										 |  |  |  |             ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |             netdev_dummy_set_admin_state__(netdev, up); | 
					
						
							|  |  |  |  |             ovs_mutex_unlock(&netdev->mutex); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-08-12 12:49:23 -07:00
										 |  |  |  |         ovs_mutex_unlock(&dummy_list_mutex); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     unixctl_command_reply(conn, "OK"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-05 16:01:17 -07:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | display_conn_state__(struct ds *s, const char *name, | 
					
						
							|  |  |  |  |                      enum dummy_netdev_conn_state state) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     ds_put_format(s, "%s: ", name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (state) { | 
					
						
							|  |  |  |  |     case CONN_STATE_CONNECTED: | 
					
						
							|  |  |  |  |         ds_put_cstr(s, "connected\n"); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case CONN_STATE_NOT_CONNECTED: | 
					
						
							|  |  |  |  |         ds_put_cstr(s, "disconnected\n"); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case CONN_STATE_UNKNOWN: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         ds_put_cstr(s, "unknown\n"); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_conn_state(struct unixctl_conn *conn, int argc, | 
					
						
							|  |  |  |  |                         const char *argv[], void *aux OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     enum dummy_netdev_conn_state state = CONN_STATE_UNKNOWN; | 
					
						
							|  |  |  |  |     struct ds s; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ds_init(&s); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (argc > 1) { | 
					
						
							|  |  |  |  |         const char *dev_name = argv[1]; | 
					
						
							|  |  |  |  |         struct netdev *netdev = netdev_from_name(dev_name); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (netdev && is_dummy_class(netdev->netdev_class)) { | 
					
						
							|  |  |  |  |             struct netdev_dummy *dummy_dev = netdev_dummy_cast(netdev); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             ovs_mutex_lock(&dummy_dev->mutex); | 
					
						
							|  |  |  |  |             state = dummy_netdev_get_conn_state(&dummy_dev->conn); | 
					
						
							|  |  |  |  |             ovs_mutex_unlock(&dummy_dev->mutex); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             netdev_close(netdev); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         display_conn_state__(&s, dev_name, state); | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         struct netdev_dummy *netdev; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ovs_mutex_lock(&dummy_list_mutex); | 
					
						
							|  |  |  |  |         LIST_FOR_EACH (netdev, list_node, &dummy_list) { | 
					
						
							|  |  |  |  |             ovs_mutex_lock(&netdev->mutex); | 
					
						
							|  |  |  |  |             state = dummy_netdev_get_conn_state(&netdev->conn); | 
					
						
							|  |  |  |  |             ovs_mutex_unlock(&netdev->mutex); | 
					
						
							|  |  |  |  |             if (state != CONN_STATE_UNKNOWN) { | 
					
						
							|  |  |  |  |                 display_conn_state__(&s, netdev->up.name, state); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         ovs_mutex_unlock(&dummy_list_mutex); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     unixctl_command_reply(conn, ds_cstr(&s)); | 
					
						
							|  |  |  |  |     ds_destroy(&s); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_ip4addr(struct unixctl_conn *conn, int argc OVS_UNUSED, | 
					
						
							|  |  |  |  |                      const char *argv[], void *aux OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev *netdev = netdev_from_name(argv[1]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (netdev && is_dummy_class(netdev->netdev_class)) { | 
					
						
							| 
									
										
										
										
											2015-12-15 18:04:20 -08:00
										 |  |  |  |         struct in_addr ip, mask; | 
					
						
							|  |  |  |  |         char *error; | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 18:04:20 -08:00
										 |  |  |  |         error = ip_parse_masked(argv[2], &ip.s_addr, &mask.s_addr); | 
					
						
							|  |  |  |  |         if (!error) { | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |             netdev_dummy_set_in4(netdev, ip, mask); | 
					
						
							|  |  |  |  |             unixctl_command_reply(conn, "OK"); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2015-12-15 18:04:20 -08:00
										 |  |  |  |             unixctl_command_reply_error(conn, error); | 
					
						
							|  |  |  |  |             free(error); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         unixctl_command_reply_error(conn, "Unknown Dummy Interface"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     netdev_close(netdev); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_ip6addr(struct unixctl_conn *conn, int argc OVS_UNUSED, | 
					
						
							|  |  |  |  |                      const char *argv[], void *aux OVS_UNUSED) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct netdev *netdev = netdev_from_name(argv[1]); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |     if (netdev && is_dummy_class(netdev->netdev_class)) { | 
					
						
							|  |  |  |  |         struct in6_addr ip6; | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |         char *error; | 
					
						
							|  |  |  |  |         uint32_t plen; | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |         error = ipv6_parse_cidr(argv[2], &ip6, &plen); | 
					
						
							|  |  |  |  |         if (!error) { | 
					
						
							|  |  |  |  |             struct in6_addr mask; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             mask = ipv6_create_mask(plen); | 
					
						
							|  |  |  |  |             netdev_dummy_set_in6(netdev, &ip6, &mask); | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |             unixctl_command_reply(conn, "OK"); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2016-03-24 09:30:57 -07:00
										 |  |  |  |             unixctl_command_reply_error(conn, error); | 
					
						
							|  |  |  |  |             free(error); | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     } else { | 
					
						
							|  |  |  |  |         unixctl_command_reply_error(conn, "Unknown Dummy Interface"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |     netdev_close(netdev); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 16:58:49 -07:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | netdev_dummy_override(const char *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (!netdev_unregister_provider(type)) { | 
					
						
							|  |  |  |  |         struct netdev_class *class; | 
					
						
							|  |  |  |  |         int error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         class = xmemdup(&dummy_class, sizeof dummy_class); | 
					
						
							|  |  |  |  |         class->type = xstrdup(type); | 
					
						
							|  |  |  |  |         error = netdev_register_provider(class); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             VLOG_ERR("%s: failed to register netdev provider (%s)", | 
					
						
							|  |  |  |  |                      type, ovs_strerror(error)); | 
					
						
							|  |  |  |  |             free(CONST_CAST(char *, class->type)); | 
					
						
							|  |  |  |  |             free(class); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-06-13 16:58:49 -07:00
										 |  |  |  | netdev_dummy_register(enum dummy_level level) | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:53 +03:00
										 |  |  |  |     unixctl_command_register("netdev-dummy/receive", | 
					
						
							| 
									
										
										
										
											2017-01-18 02:30:26 -08:00
										 |  |  |  |                              "name [--qid queue_id] packet|flow [--len packet_len]", | 
					
						
							| 
									
										
										
										
											2011-12-06 14:22:27 -08:00
										 |  |  |  |                              2, INT_MAX, netdev_dummy_receive, NULL); | 
					
						
							| 
									
										
										
										
											2012-08-09 18:08:40 -07:00
										 |  |  |  |     unixctl_command_register("netdev-dummy/set-admin-state", | 
					
						
							|  |  |  |  |                              "[netdev] up|down", 1, 2, | 
					
						
							|  |  |  |  |                              netdev_dummy_set_admin_state, NULL); | 
					
						
							| 
									
										
										
										
											2014-06-05 16:01:17 -07:00
										 |  |  |  |     unixctl_command_register("netdev-dummy/conn-state", | 
					
						
							|  |  |  |  |                              "[netdev]", 0, 1, | 
					
						
							|  |  |  |  |                              netdev_dummy_conn_state, NULL); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  |     unixctl_command_register("netdev-dummy/ip4addr", | 
					
						
							|  |  |  |  |                              "[netdev] ipaddr/mask-prefix-len", 2, 2, | 
					
						
							|  |  |  |  |                              netdev_dummy_ip4addr, NULL); | 
					
						
							| 
									
										
										
										
											2015-12-04 12:36:51 -02:00
										 |  |  |  |     unixctl_command_register("netdev-dummy/ip6addr", | 
					
						
							|  |  |  |  |                              "[netdev] ip6addr", 2, 2, | 
					
						
							|  |  |  |  |                              netdev_dummy_ip6addr, NULL); | 
					
						
							| 
									
										
										
										
											2014-11-11 11:53:47 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-13 16:58:49 -07:00
										 |  |  |  |     if (level == DUMMY_OVERRIDE_ALL) { | 
					
						
							| 
									
										
										
										
											2012-01-19 10:24:46 -08:00
										 |  |  |  |         struct sset types; | 
					
						
							|  |  |  |  |         const char *type; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         sset_init(&types); | 
					
						
							|  |  |  |  |         netdev_enumerate_types(&types); | 
					
						
							|  |  |  |  |         SSET_FOR_EACH (type, &types) { | 
					
						
							| 
									
										
										
										
											2015-06-13 16:58:49 -07:00
										 |  |  |  |             if (strcmp(type, "patch")) { | 
					
						
							|  |  |  |  |                 netdev_dummy_override(type); | 
					
						
							| 
									
										
										
										
											2012-01-19 10:24:46 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         sset_destroy(&types); | 
					
						
							| 
									
										
										
										
											2015-06-13 16:58:49 -07:00
										 |  |  |  |     } else if (level == DUMMY_OVERRIDE_SYSTEM) { | 
					
						
							|  |  |  |  |         netdev_dummy_override("system"); | 
					
						
							| 
									
										
										
										
											2012-01-19 10:24:46 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     netdev_register_provider(&dummy_class); | 
					
						
							| 
									
										
										
										
											2016-02-11 13:11:10 -08:00
										 |  |  |  |     netdev_register_provider(&dummy_internal_class); | 
					
						
							| 
									
										
										
										
											2016-05-27 16:32:48 +03:00
										 |  |  |  |     netdev_register_provider(&dummy_pmd_class); | 
					
						
							| 
									
										
										
										
											2013-01-25 13:30:40 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-18 08:27:20 -07:00
										 |  |  |  |     netdev_vport_tunnel_register(); | 
					
						
							| 
									
										
										
										
											2010-11-29 12:21:08 -08:00
										 |  |  |  | } |