File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / pimd / rp.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Mon Jun 12 07:59:37 2017 UTC (7 years, 7 months ago) by misho
Branches: pimd, MAIN
CVS tags: v2_3_2, HEAD
pimd 2.3.2

    1: /*
    2:  * Copyright (c) 1998-2001
    3:  * University of Southern California/Information Sciences Institute.
    4:  * All rights reserved.
    5:  *
    6:  * Redistribution and use in source and binary forms, with or without
    7:  * modification, are permitted provided that the following conditions
    8:  * are met:
    9:  * 1. Redistributions of source code must retain the above copyright
   10:  *    notice, this list of conditions and the following disclaimer.
   11:  * 2. Redistributions in binary form must reproduce the above copyright
   12:  *    notice, this list of conditions and the following disclaimer in the
   13:  *    documentation and/or other materials provided with the distribution.
   14:  * 3. Neither the name of the project nor the names of its contributors
   15:  *    may be used to endorse or promote products derived from this software
   16:  *    without specific prior written permission.
   17:  *
   18:  * THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
   19:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
   20:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
   21:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
   22:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
   23:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
   24:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
   25:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
   26:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
   27:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
   28:  * SUCH DAMAGE.
   29:  */
   30: 
   31: #include "defs.h"
   32: 
   33: 
   34: /*
   35:  * The hash function. Stollen from Eddy's (eddy@isi.edu)
   36:  * implementation (for compatibility ;)
   37:  */
   38: #define SEED1   1103515245
   39: #define SEED2   12345
   40: #define RP_HASH_VALUE(G, M, C) (((SEED1) * (((SEED1) * ((G) & (M)) + (SEED2)) ^ (C)) + (SEED2)) % 0x80000000)
   41: 
   42: cand_rp_t               *cand_rp_list;
   43: grp_mask_t              *grp_mask_list;
   44: cand_rp_t               *segmented_cand_rp_list;
   45: grp_mask_t              *segmented_grp_mask_list;
   46: uint16_t                 curr_bsr_fragment_tag;
   47: uint8_t                  curr_bsr_priority;
   48: uint32_t                 curr_bsr_address;
   49: uint32_t                 curr_bsr_hash_mask;
   50: uint16_t                 pim_bootstrap_timer;   /* For electing the BSR and
   51: 						 * sending Cand-RP-set msgs */
   52: uint8_t                  my_bsr_priority;
   53: uint32_t                 my_bsr_address;
   54: uint32_t                 my_bsr_hash_mask;
   55: uint8_t                  cand_bsr_flag = FALSE; /* Set to TRUE if I am
   56: 						 * a candidate BSR */
   57: uint32_t                 my_cand_rp_address;
   58: uint8_t                  my_cand_rp_priority;
   59: uint16_t                 my_cand_rp_holdtime;
   60: uint16_t                 my_cand_rp_adv_period; /* The locally configured
   61: 						 * Cand-RP adv. period. */
   62: uint16_t                 pim_cand_rp_adv_timer;
   63: uint8_t                  cand_rp_flag  = FALSE;  /* Candidate RP flag */
   64: struct cand_rp_adv_message_ cand_rp_adv_message;
   65: uint32_t                 rp_my_ipv4_hashmask;
   66: 
   67: 
   68: /*
   69:  * Local functions definition.
   70:  */
   71: static cand_rp_t  *add_cand_rp          (cand_rp_t **used_cand_rp_list, uint32_t address);
   72: static grp_mask_t *add_grp_mask         (grp_mask_t **used_grp_mask_list,
   73: 					 uint32_t group_addr,
   74: 					 uint32_t group_mask,
   75: 					 uint32_t hash_mask);
   76: static void       delete_grp_mask_entry (cand_rp_t **used_cand_rp_list,
   77: 					 grp_mask_t **used_grp_mask_list,
   78: 					 grp_mask_t *grp_mask_delete);
   79: static void       delete_rp_entry       (cand_rp_t **used_cand_rp_list,
   80: 					 grp_mask_t **used_grp_mask_list,
   81: 					 cand_rp_t *cand_rp_ptr);
   82: 
   83: 
   84: void init_rp_and_bsr(void)
   85: {
   86:     /* TODO: if the grplist is not NULL, remap all groups ASAP! */
   87:     delete_rp_list(&cand_rp_list, &grp_mask_list);
   88:     delete_rp_list(&segmented_cand_rp_list, &segmented_grp_mask_list);
   89: 
   90:     if (cand_bsr_flag == FALSE) {
   91: 	/*
   92: 	 * If I am not candidat BSR, initialize the "current BSR"
   93: 	 * as having the lowest priority.
   94: 	 */
   95: 	curr_bsr_fragment_tag = 0;
   96: 	curr_bsr_priority = 0;             /* Lowest priority */
   97: 	curr_bsr_address  = INADDR_ANY_N;  /* Lowest priority */
   98: 	MASKLEN_TO_MASK(RP_DEFAULT_IPV4_HASHMASKLEN, curr_bsr_hash_mask);
   99: 	SET_TIMER(pim_bootstrap_timer, PIM_BOOTSTRAP_TIMEOUT);
  100:     } else {
  101: 	curr_bsr_fragment_tag = RANDOM();
  102: 	curr_bsr_priority = my_bsr_priority;
  103: 	curr_bsr_address = my_bsr_address;
  104: 	curr_bsr_hash_mask = my_bsr_hash_mask;
  105: 	SET_TIMER(pim_bootstrap_timer, bootstrap_initial_delay());
  106:     }
  107: 
  108:     if (cand_rp_flag != FALSE) {
  109: 	MASKLEN_TO_MASK(RP_DEFAULT_IPV4_HASHMASKLEN, rp_my_ipv4_hashmask);
  110: 	/* Setup the Cand-RP-Adv-Timer */
  111: 	SET_TIMER(pim_cand_rp_adv_timer, RANDOM() % my_cand_rp_adv_period);
  112:     }
  113: }
  114: 
  115: 
  116: uint16_t bootstrap_initial_delay(void)
  117: {
  118:     uint32_t addr_delay;
  119:     uint32_t delay;
  120:     uint32_t log_mask;
  121:     int32_t  log_of_2;
  122:     uint8_t  best_priority;
  123: 
  124:     /*
  125:      * The bootstrap timer initial value (if Cand-BSR).
  126:      * It depends of the bootstrap router priority:
  127:      * higher priority has shorter value:
  128:      *
  129:      * delay = 5 + 2 * log_2(1 + best_priority - myPriority) + addr_delay;
  130:      *
  131:      *    best_priority = Max(storedPriority, myPriority);
  132:      *    if (best_priority == myPriority)
  133:      *        addr_delay = log_2(bestAddr - myAddr)/16;
  134:      *    else
  135:      *        addr_delay = 2 - (myAddr/2^31);
  136:      */
  137: 
  138:     best_priority = max(curr_bsr_priority, my_bsr_priority);
  139:     if (best_priority == my_bsr_priority) {
  140: 	addr_delay = ntohl(curr_bsr_address) - ntohl(my_bsr_address);
  141: 	/* Calculate the integer part of log_2 of (bestAddr - myAddr) */
  142: 	/* To do so, have to find the position number of the first bit
  143: 	 * from left which is `1`
  144: 	 */
  145: 	log_mask = sizeof(addr_delay) << 3;
  146: 	log_mask = (1 << (log_mask - 1));  /* Set the leftmost bit to `1` */
  147: 	for (log_of_2 = (sizeof(addr_delay) << 3) - 1 ; log_of_2; log_of_2--) {
  148: 	    if (addr_delay & log_mask)
  149: 		break;
  150: 
  151: 	    log_mask >>= 1;  /* Start shifting `1` on right */
  152: 	}
  153: 	addr_delay = log_of_2 / 16;
  154:     } else {
  155: 	addr_delay = 2 - (ntohl(my_bsr_address) / ( 1 << 31));
  156:     }
  157: 
  158:     delay = 1 + best_priority - my_bsr_priority;
  159:     /* Calculate log_2(delay) */
  160:     log_mask = sizeof(delay) << 3;
  161:     log_mask = (1 << (log_mask - 1));  /* Set the leftmost bit to `1` */
  162:     for (log_of_2 = (sizeof(delay) << 3) - 1 ; log_of_2; log_of_2--) {
  163: 	if (delay & log_mask)
  164: 	    break;
  165: 
  166: 	log_mask >>= 1;  /* Start shifting `1` on right */
  167:     }
  168: 
  169:     delay = 5 + 2 * log_of_2 + addr_delay;
  170: 
  171:     return (uint16_t)delay;
  172: }
  173: 
  174: 
  175: static cand_rp_t *add_cand_rp(cand_rp_t **used_cand_rp_list, uint32_t address)
  176: {
  177:     cand_rp_t *prev = NULL;
  178:     cand_rp_t *next;
  179:     cand_rp_t *ptr;
  180:     rpentry_t *entry;
  181:     uint32_t addr_h = ntohl(address);
  182: 
  183:     /* The ordering is the bigger first */
  184:     for (next = *used_cand_rp_list; next; prev = next, next = next->next) {
  185: 	if (ntohl(next->rpentry->address) > addr_h)
  186: 	    continue;
  187: 
  188: 	if (next->rpentry->address == address)
  189: 	    return next;
  190: 	else
  191: 	    break;
  192:     }
  193: 
  194:     /* Create and insert the new entry between prev and next */
  195:     ptr = (cand_rp_t *)calloc(1, sizeof(cand_rp_t));
  196:     if (!ptr)
  197: 	logit(LOG_ERR, 0, "Ran out of memory in add_cand_rp()");
  198:     ptr->rp_grp_next = NULL;
  199:     ptr->next = next;
  200:     ptr->prev = prev;
  201:     if (next)
  202: 	next->prev = ptr;
  203:     if (prev == NULL)
  204: 	*used_cand_rp_list = ptr;
  205:     else
  206: 	prev->next = ptr;
  207: 
  208:     entry = (rpentry_t *)calloc(1, sizeof(rpentry_t));
  209:     if (!entry)
  210: 	logit(LOG_ERR, 0, "Ran out of memory in add_cand_rp()");
  211:     ptr->rpentry = entry;
  212:     entry->next = NULL;
  213:     entry->prev  = NULL;
  214:     entry->address = address;
  215:     entry->mrtlink = NULL;
  216:     entry->incoming = NO_VIF;
  217:     entry->upstream = NULL;
  218:     /* TODO: setup the metric and the preference as ~0 (the lowest)? */
  219:     entry->metric = ~0;
  220:     entry->preference = ~0;
  221:     RESET_TIMER(entry->timer);
  222:     entry->cand_rp = ptr;
  223: 
  224:     /* TODO: XXX: check whether there is a route to that RP: if return value
  225:      * is FALSE, then no route.
  226:      */
  227:     if (local_address(entry->address) == NO_VIF)
  228: 	/* TODO: check for error and delete */
  229: 	set_incoming(entry, PIM_IIF_RP);
  230:     else
  231: 	/* TODO: XXX: CHECK!!! */
  232: 	entry->incoming = reg_vif_num;
  233: 
  234:     return ptr;
  235: }
  236: 
  237: 
  238: static grp_mask_t *add_grp_mask(grp_mask_t **used_grp_mask_list, uint32_t group_addr, uint32_t group_mask, uint32_t hash_mask)
  239: {
  240:     grp_mask_t *prev = NULL;
  241:     grp_mask_t *next;
  242:     grp_mask_t *ptr;
  243:     uint32_t prefix_h = ntohl(group_addr & group_mask);
  244: 
  245:     /* The ordering of group_addr is: bigger first */
  246:     for (next = *used_grp_mask_list; next; prev = next, next = next->next) {
  247: 	if (ntohl(next->group_addr & next->group_mask) > prefix_h)
  248: 	    continue;
  249: 
  250: 	/* The ordering of group_mask is: bigger (longer) first */
  251: 	if ((next->group_addr & next->group_mask) == (group_addr & group_mask)) {
  252: 	    if (ntohl(next->group_mask) > ntohl(group_mask))
  253: 		continue;
  254: 	    else if (next->group_mask == group_mask)
  255: 		return next;
  256: 	    else
  257: 		break;
  258: 	}
  259:     }
  260: 
  261:     ptr = (grp_mask_t *)calloc(1, sizeof(grp_mask_t));
  262:     if (!ptr)
  263: 	logit(LOG_ERR, 0, "Ran out of memory in add_grp_mask()");
  264: 
  265:     ptr->grp_rp_next = (rp_grp_entry_t *)NULL;
  266:     ptr->next = next;
  267:     ptr->prev = prev;
  268:     if (next)
  269: 	next->prev = ptr;
  270:     if (prev == NULL)
  271: 	*used_grp_mask_list = ptr;
  272:     else
  273: 	prev->next = ptr;
  274: 
  275:     ptr->group_addr = group_addr;
  276:     ptr->group_mask = group_mask;
  277:     ptr->hash_mask  = hash_mask;
  278:     ptr->group_rp_number = 0;
  279:     ptr->fragment_tag = 0;
  280: 
  281:     return ptr;
  282: }
  283: 
  284: 
  285: /* TODO: XXX: BUG: a remapping for some groups currently using some other
  286:  * grp_mask may be required by the addition of the new entry!!!
  287:  * Remapping all groups might be a costly process...
  288:  */
  289: rp_grp_entry_t *add_rp_grp_entry(cand_rp_t  **used_cand_rp_list,
  290: 				 grp_mask_t **used_grp_mask_list,
  291: 				 uint32_t rp_addr,
  292: 				 uint8_t  rp_priority,
  293: 				 uint16_t rp_holdtime,
  294: 				 uint32_t group_addr,
  295: 				 uint32_t group_mask,
  296: 				 uint32_t bsr_hash_mask,
  297: 				 uint16_t fragment_tag)
  298: {
  299:     cand_rp_t *cand_rp_ptr;
  300:     grp_mask_t *mask_ptr;
  301:     rpentry_t *rpentry_ptr;
  302:     rp_grp_entry_t *entry_next;
  303:     rp_grp_entry_t *entry_new;
  304:     rp_grp_entry_t *entry_prev = NULL;
  305:     grpentry_t *grpentry_ptr_prev;
  306:     grpentry_t *grpentry_ptr_next;
  307:     uint32_t rp_addr_h;
  308:     uint8_t old_highest_priority = ~0;  /* Smaller value means "higher" */
  309: 
  310:     /* Input data verification */
  311:     if (!inet_valid_host(rp_addr))
  312: 	return NULL;
  313:     if (!IN_CLASSD(ntohl(group_addr)))
  314: 	return NULL;
  315: 
  316:     mask_ptr = add_grp_mask(used_grp_mask_list, group_addr, group_mask, bsr_hash_mask);
  317:     if (mask_ptr == NULL)
  318: 	return NULL;
  319: 
  320: /* TODO: delete */
  321: #if 0
  322:     if (mask_ptr->grp_rp_next) {
  323: 	/* Check for obsolete grp_rp chain */
  324: 	if ((my_bsr_address != curr_bsr_address) && (mask_ptr->grp_rp_next->fragment_tag != fragment_tag)) {
  325: 	    /* This grp_rp chain is obsolete. Delete it. */
  326: 	    delete_grp_mask(used_cand_rp_list, used_grp_mask_list, group_addr, group_mask);
  327: 	    mask_ptr = add_grp_mask(used_grp_mask_list, group_addr, group_mask, bsr_hash_mask);
  328: 
  329: 	    if (mask_ptr == NULL)
  330: 		return NULL;
  331: 	}
  332:     }
  333: #endif /* 0 */
  334: 
  335:     cand_rp_ptr = add_cand_rp(used_cand_rp_list, rp_addr);
  336:     if (cand_rp_ptr == NULL) {
  337: 	if (mask_ptr->grp_rp_next == NULL)
  338: 	    delete_grp_mask(used_cand_rp_list, used_grp_mask_list,
  339: 			    group_addr, group_mask);
  340: 	return NULL;
  341:     }
  342: 
  343:     rpentry_ptr = cand_rp_ptr->rpentry;
  344:     SET_TIMER(rpentry_ptr->timer, rp_holdtime);
  345:     rp_addr_h = ntohl(rp_addr);
  346:     mask_ptr->fragment_tag = fragment_tag;   /* For garbage collection */
  347: 
  348:     entry_prev = NULL;
  349:     entry_next = mask_ptr->grp_rp_next;
  350:     /* TODO: improve it */
  351:     if (entry_next != NULL)
  352: 	old_highest_priority = entry_next->priority;
  353:     for ( ; entry_next; entry_prev = entry_next,
  354: 	      entry_next = entry_next->grp_rp_next) {
  355: 	/* Smaller value means higher priority. The entries are
  356: 	 * sorted with the highest priority first.
  357: 	 */
  358: 	if (entry_next->priority < rp_priority)
  359: 	    continue;
  360: 	if (entry_next->priority > rp_priority)
  361: 	    break;
  362: 
  363: 	/*
  364: 	 * Here we don't care about higher/lower addresses, because
  365: 	 * higher address does not guarantee higher hash_value,
  366: 	 * but anyway we do order with the higher address first,
  367: 	 * so it will be easier to find an existing entry and update the
  368: 	 * holdtime.
  369: 	 */
  370: 	if (ntohl(entry_next->rp->rpentry->address) > rp_addr_h)
  371: 	    continue;
  372: 	if (ntohl(entry_next->rp->rpentry->address) < rp_addr_h)
  373: 	    break;
  374: 	/* We already have this entry. Update the holdtime */
  375: 	/* TODO: We shoudn't have old existing entry, because with the
  376: 	 * current implementation all of them will be deleted
  377: 	 * (different fragment_tag). Debug and check and eventually
  378: 	 * delete.
  379: 	 */
  380: 	entry_next->holdtime = rp_holdtime;
  381: 	entry_next->fragment_tag = fragment_tag;
  382: 
  383: 	return entry_next;
  384:     }
  385: 
  386:     /* Create and link the new entry */
  387:     entry_new = (rp_grp_entry_t *)calloc(1, sizeof(rp_grp_entry_t));
  388:     if (!entry_new)
  389: 	logit(LOG_ERR, 0, "Ran out of memory in add_rp_grp_entry()");
  390:     entry_new->grp_rp_next = entry_next;
  391:     entry_new->grp_rp_prev = entry_prev;
  392:     if (entry_next)
  393: 	entry_next->grp_rp_prev = entry_new;
  394:     if (entry_prev == NULL)
  395: 	mask_ptr->grp_rp_next = entry_new;
  396:     else
  397: 	entry_prev->grp_rp_next = entry_new;
  398: 
  399:     /*
  400:      * The rp_grp_entry chain is not ordered, so just plug
  401:      * the new entry at the head.
  402:      */
  403:     entry_new->rp_grp_next = cand_rp_ptr->rp_grp_next;
  404:     if (cand_rp_ptr->rp_grp_next)
  405: 	cand_rp_ptr->rp_grp_next->rp_grp_prev = entry_new;
  406:     entry_new->rp_grp_prev = NULL;
  407:     cand_rp_ptr->rp_grp_next = entry_new;
  408: 
  409:     entry_new->holdtime = rp_holdtime;
  410:     entry_new->fragment_tag = fragment_tag;
  411:     entry_new->priority = rp_priority;
  412:     entry_new->group = mask_ptr;
  413:     entry_new->rp = cand_rp_ptr;
  414:     entry_new->grplink = NULL;
  415: 
  416:     /* If I am BSR candidate and rp_addr is NOT hacked SSM address, then log it */
  417:     if( cand_bsr_flag && rp_addr != 0x0100fea9 ) {
  418: 	uint32_t mask;
  419: 	MASK_TO_MASKLEN(group_mask, mask);
  420: 	logit(LOG_INFO, 0, "New RP candidate %s for group %s/%d, priority %d",
  421: 	      inet_fmt(rp_addr, s1, sizeof(s1)), inet_fmt(group_addr, s2, sizeof(s2)), mask, rp_priority);
  422:     }
  423: 
  424:     mask_ptr->group_rp_number++;
  425: 
  426:     if (mask_ptr->grp_rp_next->priority == rp_priority) {
  427: 	/* The first entries are with the best priority. */
  428: 	/* Adding this rp_grp_entry may result in group_to_rp remapping */
  429: 	for (entry_next = mask_ptr->grp_rp_next; entry_next; entry_next = entry_next->grp_rp_next) {
  430: 	    if (entry_next->priority > old_highest_priority)
  431: 		break;
  432: 
  433: 	    for (grpentry_ptr_prev = entry_next->grplink; grpentry_ptr_prev; ) {
  434: 		grpentry_ptr_next = grpentry_ptr_prev->rpnext;
  435: 		remap_grpentry(grpentry_ptr_prev);
  436: 		grpentry_ptr_prev = grpentry_ptr_next;
  437: 	    }
  438: 	}
  439:     }
  440: 
  441:     return entry_new;
  442: }
  443: 
  444: 
  445: void delete_rp_grp_entry(cand_rp_t **used_cand_rp_list, grp_mask_t **used_grp_mask_list, rp_grp_entry_t *entry)
  446: {
  447:     grpentry_t *ptr;
  448:     grpentry_t *ptr_next;
  449: 
  450:     if (entry == NULL)
  451: 	return;
  452:     entry->group->group_rp_number--;
  453: 
  454:     /* Free the rp_grp* and grp_rp* links */
  455:     if (entry->rp_grp_prev)
  456: 	entry->rp_grp_prev->rp_grp_next = entry->rp_grp_next;
  457:     else
  458: 	entry->rp->rp_grp_next = entry->rp_grp_next;
  459:     if (entry->rp_grp_next)
  460: 	entry->rp_grp_next->rp_grp_prev = entry->rp_grp_prev;
  461: 
  462:     if (entry->grp_rp_prev)
  463: 	entry->grp_rp_prev->grp_rp_next = entry->grp_rp_next;
  464:     else
  465: 	entry->group->grp_rp_next = entry->grp_rp_next;
  466: 
  467:     if (entry->grp_rp_next)
  468: 	entry->grp_rp_next->grp_rp_prev = entry->grp_rp_prev;
  469: 
  470:     /* Delete Cand-RP or Group-prefix if useless */
  471:     if (entry->group->grp_rp_next == NULL)
  472: 	delete_grp_mask_entry(used_cand_rp_list, used_grp_mask_list, entry->group);
  473: 
  474:     if (entry->rp->rp_grp_next == NULL)
  475: 	delete_rp_entry(used_cand_rp_list, used_grp_mask_list, entry->rp);
  476: 
  477:     /* Remap all affected groups */
  478:     for (ptr = entry->grplink; ptr; ptr = ptr_next) {
  479: 	ptr_next = ptr->rpnext;
  480: 	remap_grpentry(ptr);
  481:     }
  482: 
  483:     free((char *)entry);
  484: }
  485: 
  486: /* TODO: XXX: the affected group entries will be partially
  487:  * setup, because may have group routing entry, but NULL pointers to RP.
  488:  * After the call to this function, must remap all group entries ASAP.
  489:  */
  490: void delete_rp_list(cand_rp_t  **used_cand_rp_list, grp_mask_t **used_grp_mask_list)
  491: {
  492:     cand_rp_t      *cand_ptr,  *cand_next;
  493:     rp_grp_entry_t *entry_ptr, *entry_next;
  494:     grp_mask_t     *mask_ptr, *mask_next;
  495:     grpentry_t     *gentry_ptr, *gentry_ptr_next;
  496: 
  497:     for (cand_ptr = *used_cand_rp_list; cand_ptr; ) {
  498: 	cand_next = cand_ptr->next;
  499: 
  500: 	/* Free the mrtentry (if any) for this RP */
  501: 	if (cand_ptr->rpentry->mrtlink) {
  502: 	    if (cand_ptr->rpentry->mrtlink->flags & MRTF_KERNEL_CACHE)
  503: 		delete_mrtentry_all_kernel_cache(cand_ptr->rpentry->mrtlink);
  504: 	    FREE_MRTENTRY(cand_ptr->rpentry->mrtlink);
  505: 	}
  506: 	free(cand_ptr->rpentry);
  507: 
  508: 	/* Free the whole chain of entry for this RP */
  509: 	for (entry_ptr = cand_ptr->rp_grp_next; entry_ptr; entry_ptr = entry_next) {
  510: 	    entry_next = entry_ptr->rp_grp_next;
  511: 
  512: 	    /* Clear the RP related invalid pointers for all group entries */
  513: 	    for (gentry_ptr = entry_ptr->grplink; gentry_ptr; gentry_ptr = gentry_ptr_next) {
  514: 		gentry_ptr_next = gentry_ptr->rpnext;
  515: 		gentry_ptr->rpnext = NULL;
  516: 		gentry_ptr->rpprev = NULL;
  517: 		gentry_ptr->active_rp_grp = NULL;
  518: 		gentry_ptr->rpaddr = INADDR_ANY_N;
  519: 	    }
  520: 
  521: 	    free(entry_ptr);
  522: 	}
  523: 
  524: 	free(cand_ptr);
  525: 	cand_ptr = cand_next;
  526:     }
  527:     *used_cand_rp_list = NULL;
  528: 
  529:     for (mask_ptr = *used_grp_mask_list; mask_ptr; mask_ptr = mask_next) {
  530: 	mask_next = mask_ptr->next;
  531: 	free(mask_ptr);
  532:     }
  533:     *used_grp_mask_list = NULL;
  534: }
  535: 
  536: 
  537: void delete_grp_mask(cand_rp_t **used_cand_rp_list, grp_mask_t **used_grp_mask_list, uint32_t group_addr, uint32_t group_mask)
  538: {
  539:     grp_mask_t *ptr;
  540:     uint32_t prefix_h = ntohl(group_addr & group_mask);
  541: 
  542:     for (ptr = *used_grp_mask_list; ptr; ptr = ptr->next) {
  543: 	if (ntohl(ptr->group_addr & ptr->group_mask) > prefix_h)
  544: 	    continue;
  545: 
  546: 	if (ptr->group_addr == group_addr) {
  547: 	    if (ntohl(ptr->group_mask) > ntohl(group_mask))
  548: 		continue;
  549: 	    else if (ptr->group_mask == group_mask)
  550: 		break;
  551: 	    else
  552: 		return;   /* Not found */
  553: 	}
  554:     }
  555: 
  556:     if (ptr == (grp_mask_t *)NULL)
  557: 	return;       /* Not found */
  558: 
  559:     delete_grp_mask_entry(used_cand_rp_list, used_grp_mask_list, ptr);
  560: }
  561: 
  562: static void delete_grp_mask_entry(cand_rp_t **used_cand_rp_list, grp_mask_t **used_grp_mask_list, grp_mask_t *grp_mask_delete)
  563: {
  564:     grpentry_t *grp_ptr, *grp_ptr_next;
  565:     rp_grp_entry_t *entry_ptr;
  566:     rp_grp_entry_t *entry_next;
  567: 
  568:     if (grp_mask_delete == NULL)
  569: 	return;
  570: 
  571:     /* Remove from the grp_mask_list first */
  572:     if (grp_mask_delete->prev)
  573: 	grp_mask_delete->prev->next = grp_mask_delete->next;
  574:     else
  575: 	*used_grp_mask_list = grp_mask_delete->next;
  576: 
  577:     if (grp_mask_delete->next)
  578: 	grp_mask_delete->next->prev = grp_mask_delete->prev;
  579: 
  580:     /* Remove all grp_rp entries for this grp_mask */
  581:     for (entry_ptr = grp_mask_delete->grp_rp_next; entry_ptr; entry_ptr = entry_next) {
  582: 	entry_next = entry_ptr->grp_rp_next;
  583: 
  584: 	/* Remap all related grpentry */
  585: 	for (grp_ptr = entry_ptr->grplink; grp_ptr; grp_ptr = grp_ptr_next) {
  586: 	    grp_ptr_next = grp_ptr->rpnext;
  587: 	    remap_grpentry(grp_ptr);
  588: 	}
  589: 
  590: 	if (entry_ptr->rp_grp_prev != (rp_grp_entry_t *)NULL)
  591: 	    entry_ptr->rp_grp_prev->rp_grp_next = entry_ptr->rp_grp_next;
  592: 	else
  593: 	    entry_ptr->rp->rp_grp_next = entry_ptr->rp_grp_next;
  594: 
  595: 	if (entry_ptr->rp_grp_next != NULL)
  596: 	    entry_ptr->rp_grp_next->rp_grp_prev = entry_ptr->rp_grp_prev;
  597: 
  598: 	/* Delete the RP entry */
  599: 	if (entry_ptr->rp->rp_grp_next == NULL)
  600: 	    delete_rp_entry(used_cand_rp_list, used_grp_mask_list, entry_ptr->rp);
  601: 
  602: 	free(entry_ptr);
  603:     }
  604: 
  605:     free(grp_mask_delete);
  606: }
  607: 
  608: /*
  609:  * TODO: currently not used.
  610:  */
  611: void delete_rp(cand_rp_t **used_cand_rp_list, grp_mask_t **used_grp_mask_list, uint32_t rp_addr)
  612: {
  613:     cand_rp_t *ptr;
  614:     uint32_t rp_addr_h = ntohl(rp_addr);
  615: 
  616:     for(ptr = *used_cand_rp_list; ptr != NULL; ptr = ptr->next) {
  617: 	if (ntohl(ptr->rpentry->address) > rp_addr_h)
  618: 	    continue;
  619: 
  620: 	if (ptr->rpentry->address == rp_addr)
  621: 	    break;
  622: 	else
  623: 	    return;   /* Not found */
  624:     }
  625: 
  626:     if (ptr == NULL)
  627: 	return;       /* Not found */
  628: 
  629:     delete_rp_entry(used_cand_rp_list, used_grp_mask_list, ptr);
  630: }
  631: 
  632: 
  633: static void delete_rp_entry(cand_rp_t **used_cand_rp_list, grp_mask_t **used_grp_mask_list, cand_rp_t *cand_rp_delete)
  634: {
  635:     rp_grp_entry_t *entry_ptr;
  636:     rp_grp_entry_t *entry_next;
  637:     grpentry_t *grp_ptr;
  638:     grpentry_t *grp_ptr_next;
  639: 
  640:     if (cand_rp_delete == NULL)
  641: 	return;
  642: 
  643:     /* Remove from the cand-RP chain */
  644:     if (cand_rp_delete->prev)
  645: 	cand_rp_delete->prev->next = cand_rp_delete->next;
  646:     else
  647: 	*used_cand_rp_list = cand_rp_delete->next;
  648: 
  649:     if (cand_rp_delete->next)
  650: 	cand_rp_delete->next->prev = cand_rp_delete->prev;
  651: 
  652:     if (cand_rp_delete->rpentry->mrtlink) {
  653: 	if (cand_rp_delete->rpentry->mrtlink->flags & MRTF_KERNEL_CACHE)
  654: 	    delete_mrtentry_all_kernel_cache(cand_rp_delete->rpentry->mrtlink);
  655: 
  656: 	FREE_MRTENTRY(cand_rp_delete->rpentry->mrtlink);
  657:     }
  658:     free ((char *)cand_rp_delete->rpentry);
  659: 
  660:     /* Remove all rp_grp entries for this RP */
  661:     for (entry_ptr = cand_rp_delete->rp_grp_next; entry_ptr; entry_ptr = entry_next) {
  662: 	entry_next = entry_ptr->rp_grp_next;
  663: 	entry_ptr->group->group_rp_number--;
  664: 
  665: 	/* First take care of the grp_rp chain */
  666: 	if (entry_ptr->grp_rp_prev)
  667: 	    entry_ptr->grp_rp_prev->grp_rp_next = entry_ptr->grp_rp_next;
  668: 	else
  669: 	    entry_ptr->group->grp_rp_next = entry_ptr->grp_rp_next;
  670: 
  671: 	if (entry_ptr->grp_rp_next)
  672: 	    entry_ptr->grp_rp_next->grp_rp_prev = entry_ptr->grp_rp_prev;
  673: 
  674: 	if (entry_ptr->grp_rp_next == NULL)
  675: 	    delete_grp_mask_entry(used_cand_rp_list, used_grp_mask_list, entry_ptr->group);
  676: 
  677: 	/* Remap the related groups */
  678: 	for (grp_ptr = entry_ptr->grplink; grp_ptr; grp_ptr = grp_ptr_next) {
  679: 	    grp_ptr_next = grp_ptr->rpnext;
  680: 	    remap_grpentry(grp_ptr);
  681: 	}
  682: 
  683: 	free(entry_ptr);
  684:     }
  685: 
  686:     free((char *)cand_rp_delete);
  687: }
  688: 
  689: 
  690: /*
  691:  * Rehash the RP for the group.
  692:  * XXX: currently, every time when remap_grpentry() is called, there has
  693:  * being a good reason to change the RP, so for performancy reasons
  694:  * no check is performed whether the RP will be really different one.
  695:  */
  696: int remap_grpentry(grpentry_t *grpentry_ptr)
  697: {
  698:     rpentry_t *rpentry_ptr;
  699:     rp_grp_entry_t *entry_ptr;
  700:     mrtentry_t *grp_route;
  701:     mrtentry_t *mrtentry_ptr;
  702: 
  703:     if (grpentry_ptr == NULL)
  704: 	return FALSE;
  705: 
  706:     /* Remove from the list of all groups matching to the same RP */
  707:     if (grpentry_ptr->rpprev) {
  708: 	grpentry_ptr->rpprev->rpnext = grpentry_ptr->rpnext;
  709:     } else {
  710: 	if (grpentry_ptr->active_rp_grp)
  711: 	    grpentry_ptr->active_rp_grp->grplink = grpentry_ptr->rpnext;
  712:     }
  713: 
  714:     if (grpentry_ptr->rpnext)
  715: 	grpentry_ptr->rpnext->rpprev = grpentry_ptr->rpprev;
  716: 
  717:     entry_ptr = rp_grp_match(grpentry_ptr->group);
  718:     if (entry_ptr == NULL) {
  719: 	/* If cannot remap, delete the group */
  720: 	delete_grpentry(grpentry_ptr);
  721: 	return FALSE;
  722:     }
  723:     rpentry_ptr = entry_ptr->rp->rpentry;
  724: 
  725:     /* Add to the new chain of all groups mapping to the same RP */
  726:     grpentry_ptr->rpaddr  = rpentry_ptr->address;
  727:     grpentry_ptr->active_rp_grp = entry_ptr;
  728:     grpentry_ptr->rpnext = entry_ptr->grplink;
  729:     if (grpentry_ptr->rpnext)
  730: 	grpentry_ptr->rpnext->rpprev = grpentry_ptr;
  731:     grpentry_ptr->rpprev = NULL;
  732:     entry_ptr->grplink = grpentry_ptr;
  733: 
  734:     grp_route = grpentry_ptr->grp_route;
  735:     if (grp_route) {
  736: 	grp_route->upstream   = rpentry_ptr->upstream;
  737: 	grp_route->metric     = rpentry_ptr->metric;
  738: 	grp_route->preference = rpentry_ptr->preference;
  739: 	change_interfaces(grp_route, rpentry_ptr->incoming,
  740: 			  grp_route->joined_oifs,
  741: 			  grp_route->pruned_oifs,
  742: 			  grp_route->leaves,
  743: 			  grp_route->asserted_oifs, MFC_UPDATE_FORCE);
  744:     }
  745: 
  746:     for (mrtentry_ptr = grpentry_ptr->mrtlink; mrtentry_ptr; mrtentry_ptr = mrtentry_ptr->grpnext) {
  747: 	if (!(mrtentry_ptr->flags & MRTF_RP))
  748: 	    continue;
  749: 
  750: 	mrtentry_ptr->upstream = rpentry_ptr->upstream;
  751: 	mrtentry_ptr->metric   = rpentry_ptr->metric;
  752: 	mrtentry_ptr->preference = rpentry_ptr->preference;
  753: 	change_interfaces(mrtentry_ptr, rpentry_ptr->incoming,
  754: 			  mrtentry_ptr->joined_oifs,
  755: 			  mrtentry_ptr->pruned_oifs,
  756: 			  mrtentry_ptr->leaves,
  757: 			  mrtentry_ptr->asserted_oifs, MFC_UPDATE_FORCE);
  758:     }
  759: 
  760:     return TRUE;
  761: }
  762: 
  763: 
  764: rpentry_t *rp_match(uint32_t group)
  765: {
  766:     rp_grp_entry_t *ptr;
  767: 
  768:     ptr = rp_grp_match(group);
  769:     if (ptr)
  770: 	return ptr->rp->rpentry;
  771: 
  772:     return NULL;
  773: }
  774: 
  775: rp_grp_entry_t *rp_grp_match(uint32_t group)
  776: {
  777:     grp_mask_t *mask_ptr;
  778:     rp_grp_entry_t *entry_ptr;
  779:     rp_grp_entry_t *best_entry = NULL;
  780:     uint8_t best_priority       = ~0; /* Smaller is better */
  781:     uint32_t best_hash_value    = 0;  /* Bigger is better */
  782:     uint32_t best_address_h     = 0;  /* Bigger is better */
  783:     uint32_t curr_hash_value    = 0;
  784:     uint32_t curr_address_h     = 0;
  785:     uint32_t group_h            = ntohl(group);
  786: 
  787:     if (grp_mask_list == NULL)
  788: 	return NULL;
  789: 
  790:     for (mask_ptr = grp_mask_list; mask_ptr; mask_ptr = mask_ptr->next) {
  791: 	/* Search the grp_mask (group-prefix) list */
  792: 	if ((group_h & ntohl(mask_ptr->group_mask))
  793: 	    != ntohl(mask_ptr->group_mask & mask_ptr->group_addr))
  794: 	    continue;
  795: 
  796: 	for (entry_ptr = mask_ptr->grp_rp_next; entry_ptr; entry_ptr = entry_ptr->grp_rp_next) {
  797: 	    if (best_priority < entry_ptr->priority)
  798: 		break;
  799: 
  800: 	    curr_hash_value = RP_HASH_VALUE(group_h, mask_ptr->hash_mask, curr_address_h);
  801: 	    curr_address_h = ntohl(entry_ptr->rp->rpentry->address);
  802: 
  803: 	    if (best_priority == entry_ptr->priority) {
  804: 		/* Compare the hash_value and then the addresses */
  805: 		if (curr_hash_value < best_hash_value)
  806: 		    continue;
  807: 
  808: 		if (curr_hash_value == best_hash_value) {
  809: 		    if (curr_address_h < best_address_h)
  810: 			continue;
  811: 		}
  812: 	    }
  813: 
  814: 	    /* The current entry in the loop is preferred */
  815: 	    best_entry = entry_ptr;
  816: 	    best_priority = best_entry->priority;
  817: 	    best_address_h = curr_address_h;
  818: 	    best_hash_value = curr_hash_value;
  819: 	}
  820:     }
  821: 
  822:     return best_entry;
  823: }
  824: 
  825: 
  826: rpentry_t *rp_find(uint32_t rp_address)
  827: {
  828:     cand_rp_t *cand_rp_ptr;
  829:     uint32_t address_h = ntohl(rp_address);
  830: 
  831:     for(cand_rp_ptr = cand_rp_list; cand_rp_ptr != NULL; cand_rp_ptr = cand_rp_ptr->next) {
  832: 	if (ntohl(cand_rp_ptr->rpentry->address) > address_h)
  833: 	    continue;
  834: 
  835: 	if (cand_rp_ptr->rpentry->address == rp_address)
  836: 	    return cand_rp_ptr->rpentry;
  837: 
  838: 	return NULL;
  839:     }
  840: 
  841:     return NULL;
  842: }
  843: 
  844: 
  845: /*
  846:  * Create a bootstrap message in "send_buff" and returns the data size
  847:  * (excluding the IP header and the PIM header) Can be used both by the
  848:  * Bootstrap router to multicast the RP-set or by the DR to unicast it to
  849:  * a new neighbor. It DOES NOT change any timers.
  850:  */
  851: int create_pim_bootstrap_message(char *send_buff)
  852: {
  853:     uint8_t *data_ptr;
  854:     grp_mask_t *mask_ptr;
  855:     rp_grp_entry_t *entry_ptr;
  856:     int datalen;
  857:     uint8_t masklen;
  858: 
  859:     if (curr_bsr_address == INADDR_ANY_N)
  860: 	return 0;
  861: 
  862:     data_ptr = (uint8_t *)(send_buff + sizeof(struct ip) + sizeof(pim_header_t));
  863:     if (curr_bsr_address == my_bsr_address)
  864: 	curr_bsr_fragment_tag++;
  865: 
  866:     PUT_HOSTSHORT(curr_bsr_fragment_tag, data_ptr);
  867:     MASK_TO_MASKLEN(curr_bsr_hash_mask, masklen);
  868:     PUT_BYTE(masklen, data_ptr);
  869:     PUT_BYTE(curr_bsr_priority, data_ptr);
  870:     PUT_EUADDR(curr_bsr_address, data_ptr);
  871: 
  872:     /* TODO: XXX: No fragmentation support (yet) */
  873:     for (mask_ptr = grp_mask_list; mask_ptr; mask_ptr = mask_ptr->next) {
  874: 	if (IN_PIM_SSM_RANGE(mask_ptr->group_addr)) {
  875: 	    continue;  /* Do not advertise internal virtual RP for SSM groups */
  876: 	}
  877: 	MASK_TO_MASKLEN(mask_ptr->group_mask, masklen);
  878: 	PUT_EGADDR(mask_ptr->group_addr, masklen, 0, data_ptr);
  879: 	PUT_BYTE(mask_ptr->group_rp_number, data_ptr);
  880: 	PUT_BYTE(mask_ptr->group_rp_number, data_ptr); /* TODO: if frag.*/
  881: 	PUT_HOSTSHORT(0, data_ptr);
  882: 
  883: 	for (entry_ptr = mask_ptr->grp_rp_next; entry_ptr; entry_ptr = entry_ptr->grp_rp_next) {
  884: 	    PUT_EUADDR(entry_ptr->rp->rpentry->address, data_ptr);
  885: 	    PUT_HOSTSHORT(entry_ptr->holdtime, data_ptr);
  886: 	    PUT_BYTE(entry_ptr->priority, data_ptr);
  887: 	    PUT_BYTE(0, data_ptr);  /* The reserved field */
  888: 	}
  889:     }
  890: 
  891:     datalen = (data_ptr - (uint8_t *)send_buff) - sizeof(struct ip) - sizeof(pim_header_t);
  892: 
  893:     return datalen;
  894: }
  895: 
  896: 
  897: /*
  898:  * Check if the addr is the RP for the group corresponding to mrt.
  899:  * Return TRUE or FALSE.
  900:  */
  901: int check_mrtentry_rp(mrtentry_t *mrt, uint32_t addr)
  902: {
  903:     rp_grp_entry_t *ptr;
  904: 
  905:     if (!mrt)
  906: 	return FALSE;
  907: 
  908:     if (addr == INADDR_ANY_N)
  909: 	return FALSE;
  910: 
  911:     ptr = mrt->group->active_rp_grp;
  912:     if (!ptr)
  913: 	return FALSE;
  914: 
  915:     if (mrt->group->rpaddr == addr)
  916: 	return TRUE;
  917: 
  918:     return FALSE;
  919: }
  920: 
  921: /**
  922:  * Local Variables:
  923:  *  version-control: t
  924:  *  indent-tabs-mode: t
  925:  *  c-file-style: "ellemtel"
  926:  *  c-basic-offset: 4
  927:  * End:
  928:  */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>