| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:30:46 -08:00
										 |  |  |  * Copyright (c) 2015 Nicira, Inc. | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |  * Copyright (c) 2014 WindRiver, Inc. | 
					
						
							|  |  |  |  * Copyright (c) 2015 Avaya, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Implementation of Auto Attach.
 | 
					
						
							|  |  |  |  * Based on sample implementation in 802.1ab.  Above copyright and license | 
					
						
							|  |  |  |  * applies to all modifications. | 
					
						
							|  |  |  |  * Limitations: | 
					
						
							|  |  |  |  * - No support for multiple bridge. | 
					
						
							|  |  |  |  * - Auto Attach state machine not implemented. | 
					
						
							|  |  |  |  * - Auto Attach and LLDP code are bundled together.  The plan is to decoupled | 
					
						
							|  |  |  |  *   them. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <config.h>
 | 
					
						
							|  |  |  | #include "ovs-lldp.h"
 | 
					
						
							|  |  |  | #include <arpa/inet.h>
 | 
					
						
							|  |  |  | #include <inttypes.h>
 | 
					
						
							|  |  |  | #include <netinet/in.h>
 | 
					
						
							|  |  |  | #include <stdbool.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include "dynamic-string.h"
 | 
					
						
							|  |  |  | #include "flow.h"
 | 
					
						
							|  |  |  | #include "list.h"
 | 
					
						
							|  |  |  | #include "lldp/lldpd.h"
 | 
					
						
							|  |  |  | #include "lldp/lldpd-structs.h"
 | 
					
						
							|  |  |  | #include "netdev.h"
 | 
					
						
							|  |  |  | #include "openvswitch/types.h"
 | 
					
						
							|  |  |  | #include "packets.h"
 | 
					
						
							|  |  |  | #include "poll-loop.h"
 | 
					
						
							|  |  |  | #include "smap.h"
 | 
					
						
							|  |  |  | #include "unixctl.h"
 | 
					
						
							|  |  |  | #include "util.h"
 | 
					
						
							|  |  |  | #include "openvswitch/vlog.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VLOG_DEFINE_THIS_MODULE(ovs_lldp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define LLDP_PROTOCOL_ID        0x0000
 | 
					
						
							|  |  |  | #define LLDP_PROTOCOL_VERSION   0x00
 | 
					
						
							|  |  |  | #define LLDP_TYPE_CONFIG        0x00
 | 
					
						
							|  |  |  | #define LLDP_CHASSIS_TTL        120
 | 
					
						
							|  |  |  | #define ETH_TYPE_LLDP           0x88cc
 | 
					
						
							|  |  |  | #define MINIMUM_ETH_PACKET_SIZE 68
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define AA_STATUS_MULTIPLE \
 | 
					
						
							|  |  |  |     AA_STATUS(ACTIVE,2,Active) \ | 
					
						
							|  |  |  |     AA_STATUS(REJECT_GENERIC,3,Reject (Generic)) \ | 
					
						
							|  |  |  |     AA_STATUS(REJECT_AA_RES_NOTAVAIL,4,Reject (AA resources unavailable)) \ | 
					
						
							|  |  |  |     AA_STATUS(REJECT_INVALID,6,Reject (Invalid)) \ | 
					
						
							|  |  |  |     AA_STATUS(REJECT_VLAN_RES_UNAVAIL,8,Reject (VLAN resources unavailable)) \ | 
					
						
							|  |  |  |     AA_STATUS(REJECT_VLAN_APP_ISSUE,9,Reject (Application interaction issue)) \ | 
					
						
							|  |  |  |     AA_STATUS(PENDING,255,Pending) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum aa_status { | 
					
						
							|  |  |  | #define AA_STATUS(NAME, VALUE, STR) AA_STATUS_##NAME = VALUE,
 | 
					
						
							|  |  |  |     AA_STATUS_MULTIPLE | 
					
						
							|  |  |  | #undef AA_STATUS
 | 
					
						
							|  |  |  |     AA_STATUS_N_MULTIPLE | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Internal structure for an Auto Attach mapping.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct aa_mapping_internal { | 
					
						
							|  |  |  |     struct hmap_node hmap_node_isid; | 
					
						
							|  |  |  |     struct hmap_node hmap_node_aux; | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |     uint32_t         isid; | 
					
						
							|  |  |  |     uint16_t         vlan; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     void             *aux; | 
					
						
							|  |  |  |     enum aa_status   status; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct ovs_mutex mutex = OVS_MUTEX_INITIALIZER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Hash map of all LLDP instances keyed by name (port at the moment).
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct hmap all_lldps__ = HMAP_INITIALIZER(&all_lldps__); | 
					
						
							|  |  |  | static struct hmap *const all_lldps OVS_GUARDED_BY(mutex) = &all_lldps__; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Hash map of all the Auto Attach mappings.  Global at the moment (but will
 | 
					
						
							|  |  |  |  * be per bridge).  Used when adding a new port to a bridge so that we can | 
					
						
							|  |  |  |  * properly install all the configured mapping on the port and export them | 
					
						
							|  |  |  |  * To the Auto Attach server via LLDP. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct hmap all_mappings__ = HMAP_INITIALIZER(&all_mappings__); | 
					
						
							|  |  |  | static struct hmap *const all_mappings OVS_GUARDED_BY(mutex) = &all_mappings__; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct lldp_aa_element_system_id system_id_null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Convert an LLDP chassis ID to a string.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | chassisid_to_string(uint8_t *array, size_t len, char **str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *str = xmalloc(len * 3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < len; i++) { | 
					
						
							|  |  |  |         snprintf(&(*str)[i * 3], 4, "%02x:", array[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     (*str)[(i * 3) - 1] = '\0'; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Find an Auto Attach mapping keyed by I-SID.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct aa_mapping_internal * | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  | mapping_find_by_isid(struct lldp *lldp, uint32_t isid) | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct aa_mapping_internal *m; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |     HMAP_FOR_EACH_IN_BUCKET (m, hmap_node_isid, hash_int(isid, 0), | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                              &lldp->mappings_by_isid) { | 
					
						
							|  |  |  |         if (isid == m->isid) { | 
					
						
							|  |  |  |             return m; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Find an Auto Attach mapping keyed by aux.  aux is an opaque pointer created
 | 
					
						
							|  |  |  |  * by the bridge that refers to an OVSDB mapping record. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct aa_mapping_internal * | 
					
						
							|  |  |  | mapping_find_by_aux(struct lldp *lldp, const void *aux) OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct aa_mapping_internal *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH_IN_BUCKET (m, hmap_node_aux, hash_pointer(aux, 0), | 
					
						
							|  |  |  |                              &lldp->mappings_by_aux) { | 
					
						
							|  |  |  |         if (aux == m->aux) { | 
					
						
							|  |  |  |             return m; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Convert an Auto Attach request status to a string.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | aa_status_to_str(uint8_t status) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (status) { | 
					
						
							|  |  |  | #define AA_STATUS(NAME, VALUE, STR) case AA_STATUS_##NAME: return #STR;
 | 
					
						
							|  |  |  |         AA_STATUS_MULTIPLE | 
					
						
							|  |  |  | #undef AA_STATUS
 | 
					
						
							|  |  |  |         default: return "Undefined"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Display LLDP and Auto Attach statistics.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_print_lldp_and_aa_stats(struct ds *ds, struct lldp *lldp) | 
					
						
							|  |  |  |     OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_hardware *hw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ds_put_format(ds, "Statistics: %s\n", lldp->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!lldp->lldpd) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |     LIST_FOR_EACH (hw, h_entries, &lldp->lldpd->g_hardware) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         ds_put_format(ds, "\ttx cnt: %"PRIu64"\n", hw->h_tx_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\trx cnt: %"PRIu64"\n", hw->h_rx_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\trx discarded cnt: %"PRIu64"\n", | 
					
						
							|  |  |  |                       hw->h_rx_discarded_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\trx unrecognized cnt: %"PRIu64"\n", | 
					
						
							|  |  |  |                       hw->h_rx_unrecognized_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\tageout cnt: %"PRIu64"\n", hw->h_ageout_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\tinsert cnt: %"PRIu64"\n", hw->h_insert_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\tdelete cnt: %"PRIu64"\n", hw->h_delete_cnt); | 
					
						
							|  |  |  |         ds_put_format(ds, "\tdrop cnt: %"PRIu64"\n", hw->h_drop_cnt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_print_element_status_port(struct ds *ds, struct lldpd_hardware *hw) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_port *port; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:39:54 -08:00
										 |  |  |     LIST_FOR_EACH (port, p_entries, &hw->h_rports) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         if (memcmp(&port->p_element.system_id, | 
					
						
							|  |  |  |                    &system_id_null, | 
					
						
							|  |  |  |                    sizeof port->p_element.system_id)) { | 
					
						
							| 
									
										
										
										
											2015-04-16 08:52:29 -07:00
										 |  |  |             const char *none_str = "<None>"; | 
					
						
							|  |  |  |             const char *descr = NULL; | 
					
						
							|  |  |  |             char *id = NULL; | 
					
						
							|  |  |  |             char *system; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (port->p_chassis) { | 
					
						
							|  |  |  |                 if (port->p_chassis->c_id_len > 0) { | 
					
						
							| 
									
										
										
										
											2015-02-22 14:41:18 -08:00
										 |  |  |                     chassisid_to_string(port->p_chassis->c_id, | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                                         port->p_chassis->c_id_len, &id); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 08:52:29 -07:00
										 |  |  |                 descr = port->p_chassis->c_descr; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             chassisid_to_string((uint8_t *) &port->p_element.system_id, | 
					
						
							|  |  |  |                 sizeof port->p_element.system_id, &system); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 08:52:29 -07:00
										 |  |  |             ds_put_format(ds, "\tAuto Attach Primary Server Id: %s\n", | 
					
						
							|  |  |  |                           id ? id : none_str); | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |             ds_put_format(ds, "\tAuto Attach Primary Server Descr: %s\n", | 
					
						
							| 
									
										
										
										
											2015-04-16 08:52:29 -07:00
										 |  |  |                           descr ? descr : none_str); | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |             ds_put_format(ds, "\tAuto Attach Primary Server System Id: %s\n", | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                           system); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             free(id); | 
					
						
							|  |  |  |             free(system); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Auto Attach server broadcast an LLDP message periodically.  Display
 | 
					
						
							|  |  |  |  * the discovered server. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_print_element_status(struct ds *ds, struct lldp *lldp) OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_hardware *hw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ds_put_format(ds, "LLDP: %s\n", lldp->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!lldp->lldpd) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |     LIST_FOR_EACH (hw, h_entries, &lldp->lldpd->g_hardware) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         aa_print_element_status_port(ds, hw); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_print_isid_status_port_isid(struct lldp *lldp, struct lldpd_port *port) | 
					
						
							|  |  |  |     OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_aa_isid_vlan_maps_tlv *mapping; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:35:00 -08:00
										 |  |  |     if (list_is_empty(&port->p_isid_vlan_maps)) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:35:00 -08:00
										 |  |  |     LIST_FOR_EACH (mapping, m_entries, &port->p_isid_vlan_maps) { | 
					
						
							| 
									
										
										
										
											2015-03-03 15:32:59 -08:00
										 |  |  |         uint32_t isid = mapping->isid_vlan_data.isid; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         struct aa_mapping_internal *m = mapping_find_by_isid(lldp, isid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         VLOG_INFO("h_rport: isid=%u, vlan=%u, status=%d", | 
					
						
							|  |  |  |                   isid, | 
					
						
							|  |  |  |                   mapping->isid_vlan_data.vlan, | 
					
						
							|  |  |  |                   mapping->isid_vlan_data.status); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 12:37:22 -08:00
										 |  |  |         /* Update the status of our internal state for the mapping. */ | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         if (m) { | 
					
						
							| 
									
										
										
										
											2015-02-22 12:37:22 -08:00
										 |  |  |             VLOG_INFO("Setting status for ISID=%"PRIu32" to %"PRIu16, | 
					
						
							|  |  |  |                       isid, mapping->isid_vlan_data.status); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |             m->status = mapping->isid_vlan_data.status; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2015-02-22 12:37:22 -08:00
										 |  |  |             VLOG_WARN("Couldn't find mapping for I-SID=%"PRIu32, isid); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_print_isid_status_port(struct lldp *lldp, struct lldpd_hardware *hw) | 
					
						
							|  |  |  |     OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_port *port; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:39:54 -08:00
										 |  |  |     LIST_FOR_EACH (port, p_entries, &hw->h_rports) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         aa_print_isid_status_port_isid(lldp, port); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* The Auto Attach server will broadcast the status of the configured mappings
 | 
					
						
							|  |  |  |  * via LLDP.  Display the status. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_print_isid_status(struct ds *ds, struct lldp *lldp) OVS_REQUIRES(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_hardware *hw; | 
					
						
							|  |  |  |     struct aa_mapping_internal *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!lldp->lldpd) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ds_put_format(ds, "LLDP: %s\n", lldp->name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |     LIST_FOR_EACH (hw, h_entries, &lldp->lldpd->g_hardware) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         aa_print_isid_status_port(lldp, hw); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ds_put_format(ds, "%-8s %-4s %-11s %-8s\n", | 
					
						
							|  |  |  |                       "I-SID", | 
					
						
							|  |  |  |                       "VLAN", | 
					
						
							|  |  |  |                       "Source", | 
					
						
							|  |  |  |                       "Status"); | 
					
						
							|  |  |  |     ds_put_format(ds, "-------- ---- ----------- --------\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH (m, hmap_node_isid, &lldp->mappings_by_isid) { | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |         ds_put_format(ds, "%-8"PRIu32" %-4"PRIu16" %-11s %-11s\n", | 
					
						
							|  |  |  |                       m->isid, m->vlan, "Switch", aa_status_to_str(m->status)); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_unixctl_status(struct unixctl_conn *conn, int argc OVS_UNUSED, | 
					
						
							|  |  |  |                   const char *argv[] OVS_UNUSED, void *aux OVS_UNUSED) | 
					
						
							|  |  |  |     OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  |     struct ds ds = DS_EMPTY_INITIALIZER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         aa_print_element_status(&ds, lldp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     unixctl_command_reply(conn, ds_cstr(&ds)); | 
					
						
							|  |  |  |     ds_destroy(&ds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_unixctl_show_isid(struct unixctl_conn *conn, int argc OVS_UNUSED, | 
					
						
							|  |  |  |                      const char *argv[] OVS_UNUSED, void *aux OVS_UNUSED) | 
					
						
							|  |  |  |     OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  |     struct ds ds = DS_EMPTY_INITIALIZER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         aa_print_isid_status(&ds, lldp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     unixctl_command_reply(conn, ds_cstr(&ds)); | 
					
						
							|  |  |  |     ds_destroy(&ds); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_unixctl_statistics(struct unixctl_conn *conn, int argc OVS_UNUSED, | 
					
						
							|  |  |  |                       const char *argv[] OVS_UNUSED, void *aux OVS_UNUSED) | 
					
						
							|  |  |  |     OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct ds ds = DS_EMPTY_INITIALIZER; | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Cycle through all ports and dump the stats for each one */ | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         aa_print_lldp_and_aa_stats(&ds, lldp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unixctl_command_reply(conn, ds_cstr(&ds)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* An Auto Attach mapping was configured.  Populate the corresponding
 | 
					
						
							|  |  |  |  * structures in the LLDP hardware. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | update_mapping_on_lldp(struct lldp *lldp, struct lldpd_hardware *hardware, | 
					
						
							|  |  |  |                        struct aa_mapping_internal *m) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_aa_isid_vlan_maps_tlv *lm = xzalloc(sizeof *lm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hardware->h_ifname) { | 
					
						
							|  |  |  |         VLOG_INFO("\t\t hardware->h_ifname=%s", hardware->h_ifname); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:32:59 -08:00
										 |  |  |     lm->isid_vlan_data.isid = m->isid; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     lm->isid_vlan_data.vlan = m->vlan; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:35:00 -08:00
										 |  |  |     list_push_back(&hardware->h_lport.p_isid_vlan_maps, &lm->m_entries); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* TODO Should be done in the Auto Attach state machine when a mapping goes
 | 
					
						
							|  |  |  |      * from "pending" to "active". | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |     struct bridge_aa_vlan *node = xmalloc(sizeof *node); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |     node->port_name = xstrdup(hardware->h_ifname); | 
					
						
							|  |  |  |     node->vlan = m->vlan; | 
					
						
							|  |  |  |     node->oper = BRIDGE_AA_VLAN_OPER_ADD; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |     list_push_back(&lldp->active_mapping_queue, &node->list_node); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Bridge will poll the list of VLAN that needs to be auto configure based on
 | 
					
						
							|  |  |  |  * the Auto Attach mappings that have been exchanged with the server. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | aa_get_vlan_queued(struct ovs_list *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							| 
									
										
										
										
											2015-04-06 14:02:28 -07:00
										 |  |  |         struct bridge_aa_vlan *node; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-06 14:02:28 -07:00
										 |  |  |         LIST_FOR_EACH_POP (node, list_node, &lldp->active_mapping_queue) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |             struct bridge_aa_vlan *copy; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             copy = xmalloc(sizeof *copy); | 
					
						
							|  |  |  |             copy->port_name = xstrdup(node->port_name); | 
					
						
							|  |  |  |             copy->vlan = node->vlan; | 
					
						
							|  |  |  |             copy->oper = node->oper; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             list_push_back(list, ©->list_node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* Cleanup */ | 
					
						
							|  |  |  |             free(node->port_name); | 
					
						
							|  |  |  |             free(node); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Bridge will poll whether or not VLAN have been auto-configured.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | unsigned int | 
					
						
							|  |  |  | aa_get_vlan_queue_size(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  |     unsigned int size = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         size += list_size(&lldp->active_mapping_queue); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Configure Auto Attach.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | aa_configure(const struct aa_settings *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* TODO Change all instances for now */ | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         struct lldpd_chassis *chassis; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 16:47:01 -08:00
										 |  |  |         LIST_FOR_EACH (chassis, list, &lldp->lldpd->g_chassis) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |             /* System Description */ | 
					
						
							| 
									
										
										
										
											2015-02-22 16:13:14 -08:00
										 |  |  |             free(chassis->c_descr); | 
					
						
							| 
									
										
										
										
											2015-04-06 10:59:19 -04:00
										 |  |  |             chassis->c_descr = s && s->system_description[0] ? | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                 xstrdup(s->system_description) : xstrdup(PACKAGE_STRING); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* System Name */ | 
					
						
							| 
									
										
										
										
											2015-04-06 10:59:19 -04:00
										 |  |  |             if (s) { | 
					
						
							|  |  |  |                 free(chassis->c_name); | 
					
						
							|  |  |  |                 chassis->c_name = xstrdup(s->system_name); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Add a new Auto Attach mapping.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | aa_mapping_register(void *aux, const struct aa_mapping_settings *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct aa_mapping_internal *bridge_m; | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |     VLOG_INFO("Adding mapping ISID=%"PRIu32", VLAN=%"PRIu16", aux=%p", | 
					
						
							|  |  |  |               s->isid, s->vlan, aux); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* TODO These mappings should be stores per bridge.  This is used
 | 
					
						
							|  |  |  |      * When a port is added.  Auto Attach mappings need to be added on this | 
					
						
							|  |  |  |      * port. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     bridge_m = xzalloc(sizeof *bridge_m); | 
					
						
							|  |  |  |     bridge_m->isid = s->isid; | 
					
						
							|  |  |  |     bridge_m->vlan = s->vlan; | 
					
						
							|  |  |  |     bridge_m->aux = aux; | 
					
						
							|  |  |  |     bridge_m->status = AA_STATUS_PENDING; | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |     hmap_insert(all_mappings, &bridge_m->hmap_node_isid, | 
					
						
							|  |  |  |                 hash_int(bridge_m->isid, 0)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Update mapping on the all the LLDP instances. */ | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         struct lldpd_hardware *hw; | 
					
						
							|  |  |  |         struct aa_mapping_internal *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         VLOG_INFO("\t lldp->name=%s", lldp->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (mapping_find_by_isid(lldp, s->isid)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m = xzalloc(sizeof *m); | 
					
						
							|  |  |  |         m->isid = s->isid; | 
					
						
							|  |  |  |         m->vlan = s->vlan; | 
					
						
							|  |  |  |         m->status = AA_STATUS_PENDING; | 
					
						
							|  |  |  |         m->aux = aux; | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |         hmap_insert(&lldp->mappings_by_isid, &m->hmap_node_isid, | 
					
						
							|  |  |  |                     hash_int(m->isid, 0)); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         hmap_insert(&lldp->mappings_by_aux, | 
					
						
							|  |  |  |                     &m->hmap_node_aux, | 
					
						
							|  |  |  |                     hash_pointer(m->aux, 0)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Configure the mapping on each port of the LLDP stack. */ | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |         LIST_FOR_EACH (hw, h_entries, &lldp->lldpd->g_hardware) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |             update_mapping_on_lldp(lldp, hw, m); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | aa_mapping_unregister_mapping(struct lldp *lldp, | 
					
						
							|  |  |  |                               struct lldpd_hardware *hw, | 
					
						
							|  |  |  |                               struct aa_mapping_internal *m) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd_aa_isid_vlan_maps_tlv *lm, *lm_next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:35:00 -08:00
										 |  |  |     LIST_FOR_EACH_SAFE (lm, lm_next, m_entries, | 
					
						
							|  |  |  |                         &hw->h_lport.p_isid_vlan_maps) { | 
					
						
							| 
									
										
										
										
											2015-03-03 15:32:59 -08:00
										 |  |  |         uint32_t isid = lm->isid_vlan_data.isid; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |         if (isid == m->isid) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |             VLOG_INFO("\t\t Removing lport, isid=%u, vlan=%u", | 
					
						
							|  |  |  |                       isid, | 
					
						
							|  |  |  |                       lm->isid_vlan_data.vlan); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             list_remove(&lm->m_entries); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* TODO Should be done in the AA SM when a mapping goes
 | 
					
						
							|  |  |  |              * from "pending" to "active". | 
					
						
							|  |  |  |              */ | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |             struct bridge_aa_vlan *node = xmalloc(sizeof *node); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |             node->port_name = xstrdup(hw->h_ifname); | 
					
						
							|  |  |  |             node->vlan = m->vlan; | 
					
						
							|  |  |  |             node->oper = BRIDGE_AA_VLAN_OPER_REMOVE; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 15:59:21 -08:00
										 |  |  |             list_push_back(&lldp->active_mapping_queue, &node->list_node); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Remove an existing Auto Attach mapping.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | aa_mapping_unregister(void *aux) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VLOG_INFO("Removing mapping aux=%p", aux); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HMAP_FOR_EACH (lldp, hmap_node, all_lldps) { | 
					
						
							|  |  |  |         struct lldpd_hardware *hw; | 
					
						
							|  |  |  |         struct aa_mapping_internal *m = mapping_find_by_aux(lldp, aux); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Remove from internal hash tables. */ | 
					
						
							|  |  |  |         if (m) { | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |             uint32_t isid = m->isid; | 
					
						
							|  |  |  |             uint16_t vlan = m->vlan; | 
					
						
							|  |  |  |             struct aa_mapping_internal *p = mapping_find_by_isid(lldp, isid); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |             VLOG_INFO("\t Removing mapping ISID=%"PRIu32", VLAN=%"PRIu16 | 
					
						
							|  |  |  |                       " (lldp->name=%s)", isid, vlan, lldp->name); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (p) { | 
					
						
							|  |  |  |                 hmap_remove(&lldp->mappings_by_isid, &p->hmap_node_isid); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             hmap_remove(&lldp->mappings_by_aux, &m->hmap_node_aux); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* Remove from all the lldp instances */ | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |             LIST_FOR_EACH (hw, h_entries, &lldp->lldpd->g_hardware) { | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                 if (hw->h_ifname) { | 
					
						
							|  |  |  |                     VLOG_INFO("\t\t hardware->h_ifname=%s", hw->h_ifname); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 aa_mapping_unregister_mapping(lldp, hw, m); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-04-16 14:46:27 -07:00
										 |  |  |             free(m); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |             /* Remove from the all_mappings */ | 
					
						
							|  |  |  |             HMAP_FOR_EACH (m, hmap_node_isid, all_mappings) { | 
					
						
							|  |  |  |                 if (m && isid == m->isid && vlan == m->vlan) { | 
					
						
							|  |  |  |                     hmap_remove(all_mappings, &m->hmap_node_isid); | 
					
						
							|  |  |  |                     break; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | lldp_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unixctl_command_register("autoattach/status", "[bridge]", 0, 1, | 
					
						
							|  |  |  |                              aa_unixctl_status, NULL); | 
					
						
							|  |  |  |     unixctl_command_register("autoattach/show-isid", "[bridge]", 0, 1, | 
					
						
							|  |  |  |                              aa_unixctl_show_isid, NULL); | 
					
						
							|  |  |  |     unixctl_command_register("autoattach/statistics", "[bridge]", 0, 1, | 
					
						
							|  |  |  |                              aa_unixctl_statistics, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Returns true if 'lldp' should process packets from 'flow'.  Sets
 | 
					
						
							|  |  |  |  * fields in 'wc' that were used to make the determination. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool | 
					
						
							|  |  |  | lldp_should_process_flow(const struct flow *flow) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (flow->dl_type == htons(ETH_TYPE_LLDP)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Process an LLDP packet that was received on a bridge port.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:10 -05:00
										 |  |  | lldp_process_packet(struct lldp *lldp, const struct dp_packet *p) | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | { | 
					
						
							|  |  |  |     if (lldp) { | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |         lldpd_recv(lldp->lldpd, lldpd_first_hardware(lldp->lldpd), | 
					
						
							| 
									
										
										
										
											2015-03-03 14:08:58 -08:00
										 |  |  |                    (char *) dp_packet_data(p), dp_packet_size(p)); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This code is called periodically to check if the LLDP module has an LLDP
 | 
					
						
							|  |  |  |  * message it wishes to send.  It is called several times every second. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool | 
					
						
							|  |  |  | lldp_should_send_packet(struct lldp *cfg) OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  |     ret = timer_expired(&cfg->tx_timer); | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Returns the next wake up time.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | long long int | 
					
						
							|  |  |  | lldp_wake_time(const struct lldp *lldp) OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     long long int retval; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!lldp) { | 
					
						
							|  |  |  |         return LLONG_MAX; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  |     retval = lldp->tx_timer.t; | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return retval; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Put the monitor thread to sleep until it's next wake time.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | long long int | 
					
						
							|  |  |  | lldp_wait(struct lldp *lldp) OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     long long int wake_time = lldp_wake_time(lldp); | 
					
						
							|  |  |  |     poll_timer_wait_until(wake_time); | 
					
						
							|  |  |  |     return wake_time; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Prepare the LLDP packet to be sent on a bridge port.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:10 -05:00
										 |  |  | lldp_put_packet(struct lldp *lldp, struct dp_packet *packet, | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |                 uint8_t eth_src[ETH_ADDR_LEN]) OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldpd *mylldpd = lldp->lldpd; | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |     struct lldpd_hardware *hw = lldpd_first_hardware(mylldpd); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     static const uint8_t eth_addr_lldp[6] = | 
					
						
							|  |  |  |         {0x01, 0x80, 0xC2, 0x00, 0x00, 0x0e}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     eth_compose(packet, eth_addr_lldp, eth_src, ETH_TYPE_LLDP, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 13:44:53 -07:00
										 |  |  |     lldpd_send(hw, packet); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     timer_set_duration(&lldp->tx_timer, lldp->lldpd->g_config.c_tx_interval); | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Configures the LLDP stack.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | bool | 
					
						
							|  |  |  | lldp_configure(struct lldp *lldp) OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (lldp) { | 
					
						
							|  |  |  |         ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  |         timer_set_expired(&lldp->tx_timer); | 
					
						
							|  |  |  |         timer_set_duration(&lldp->tx_timer, LLDP_DEFAULT_TRANSMIT_INTERVAL_MS); | 
					
						
							|  |  |  |         lldp->lldpd->g_config.c_tx_interval = | 
					
						
							|  |  |  |             LLDP_DEFAULT_TRANSMIT_INTERVAL_MS; | 
					
						
							|  |  |  |         ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Create an LLDP stack instance.  At the moment there is one per bridge port.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct lldp * | 
					
						
							|  |  |  | lldp_create(const struct netdev *netdev, | 
					
						
							|  |  |  |             const uint32_t mtu, | 
					
						
							|  |  |  |             const struct smap *cfg) OVS_EXCLUDED(mutex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  |     struct lldpd_chassis *lchassis; | 
					
						
							|  |  |  |     struct lldpd_hardware *hw; | 
					
						
							|  |  |  |     struct aa_mapping_internal *m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!cfg || !smap_get_bool(cfg, "enable", false)) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lldp = xzalloc(sizeof *lldp); | 
					
						
							|  |  |  |     lldp->name = xstrdup(netdev_get_name(netdev)); | 
					
						
							|  |  |  |     lldp->lldpd = xzalloc(sizeof *lldp->lldpd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hmap_init(&lldp->mappings_by_isid); | 
					
						
							|  |  |  |     hmap_init(&lldp->mappings_by_aux); | 
					
						
							|  |  |  |     list_init(&lldp->active_mapping_queue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lchassis = xzalloc(sizeof *lchassis); | 
					
						
							|  |  |  |     lchassis->c_cap_available = LLDP_CAP_BRIDGE; | 
					
						
							|  |  |  |     lchassis->c_cap_enabled = LLDP_CAP_BRIDGE; | 
					
						
							|  |  |  |     lchassis->c_id_subtype = LLDP_CHASSISID_SUBTYPE_LLADDR; | 
					
						
							|  |  |  |     lchassis->c_id_len = ETH_ADDR_LEN; | 
					
						
							|  |  |  |     lchassis->c_id = xmalloc(ETH_ADDR_LEN); | 
					
						
							| 
									
										
										
										
											2015-02-22 14:41:18 -08:00
										 |  |  |     netdev_get_etheraddr(netdev, lchassis->c_id); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:37:18 -08:00
										 |  |  |     list_init(&lchassis->c_mgmt); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     lchassis->c_ttl = lldp->lldpd->g_config.c_tx_interval * | 
					
						
							|  |  |  |                       lldp->lldpd->g_config.c_tx_hold; | 
					
						
							|  |  |  |     lchassis->c_ttl = LLDP_CHASSIS_TTL; | 
					
						
							|  |  |  |     lldpd_assign_cfg_to_protocols(lldp->lldpd); | 
					
						
							| 
									
										
										
										
											2015-02-22 16:47:01 -08:00
										 |  |  |     list_init(&lldp->lldpd->g_chassis); | 
					
						
							|  |  |  |     list_push_back(&lldp->lldpd->g_chassis, &lchassis->list); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if ((hw = lldpd_alloc_hardware(lldp->lldpd, | 
					
						
							|  |  |  |                                    (char *) netdev_get_name(netdev), | 
					
						
							|  |  |  |                                    0)) == NULL) { | 
					
						
							|  |  |  |         VLOG_WARN("Unable to allocate space for %s", | 
					
						
							|  |  |  |                   (char *) netdev_get_name(netdev)); | 
					
						
							|  |  |  |         out_of_memory(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_refcount_init(&lldp->ref_cnt); | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  |     hw->h_flags |= IFF_RUNNING; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     hw->h_mtu = mtu; | 
					
						
							|  |  |  |     hw->h_lport.p_id_subtype = LLDP_PORTID_SUBTYPE_IFNAME; | 
					
						
							|  |  |  |     hw->h_lport.p_id = xstrdup(netdev_get_name(netdev)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* p_id is not necessarily a null terminated string. */ | 
					
						
							|  |  |  |     hw->h_lport.p_id_len = strlen(netdev_get_name(netdev)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Auto Attach element tlv */ | 
					
						
							| 
									
										
										
										
											2015-03-18 14:47:14 -04:00
										 |  |  |     hw->h_lport.p_element.type = LLDP_TLV_AA_ELEM_TYPE_CLIENT_VIRTUAL_SWITCH; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     hw->h_lport.p_element.mgmt_vlan = 0; | 
					
						
							|  |  |  |     memcpy(&hw->h_lport.p_element.system_id.system_mac, | 
					
						
							|  |  |  |            lchassis->c_id, lchassis->c_id_len); | 
					
						
							|  |  |  |     hw->h_lport.p_element.system_id.conn_type = | 
					
						
							|  |  |  |         LLDP_TLV_AA_ELEM_CONN_TYPE_SINGLE; | 
					
						
							| 
									
										
										
										
											2015-03-18 14:47:14 -04:00
										 |  |  |     hw->h_lport.p_element.system_id.rsvd = 0; | 
					
						
							|  |  |  |     hw->h_lport.p_element.system_id.rsvd2[0] = 0; | 
					
						
							|  |  |  |     hw->h_lport.p_element.system_id.rsvd2[1] = 0; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:35:00 -08:00
										 |  |  |     list_init(&hw->h_lport.p_isid_vlan_maps); | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |     list_init(&lldp->lldpd->g_hardware); | 
					
						
							|  |  |  |     list_push_back(&lldp->lldpd->g_hardware, &hw->h_entries); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Update port with Auto Attach mappings configured. */ | 
					
						
							|  |  |  |     HMAP_FOR_EACH (m, hmap_node_isid, all_mappings) { | 
					
						
							|  |  |  |         struct aa_mapping_internal *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (mapping_find_by_isid(lldp, m->isid)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         p = xmemdup(m, sizeof *p); | 
					
						
							| 
									
										
										
										
											2015-03-03 15:36:28 -08:00
										 |  |  |         hmap_insert(&lldp->mappings_by_isid, &p->hmap_node_isid, | 
					
						
							|  |  |  |                     hash_int(p->isid, 0)); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |         hmap_insert(&lldp->mappings_by_aux, | 
					
						
							|  |  |  |                     &p->hmap_node_aux, | 
					
						
							|  |  |  |                     hash_pointer(p->aux, 0)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         update_mapping_on_lldp(lldp, hw, p); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hmap_insert(all_lldps, &lldp->hmap_node, | 
					
						
							|  |  |  |                 hash_string(netdev_get_name(netdev), 0)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return lldp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct lldp * | 
					
						
							|  |  |  | lldp_create_dummy(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp; | 
					
						
							|  |  |  |     struct lldpd_chassis *lchassis; | 
					
						
							|  |  |  |     struct lldpd_hardware *hw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lldp = xzalloc(sizeof *lldp); | 
					
						
							|  |  |  |     lldp->name = "dummy-lldp"; | 
					
						
							|  |  |  |     lldp->lldpd = xzalloc(sizeof *lldp->lldpd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hmap_init(&lldp->mappings_by_isid); | 
					
						
							|  |  |  |     hmap_init(&lldp->mappings_by_aux); | 
					
						
							|  |  |  |     list_init(&lldp->active_mapping_queue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lchassis = xzalloc(sizeof *lchassis); | 
					
						
							|  |  |  |     lchassis->c_cap_available = LLDP_CAP_BRIDGE; | 
					
						
							|  |  |  |     lchassis->c_cap_enabled = LLDP_CAP_BRIDGE; | 
					
						
							|  |  |  |     lchassis->c_id_subtype = LLDP_CHASSISID_SUBTYPE_LLADDR; | 
					
						
							|  |  |  |     lchassis->c_id_len = ETH_ADDR_LEN; | 
					
						
							|  |  |  |     lchassis->c_id = xmalloc(ETH_ADDR_LEN); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:37:18 -08:00
										 |  |  |     list_init(&lchassis->c_mgmt); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     lchassis->c_ttl = LLDP_CHASSIS_TTL; | 
					
						
							|  |  |  |     lldpd_assign_cfg_to_protocols(lldp->lldpd); | 
					
						
							| 
									
										
										
										
											2015-02-22 16:47:01 -08:00
										 |  |  |     list_init(&lldp->lldpd->g_chassis); | 
					
						
							|  |  |  |     list_push_back(&lldp->lldpd->g_chassis, &lchassis->list); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-03 15:30:46 -08:00
										 |  |  |     hw = lldpd_alloc_hardware(lldp->lldpd, "dummy-hw", 0); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ovs_refcount_init(&lldp->ref_cnt); | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  |     hw->h_flags |= IFF_RUNNING; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     hw->h_mtu = 1500; | 
					
						
							|  |  |  |     hw->h_lport.p_id_subtype = LLDP_PORTID_SUBTYPE_IFNAME; | 
					
						
							|  |  |  |     hw->h_lport.p_id = "dummy-port"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* p_id is not necessarily a null terminated string. */ | 
					
						
							|  |  |  |     hw->h_lport.p_id_len = strlen(hw->h_lport.p_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Auto Attach element tlv */ | 
					
						
							| 
									
										
										
										
											2015-03-18 14:47:14 -04:00
										 |  |  |     hw->h_lport.p_element.type = LLDP_TLV_AA_ELEM_TYPE_CLIENT_VIRTUAL_SWITCH; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  |     hw->h_lport.p_element.mgmt_vlan = 0; | 
					
						
							|  |  |  |     memcpy(&hw->h_lport.p_element.system_id.system_mac, | 
					
						
							|  |  |  |            lchassis->c_id, lchassis->c_id_len); | 
					
						
							|  |  |  |     hw->h_lport.p_element.system_id.conn_type = | 
					
						
							|  |  |  |         LLDP_TLV_AA_ELEM_CONN_TYPE_SINGLE; | 
					
						
							| 
									
										
										
										
											2015-03-18 14:47:14 -04:00
										 |  |  |     hw->h_lport.p_element.system_id.rsvd = 0; | 
					
						
							|  |  |  |     hw->h_lport.p_element.system_id.rsvd2[0] = 0; | 
					
						
							|  |  |  |     hw->h_lport.p_element.system_id.rsvd2[1] = 0; | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-22 13:35:00 -08:00
										 |  |  |     list_init(&hw->h_lport.p_isid_vlan_maps); | 
					
						
							| 
									
										
										
										
											2015-02-22 16:55:34 -08:00
										 |  |  |     list_init(&lldp->lldpd->g_hardware); | 
					
						
							|  |  |  |     list_push_back(&lldp->lldpd->g_hardware, &hw->h_entries); | 
					
						
							| 
									
										
										
										
											2015-02-20 14:17:09 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return lldp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Unreference a specific LLDP instance.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | lldp_unref(struct lldp *lldp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!lldp) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ovs_mutex_lock(&mutex); | 
					
						
							|  |  |  |     if (ovs_refcount_unref_relaxed(&lldp->ref_cnt) != 1) { | 
					
						
							|  |  |  |         ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hmap_remove(all_lldps, &lldp->hmap_node); | 
					
						
							|  |  |  |     ovs_mutex_unlock(&mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lldpd_cleanup(lldp->lldpd); | 
					
						
							|  |  |  |     free(lldp->lldpd); | 
					
						
							|  |  |  |     free(lldp->name); | 
					
						
							|  |  |  |     free(lldp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Unreference a specific LLDP instance.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct lldp * | 
					
						
							|  |  |  | lldp_ref(const struct lldp *lldp_) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct lldp *lldp = CONST_CAST(struct lldp *, lldp_); | 
					
						
							|  |  |  |     if (lldp) { | 
					
						
							|  |  |  |         ovs_refcount_ref(&lldp->ref_cnt); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return lldp; | 
					
						
							|  |  |  | } |