/* $NetBSD: bridgestp.c,v 1.26 2018/02/28 13:12:53 ozaki-r Exp $ */ /* * Copyright (c) 2000 Jason L. Wright (jason@thought.net) * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by Jason L. Wright * 4. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * * OpenBSD: bridgestp.c,v 1.5 2001/03/22 03:48:29 jason Exp */ /* * Implementation of the spanning tree protocol as defined in * ISO/IEC Final DIS 15802-3 (IEEE P802.1D/D17), May 25, 1998. * (In English: IEEE 802.1D, Draft 17, 1998) */ #include __KERNEL_RCSID(0, "$NetBSD: bridgestp.c,v 1.26 2018/02/28 13:12:53 ozaki-r Exp $"); #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* BPDU message types */ #define BSTP_MSGTYPE_CFG 0x00 /* Configuration */ #define BSTP_MSGTYPE_TCN 0x80 /* Topology chg notification */ /* BPDU flags */ #define BSTP_FLAG_TC 0x01 /* Topology change */ #define BSTP_FLAG_TCA 0x80 /* Topology change ack */ #define BSTP_MESSAGE_AGE_INCR (1 * 256) /* in 256ths of a second */ #define BSTP_TICK_VAL (1 * 256) /* in 256ths of a second */ /* * Because BPDU's do not make nicely aligned structures, two different * declarations are used: bstp_?bpdu (wire representation, packed) and * bstp_*_unit (internal, nicely aligned version). */ /* configuration bridge protocol data unit */ struct bstp_cbpdu { uint8_t cbu_dsap; /* LLC: destination sap */ uint8_t cbu_ssap; /* LLC: source sap */ uint8_t cbu_ctl; /* LLC: control */ uint16_t cbu_protoid; /* protocol id */ uint8_t cbu_protover; /* protocol version */ uint8_t cbu_bpdutype; /* message type */ uint8_t cbu_flags; /* flags (below) */ /* root id */ uint16_t cbu_rootpri; /* root priority */ uint8_t cbu_rootaddr[6]; /* root address */ uint32_t cbu_rootpathcost; /* root path cost */ /* bridge id */ uint16_t cbu_bridgepri; /* bridge priority */ uint8_t cbu_bridgeaddr[6]; /* bridge address */ uint16_t cbu_portid; /* port id */ uint16_t cbu_messageage; /* current message age */ uint16_t cbu_maxage; /* maximum age */ uint16_t cbu_hellotime; /* hello time */ uint16_t cbu_forwarddelay; /* forwarding delay */ } __packed; /* topology change notification bridge protocol data unit */ struct bstp_tbpdu { uint8_t tbu_dsap; /* LLC: destination sap */ uint8_t tbu_ssap; /* LLC: source sap */ uint8_t tbu_ctl; /* LLC: control */ uint16_t tbu_protoid; /* protocol id */ uint8_t tbu_protover; /* protocol version */ uint8_t tbu_bpdutype; /* message type */ } __packed; const uint8_t bstp_etheraddr[] = { 0x01, 0x80, 0xc2, 0x00, 0x00, 0x00 }; static void bstp_initialize_port(struct bridge_softc *, struct bridge_iflist *); static void bstp_ifupdstatus(struct bridge_softc *, struct bridge_iflist *); static void bstp_enable_port(struct bridge_softc *, struct bridge_iflist *); static void bstp_disable_port(struct bridge_softc *, struct bridge_iflist *); static int bstp_root_bridge(struct bridge_softc *sc); static int bstp_supersedes_port_info(struct bridge_softc *, struct bridge_iflist *, struct bstp_config_unit *); static int bstp_designated_port(struct bridge_softc *, struct bridge_iflist *); static int bstp_designated_for_some_port(struct bridge_softc *); static void bstp_transmit_config(struct bridge_softc *, struct bridge_iflist *); static void bstp_transmit_tcn(struct bridge_softc *); static void bstp_received_config_bpdu(struct bridge_softc *, struct bridge_iflist *, struct bstp_config_unit *); static void bstp_received_tcn_bpdu(struct bridge_softc *, struct bridge_iflist *, struct bstp_tcn_unit *); static void bstp_record_config_information(struct bridge_softc *, struct bridge_iflist *, struct bstp_config_unit *); static void bstp_record_config_timeout_values(struct bridge_softc *, struct bstp_config_unit *); static void bstp_config_bpdu_generation(struct bridge_softc *); static void bstp_send_config_bpdu(struct bridge_softc *, struct bridge_iflist *, struct bstp_config_unit *); static void bstp_configuration_update(struct bridge_softc *); static void bstp_root_selection(struct bridge_softc *); static void bstp_designated_port_selection(struct bridge_softc *); static void bstp_become_designated_port(struct bridge_softc *, struct bridge_iflist *); static void bstp_port_state_selection(struct bridge_softc *); static void bstp_make_forwarding(struct bridge_softc *, struct bridge_iflist *); static void bstp_make_blocking(struct bridge_softc *, struct bridge_iflist *); static void bstp_set_port_state(struct bridge_iflist *, uint8_t); #if notused static void bstp_set_bridge_priority(struct bridge_softc *, uint64_t); static void bstp_set_port_priority(struct bridge_softc *, struct bridge_iflist *, uint16_t); static void bstp_set_path_cost(struct bridge_softc *, struct bridge_iflist *, uint32_t); #endif static void bstp_topology_change_detection(struct bridge_softc *); static void bstp_topology_change_acknowledged(struct bridge_softc *); static void bstp_acknowledge_topology_change(struct bridge_softc *, struct bridge_iflist *); static void bstp_tick(void *); static void bstp_timer_start(struct bridge_timer *, uint16_t); static void bstp_timer_stop(struct bridge_timer *); static int bstp_timer_expired(struct bridge_timer *, uint16_t); static void bstp_hold_timer_expiry(struct bridge_softc *, struct bridge_iflist *); static void bstp_message_age_timer_expiry(struct bridge_softc *, struct bridge_iflist *); static void bstp_forward_delay_timer_expiry(struct bridge_softc *, struct bridge_iflist *); static void bstp_topology_change_timer_expiry(struct bridge_softc *); static void bstp_tcn_timer_expiry(struct bridge_softc *); static void bstp_hello_timer_expiry(struct bridge_softc *); static void bstp_transmit_config(struct bridge_softc *sc, struct bridge_iflist *bif) { if (bif->bif_hold_timer.active) { bif->bif_config_pending = 1; return; } bif->bif_config_bpdu.cu_message_type = BSTP_MSGTYPE_CFG; bif->bif_config_bpdu.cu_rootid = sc->sc_designated_root; bif->bif_config_bpdu.cu_root_path_cost = sc->sc_root_path_cost; bif->bif_config_bpdu.cu_bridge_id = sc->sc_bridge_id; bif->bif_config_bpdu.cu_port_id = bif->bif_port_id; if (bstp_root_bridge(sc)) bif->bif_config_bpdu.cu_message_age = 0; else bif->bif_config_bpdu.cu_message_age = sc->sc_root_port->bif_message_age_timer.value + BSTP_MESSAGE_AGE_INCR; bif->bif_config_bpdu.cu_max_age = sc->sc_max_age; bif->bif_config_bpdu.cu_hello_time = sc->sc_hello_time; bif->bif_config_bpdu.cu_forward_delay = sc->sc_forward_delay; bif->bif_config_bpdu.cu_topology_change_acknowledgment = bif->bif_topology_change_acknowledge; bif->bif_config_bpdu.cu_topology_change = sc->sc_topology_change; if (bif->bif_config_bpdu.cu_message_age < sc->sc_max_age) { bif->bif_topology_change_acknowledge = 0; bif->bif_config_pending = 0; bstp_send_config_bpdu(sc, bif, &bif->bif_config_bpdu); bstp_timer_start(&bif->bif_hold_timer, 0); } } static void bstp_send_config_bpdu(struct bridge_softc *sc, struct bridge_iflist *bif, struct bstp_config_unit *cu) { struct ifnet *ifp; struct mbuf *m; struct ether_header *eh; struct bstp_cbpdu bpdu; KASSERT(BRIDGE_LOCKED(sc)); ifp = bif->bif_ifp; if ((ifp->if_flags & IFF_RUNNING) == 0) return; MGETHDR(m, M_DONTWAIT, MT_DATA); if (m == NULL) return; eh = mtod(m, struct ether_header *); m_set_rcvif(m, ifp); m->m_pkthdr.len = sizeof(*eh) + sizeof(bpdu); m->m_len = m->m_pkthdr.len; bpdu.cbu_ssap = bpdu.cbu_dsap = LLC_8021D_LSAP; bpdu.cbu_ctl = LLC_UI; bpdu.cbu_protoid = htons(0); bpdu.cbu_protover = 0; bpdu.cbu_bpdutype = cu->cu_message_type; bpdu.cbu_flags = (cu->cu_topology_change ? BSTP_FLAG_TC : 0) | (cu->cu_topology_change_acknowledgment ? BSTP_FLAG_TCA : 0); bpdu.cbu_rootpri = htons(cu->cu_rootid >> 48); bpdu.cbu_rootaddr[0] = cu->cu_rootid >> 40; bpdu.cbu_rootaddr[1] = cu->cu_rootid >> 32; bpdu.cbu_rootaddr[2] = cu->cu_rootid >> 24; bpdu.cbu_rootaddr[3] = cu->cu_rootid >> 16; bpdu.cbu_rootaddr[4] = cu->cu_rootid >> 8; bpdu.cbu_rootaddr[5] = cu->cu_rootid >> 0; bpdu.cbu_rootpathcost = htonl(cu->cu_root_path_cost); bpdu.cbu_bridgepri = htons(cu->cu_rootid >> 48); bpdu.cbu_bridgeaddr[0] = cu->cu_rootid >> 40; bpdu.cbu_bridgeaddr[1] = cu->cu_rootid >> 32; bpdu.cbu_bridgeaddr[2] = cu->cu_rootid >> 24; bpdu.cbu_bridgeaddr[3] = cu->cu_rootid >> 16; bpdu.cbu_bridgeaddr[4] = cu->cu_rootid >> 8; bpdu.cbu_bridgeaddr[5] = cu->cu_rootid >> 0; bpdu.cbu_portid = htons(cu->cu_port_id); bpdu.cbu_messageage = htons(cu->cu_message_age); bpdu.cbu_maxage = htons(cu->cu_max_age); bpdu.cbu_hellotime = htons(cu->cu_hello_time); bpdu.cbu_forwarddelay = htons(cu->cu_forward_delay); memcpy(eh->ether_shost, CLLADDR(ifp->if_sadl), ETHER_ADDR_LEN); memcpy(eh->ether_dhost, bstp_etheraddr, ETHER_ADDR_LEN); eh->ether_type = htons(sizeof(bpdu)); memcpy(mtod(m, char *) + sizeof(*eh), &bpdu, sizeof(bpdu)); BRIDGE_UNLOCK(sc); bridge_enqueue(sc, ifp, m, 0); BRIDGE_LOCK(sc); } static int bstp_root_bridge(struct bridge_softc *sc) { return (sc->sc_designated_root == sc->sc_bridge_id); } static int bstp_supersedes_port_info(struct bridge_softc *sc, struct bridge_iflist *bif, struct bstp_config_unit *cu) { if (cu->cu_rootid < bif->bif_designated_root) return (1); if (cu->cu_rootid > bif->bif_designated_root) return (0); if (cu->cu_root_path_cost < bif->bif_designated_cost) return (1); if (cu->cu_root_path_cost > bif->bif_designated_cost) return (0); if (cu->cu_bridge_id < bif->bif_designated_bridge) return (1); if (cu->cu_bridge_id > bif->bif_designated_bridge) return (0); if (sc->sc_bridge_id != cu->cu_bridge_id) return (1); if (cu->cu_port_id <= bif->bif_designated_port) return (1); return (0); } static void bstp_record_config_information(struct bridge_softc *sc, struct bridge_iflist *bif, struct bstp_config_unit *cu) { bif->bif_designated_root = cu->cu_rootid; bif->bif_designated_cost = cu->cu_root_path_cost; bif->bif_designated_bridge = cu->cu_bridge_id; bif->bif_designated_port = cu->cu_port_id; bstp_timer_start(&bif->bif_message_age_timer, cu->cu_message_age); } static void bstp_record_config_timeout_values(struct bridge_softc *sc, struct bstp_config_unit *config) { sc->sc_max_age = config->cu_max_age; sc->sc_hello_time = config->cu_hello_time; sc->sc_forward_delay = config->cu_forward_delay; sc->sc_topology_change = config->cu_topology_change; } static void bstp_config_bpdu_generation(struct bridge_softc *sc) { struct bridge_iflist *bif; BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bstp_designated_port(sc, bif) && (bif->bif_state != BSTP_IFSTATE_DISABLED)) bstp_transmit_config(sc, bif); } } static int bstp_designated_port(struct bridge_softc *sc, struct bridge_iflist *bif) { return ((bif->bif_designated_bridge == sc->sc_bridge_id) && (bif->bif_designated_port == bif->bif_port_id)); } static void bstp_transmit_tcn(struct bridge_softc *sc) { struct bstp_tbpdu bpdu; struct bridge_iflist *bif = sc->sc_root_port; struct ifnet *ifp; struct ether_header *eh; struct mbuf *m; KASSERT(BRIDGE_LOCKED(sc)); KASSERT(bif != NULL); ifp = bif->bif_ifp; if ((ifp->if_flags & IFF_RUNNING) == 0) return; MGETHDR(m, M_DONTWAIT, MT_DATA); if (m == NULL) return; m_set_rcvif(m, ifp); m->m_pkthdr.len = sizeof(*eh) + sizeof(bpdu); m->m_len = m->m_pkthdr.len; eh = mtod(m, struct ether_header *); memcpy(eh->ether_shost, CLLADDR(ifp->if_sadl), ETHER_ADDR_LEN); memcpy(eh->ether_dhost, bstp_etheraddr, ETHER_ADDR_LEN); eh->ether_type = htons(sizeof(bpdu)); bpdu.tbu_ssap = bpdu.tbu_dsap = LLC_8021D_LSAP; bpdu.tbu_ctl = LLC_UI; bpdu.tbu_protoid = 0; bpdu.tbu_protover = 0; bpdu.tbu_bpdutype = BSTP_MSGTYPE_TCN; memcpy(mtod(m, char *) + sizeof(*eh), &bpdu, sizeof(bpdu)); BRIDGE_UNLOCK(sc); bridge_enqueue(sc, ifp, m, 0); BRIDGE_LOCK(sc); } static void bstp_configuration_update(struct bridge_softc *sc) { bstp_root_selection(sc); bstp_designated_port_selection(sc); } static void bstp_root_selection(struct bridge_softc *sc) { struct bridge_iflist *root_port = NULL, *bif; BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bstp_designated_port(sc, bif)) continue; if (bif->bif_state == BSTP_IFSTATE_DISABLED) continue; if (bif->bif_designated_root >= sc->sc_bridge_id) continue; if (root_port == NULL) goto set_port; if (bif->bif_designated_root < root_port->bif_designated_root) goto set_port; if (bif->bif_designated_root > root_port->bif_designated_root) continue; if ((bif->bif_designated_cost + bif->bif_path_cost) < (root_port->bif_designated_cost + root_port->bif_path_cost)) goto set_port; if ((bif->bif_designated_cost + bif->bif_path_cost) > (root_port->bif_designated_cost + root_port->bif_path_cost)) continue; if (bif->bif_designated_bridge < root_port->bif_designated_bridge) goto set_port; if (bif->bif_designated_bridge > root_port->bif_designated_bridge) continue; if (bif->bif_designated_port < root_port->bif_designated_port) goto set_port; if (bif->bif_designated_port > root_port->bif_designated_port) continue; if (bif->bif_port_id >= root_port->bif_port_id) continue; set_port: root_port = bif; } sc->sc_root_port = root_port; if (root_port == NULL) { sc->sc_designated_root = sc->sc_bridge_id; sc->sc_root_path_cost = 0; } else { sc->sc_designated_root = root_port->bif_designated_root; sc->sc_root_path_cost = root_port->bif_designated_cost + root_port->bif_path_cost; } } static void bstp_designated_port_selection(struct bridge_softc *sc) { struct bridge_iflist *bif; BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bstp_designated_port(sc, bif)) goto designated; if (bif->bif_designated_root != sc->sc_designated_root) goto designated; if (sc->sc_root_path_cost < bif->bif_designated_cost) goto designated; if (sc->sc_root_path_cost > bif->bif_designated_cost) continue; if (sc->sc_bridge_id < bif->bif_designated_bridge) goto designated; if (sc->sc_bridge_id > bif->bif_designated_bridge) continue; if (bif->bif_port_id > bif->bif_designated_port) continue; designated: bstp_become_designated_port(sc, bif); } } static void bstp_become_designated_port(struct bridge_softc *sc, struct bridge_iflist *bif) { bif->bif_designated_root = sc->sc_designated_root; bif->bif_designated_cost = sc->sc_root_path_cost; bif->bif_designated_bridge = sc->sc_bridge_id; bif->bif_designated_port = bif->bif_port_id; } static void bstp_port_state_selection(struct bridge_softc *sc) { struct bridge_iflist *bif; BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bif == sc->sc_root_port) { bif->bif_config_pending = 0; bif->bif_topology_change_acknowledge = 0; bstp_make_forwarding(sc, bif); } else if (bstp_designated_port(sc, bif)) { bstp_timer_stop(&bif->bif_message_age_timer); bstp_make_forwarding(sc, bif); } else { bif->bif_config_pending = 0; bif->bif_topology_change_acknowledge = 0; bstp_make_blocking(sc, bif); } } } static void bstp_make_forwarding(struct bridge_softc *sc, struct bridge_iflist *bif) { if (bif->bif_state == BSTP_IFSTATE_BLOCKING) { bstp_set_port_state(bif, BSTP_IFSTATE_LISTENING); bstp_timer_start(&bif->bif_forward_delay_timer, 0); } } static void bstp_make_blocking(struct bridge_softc *sc, struct bridge_iflist *bif) { if ((bif->bif_state != BSTP_IFSTATE_DISABLED) && (bif->bif_state != BSTP_IFSTATE_BLOCKING)) { if ((bif->bif_state == BSTP_IFSTATE_FORWARDING) || (bif->bif_state == BSTP_IFSTATE_LEARNING)) { if (bif->bif_change_detection_enabled) { bstp_topology_change_detection(sc); } } bstp_set_port_state(bif, BSTP_IFSTATE_BLOCKING); bstp_timer_stop(&bif->bif_forward_delay_timer); } } static void bstp_set_port_state(struct bridge_iflist *bif, uint8_t state) { bif->bif_state = state; } static void bstp_topology_change_detection(struct bridge_softc *sc) { if (bstp_root_bridge(sc)) { sc->sc_topology_change = 1; bstp_timer_start(&sc->sc_topology_change_timer, 0); } else if (!sc->sc_topology_change_detected) { bstp_transmit_tcn(sc); bstp_timer_start(&sc->sc_tcn_timer, 0); } sc->sc_topology_change_detected = 1; } static void bstp_topology_change_acknowledged(struct bridge_softc *sc) { sc->sc_topology_change_detected = 0; bstp_timer_stop(&sc->sc_tcn_timer); } static void bstp_acknowledge_topology_change(struct bridge_softc *sc, struct bridge_iflist *bif) { bif->bif_topology_change_acknowledge = 1; bstp_transmit_config(sc, bif); } void bstp_input(struct bridge_softc *sc, struct bridge_iflist *bif, struct mbuf *m) { struct ether_header *eh; struct bstp_tbpdu tpdu; struct bstp_cbpdu cpdu; struct bstp_config_unit cu; struct bstp_tcn_unit tu; uint16_t len; eh = mtod(m, struct ether_header *); if ((bif->bif_flags & IFBIF_STP) == 0) goto out; len = ntohs(eh->ether_type); if (len < sizeof(tpdu)) goto out; m_adj(m, ETHER_HDR_LEN); if (m->m_pkthdr.len > len) m_adj(m, len - m->m_pkthdr.len); if (m->m_len < sizeof(tpdu) && (m = m_pullup(m, sizeof(tpdu))) == NULL) goto out; memcpy(&tpdu, mtod(m, void *), sizeof(tpdu)); if (tpdu.tbu_dsap != LLC_8021D_LSAP || tpdu.tbu_ssap != LLC_8021D_LSAP || tpdu.tbu_ctl != LLC_UI) goto out; if (tpdu.tbu_protoid != 0 || tpdu.tbu_protover != 0) goto out; switch (tpdu.tbu_bpdutype) { case BSTP_MSGTYPE_TCN: tu.tu_message_type = tpdu.tbu_bpdutype; BRIDGE_LOCK(sc); bstp_received_tcn_bpdu(sc, bif, &tu); BRIDGE_UNLOCK(sc); break; case BSTP_MSGTYPE_CFG: if (m->m_len < sizeof(cpdu) && (m = m_pullup(m, sizeof(cpdu))) == NULL) goto out; memcpy(&cpdu, mtod(m, void *), sizeof(cpdu)); cu.cu_rootid = (((uint64_t)ntohs(cpdu.cbu_rootpri)) << 48) | (((uint64_t)cpdu.cbu_rootaddr[0]) << 40) | (((uint64_t)cpdu.cbu_rootaddr[1]) << 32) | (((uint64_t)cpdu.cbu_rootaddr[2]) << 24) | (((uint64_t)cpdu.cbu_rootaddr[3]) << 16) | (((uint64_t)cpdu.cbu_rootaddr[4]) << 8) | (((uint64_t)cpdu.cbu_rootaddr[5]) << 0); cu.cu_bridge_id = (((uint64_t)ntohs(cpdu.cbu_bridgepri)) << 48) | (((uint64_t)cpdu.cbu_bridgeaddr[0]) << 40) | (((uint64_t)cpdu.cbu_bridgeaddr[1]) << 32) | (((uint64_t)cpdu.cbu_bridgeaddr[2]) << 24) | (((uint64_t)cpdu.cbu_bridgeaddr[3]) << 16) | (((uint64_t)cpdu.cbu_bridgeaddr[4]) << 8) | (((uint64_t)cpdu.cbu_bridgeaddr[5]) << 0); cu.cu_root_path_cost = ntohl(cpdu.cbu_rootpathcost); cu.cu_message_age = ntohs(cpdu.cbu_messageage); cu.cu_max_age = ntohs(cpdu.cbu_maxage); cu.cu_hello_time = ntohs(cpdu.cbu_hellotime); cu.cu_forward_delay = ntohs(cpdu.cbu_forwarddelay); cu.cu_port_id = ntohs(cpdu.cbu_portid); cu.cu_message_type = cpdu.cbu_bpdutype; cu.cu_topology_change_acknowledgment = (cpdu.cbu_flags & BSTP_FLAG_TCA) ? 1 : 0; cu.cu_topology_change = (cpdu.cbu_flags & BSTP_FLAG_TC) ? 1 : 0; BRIDGE_LOCK(sc); bstp_received_config_bpdu(sc, bif, &cu); BRIDGE_UNLOCK(sc); break; default: goto out; } out: if (m) m_freem(m); return; } static void bstp_received_config_bpdu(struct bridge_softc *sc, struct bridge_iflist *bif, struct bstp_config_unit *cu) { int root; root = bstp_root_bridge(sc); if (bif->bif_state != BSTP_IFSTATE_DISABLED) { if (bstp_supersedes_port_info(sc, bif, cu)) { bstp_record_config_information(sc, bif, cu); bstp_configuration_update(sc); bstp_port_state_selection(sc); if ((bstp_root_bridge(sc) == 0) && root) { bstp_timer_stop(&sc->sc_hello_timer); if (sc->sc_topology_change_detected) { bstp_timer_stop( &sc->sc_topology_change_timer); bstp_transmit_tcn(sc); bstp_timer_start(&sc->sc_tcn_timer, 0); } } if (bif == sc->sc_root_port) { bstp_record_config_timeout_values(sc, cu); bstp_config_bpdu_generation(sc); if (cu->cu_topology_change_acknowledgment) bstp_topology_change_acknowledged(sc); } } else if (bstp_designated_port(sc, bif)) bstp_transmit_config(sc, bif); } } static void bstp_received_tcn_bpdu(struct bridge_softc *sc, struct bridge_iflist *bif, struct bstp_tcn_unit *tcn) { if (bif->bif_state != BSTP_IFSTATE_DISABLED && bstp_designated_port(sc, bif)) { bstp_topology_change_detection(sc); bstp_acknowledge_topology_change(sc, bif); } } static void bstp_hello_timer_expiry(struct bridge_softc *sc) { bstp_config_bpdu_generation(sc); bstp_timer_start(&sc->sc_hello_timer, 0); } static void bstp_message_age_timer_expiry(struct bridge_softc *sc, struct bridge_iflist *bif) { int root; root = bstp_root_bridge(sc); bstp_become_designated_port(sc, bif); bstp_configuration_update(sc); bstp_port_state_selection(sc); if ((bstp_root_bridge(sc)) && (root == 0)) { sc->sc_max_age = sc->sc_bridge_max_age; sc->sc_hello_time = sc->sc_bridge_hello_time; sc->sc_forward_delay = sc->sc_bridge_forward_delay; bstp_topology_change_detection(sc); bstp_timer_stop(&sc->sc_tcn_timer); bstp_config_bpdu_generation(sc); bstp_timer_start(&sc->sc_hello_timer, 0); } } static void bstp_forward_delay_timer_expiry(struct bridge_softc *sc, struct bridge_iflist *bif) { if (bif->bif_state == BSTP_IFSTATE_LISTENING) { bstp_set_port_state(bif, BSTP_IFSTATE_LEARNING); bstp_timer_start(&bif->bif_forward_delay_timer, 0); } else if (bif->bif_state == BSTP_IFSTATE_LEARNING) { bstp_set_port_state(bif, BSTP_IFSTATE_FORWARDING); if (bstp_designated_for_some_port(sc) && bif->bif_change_detection_enabled) bstp_topology_change_detection(sc); } } static int bstp_designated_for_some_port(struct bridge_softc *sc) { struct bridge_iflist *bif; BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bif->bif_designated_bridge == sc->sc_bridge_id) return (1); } return (0); } static void bstp_tcn_timer_expiry(struct bridge_softc *sc) { bstp_transmit_tcn(sc); bstp_timer_start(&sc->sc_tcn_timer, 0); } static void bstp_topology_change_timer_expiry(struct bridge_softc *sc) { sc->sc_topology_change_detected = 0; sc->sc_topology_change = 0; } static void bstp_hold_timer_expiry(struct bridge_softc *sc, struct bridge_iflist *bif) { if (bif->bif_config_pending) bstp_transmit_config(sc, bif); } void bstp_initialization(struct bridge_softc *sc) { struct bridge_iflist *bif, *mif; mif = NULL; BRIDGE_LOCK(sc); BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bif->bif_ifp->if_type != IFT_ETHER) continue; bif->bif_port_id = (bif->bif_priority << 8) | (bif->bif_ifp->if_index & 0xff); if (mif == NULL) { mif = bif; continue; } if (memcmp(CLLADDR(bif->bif_ifp->if_sadl), CLLADDR(mif->bif_ifp->if_sadl), ETHER_ADDR_LEN) < 0) { mif = bif; continue; } } if (mif == NULL) { BRIDGE_UNLOCK(sc); bstp_stop(sc); return; } sc->sc_bridge_id = (((uint64_t)sc->sc_bridge_priority) << 48) | (((uint64_t)(uint8_t)CLLADDR(mif->bif_ifp->if_sadl)[0]) << 40) | (((uint64_t)(uint8_t)CLLADDR(mif->bif_ifp->if_sadl)[1]) << 32) | (((uint64_t)(uint8_t)CLLADDR(mif->bif_ifp->if_sadl)[2]) << 24) | (((uint64_t)(uint8_t)CLLADDR(mif->bif_ifp->if_sadl)[3]) << 16) | (((uint64_t)(uint8_t)CLLADDR(mif->bif_ifp->if_sadl)[4]) << 8) | (((uint64_t)(uint8_t)CLLADDR(mif->bif_ifp->if_sadl)[5]) << 0); BRIDGE_UNLOCK(sc); sc->sc_designated_root = sc->sc_bridge_id; sc->sc_root_path_cost = 0; sc->sc_root_port = NULL; sc->sc_max_age = sc->sc_bridge_max_age; sc->sc_hello_time = sc->sc_bridge_hello_time; sc->sc_forward_delay = sc->sc_bridge_forward_delay; sc->sc_topology_change_detected = 0; sc->sc_topology_change = 0; bstp_timer_stop(&sc->sc_tcn_timer); bstp_timer_stop(&sc->sc_topology_change_timer); if (callout_pending(&sc->sc_bstpcallout) == 0) callout_reset(&sc->sc_bstpcallout, hz, bstp_tick, sc); BRIDGE_LOCK(sc); BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if (bif->bif_flags & IFBIF_STP) bstp_enable_port(sc, bif); else bstp_disable_port(sc, bif); } bstp_port_state_selection(sc); bstp_config_bpdu_generation(sc); bstp_timer_start(&sc->sc_hello_timer, 0); BRIDGE_UNLOCK(sc); } void bstp_stop(struct bridge_softc *sc) { struct bridge_iflist *bif; BRIDGE_LOCK(sc); BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { bstp_set_port_state(bif, BSTP_IFSTATE_DISABLED); bstp_timer_stop(&bif->bif_hold_timer); bstp_timer_stop(&bif->bif_message_age_timer); bstp_timer_stop(&bif->bif_forward_delay_timer); } BRIDGE_UNLOCK(sc); callout_stop(&sc->sc_bstpcallout); bstp_timer_stop(&sc->sc_topology_change_timer); bstp_timer_stop(&sc->sc_tcn_timer); bstp_timer_stop(&sc->sc_hello_timer); } static void bstp_initialize_port(struct bridge_softc *sc, struct bridge_iflist *bif) { bstp_become_designated_port(sc, bif); bstp_set_port_state(bif, BSTP_IFSTATE_BLOCKING); bif->bif_topology_change_acknowledge = 0; bif->bif_config_pending = 0; bif->bif_change_detection_enabled = 1; bstp_timer_stop(&bif->bif_message_age_timer); bstp_timer_stop(&bif->bif_forward_delay_timer); bstp_timer_stop(&bif->bif_hold_timer); } static void bstp_enable_port(struct bridge_softc *sc, struct bridge_iflist *bif) { bstp_initialize_port(sc, bif); bstp_port_state_selection(sc); } static void bstp_disable_port(struct bridge_softc *sc, struct bridge_iflist *bif) { int root; root = bstp_root_bridge(sc); bstp_become_designated_port(sc, bif); bstp_set_port_state(bif, BSTP_IFSTATE_DISABLED); bif->bif_topology_change_acknowledge = 0; bif->bif_config_pending = 0; bstp_timer_stop(&bif->bif_message_age_timer); bstp_timer_stop(&bif->bif_forward_delay_timer); bstp_configuration_update(sc); bstp_port_state_selection(sc); if (bstp_root_bridge(sc) && (root == 0)) { sc->sc_max_age = sc->sc_bridge_max_age; sc->sc_hello_time = sc->sc_bridge_hello_time; sc->sc_forward_delay = sc->sc_bridge_forward_delay; bstp_topology_change_detection(sc); bstp_timer_stop(&sc->sc_tcn_timer); bstp_config_bpdu_generation(sc); bstp_timer_start(&sc->sc_hello_timer, 0); } } #if notused static void bstp_set_bridge_priority(struct bridge_softc *sc, uint64_t new_bridge_id) { struct bridge_iflist *bif; int root; root = bstp_root_bridge(sc); BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bstp_designated_port(sc, bif)) bif->bif_designated_bridge = new_bridge_id; } sc->sc_bridge_id = new_bridge_id; bstp_configuration_update(sc); bstp_port_state_selection(sc); if (bstp_root_bridge(sc) && (root == 0)) { sc->sc_max_age = sc->sc_bridge_max_age; sc->sc_hello_time = sc->sc_bridge_hello_time; sc->sc_forward_delay = sc->sc_bridge_forward_delay; bstp_topology_change_detection(sc); bstp_timer_stop(&sc->sc_tcn_timer); bstp_config_bpdu_generation(sc); bstp_timer_start(&sc->sc_hello_timer, 0); } } static void bstp_set_port_priority(struct bridge_softc *sc, struct bridge_iflist *bif, uint16_t new_port_id) { if (bstp_designated_port(sc, bif)) bif->bif_designated_port = new_port_id; bif->bif_port_id = new_port_id; if ((sc->sc_bridge_id == bif->bif_designated_bridge) && (bif->bif_port_id < bif->bif_designated_port)) { bstp_become_designated_port(sc, bif); bstp_port_state_selection(sc); } } static void bstp_set_path_cost(struct bridge_softc *sc, struct bridge_iflist *bif, uint32_t path_cost) { bif->bif_path_cost = path_cost; bstp_configuration_update(sc); bstp_port_state_selection(sc); } #endif static void bstp_ifupdstatus(struct bridge_softc *sc, struct bridge_iflist *bif) { struct ifnet *ifp = bif->bif_ifp; if (ifp->if_flags & IFF_UP) { switch (ifp->if_link_state) { case LINK_STATE_UNKNOWN: /* * Just enable the port if the link state is * unknown. */ if (bif->bif_state == BSTP_IFSTATE_DISABLED) bstp_enable_port(sc, bif); break; case LINK_STATE_UP: if (bif->bif_state == BSTP_IFSTATE_DISABLED) bstp_enable_port(sc, bif); break; case LINK_STATE_DOWN: if (bif->bif_state != BSTP_IFSTATE_DISABLED) bstp_disable_port(sc, bif); break; } return; } if (bif->bif_state != BSTP_IFSTATE_DISABLED) bstp_disable_port(sc, bif); } static void bstp_tick(void *arg) { struct bridge_softc *sc = arg; struct bridge_iflist *bif; BRIDGE_LOCK(sc); BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; /* * XXX This can cause a lag in "link does away" * XXX and "spanning tree gets updated". We need * XXX come sort of callback from the link state * XXX update code to kick spanning tree. * XXX --thorpej@NetBSD.org */ bstp_ifupdstatus(sc, bif); } if (bstp_timer_expired(&sc->sc_hello_timer, sc->sc_hello_time)) bstp_hello_timer_expiry(sc); if (bstp_timer_expired(&sc->sc_tcn_timer, sc->sc_bridge_hello_time)) bstp_tcn_timer_expiry(sc); if (bstp_timer_expired(&sc->sc_topology_change_timer, sc->sc_topology_change_time)) bstp_topology_change_timer_expiry(sc); BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bstp_timer_expired(&bif->bif_message_age_timer, sc->sc_max_age)) bstp_message_age_timer_expiry(sc, bif); } BRIDGE_IFLIST_WRITER_FOREACH(bif, sc) { if ((bif->bif_flags & IFBIF_STP) == 0) continue; if (bstp_timer_expired(&bif->bif_forward_delay_timer, sc->sc_forward_delay)) bstp_forward_delay_timer_expiry(sc, bif); if (bstp_timer_expired(&bif->bif_hold_timer, sc->sc_hold_time)) bstp_hold_timer_expiry(sc, bif); } if (sc->sc_if.if_flags & IFF_RUNNING) callout_reset(&sc->sc_bstpcallout, hz, bstp_tick, sc); BRIDGE_UNLOCK(sc); } static void bstp_timer_start(struct bridge_timer *t, uint16_t v) { t->value = v; t->active = 1; } static void bstp_timer_stop(struct bridge_timer *t) { t->value = 0; t->active = 0; } static int bstp_timer_expired(struct bridge_timer *t, uint16_t v) { if (t->active == 0) return (0); t->value += BSTP_TICK_VAL; if (t->value >= v) { bstp_timer_stop(t); return (1); } return (0); }