X-Git-Url: http://git.onelab.eu/?a=blobdiff_plain;f=vswitchd%2Fbridge.c;h=3faeee1e897869faacaa26f1f3d8df1e8f904f8c;hb=b2fda3effc787f265b5ad5dfa967ac00627bd075;hp=761d02ba36bdaaecfc144786d2a2f5c77c730bb1;hpb=8b05efc6b4a9aa77be588310e223f9826401208b;p=sliver-openvswitch.git diff --git a/vswitchd/bridge.c b/vswitchd/bridge.c index 761d02ba3..3faeee1e8 100644 --- a/vswitchd/bridge.c +++ b/vswitchd/bridge.c @@ -175,7 +175,8 @@ static struct port *port_lookup(const struct bridge *, const char *name); static void port_configure(struct port *); static struct lacp_settings *port_configure_lacp(struct port *, struct lacp_settings *); -static void port_configure_bond(struct port *, struct bond_settings *); +static void port_configure_bond(struct port *, struct bond_settings *, + uint32_t *bond_stable_ids); static void bridge_configure_mirrors(struct bridge *); static struct mirror *mirror_create(struct bridge *, @@ -521,10 +522,12 @@ port_configure(struct port *port) /* Get bond settings. */ if (s.n_slaves > 1) { - port_configure_bond(port, &bond_settings); s.bond = &bond_settings; + s.bond_stable_ids = xmalloc(s.n_slaves * sizeof *s.bond_stable_ids); + port_configure_bond(port, &bond_settings, s.bond_stable_ids); } else { s.bond = NULL; + s.bond_stable_ids = NULL; } /* Register. */ @@ -533,6 +536,7 @@ port_configure(struct port *port) /* Clean up. */ free(s.trunks); free(s.lacp_slaves); + free(s.bond_stable_ids); } /* Pick local port hardware address and datapath ID for 'br'. */ @@ -787,7 +791,7 @@ iface_set_ofp_port(struct iface *iface, int ofp_port) assert(iface->ofp_port < 0 && ofp_port >= 0); iface->ofp_port = ofp_port; hmap_insert(&br->ifaces, &iface->ofp_port_node, hash_int(ofp_port, 0)); - + iface_set_ofport(iface->cfg, ofp_port); } static void @@ -923,7 +927,6 @@ bridge_add_ofproto_ports(struct bridge *br) /* Already exists, nothing to do. */ ofproto_port_destroy(&ofproto_port); } - ofproto_port_destroy(&ofproto_port); } } } @@ -1392,7 +1395,7 @@ bridge_run(void) /* (Re)configure if necessary. */ database_changed = ovsdb_idl_run(idl); cfg = ovsrec_open_vswitch_first(idl); -#ifdef HAVE_OPENSSL + /* Re-configure SSL. We do this on every trip through the main loop, * instead of just when the database changes, because the contents of the * key and certificate files can change without the database changing. @@ -1405,7 +1408,7 @@ bridge_run(void) stream_ssl_set_key_and_cert(ssl->private_key, ssl->certificate); stream_ssl_set_ca_cert_file(ssl->ca_cert, ssl->bootstrap_ca_cert); } -#endif + if (database_changed || datapath_destroyed) { if (cfg) { struct ovsdb_idl_txn *txn = ovsdb_idl_txn_create(idl); @@ -1626,7 +1629,11 @@ bridge_create(const struct ovsrec_bridge *br_cfg) br->name = xstrdup(br_cfg->name); br->type = xstrdup(ofproto_normalize_type(br_cfg->datapath_type)); br->cfg = br_cfg; - eth_addr_nicira_random(br->default_ea); + + /* Derive the default Ethernet address from the bridge's UUID. This should + * be unique and it will be stable between ovs-vswitchd runs. */ + memcpy(br->default_ea, &br_cfg->header_.uuid, ETH_ADDR_LEN); + eth_addr_mark_random(br->default_ea); hmap_init(&br->ports); hmap_init(&br->ifaces); @@ -2021,7 +2028,10 @@ port_del_ifaces(struct port *port) sset_init(&new_ifaces); for (i = 0; i < port->cfg->n_interfaces; i++) { const char *name = port->cfg->interfaces[i]->name; - sset_add(&new_ifaces, name); + const char *type = port->cfg->interfaces[i]->name; + if (strcmp(type, "null")) { + sset_add(&new_ifaces, name); + } } /* Get rid of deleted interfaces. */ @@ -2047,7 +2057,8 @@ port_add_ifaces(struct port *port) shash_init(&new_ifaces); for (i = 0; i < port->cfg->n_interfaces; i++) { const struct ovsrec_interface *cfg = port->cfg->interfaces[i]; - if (!shash_add_once(&new_ifaces, cfg->name, cfg)) { + if (strcmp(cfg->type, "null") + && !shash_add_once(&new_ifaces, cfg->name, cfg)) { VLOG_WARN("port %s: %s specified twice as port interface", port->name, cfg->name); iface_set_ofport(cfg, -1); @@ -2159,9 +2170,10 @@ port_configure_lacp(struct port *port, struct lacp_settings *s) ? priority : UINT16_MAX - !list_is_short(&port->ifaces)); - s->strict = !strcmp(get_port_other_config(port->cfg, "lacp-strict", - "false"), - "true"); + s->heartbeat = !strcmp(get_port_other_config(port->cfg, + "lacp-heartbeat", + "false"), "true"); + lacp_time = get_port_other_config(port->cfg, "lacp-time", "slow"); custom_time = atoi(lacp_time); @@ -2182,11 +2194,13 @@ port_configure_lacp(struct port *port, struct lacp_settings *s) static void iface_configure_lacp(struct iface *iface, struct lacp_slave_settings *s) { - int priority, portid; + int priority, portid, key; portid = atoi(get_interface_other_config(iface->cfg, "lacp-port-id", "0")); priority = atoi(get_interface_other_config(iface->cfg, "lacp-port-priority", "0")); + key = atoi(get_interface_other_config(iface->cfg, "lacp-aggregation-key", + "0")); if (portid <= 0 || portid > UINT16_MAX) { portid = iface->ofp_port; @@ -2196,15 +2210,23 @@ iface_configure_lacp(struct iface *iface, struct lacp_slave_settings *s) priority = UINT16_MAX; } + if (key < 0 || key > UINT16_MAX) { + key = 0; + } + s->name = iface->name; s->id = portid; s->priority = priority; + s->key = key; } static void -port_configure_bond(struct port *port, struct bond_settings *s) +port_configure_bond(struct port *port, struct bond_settings *s, + uint32_t *bond_stable_ids) { const char *detect_s; + struct iface *iface; + size_t i; s->name = port->name; s->balance = BM_SLB; @@ -2231,6 +2253,7 @@ port_configure_bond(struct port *port, struct bond_settings *s) s->up_delay = MAX(0, port->cfg->bond_updelay); s->down_delay = MAX(0, port->cfg->bond_downdelay); + s->basis = atoi(get_port_other_config(port->cfg, "bond-hash-basis", "0")); s->rebalance_interval = atoi( get_port_other_config(port->cfg, "bond-rebalance-interval", "10000")); if (s->rebalance_interval < 1000) { @@ -2238,6 +2261,18 @@ port_configure_bond(struct port *port, struct bond_settings *s) } s->fake_iface = port->cfg->bond_fake_iface; + + i = 0; + LIST_FOR_EACH (iface, port_elem, &port->ifaces) { + long long stable_id; + + stable_id = atoll(get_interface_other_config(iface->cfg, + "bond-stable-id", "0")); + if (stable_id <= 0 || stable_id >= UINT32_MAX) { + stable_id = iface->ofp_port; + } + bond_stable_ids[i++] = stable_id; + } } /* Interface functions. */ @@ -2446,7 +2481,7 @@ iface_delete_queues(unsigned int queue_id, static void iface_configure_qos(struct iface *iface, const struct ovsrec_qos *qos) { - if (!qos || qos->type[0] == '\0') { + if (!qos || qos->type[0] == '\0' || qos->n_queues < 1) { netdev_set_qos(iface->netdev, NULL, NULL); } else { struct iface_delete_queues_cbdata cbdata;