1*4eaa4710SRishi Srivatsavai /************************************************************************ 2*4eaa4710SRishi Srivatsavai * RSTP library - Rapid Spanning Tree (802.1t, 802.1w) 3*4eaa4710SRishi Srivatsavai * Copyright (C) 2001-2003 Optical Access 4*4eaa4710SRishi Srivatsavai * Author: Alex Rozin 5*4eaa4710SRishi Srivatsavai * 6*4eaa4710SRishi Srivatsavai * This file is part of RSTP library. 7*4eaa4710SRishi Srivatsavai * 8*4eaa4710SRishi Srivatsavai * RSTP library is free software; you can redistribute it and/or modify it 9*4eaa4710SRishi Srivatsavai * under the terms of the GNU Lesser General Public License as published by the 10*4eaa4710SRishi Srivatsavai * Free Software Foundation; version 2.1 11*4eaa4710SRishi Srivatsavai * 12*4eaa4710SRishi Srivatsavai * RSTP library is distributed in the hope that it will be useful, but 13*4eaa4710SRishi Srivatsavai * WITHOUT ANY WARRANTY; without even the implied warranty of 14*4eaa4710SRishi Srivatsavai * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser 15*4eaa4710SRishi Srivatsavai * General Public License for more details. 16*4eaa4710SRishi Srivatsavai * 17*4eaa4710SRishi Srivatsavai * You should have received a copy of the GNU Lesser General Public License 18*4eaa4710SRishi Srivatsavai * along with RSTP library; see the file COPYING. If not, write to the Free 19*4eaa4710SRishi Srivatsavai * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 20*4eaa4710SRishi Srivatsavai * 02111-1307, USA. 21*4eaa4710SRishi Srivatsavai **********************************************************************/ 22*4eaa4710SRishi Srivatsavai 23*4eaa4710SRishi Srivatsavai /* STP priority vectors API : 17.4.2 */ 24*4eaa4710SRishi Srivatsavai 25*4eaa4710SRishi Srivatsavai #include "base.h" 26*4eaa4710SRishi Srivatsavai #include "stp_bpdu.h" 27*4eaa4710SRishi Srivatsavai #include "vector.h" 28*4eaa4710SRishi Srivatsavai #include "stp_vectors.h" 29*4eaa4710SRishi Srivatsavai 30*4eaa4710SRishi Srivatsavai int 31*4eaa4710SRishi Srivatsavai STP_VECT_compare_bridge_id (BRIDGE_ID* b1, BRIDGE_ID* b2) 32*4eaa4710SRishi Srivatsavai { 33*4eaa4710SRishi Srivatsavai if (b1->prio < b2->prio) 34*4eaa4710SRishi Srivatsavai return -1; 35*4eaa4710SRishi Srivatsavai 36*4eaa4710SRishi Srivatsavai if (b1->prio > b2->prio) 37*4eaa4710SRishi Srivatsavai return 1; 38*4eaa4710SRishi Srivatsavai return memcmp (b1->addr, b2->addr, 6); 39*4eaa4710SRishi Srivatsavai } 40*4eaa4710SRishi Srivatsavai 41*4eaa4710SRishi Srivatsavai void 42*4eaa4710SRishi Srivatsavai STP_VECT_copy (OUT PRIO_VECTOR_T* t, IN PRIO_VECTOR_T* f) 43*4eaa4710SRishi Srivatsavai { 44*4eaa4710SRishi Srivatsavai (void) memcpy (t, f, sizeof (PRIO_VECTOR_T)); 45*4eaa4710SRishi Srivatsavai } 46*4eaa4710SRishi Srivatsavai 47*4eaa4710SRishi Srivatsavai void 48*4eaa4710SRishi Srivatsavai STP_VECT_create (OUT PRIO_VECTOR_T* t, 49*4eaa4710SRishi Srivatsavai IN BRIDGE_ID* root_br, 50*4eaa4710SRishi Srivatsavai IN unsigned long root_path_cost, 51*4eaa4710SRishi Srivatsavai IN BRIDGE_ID* design_bridge, 52*4eaa4710SRishi Srivatsavai IN PORT_ID design_port, 53*4eaa4710SRishi Srivatsavai IN PORT_ID bridge_port) 54*4eaa4710SRishi Srivatsavai { 55*4eaa4710SRishi Srivatsavai (void) memcpy (&t->root_bridge, root_br, sizeof (BRIDGE_ID)); 56*4eaa4710SRishi Srivatsavai t->root_path_cost = root_path_cost; 57*4eaa4710SRishi Srivatsavai (void) memcpy (&t->design_bridge, design_bridge, sizeof (BRIDGE_ID)); 58*4eaa4710SRishi Srivatsavai t->design_port = design_port; 59*4eaa4710SRishi Srivatsavai t->bridge_port = bridge_port; 60*4eaa4710SRishi Srivatsavai } 61*4eaa4710SRishi Srivatsavai 62*4eaa4710SRishi Srivatsavai int 63*4eaa4710SRishi Srivatsavai STP_VECT_compare_vector (PRIO_VECTOR_T* v1, PRIO_VECTOR_T* v2) 64*4eaa4710SRishi Srivatsavai { 65*4eaa4710SRishi Srivatsavai int bridcmp; 66*4eaa4710SRishi Srivatsavai 67*4eaa4710SRishi Srivatsavai bridcmp = STP_VECT_compare_bridge_id (&v1->root_bridge, &v2->root_bridge); 68*4eaa4710SRishi Srivatsavai if (bridcmp < 0) return bridcmp; 69*4eaa4710SRishi Srivatsavai 70*4eaa4710SRishi Srivatsavai if (! bridcmp) { 71*4eaa4710SRishi Srivatsavai bridcmp = v1->root_path_cost - v2->root_path_cost; 72*4eaa4710SRishi Srivatsavai if (bridcmp < 0) return bridcmp; 73*4eaa4710SRishi Srivatsavai if (! bridcmp) { 74*4eaa4710SRishi Srivatsavai bridcmp = STP_VECT_compare_bridge_id (&v1->design_bridge, &v2->design_bridge); 75*4eaa4710SRishi Srivatsavai if (bridcmp < 0) return bridcmp; 76*4eaa4710SRishi Srivatsavai if (! bridcmp) { 77*4eaa4710SRishi Srivatsavai bridcmp = v1->design_port - v2->design_port; 78*4eaa4710SRishi Srivatsavai if (bridcmp < 0) return bridcmp; 79*4eaa4710SRishi Srivatsavai if (! bridcmp) 80*4eaa4710SRishi Srivatsavai return v1->bridge_port - v2->bridge_port; 81*4eaa4710SRishi Srivatsavai } 82*4eaa4710SRishi Srivatsavai } 83*4eaa4710SRishi Srivatsavai } 84*4eaa4710SRishi Srivatsavai 85*4eaa4710SRishi Srivatsavai return bridcmp; 86*4eaa4710SRishi Srivatsavai } 87*4eaa4710SRishi Srivatsavai 88*4eaa4710SRishi Srivatsavai static unsigned short 89*4eaa4710SRishi Srivatsavai stp_vect_get_short (IN unsigned char* f) 90*4eaa4710SRishi Srivatsavai { 91*4eaa4710SRishi Srivatsavai /* LINTED: alignment */ 92*4eaa4710SRishi Srivatsavai return ntohs (*(unsigned short *)f); 93*4eaa4710SRishi Srivatsavai } 94*4eaa4710SRishi Srivatsavai 95*4eaa4710SRishi Srivatsavai static void 96*4eaa4710SRishi Srivatsavai stp_vect_set_short (IN unsigned short f, OUT unsigned char* t) 97*4eaa4710SRishi Srivatsavai { 98*4eaa4710SRishi Srivatsavai /* LINTED: alignment */ 99*4eaa4710SRishi Srivatsavai *(unsigned short *)t = htons (f); 100*4eaa4710SRishi Srivatsavai } 101*4eaa4710SRishi Srivatsavai 102*4eaa4710SRishi Srivatsavai static void 103*4eaa4710SRishi Srivatsavai stp_vect_get_bridge_id (IN unsigned char* c_br, OUT BRIDGE_ID* bridge_id) 104*4eaa4710SRishi Srivatsavai { 105*4eaa4710SRishi Srivatsavai bridge_id->prio = stp_vect_get_short (c_br); 106*4eaa4710SRishi Srivatsavai (void) memcpy (bridge_id->addr, c_br + 2, 6); 107*4eaa4710SRishi Srivatsavai } 108*4eaa4710SRishi Srivatsavai 109*4eaa4710SRishi Srivatsavai static void 110*4eaa4710SRishi Srivatsavai stp_vect_set_bridge_id (IN BRIDGE_ID* bridge_id, OUT unsigned char* c_br) 111*4eaa4710SRishi Srivatsavai { 112*4eaa4710SRishi Srivatsavai stp_vect_set_short (bridge_id->prio, c_br); 113*4eaa4710SRishi Srivatsavai (void) memcpy (c_br + 2, bridge_id->addr, 6); 114*4eaa4710SRishi Srivatsavai } 115*4eaa4710SRishi Srivatsavai 116*4eaa4710SRishi Srivatsavai void 117*4eaa4710SRishi Srivatsavai STP_VECT_get_vector (IN BPDU_BODY_T* b, OUT PRIO_VECTOR_T* v) 118*4eaa4710SRishi Srivatsavai { 119*4eaa4710SRishi Srivatsavai stp_vect_get_bridge_id (b->root_id, &v->root_bridge); 120*4eaa4710SRishi Srivatsavai 121*4eaa4710SRishi Srivatsavai /* LINTED: alignment */ 122*4eaa4710SRishi Srivatsavai v->root_path_cost = ntohl (*((long*) b->root_path_cost)); 123*4eaa4710SRishi Srivatsavai 124*4eaa4710SRishi Srivatsavai stp_vect_get_bridge_id (b->bridge_id, &v->design_bridge); 125*4eaa4710SRishi Srivatsavai 126*4eaa4710SRishi Srivatsavai v->design_port = stp_vect_get_short (b->port_id); 127*4eaa4710SRishi Srivatsavai } 128*4eaa4710SRishi Srivatsavai 129*4eaa4710SRishi Srivatsavai void 130*4eaa4710SRishi Srivatsavai STP_VECT_set_vector (IN PRIO_VECTOR_T* v, OUT BPDU_BODY_T* b) 131*4eaa4710SRishi Srivatsavai { 132*4eaa4710SRishi Srivatsavai unsigned long root_path_cost; 133*4eaa4710SRishi Srivatsavai 134*4eaa4710SRishi Srivatsavai stp_vect_set_bridge_id (&v->root_bridge, b->root_id); 135*4eaa4710SRishi Srivatsavai 136*4eaa4710SRishi Srivatsavai root_path_cost = htonl (v->root_path_cost); 137*4eaa4710SRishi Srivatsavai (void) memcpy (b->root_path_cost, &root_path_cost, 4); 138*4eaa4710SRishi Srivatsavai 139*4eaa4710SRishi Srivatsavai stp_vect_set_bridge_id (&v->design_bridge, b->bridge_id); 140*4eaa4710SRishi Srivatsavai 141*4eaa4710SRishi Srivatsavai stp_vect_set_short (v->design_port, b->port_id); 142*4eaa4710SRishi Srivatsavai } 143*4eaa4710SRishi Srivatsavai 144*4eaa4710SRishi Srivatsavai #ifdef STP_DBG 145*4eaa4710SRishi Srivatsavai 146*4eaa4710SRishi Srivatsavai /*ARGSUSED*/ 147*4eaa4710SRishi Srivatsavai void 148*4eaa4710SRishi Srivatsavai STP_VECT_br_id_print (IN char *title, IN BRIDGE_ID* br_id, IN Bool cr) 149*4eaa4710SRishi Srivatsavai { 150*4eaa4710SRishi Srivatsavai stp_trace ("%s=%04lX-%02x%02x%02x%02x%02x%02x", 151*4eaa4710SRishi Srivatsavai title, 152*4eaa4710SRishi Srivatsavai (unsigned long) br_id->prio, 153*4eaa4710SRishi Srivatsavai (unsigned char) br_id->addr[0], 154*4eaa4710SRishi Srivatsavai (unsigned char) br_id->addr[1], 155*4eaa4710SRishi Srivatsavai (unsigned char) br_id->addr[2], 156*4eaa4710SRishi Srivatsavai (unsigned char) br_id->addr[3], 157*4eaa4710SRishi Srivatsavai (unsigned char) br_id->addr[4], 158*4eaa4710SRishi Srivatsavai (unsigned char) br_id->addr[5]); 159*4eaa4710SRishi Srivatsavai #ifndef __SUN__ 160*4eaa4710SRishi Srivatsavai stp_trace (cr ? "\n" : " "); 161*4eaa4710SRishi Srivatsavai #endif 162*4eaa4710SRishi Srivatsavai } 163*4eaa4710SRishi Srivatsavai 164*4eaa4710SRishi Srivatsavai void 165*4eaa4710SRishi Srivatsavai STP_VECT_print (IN char *title, IN PRIO_VECTOR_T *v) 166*4eaa4710SRishi Srivatsavai { 167*4eaa4710SRishi Srivatsavai stp_trace ("%s:", title); 168*4eaa4710SRishi Srivatsavai STP_VECT_br_id_print ("rootBr", &v->root_bridge, False); 169*4eaa4710SRishi Srivatsavai 170*4eaa4710SRishi Srivatsavai /**** 171*4eaa4710SRishi Srivatsavai stp_trace (" rpc=%ld ", (long) v->root_path_cost); 172*4eaa4710SRishi Srivatsavai ****/ 173*4eaa4710SRishi Srivatsavai 174*4eaa4710SRishi Srivatsavai STP_VECT_br_id_print ("designBr", &v->design_bridge, False); 175*4eaa4710SRishi Srivatsavai 176*4eaa4710SRishi Srivatsavai /****/ 177*4eaa4710SRishi Srivatsavai stp_trace (" dp=%lx bp=%lx ", 178*4eaa4710SRishi Srivatsavai (unsigned long) v->design_port, 179*4eaa4710SRishi Srivatsavai (unsigned long) v->bridge_port); 180*4eaa4710SRishi Srivatsavai /***********/ 181*4eaa4710SRishi Srivatsavai stp_trace ("\n"); 182*4eaa4710SRishi Srivatsavai } 183*4eaa4710SRishi Srivatsavai #endif 184