prune.c revision 1.2       1  1.2  thorpej /*	$NetBSD: prune.c,v 1.2 1995/10/09 03:51:49 thorpej Exp $	*/
      2  1.2  thorpej 
      3  1.1  mycroft /*
      4  1.1  mycroft  * The mrouted program is covered by the license in the accompanying file
      5  1.1  mycroft  * named "LICENSE".  Use of the mrouted program represents acceptance of
      6  1.1  mycroft  * the terms and conditions listed in that file.
      7  1.1  mycroft  *
      8  1.1  mycroft  * The mrouted program is COPYRIGHT 1989 by The Board of Trustees of
      9  1.1  mycroft  * Leland Stanford Junior University.
     10  1.1  mycroft  */
     11  1.1  mycroft 
     12  1.1  mycroft 
     13  1.1  mycroft #include "defs.h"
     14  1.1  mycroft 
     15  1.1  mycroft extern int cache_lifetime;
     16  1.1  mycroft extern int max_prune_lifetime;
     17  1.1  mycroft extern struct rtentry *routing_table;
     18  1.1  mycroft 
     19  1.1  mycroft /*
     20  1.1  mycroft  * dither cache lifetime to obtain a value between x and 2*x
     21  1.1  mycroft  */
     22  1.1  mycroft #ifdef SYSV
     23  1.1  mycroft #define CACHE_LIFETIME(x) ((x) + (lrand48() % (x)))
     24  1.1  mycroft #else
     25  1.1  mycroft #define CACHE_LIFETIME(x) ((x) + (random() % (x)))
     26  1.1  mycroft #endif
     27  1.1  mycroft 
     28  1.1  mycroft #define CHK_GS(x, y) {	\
     29  1.1  mycroft 		switch(x) { \
     30  1.1  mycroft 			case 2:	\
     31  1.1  mycroft 			case 4:	\
     32  1.1  mycroft 			case 8:	\
     33  1.1  mycroft 			case 16: \
     34  1.1  mycroft 			case 32: \
     35  1.1  mycroft 			case 64: \
     36  1.1  mycroft 			case 128: \
     37  1.1  mycroft 			case 256: y = 1; \
     38  1.1  mycroft 				  break; \
     39  1.1  mycroft 			default:  y = 0; \
     40  1.1  mycroft 		} \
     41  1.1  mycroft 	}
     42  1.1  mycroft 
     43  1.1  mycroft struct gtable *kernel_table;		/* ptr to list of kernel grp entries*/
     44  1.1  mycroft static struct gtable *kernel_no_route;	/* list of grp entries w/o routes   */
     45  1.1  mycroft struct gtable *gtp;			/* pointer for kernel rt entries    */
     46  1.1  mycroft unsigned int kroutes;			/* current number of cache entries  */
     47  1.1  mycroft 
     48  1.1  mycroft /****************************************************************************
     49  1.1  mycroft                        Functions that are local to prune.c
     50  1.1  mycroft ****************************************************************************/
     51  1.1  mycroft 
     52  1.1  mycroft /*
     53  1.1  mycroft  * Updates the ttl values for each vif.
     54  1.1  mycroft  */
     55  1.1  mycroft static void
     56  1.1  mycroft prun_add_ttls(gt)
     57  1.1  mycroft     struct gtable *gt;
     58  1.1  mycroft {
     59  1.1  mycroft     struct uvif *v;
     60  1.1  mycroft     vifi_t vifi;
     61  1.1  mycroft 
     62  1.1  mycroft     for (vifi = 0, v = uvifs; vifi < numvifs; ++vifi, ++v) {
     63  1.1  mycroft 	if (VIFM_ISSET(vifi, gt->gt_grpmems))
     64  1.1  mycroft 	    gt->gt_ttls[vifi] = v->uv_threshold;
     65  1.1  mycroft 	else
     66  1.1  mycroft 	    gt->gt_ttls[vifi] = 0;
     67  1.1  mycroft     }
     68  1.1  mycroft }
     69  1.1  mycroft 
     70  1.1  mycroft /*
     71  1.1  mycroft  * checks for scoped multicast addresses
     72  1.1  mycroft  */
     73  1.1  mycroft #define GET_SCOPE(gt) { \
     74  1.1  mycroft 	register int _i; \
     75  1.1  mycroft 	if ((ntohl((gt)->gt_mcastgrp) & 0xff000000) == 0xef000000) \
     76  1.1  mycroft 	    for (_i = 0; _i < numvifs; _i++) \
     77  1.1  mycroft 		if (scoped_addr(_i, (gt)->gt_mcastgrp)) \
     78  1.1  mycroft 		    VIFM_SET(_i, (gt)->gt_scope); \
     79  1.1  mycroft 	}
     80  1.1  mycroft 
     81  1.1  mycroft int
     82  1.1  mycroft scoped_addr(vifi, addr)
     83  1.1  mycroft     vifi_t vifi;
     84  1.1  mycroft     u_int32_t addr;
     85  1.1  mycroft {
     86  1.1  mycroft     struct vif_acl *acl;
     87  1.1  mycroft 
     88  1.1  mycroft     for (acl = uvifs[vifi].uv_acl; acl; acl = acl->acl_next)
     89  1.1  mycroft 	if ((addr & acl->acl_mask) == acl->acl_addr)
     90  1.1  mycroft 	    return 1;
     91  1.1  mycroft 
     92  1.1  mycroft     return 0;
     93  1.1  mycroft }
     94  1.1  mycroft 
     95  1.1  mycroft /*
     96  1.1  mycroft  * Determine if mcastgrp has a listener on vifi
     97  1.1  mycroft  */
     98  1.1  mycroft int
     99  1.1  mycroft grplst_mem(vifi, mcastgrp)
    100  1.1  mycroft     vifi_t vifi;
    101  1.1  mycroft     u_int32_t mcastgrp;
    102  1.1  mycroft {
    103  1.1  mycroft     register struct listaddr *g;
    104  1.1  mycroft     register struct uvif *v;
    105  1.1  mycroft 
    106  1.1  mycroft     v = &uvifs[vifi];
    107  1.1  mycroft 
    108  1.1  mycroft     for (g = v->uv_groups; g != NULL; g = g->al_next)
    109  1.1  mycroft 	if (mcastgrp == g->al_addr)
    110  1.1  mycroft 	    return 1;
    111  1.1  mycroft 
    112  1.1  mycroft     return 0;
    113  1.1  mycroft }
    114  1.1  mycroft 
    115  1.1  mycroft /*
    116  1.1  mycroft  * Finds the group entry with the specified source and netmask.
    117  1.1  mycroft  * If netmask is 0, it uses the route's netmask.
    118  1.1  mycroft  *
    119  1.1  mycroft  * Returns TRUE if found a match, and the global variable gtp is left
    120  1.1  mycroft  * pointing to entry before the found entry.
    121  1.1  mycroft  * Returns FALSE if no exact match found, gtp is left pointing to before
    122  1.1  mycroft  * the entry in question belongs, or is NULL if the it belongs at the
    123  1.1  mycroft  * head of the list.
    124  1.1  mycroft  */
    125  1.1  mycroft int
    126  1.1  mycroft find_src_grp(src, mask, grp)
    127  1.1  mycroft    u_int32_t src;
    128  1.1  mycroft    u_int32_t mask;
    129  1.1  mycroft    u_int32_t grp;
    130  1.1  mycroft {
    131  1.1  mycroft     struct gtable *gt;
    132  1.1  mycroft 
    133  1.1  mycroft     gtp = NULL;
    134  1.1  mycroft     gt = kernel_table;
    135  1.1  mycroft     while (gt != NULL) {
    136  1.1  mycroft 	if (grp == gt->gt_mcastgrp &&
    137  1.1  mycroft 	    (mask ? (gt->gt_route->rt_origin == src &&
    138  1.1  mycroft 		     gt->gt_route->rt_originmask == mask) :
    139  1.1  mycroft 		    ((src & gt->gt_route->rt_originmask) ==
    140  1.1  mycroft 		     gt->gt_route->rt_origin)))
    141  1.1  mycroft 	    return TRUE;
    142  1.1  mycroft 	if (ntohl(grp) > ntohl(gt->gt_mcastgrp) ||
    143  1.1  mycroft 	    (grp == gt->gt_mcastgrp &&
    144  1.1  mycroft 	     (ntohl(mask) < ntohl(gt->gt_route->rt_originmask) ||
    145  1.1  mycroft 	      (mask == gt->gt_route->rt_originmask &&
    146  1.1  mycroft 	       (ntohl(src) > ntohl(gt->gt_route->rt_origin)))))) {
    147  1.1  mycroft 	    gtp = gt;
    148  1.1  mycroft 	    gt = gt->gt_gnext;
    149  1.1  mycroft 	}
    150  1.1  mycroft 	else break;
    151  1.1  mycroft     }
    152  1.1  mycroft     return FALSE;
    153  1.1  mycroft }
    154  1.1  mycroft 
    155  1.1  mycroft /*
    156  1.1  mycroft  * Check if the neighbor supports pruning
    157  1.1  mycroft  */
    158  1.1  mycroft static int
    159  1.1  mycroft pruning_neighbor(vifi, addr)
    160  1.1  mycroft     vifi_t vifi;
    161  1.1  mycroft     u_int32_t addr;
    162  1.1  mycroft {
    163  1.1  mycroft     struct listaddr *n = neighbor_info(vifi, addr);
    164  1.1  mycroft     int vers;
    165  1.1  mycroft 
    166  1.1  mycroft     if (n == NULL)
    167  1.1  mycroft 	return 0;
    168  1.1  mycroft 
    169  1.1  mycroft     if (n->al_flags & NF_PRUNE)
    170  1.1  mycroft 	return 1;
    171  1.1  mycroft 
    172  1.1  mycroft     /*
    173  1.1  mycroft      * Versions from 3.0 to 3.4 relied on the version number to identify
    174  1.1  mycroft      * that they could handle pruning.
    175  1.1  mycroft      */
    176  1.1  mycroft     vers = NBR_VERS(n);
    177  1.1  mycroft     return (vers >= 0x0300 && vers <= 0x0304);
    178  1.1  mycroft }
    179  1.1  mycroft 
    180  1.1  mycroft /*
    181  1.1  mycroft  * Can the neighbor in question handle multicast traceroute?
    182  1.1  mycroft  */
    183  1.1  mycroft static int
    184  1.1  mycroft can_mtrace(vifi, addr)
    185  1.1  mycroft     vifi_t	vifi;
    186  1.1  mycroft     u_int32_t	addr;
    187  1.1  mycroft {
    188  1.1  mycroft     struct listaddr *n = neighbor_info(vifi, addr);
    189  1.1  mycroft     int vers;
    190  1.1  mycroft 
    191  1.1  mycroft     if (n == NULL)
    192  1.1  mycroft 	return 0;
    193  1.1  mycroft 
    194  1.1  mycroft     if (n->al_flags & NF_MTRACE)
    195  1.1  mycroft 	return 1;
    196  1.1  mycroft 
    197  1.1  mycroft     /*
    198  1.1  mycroft      * Versions 3.3 and 3.4 relied on the version number to identify
    199  1.1  mycroft      * that they could handle traceroute.
    200  1.1  mycroft      */
    201  1.1  mycroft     vers = NBR_VERS(n);
    202  1.1  mycroft     return (vers >= 0x0303 && vers <= 0x0304);
    203  1.1  mycroft }
    204  1.1  mycroft 
    205  1.1  mycroft /*
    206  1.1  mycroft  * Returns the prune entry of the router, or NULL if none exists
    207  1.1  mycroft  */
    208  1.1  mycroft static struct ptable *
    209  1.1  mycroft find_prune_entry(vr, pt)
    210  1.1  mycroft     u_int32_t vr;
    211  1.1  mycroft     struct ptable *pt;
    212  1.1  mycroft {
    213  1.1  mycroft     while (pt) {
    214  1.1  mycroft 	if (pt->pt_router == vr)
    215  1.1  mycroft 	    return pt;
    216  1.1  mycroft 	pt = pt->pt_next;
    217  1.1  mycroft     }
    218  1.1  mycroft 
    219  1.1  mycroft     return NULL;
    220  1.1  mycroft }
    221  1.1  mycroft 
    222  1.1  mycroft /*
    223  1.1  mycroft  * Send a prune message to the dominant router for
    224  1.1  mycroft  * this source.
    225  1.1  mycroft  *
    226  1.1  mycroft  * Record an entry that a prune was sent for this group
    227  1.1  mycroft  */
    228  1.1  mycroft static void
    229  1.1  mycroft send_prune(gt)
    230  1.1  mycroft     struct gtable *gt;
    231  1.1  mycroft {
    232  1.1  mycroft     struct ptable *pt;
    233  1.1  mycroft     char *p;
    234  1.1  mycroft     int i;
    235  1.1  mycroft     int datalen;
    236  1.1  mycroft     u_int32_t src;
    237  1.1  mycroft     u_int32_t dst;
    238  1.1  mycroft     u_int32_t tmp;
    239  1.1  mycroft 
    240  1.1  mycroft     /* Don't process any prunes if router is not pruning */
    241  1.1  mycroft     if (pruning == 0)
    242  1.1  mycroft 	return;
    243  1.1  mycroft 
    244  1.1  mycroft     /* Can't process a prune if we don't have an associated route */
    245  1.1  mycroft     if (gt->gt_route == NULL)
    246  1.1  mycroft 	return;
    247  1.1  mycroft 
    248  1.1  mycroft     /* Don't send a prune to a non-pruning router */
    249  1.1  mycroft     if (!pruning_neighbor(gt->gt_route->rt_parent, gt->gt_route->rt_gateway))
    250  1.1  mycroft 	return;
    251  1.1  mycroft 
    252  1.1  mycroft     /*
    253  1.1  mycroft      * sends a prune message to the router upstream.
    254  1.1  mycroft      */
    255  1.1  mycroft     src = uvifs[gt->gt_route->rt_parent].uv_lcl_addr;
    256  1.1  mycroft     dst = gt->gt_route->rt_gateway;
    257  1.1  mycroft 
    258  1.1  mycroft     p = send_buf + MIN_IP_HEADER_LEN + IGMP_MINLEN;
    259  1.1  mycroft     datalen = 0;
    260  1.1  mycroft 
    261  1.1  mycroft     /*
    262  1.1  mycroft      * determine prune lifetime
    263  1.1  mycroft      */
    264  1.1  mycroft     gt->gt_prsent_timer = gt->gt_timer;
    265  1.1  mycroft     for (pt = gt->gt_pruntbl; pt; pt = pt->pt_next)
    266  1.1  mycroft 	if (pt->pt_timer < gt->gt_prsent_timer)
    267  1.1  mycroft 	    gt->gt_prsent_timer = pt->pt_timer;
    268  1.1  mycroft 
    269  1.1  mycroft     /*
    270  1.1  mycroft      * If we have a graft pending, cancel graft retransmission
    271  1.1  mycroft      */
    272  1.1  mycroft     gt->gt_grftsnt = 0;
    273  1.1  mycroft 
    274  1.1  mycroft     for (i = 0; i < 4; i++)
    275  1.1  mycroft 	*p++ = ((char *)&(gt->gt_route->rt_origin))[i];
    276  1.1  mycroft     for (i = 0; i < 4; i++)
    277  1.1  mycroft 	*p++ = ((char *)&(gt->gt_mcastgrp))[i];
    278  1.1  mycroft     tmp = htonl(gt->gt_prsent_timer);
    279  1.1  mycroft     for (i = 0; i < 4; i++)
    280  1.1  mycroft 	*p++ = ((char *)&(tmp))[i];
    281  1.1  mycroft     datalen += 12;
    282  1.1  mycroft 
    283  1.1  mycroft     send_igmp(src, dst, IGMP_DVMRP, DVMRP_PRUNE,
    284  1.1  mycroft 	      htonl(MROUTED_LEVEL), datalen);
    285  1.1  mycroft 
    286  1.1  mycroft     log(LOG_DEBUG, 0, "sent prune for (%s %s)/%d on vif %d to %s",
    287  1.1  mycroft       inet_fmts(gt->gt_route->rt_origin, gt->gt_route->rt_originmask, s1),
    288  1.1  mycroft       inet_fmt(gt->gt_mcastgrp, s2),
    289  1.1  mycroft       gt->gt_prsent_timer, gt->gt_route->rt_parent,
    290  1.1  mycroft       inet_fmt(gt->gt_route->rt_gateway, s3));
    291  1.1  mycroft }
    292  1.1  mycroft 
    293  1.1  mycroft /*
    294  1.1  mycroft  * a prune was sent upstream
    295  1.1  mycroft  * so, a graft has to be sent to annul the prune
    296  1.1  mycroft  * set up a graft timer so that if an ack is not
    297  1.1  mycroft  * heard within that time, another graft request
    298  1.1  mycroft  * is sent out.
    299  1.1  mycroft  */
    300  1.1  mycroft static void
    301  1.1  mycroft send_graft(gt)
    302  1.1  mycroft     struct gtable *gt;
    303  1.1  mycroft {
    304  1.1  mycroft     register char *p;
    305  1.1  mycroft     register int i;
    306  1.1  mycroft     int datalen;
    307  1.1  mycroft     u_int32_t src;
    308  1.1  mycroft     u_int32_t dst;
    309  1.1  mycroft 
    310  1.1  mycroft     /* Can't send a graft without an associated route */
    311  1.1  mycroft     if (gt->gt_route == NULL)
    312  1.1  mycroft 	return;
    313  1.1  mycroft 
    314  1.1  mycroft     src = uvifs[gt->gt_route->rt_parent].uv_lcl_addr;
    315  1.1  mycroft     dst = gt->gt_route->rt_gateway;
    316  1.1  mycroft 
    317  1.1  mycroft     p = send_buf + MIN_IP_HEADER_LEN + IGMP_MINLEN;
    318  1.1  mycroft     datalen = 0;
    319  1.1  mycroft 
    320  1.1  mycroft     for (i = 0; i < 4; i++)
    321  1.1  mycroft 	*p++ = ((char *)&(gt->gt_route->rt_origin))[i];
    322  1.1  mycroft     for (i = 0; i < 4; i++)
    323  1.1  mycroft 	*p++ = ((char *)&(gt->gt_mcastgrp))[i];
    324  1.1  mycroft     datalen += 8;
    325  1.1  mycroft 
    326  1.1  mycroft     if (datalen != 0) {
    327  1.1  mycroft 	send_igmp(src, dst, IGMP_DVMRP, DVMRP_GRAFT,
    328  1.1  mycroft 		  htonl(MROUTED_LEVEL), datalen);
    329  1.1  mycroft     }
    330  1.1  mycroft     log(LOG_DEBUG, 0, "sent graft for (%s %s) to %s on vif %d",
    331  1.1  mycroft 	inet_fmts(gt->gt_route->rt_origin, gt->gt_route->rt_originmask, s1),
    332  1.1  mycroft 	inet_fmt(gt->gt_mcastgrp, s2),
    333  1.1  mycroft 	inet_fmt(gt->gt_route->rt_gateway, s3), gt->gt_route->rt_parent);
    334  1.1  mycroft }
    335  1.1  mycroft 
    336  1.1  mycroft /*
    337  1.1  mycroft  * Send an ack that a graft was received
    338  1.1  mycroft  */
    339  1.1  mycroft static void
    340  1.1  mycroft send_graft_ack(src, dst, origin, grp)
    341  1.1  mycroft     u_int32_t src;
    342  1.1  mycroft     u_int32_t dst;
    343  1.1  mycroft     u_int32_t origin;
    344  1.1  mycroft     u_int32_t grp;
    345  1.1  mycroft {
    346  1.1  mycroft     register char *p;
    347  1.1  mycroft     register int i;
    348  1.1  mycroft     int datalen;
    349  1.1  mycroft 
    350  1.1  mycroft     p = send_buf + MIN_IP_HEADER_LEN + IGMP_MINLEN;
    351  1.1  mycroft     datalen = 0;
    352  1.1  mycroft 
    353  1.1  mycroft     for (i = 0; i < 4; i++)
    354  1.1  mycroft 	*p++ = ((char *)&(origin))[i];
    355  1.1  mycroft     for (i = 0; i < 4; i++)
    356  1.1  mycroft 	*p++ = ((char *)&(grp))[i];
    357  1.1  mycroft     datalen += 8;
    358  1.1  mycroft 
    359  1.1  mycroft     send_igmp(src, dst, IGMP_DVMRP, DVMRP_GRAFT_ACK,
    360  1.1  mycroft 	      htonl(MROUTED_LEVEL), datalen);
    361  1.1  mycroft 
    362  1.1  mycroft     log(LOG_DEBUG, 0, "sent graft ack for (%s, %s) to %s",
    363  1.1  mycroft 	inet_fmt(origin, s1), inet_fmt(grp, s2), inet_fmt(dst, s3));
    364  1.1  mycroft }
    365  1.1  mycroft 
    366  1.1  mycroft /*
    367  1.1  mycroft  * Update the kernel cache with all the routes hanging off the group entry
    368  1.1  mycroft  */
    369  1.1  mycroft static void
    370  1.1  mycroft update_kernel(g)
    371  1.1  mycroft     struct gtable *g;
    372  1.1  mycroft {
    373  1.1  mycroft     struct stable *st;
    374  1.1  mycroft 
    375  1.1  mycroft     for (st = g->gt_srctbl; st; st = st->st_next)
    376  1.1  mycroft 	k_add_rg(st->st_origin, g);
    377  1.1  mycroft }
    378  1.1  mycroft 
    379  1.1  mycroft /****************************************************************************
    380  1.1  mycroft                           Functions that are used externally
    381  1.1  mycroft ****************************************************************************/
    382  1.1  mycroft 
    383  1.1  mycroft #ifdef SNMP
    384  1.1  mycroft #include <sys/types.h>
    385  1.1  mycroft #include "snmp.h"
    386  1.1  mycroft 
    387  1.1  mycroft /*
    388  1.1  mycroft  * Find a specific group entry in the group table
    389  1.1  mycroft  */
    390  1.1  mycroft struct gtable *
    391  1.1  mycroft find_grp(grp)
    392  1.1  mycroft    u_long grp;
    393  1.1  mycroft {
    394  1.1  mycroft    struct gtable *gt;
    395  1.1  mycroft 
    396  1.1  mycroft    for (gt = kernel_table; gt; gt = gt->gt_gnext) {
    397  1.1  mycroft       if (ntohl(grp) < ntohl(gt->gt_mcastgrp))
    398  1.1  mycroft       	 break;
    399  1.1  mycroft       if (gt->gt_mcastgrp == grp)
    400  1.1  mycroft          return gt;
    401  1.1  mycroft    }
    402  1.1  mycroft    return NULL;
    403  1.1  mycroft }
    404  1.1  mycroft 
    405  1.1  mycroft /*
    406  1.1  mycroft  * Given a group entry and source, find the corresponding source table
    407  1.1  mycroft  * entry
    408  1.1  mycroft  */
    409  1.1  mycroft struct stable *
    410  1.1  mycroft find_grp_src(gt, src)
    411  1.1  mycroft    struct gtable *gt;
    412  1.1  mycroft    u_long src;
    413  1.1  mycroft {
    414  1.1  mycroft    struct stable *st;
    415  1.1  mycroft    u_long grp = gt->gt_mcastgrp;
    416  1.1  mycroft    struct gtable *gtcurr;
    417  1.1  mycroft 
    418  1.1  mycroft    for (gtcurr = gt; gtcurr->gt_mcastgrp == grp; gtcurr = gtcurr->gt_gnext) {
    419  1.1  mycroft       for (st = gtcurr->gt_srctbl; st; st = st->st_next)
    420  1.1  mycroft 	 if (st->st_origin == src)
    421  1.1  mycroft 	    return st;
    422  1.1  mycroft    }
    423  1.1  mycroft    return NULL;
    424  1.1  mycroft }
    425  1.1  mycroft 
    426  1.1  mycroft /*
    427  1.1  mycroft  * Find next entry > specification
    428  1.1  mycroft  */
    429  1.1  mycroft int
    430  1.1  mycroft next_grp_src_mask(gtpp, stpp, grp, src, mask)
    431  1.1  mycroft    struct gtable **gtpp;   /* ordered by group  */
    432  1.1  mycroft    struct stable **stpp;   /* ordered by source */
    433  1.1  mycroft    u_long grp;
    434  1.1  mycroft    u_long src;
    435  1.1  mycroft    u_long mask;
    436  1.1  mycroft {
    437  1.1  mycroft    struct gtable *gt, *gbest = NULL;
    438  1.1  mycroft    struct stable *st, *sbest = NULL;
    439  1.1  mycroft 
    440  1.1  mycroft    /* Find first group entry >= grp spec */
    441  1.1  mycroft    (*gtpp) = kernel_table;
    442  1.1  mycroft    while ((*gtpp) && ntohl((*gtpp)->gt_mcastgrp) < ntohl(grp))
    443  1.1  mycroft       (*gtpp)=(*gtpp)->gt_gnext;
    444  1.1  mycroft    if (!(*gtpp))
    445  1.1  mycroft       return 0; /* no more groups */
    446  1.1  mycroft 
    447  1.1  mycroft    for (gt = kernel_table; gt; gt=gt->gt_gnext) {
    448  1.1  mycroft       /* Since grps are ordered, we can stop when group changes from gbest */
    449  1.1  mycroft       if (gbest && gbest->gt_mcastgrp != gt->gt_mcastgrp)
    450  1.1  mycroft          break;
    451  1.1  mycroft       for (st = gt->gt_srctbl; st; st=st->st_next) {
    452  1.1  mycroft 
    453  1.1  mycroft          /* Among those entries > spec, find "lowest" one */
    454  1.1  mycroft          if (((ntohl(gt->gt_mcastgrp)> ntohl(grp))
    455  1.1  mycroft            || (ntohl(gt->gt_mcastgrp)==ntohl(grp)
    456  1.1  mycroft               && ntohl(st->st_origin)> ntohl(src))
    457  1.1  mycroft            || (ntohl(gt->gt_mcastgrp)==ntohl(grp)
    458  1.1  mycroft               && ntohl(st->st_origin)==src && 0xFFFFFFFF>ntohl(mask)))
    459  1.1  mycroft           && (!gbest
    460  1.1  mycroft            || (ntohl(gt->gt_mcastgrp)< ntohl(gbest->gt_mcastgrp))
    461  1.1  mycroft            || (ntohl(gt->gt_mcastgrp)==ntohl(gbest->gt_mcastgrp)
    462  1.1  mycroft               && ntohl(st->st_origin)< ntohl(sbest->st_origin)))) {
    463  1.1  mycroft                gbest = gt;
    464  1.1  mycroft                sbest = st;
    465  1.1  mycroft          }
    466  1.1  mycroft       }
    467  1.1  mycroft    }
    468  1.1  mycroft    (*gtpp) = gbest;
    469  1.1  mycroft    (*stpp) = sbest;
    470  1.1  mycroft    return (*gtpp)!=0;
    471  1.1  mycroft }
    472  1.1  mycroft 
    473  1.1  mycroft /*
    474  1.1  mycroft  * Ensure that sg contains current information for the given group,source.
    475  1.1  mycroft  * This is fetched from the kernel as a unit so that counts for the entry
    476  1.1  mycroft  * are consistent, i.e. packet and byte counts for the same entry are
    477  1.1  mycroft  * read at the same time.
    478  1.1  mycroft  */
    479  1.1  mycroft void
    480  1.1  mycroft refresh_sg(sg, gt, st)
    481  1.1  mycroft    struct sioc_sg_req *sg;
    482  1.1  mycroft    struct gtable *gt;
    483  1.1  mycroft    struct stable *st;
    484  1.1  mycroft {
    485  1.1  mycroft    static   int lastq = -1;
    486  1.1  mycroft 
    487  1.1  mycroft    if (quantum != lastq || sg->src.s_addr!=st->st_origin
    488  1.1  mycroft     || sg->grp.s_addr!=gt->gt_mcastgrp) {
    489  1.1  mycroft        lastq = quantum;
    490  1.1  mycroft        sg->src.s_addr = st->st_origin;
    491  1.1  mycroft        sg->grp.s_addr = gt->gt_mcastgrp;
    492  1.1  mycroft        ioctl(udp_socket, SIOCGETSGCNT, (char *)sg);
    493  1.1  mycroft    }
    494  1.1  mycroft }
    495  1.1  mycroft 
    496  1.1  mycroft /*
    497  1.1  mycroft  * Return pointer to a specific route entry.  This must be a separate
    498  1.1  mycroft  * function from find_route() which modifies rtp.
    499  1.1  mycroft  */
    500  1.1  mycroft struct rtentry *
    501  1.1  mycroft snmp_find_route(src, mask)
    502  1.1  mycroft     register u_long src, mask;
    503  1.1  mycroft {
    504  1.1  mycroft     register struct rtentry *rt;
    505  1.1  mycroft 
    506  1.1  mycroft    for (rt = routing_table; rt; rt = rt->rt_next) {
    507  1.1  mycroft       if (src == rt->rt_origin && mask == rt->rt_originmask)
    508  1.1  mycroft          return rt;
    509  1.1  mycroft    }
    510  1.1  mycroft    return NULL;
    511  1.1  mycroft }
    512  1.1  mycroft 
    513  1.1  mycroft /*
    514  1.1  mycroft  * Find next route entry > specification
    515  1.1  mycroft  */
    516  1.1  mycroft int
    517  1.1  mycroft next_route(rtpp, src, mask)
    518  1.1  mycroft    struct rtentry **rtpp;
    519  1.1  mycroft    u_long src;
    520  1.1  mycroft    u_long mask;
    521  1.1  mycroft {
    522  1.1  mycroft    struct rtentry *rt, *rbest = NULL;
    523  1.1  mycroft 
    524  1.1  mycroft    /* Among all entries > spec, find "lowest" one in order */
    525  1.1  mycroft    for (rt = routing_table; rt; rt=rt->rt_next) {
    526  1.1  mycroft       if ((ntohl(rt->rt_origin) > ntohl(src)
    527  1.1  mycroft           || (ntohl(rt->rt_origin) == ntohl(src)
    528  1.1  mycroft              && ntohl(rt->rt_originmask) > ntohl(mask)))
    529  1.1  mycroft        && (!rbest || (ntohl(rt->rt_origin) < ntohl(rbest->rt_origin))
    530  1.1  mycroft           || (ntohl(rt->rt_origin) == ntohl(rbest->rt_origin)
    531  1.1  mycroft              && ntohl(rt->rt_originmask) < ntohl(rbest->rt_originmask))))
    532  1.1  mycroft                rbest = rt;
    533  1.1  mycroft    }
    534  1.1  mycroft    (*rtpp) = rbest;
    535  1.1  mycroft    return (*rtpp)!=0;
    536  1.1  mycroft }
    537  1.1  mycroft 
    538  1.1  mycroft /*
    539  1.1  mycroft  * Given a routing table entry, and a vifi, find the next vifi/entry
    540  1.1  mycroft  */
    541  1.1  mycroft int
    542  1.1  mycroft next_route_child(rtpp, src, mask, vifi)
    543  1.1  mycroft    struct rtentry **rtpp;
    544  1.1  mycroft    u_long    src;
    545  1.1  mycroft    u_long    mask;
    546  1.1  mycroft    vifi_t   *vifi;     /* vif at which to start looking */
    547  1.1  mycroft {
    548  1.1  mycroft    struct rtentry *rt;
    549  1.1  mycroft 
    550  1.1  mycroft    /* Get (S,M) entry */
    551  1.1  mycroft    if (!((*rtpp) = snmp_find_route(src,mask)))
    552  1.1  mycroft       if (!next_route(rtpp, src, mask))
    553  1.1  mycroft          return 0;
    554  1.1  mycroft 
    555  1.1  mycroft    /* Continue until we get one with a valid next vif */
    556  1.1  mycroft    do {
    557  1.1  mycroft       for (; (*rtpp)->rt_children && *vifi<numvifs; (*vifi)++)
    558  1.1  mycroft          if (VIFM_ISSET(*vifi, (*rtpp)->rt_children))
    559  1.1  mycroft             return 1;
    560  1.1  mycroft       *vifi = 0;
    561  1.1  mycroft    } while( next_route(rtpp, (*rtpp)->rt_origin, (*rtpp)->rt_originmask) );
    562  1.1  mycroft 
    563  1.1  mycroft    return 0;
    564  1.1  mycroft }
    565  1.1  mycroft 
    566  1.1  mycroft /*
    567  1.1  mycroft  * Given a routing table entry, and a vifi, find the next entry
    568  1.1  mycroft  * equal to or greater than those
    569  1.1  mycroft  */
    570  1.1  mycroft int
    571  1.1  mycroft next_child(gtpp, stpp, grp, src, mask, vifi)
    572  1.1  mycroft    struct gtable **gtpp;
    573  1.1  mycroft    struct stable **stpp;
    574  1.1  mycroft    u_long    grp;
    575  1.1  mycroft    u_long    src;
    576  1.1  mycroft    u_long    mask;
    577  1.1  mycroft    vifi_t   *vifi;     /* vif at which to start looking */
    578  1.1  mycroft {
    579  1.1  mycroft    struct stable *st;
    580  1.1  mycroft 
    581  1.1  mycroft    /* Get (G,S,M) entry */
    582  1.1  mycroft    if (mask!=0xFFFFFFFF
    583  1.1  mycroft     || !((*gtpp) = find_grp(grp))
    584  1.1  mycroft     || !((*stpp) = find_grp_src((*gtpp),src)))
    585  1.1  mycroft       if (!next_grp_src_mask(gtpp, stpp, grp, src, mask))
    586  1.1  mycroft          return 0;
    587  1.1  mycroft 
    588  1.1  mycroft    /* Continue until we get one with a valid next vif */
    589  1.1  mycroft    do {
    590  1.1  mycroft       for (; (*gtpp)->gt_route->rt_children && *vifi<numvifs; (*vifi)++)
    591  1.1  mycroft          if (VIFM_ISSET(*vifi, (*gtpp)->gt_route->rt_children))
    592  1.1  mycroft             return 1;
    593  1.1  mycroft       *vifi = 0;
    594  1.1  mycroft    } while (next_grp_src_mask(gtpp, stpp, (*gtpp)->gt_mcastgrp,
    595  1.1  mycroft 		(*stpp)->st_origin, 0xFFFFFFFF) );
    596  1.1  mycroft 
    597  1.1  mycroft    return 0;
    598  1.1  mycroft }
    599  1.1  mycroft #endif /* SNMP */
    600  1.1  mycroft 
    601  1.1  mycroft /*
    602  1.1  mycroft  * Initialize the kernel table structure
    603  1.1  mycroft  */
    604  1.1  mycroft void
    605  1.1  mycroft init_ktable()
    606  1.1  mycroft {
    607  1.1  mycroft     kernel_table 	= NULL;
    608  1.1  mycroft     kernel_no_route	= NULL;
    609  1.1  mycroft     kroutes		= 0;
    610  1.1  mycroft }
    611  1.1  mycroft 
    612  1.1  mycroft /*
    613  1.1  mycroft  * Add a new table entry for (origin, mcastgrp)
    614  1.1  mycroft  */
    615  1.1  mycroft void
    616  1.1  mycroft add_table_entry(origin, mcastgrp)
    617  1.1  mycroft     u_int32_t origin;
    618  1.1  mycroft     u_int32_t mcastgrp;
    619  1.1  mycroft {
    620  1.1  mycroft     struct rtentry *r;
    621  1.1  mycroft     struct gtable *gt,**gtnp,*prev_gt;
    622  1.1  mycroft     struct stable *st,**stnp;
    623  1.1  mycroft     int i;
    624  1.1  mycroft 
    625  1.1  mycroft     r = determine_route(origin);
    626  1.1  mycroft     prev_gt = NULL;
    627  1.1  mycroft     if (r == NULL) {
    628  1.1  mycroft 	/*
    629  1.1  mycroft 	 * Look for it on the no_route table; if it is found then
    630  1.1  mycroft 	 * it will be detected as a duplicate below.
    631  1.1  mycroft 	 */
    632  1.1  mycroft 	for (gt = kernel_no_route; gt; gt = gt->gt_next)
    633  1.1  mycroft 	    if (mcastgrp == gt->gt_mcastgrp &&
    634  1.1  mycroft 		gt->gt_srctbl && gt->gt_srctbl->st_origin == origin)
    635  1.1  mycroft 			break;
    636  1.1  mycroft 	gtnp = &kernel_no_route;
    637  1.1  mycroft     } else {
    638  1.1  mycroft 	gtnp = &r->rt_groups;
    639  1.1  mycroft 	while ((gt = *gtnp) != NULL) {
    640  1.1  mycroft 	    if (gt->gt_mcastgrp >= mcastgrp)
    641  1.1  mycroft 		break;
    642  1.1  mycroft 	    gtnp = >->gt_next;
    643  1.1  mycroft 	    prev_gt = gt;
    644  1.1  mycroft 	}
    645  1.1  mycroft     }
    646  1.1  mycroft 
    647  1.1  mycroft     if (gt == NULL || gt->gt_mcastgrp != mcastgrp) {
    648  1.1  mycroft 	gt = (struct gtable *)malloc(sizeof(struct gtable));
    649  1.1  mycroft 	if (gt == NULL)
    650  1.1  mycroft 	    log(LOG_ERR, 0, "ran out of memory");
    651  1.1  mycroft 
    652  1.1  mycroft 	gt->gt_mcastgrp	    = mcastgrp;
    653  1.1  mycroft 	gt->gt_timer   	    = CACHE_LIFETIME(cache_lifetime);
    654  1.1  mycroft 	time(>->gt_ctime);
    655  1.1  mycroft 	gt->gt_grpmems	    = 0;
    656  1.1  mycroft 	gt->gt_scope	    = 0;
    657  1.1  mycroft 	gt->gt_prsent_timer = 0;
    658  1.1  mycroft 	gt->gt_grftsnt	    = 0;
    659  1.1  mycroft 	gt->gt_srctbl	    = NULL;
    660  1.1  mycroft 	gt->gt_pruntbl	    = NULL;
    661  1.1  mycroft 	gt->gt_route	    = r;
    662  1.1  mycroft 
    663  1.1  mycroft 	if (r != NULL) {
    664  1.1  mycroft 	    /* obtain the multicast group membership list */
    665  1.1  mycroft 	    for (i = 0; i < numvifs; i++) {
    666  1.1  mycroft 		if (VIFM_ISSET(i, r->rt_children) &&
    667  1.1  mycroft 		    !(VIFM_ISSET(i, r->rt_leaves)))
    668  1.1  mycroft 		    VIFM_SET(i, gt->gt_grpmems);
    669  1.1  mycroft 
    670  1.1  mycroft 		if (VIFM_ISSET(i, r->rt_leaves) && grplst_mem(i, mcastgrp))
    671  1.1  mycroft 		    VIFM_SET(i, gt->gt_grpmems);
    672  1.1  mycroft 	    }
    673  1.1  mycroft 	    GET_SCOPE(gt);
    674  1.1  mycroft 	    if (VIFM_ISSET(r->rt_parent, gt->gt_scope))
    675  1.1  mycroft 		gt->gt_scope = -1;
    676  1.1  mycroft 	    gt->gt_grpmems &= ~gt->gt_scope;
    677  1.1  mycroft 	} else {
    678  1.1  mycroft 	    gt->gt_scope = -1;
    679  1.1  mycroft 	    gt->gt_grpmems = 0;
    680  1.1  mycroft 	}
    681  1.1  mycroft 
    682  1.1  mycroft 	/* update ttls */
    683  1.1  mycroft 	prun_add_ttls(gt);
    684  1.1  mycroft 
    685  1.1  mycroft 	gt->gt_next = *gtnp;
    686  1.1  mycroft 	*gtnp = gt;
    687  1.1  mycroft 	if (gt->gt_next)
    688  1.1  mycroft 	    gt->gt_next->gt_prev = gt;
    689  1.1  mycroft 	gt->gt_prev = prev_gt;
    690  1.1  mycroft 
    691  1.1  mycroft 	if (r) {
    692  1.1  mycroft 	    if (find_src_grp(r->rt_origin, r->rt_originmask, gt->gt_mcastgrp)) {
    693  1.1  mycroft 		struct gtable *g;
    694  1.1  mycroft 
    695  1.1  mycroft 		g = gtp ? gtp->gt_gnext : kernel_table;
    696  1.1  mycroft 		log(LOG_WARNING, 0, "Entry for (%s %s) (rt:%x) exists (rt:%x)",
    697  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
    698  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2),
    699  1.1  mycroft 		    r, g->gt_route);
    700  1.1  mycroft 	    } else {
    701  1.1  mycroft 		if (gtp) {
    702  1.1  mycroft 		    gt->gt_gnext = gtp->gt_gnext;
    703  1.1  mycroft 		    gt->gt_gprev = gtp;
    704  1.1  mycroft 		    gtp->gt_gnext = gt;
    705  1.1  mycroft 		} else {
    706  1.1  mycroft 		    gt->gt_gnext = kernel_table;
    707  1.1  mycroft 		    gt->gt_gprev = NULL;
    708  1.1  mycroft 		    kernel_table = gt;
    709  1.1  mycroft 		}
    710  1.1  mycroft 		if (gt->gt_gnext)
    711  1.1  mycroft 		    gt->gt_gnext->gt_gprev = gt;
    712  1.1  mycroft 	    }
    713  1.1  mycroft 	} else {
    714  1.1  mycroft 	    gt->gt_gnext = gt->gt_prev = NULL;
    715  1.1  mycroft 	}
    716  1.1  mycroft     }
    717  1.1  mycroft 
    718  1.1  mycroft     stnp = >->gt_srctbl;
    719  1.1  mycroft     while ((st = *stnp) != NULL) {
    720  1.1  mycroft 	if (ntohl(st->st_origin) >= ntohl(origin))
    721  1.1  mycroft 	    break;
    722  1.1  mycroft 	stnp = &st->st_next;
    723  1.1  mycroft     }
    724  1.1  mycroft 
    725  1.1  mycroft     if (st == NULL || st->st_origin != origin) {
    726  1.1  mycroft 	st = (struct stable *)malloc(sizeof(struct stable));
    727  1.1  mycroft 	if (st == NULL)
    728  1.1  mycroft 	    log(LOG_ERR, 0, "ran out of memory");
    729  1.1  mycroft 
    730  1.1  mycroft 	st->st_origin = origin;
    731  1.1  mycroft 	st->st_pktcnt = 0;
    732  1.1  mycroft 	st->st_next = *stnp;
    733  1.1  mycroft 	*stnp = st;
    734  1.1  mycroft     } else {
    735  1.1  mycroft 	log(LOG_WARNING, 0, "kernel entry already exists for (%s %s)",
    736  1.1  mycroft 		inet_fmt(origin, s1), inet_fmt(mcastgrp, s2));
    737  1.1  mycroft 	return;
    738  1.1  mycroft     }
    739  1.1  mycroft 
    740  1.1  mycroft     kroutes++;
    741  1.1  mycroft     k_add_rg(origin, gt);
    742  1.1  mycroft 
    743  1.1  mycroft     log(LOG_DEBUG, 0, "add cache entry (%s %s) gm:%x, parent-vif:%d",
    744  1.1  mycroft 	inet_fmt(origin, s1),
    745  1.1  mycroft 	inet_fmt(mcastgrp, s2),
    746  1.1  mycroft 	gt->gt_grpmems, r ? r->rt_parent : -1);
    747  1.1  mycroft 
    748  1.1  mycroft     /* If there are no leaf vifs
    749  1.1  mycroft      * which have this group, then
    750  1.1  mycroft      * mark this src-grp as a prune candidate.
    751  1.1  mycroft      */
    752  1.1  mycroft     if (!gt->gt_prsent_timer && !gt->gt_grpmems && r && r->rt_gateway)
    753  1.1  mycroft 	send_prune(gt);
    754  1.1  mycroft }
    755  1.1  mycroft 
    756  1.1  mycroft /*
    757  1.1  mycroft  * An mrouter has gone down and come up on an interface
    758  1.1  mycroft  * Forward on that interface immediately
    759  1.1  mycroft  */
    760  1.1  mycroft void
    761  1.1  mycroft reset_neighbor_state(vifi, addr)
    762  1.1  mycroft     vifi_t vifi;
    763  1.1  mycroft     u_int32_t addr;
    764  1.1  mycroft {
    765  1.1  mycroft     struct rtentry *r;
    766  1.1  mycroft     struct gtable *g;
    767  1.1  mycroft     struct ptable *pt, *prev_pt;
    768  1.1  mycroft     struct stable *st, *prev_st;
    769  1.1  mycroft 
    770  1.1  mycroft     for (g = kernel_table; g; g = g->gt_gnext) {
    771  1.1  mycroft 	r = g->gt_route;
    772  1.1  mycroft 
    773  1.1  mycroft 	/*
    774  1.1  mycroft 	 * If neighbor was the parent, remove the prune sent state
    775  1.1  mycroft 	 * Don't send any grafts upstream.
    776  1.1  mycroft 	 */
    777  1.1  mycroft 	if (vifi == r->rt_parent) {
    778  1.1  mycroft 	    if (addr == r->rt_gateway) {
    779  1.1  mycroft 		log(LOG_DEBUG, 0, "reset_neighbor_state del prunes (%s %s)",
    780  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
    781  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2));
    782  1.1  mycroft 
    783  1.1  mycroft 		pt = g->gt_pruntbl;
    784  1.1  mycroft 		while (pt) {
    785  1.1  mycroft 		    /*
    786  1.1  mycroft 		     * Expire prune, send again on this vif.
    787  1.1  mycroft 		     */
    788  1.1  mycroft 		    VIFM_SET(pt->pt_vifi, g->gt_grpmems);
    789  1.1  mycroft 		    prev_pt = pt;
    790  1.1  mycroft 		    pt = prev_pt->pt_next;
    791  1.1  mycroft 		    free(prev_pt);
    792  1.1  mycroft 		}
    793  1.1  mycroft 		g->gt_pruntbl = NULL;
    794  1.1  mycroft 
    795  1.1  mycroft 		st = g->gt_srctbl;
    796  1.1  mycroft 		while (st) {
    797  1.1  mycroft 		    log(LOG_DEBUG, 0, "reset_neighbor_state del sg (%s %s)",
    798  1.1  mycroft 			inet_fmt(st->st_origin, s1),
    799  1.1  mycroft 			inet_fmt(g->gt_mcastgrp, s2));
    800  1.1  mycroft 
    801  1.1  mycroft 		    if (k_del_rg(st->st_origin, g) < 0) {
    802  1.1  mycroft 			log(LOG_WARNING, errno,
    803  1.1  mycroft 			    "reset_neighbor_state trying to delete (%s %s)",
    804  1.1  mycroft 			    inet_fmt(st->st_origin, s1),
    805  1.1  mycroft 			    inet_fmt(g->gt_mcastgrp, s2));
    806  1.1  mycroft 		    }
    807  1.1  mycroft 		    kroutes--;
    808  1.1  mycroft 		    prev_st = st;
    809  1.1  mycroft 		    st = prev_st->st_next;
    810  1.1  mycroft 		    free(prev_st);
    811  1.1  mycroft 		}
    812  1.1  mycroft 		g->gt_srctbl = NULL;
    813  1.1  mycroft 		/*
    814  1.1  mycroft 		 * Keep the group entries themselves around since the
    815  1.1  mycroft 		 * state will likely just come right back, and if not,
    816  1.1  mycroft 		 * the group entries will time out with no kernel entries
    817  1.1  mycroft 		 * and no prune state.
    818  1.1  mycroft 		 */
    819  1.1  mycroft 		g->gt_prsent_timer = 0;
    820  1.1  mycroft 		g->gt_grftsnt = 0;
    821  1.1  mycroft 	    }
    822  1.1  mycroft 	} else {
    823  1.1  mycroft 	    /*
    824  1.1  mycroft 	     * Neighbor was not the parent, send grafts to join the groups
    825  1.1  mycroft 	     */
    826  1.1  mycroft 	    if (g->gt_prsent_timer) {
    827  1.1  mycroft 		g->gt_grftsnt = 1;
    828  1.1  mycroft 		send_graft(g);
    829  1.1  mycroft 		g->gt_prsent_timer = 0;
    830  1.1  mycroft 	    }
    831  1.1  mycroft 
    832  1.1  mycroft 	    /*
    833  1.1  mycroft 	     * Remove any prunes that this router has sent us.
    834  1.1  mycroft 	     */
    835  1.1  mycroft 	    prev_pt = (struct ptable *)&g->gt_pruntbl;
    836  1.1  mycroft 	    for (pt = g->gt_pruntbl; pt; pt = pt->pt_next) {
    837  1.1  mycroft 		if (pt->pt_vifi == vifi && pt->pt_router == addr) {
    838  1.1  mycroft 		    prev_pt->pt_next = pt->pt_next;
    839  1.1  mycroft 		    free(pt);
    840  1.1  mycroft 		} else
    841  1.1  mycroft 		    prev_pt = pt;
    842  1.1  mycroft 	    }
    843  1.1  mycroft 
    844  1.1  mycroft 	    /*
    845  1.1  mycroft 	     * And see if we want to forward again.
    846  1.1  mycroft 	     */
    847  1.1  mycroft 	    if (!VIFM_ISSET(vifi, g->gt_grpmems)) {
    848  1.1  mycroft 		if (VIFM_ISSET(vifi, r->rt_children) &&
    849  1.1  mycroft 		    !(VIFM_ISSET(vifi, r->rt_leaves)))
    850  1.1  mycroft 		    VIFM_SET(vifi, g->gt_grpmems);
    851  1.1  mycroft 
    852  1.1  mycroft 		if (VIFM_ISSET(vifi, r->rt_leaves) &&
    853  1.1  mycroft 		    grplst_mem(vifi, g->gt_mcastgrp))
    854  1.1  mycroft 		    VIFM_SET(vifi, g->gt_grpmems);
    855  1.1  mycroft 
    856  1.1  mycroft 		g->gt_grpmems &= ~g->gt_scope;
    857  1.1  mycroft 		prun_add_ttls(g);
    858  1.1  mycroft 
    859  1.1  mycroft 		/* Update kernel state */
    860  1.1  mycroft 		update_kernel(g);
    861  1.1  mycroft 
    862  1.1  mycroft 		log(LOG_DEBUG, 0, "reset member state (%s %s) gm:%x",
    863  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
    864  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2), g->gt_grpmems);
    865  1.1  mycroft 	    }
    866  1.1  mycroft 	}
    867  1.1  mycroft     }
    868  1.1  mycroft }
    869  1.1  mycroft 
    870  1.1  mycroft /*
    871  1.1  mycroft  * Delete table entry from the kernel
    872  1.1  mycroft  * del_flag determines how many entries to delete
    873  1.1  mycroft  */
    874  1.1  mycroft void
    875  1.1  mycroft del_table_entry(r, mcastgrp, del_flag)
    876  1.1  mycroft     struct rtentry *r;
    877  1.1  mycroft     u_int32_t mcastgrp;
    878  1.1  mycroft     u_int  del_flag;
    879  1.1  mycroft {
    880  1.1  mycroft     struct gtable *g, *prev_g;
    881  1.1  mycroft     struct stable *st, *prev_st;
    882  1.1  mycroft     struct ptable *pt, *prev_pt;
    883  1.1  mycroft 
    884  1.1  mycroft     if (del_flag == DEL_ALL_ROUTES) {
    885  1.1  mycroft 	g = r->rt_groups;
    886  1.1  mycroft 	while (g) {
    887  1.1  mycroft 	    log(LOG_DEBUG, 0, "del_table_entry deleting (%s %s)",
    888  1.1  mycroft 		inet_fmts(r->rt_origin, r->rt_originmask, s1),
    889  1.1  mycroft 		inet_fmt(g->gt_mcastgrp, s2));
    890  1.1  mycroft 	    st = g->gt_srctbl;
    891  1.1  mycroft 	    while (st) {
    892  1.1  mycroft 		if (k_del_rg(st->st_origin, g) < 0) {
    893  1.1  mycroft 		    log(LOG_WARNING, errno,
    894  1.1  mycroft 			"del_table_entry trying to delete (%s, %s)",
    895  1.1  mycroft 			inet_fmt(st->st_origin, s1),
    896  1.1  mycroft 			inet_fmt(g->gt_mcastgrp, s2));
    897  1.1  mycroft 		}
    898  1.1  mycroft 		kroutes--;
    899  1.1  mycroft 		prev_st = st;
    900  1.1  mycroft 		st = st->st_next;
    901  1.1  mycroft 		free(prev_st);
    902  1.1  mycroft 	    }
    903  1.1  mycroft 	    g->gt_srctbl = NULL;
    904  1.1  mycroft 
    905  1.1  mycroft 	    pt = g->gt_pruntbl;
    906  1.1  mycroft 	    while (pt) {
    907  1.1  mycroft 		prev_pt = pt->pt_next;
    908  1.1  mycroft 		free(pt);
    909  1.1  mycroft 		pt = prev_pt;
    910  1.1  mycroft 	    }
    911  1.1  mycroft 	    g->gt_pruntbl = NULL;
    912  1.1  mycroft 
    913  1.1  mycroft 	    if (g->gt_gnext)
    914  1.1  mycroft 		g->gt_gnext->gt_gprev = g->gt_gprev;
    915  1.1  mycroft 	    if (g->gt_gprev)
    916  1.1  mycroft 		g->gt_gprev->gt_gnext = g->gt_gnext;
    917  1.1  mycroft 	    else
    918  1.1  mycroft 		kernel_table = g->gt_gnext;
    919  1.1  mycroft 
    920  1.1  mycroft 	    prev_g = g->gt_next;
    921  1.1  mycroft 	    free(g);
    922  1.1  mycroft 	    g = prev_g;
    923  1.1  mycroft 	}
    924  1.1  mycroft 	r->rt_groups = NULL;
    925  1.1  mycroft     }
    926  1.1  mycroft 
    927  1.1  mycroft     /*
    928  1.1  mycroft      * Dummy routine - someday this may be needed, so it is just there
    929  1.1  mycroft      */
    930  1.1  mycroft     if (del_flag == DEL_RTE_GROUP) {
    931  1.1  mycroft 	prev_g = (struct gtable *)&r->rt_groups;
    932  1.1  mycroft 	for (g = r->rt_groups; g; g = g->gt_next) {
    933  1.1  mycroft 	    if (g->gt_mcastgrp == mcastgrp) {
    934  1.1  mycroft 		log(LOG_DEBUG, 0, "del_table_entry deleting (%s %s)",
    935  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
    936  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2));
    937  1.1  mycroft 		st = g->gt_srctbl;
    938  1.1  mycroft 		while (st) {
    939  1.1  mycroft 		    if (k_del_rg(st->st_origin, g) < 0) {
    940  1.1  mycroft 			log(LOG_WARNING, errno,
    941  1.1  mycroft 			    "del_table_entry trying to delete (%s, %s)",
    942  1.1  mycroft 			    inet_fmt(st->st_origin, s1),
    943  1.1  mycroft 			    inet_fmt(g->gt_mcastgrp, s2));
    944  1.1  mycroft 		    }
    945  1.1  mycroft 		    kroutes--;
    946  1.1  mycroft 		    prev_st = st->st_next;
    947  1.1  mycroft 		    free(st);
    948  1.1  mycroft 		    st = prev_st;
    949  1.1  mycroft 		}
    950  1.1  mycroft 		g->gt_srctbl = NULL;
    951  1.1  mycroft 
    952  1.1  mycroft 		pt = g->gt_pruntbl;
    953  1.1  mycroft 		while (pt) {
    954  1.1  mycroft 		    prev_pt = pt->pt_next;
    955  1.1  mycroft 		    free(pt);
    956  1.1  mycroft 		    pt = prev_pt;
    957  1.1  mycroft 		}
    958  1.1  mycroft 		g->gt_pruntbl = NULL;
    959  1.1  mycroft 
    960  1.1  mycroft 		if (g->gt_gnext)
    961  1.1  mycroft 		    g->gt_gnext->gt_gprev = g->gt_gprev;
    962  1.1  mycroft 		if (g->gt_gprev)
    963  1.1  mycroft 		    g->gt_gprev->gt_gnext = g->gt_gnext;
    964  1.1  mycroft 		else
    965  1.1  mycroft 		    kernel_table = g->gt_gnext;
    966  1.1  mycroft 
    967  1.1  mycroft 		if (prev_g != (struct gtable *)&r->rt_groups)
    968  1.1  mycroft 		    g->gt_next->gt_prev = prev_g;
    969  1.1  mycroft 		else
    970  1.1  mycroft 		    g->gt_next->gt_prev = NULL;
    971  1.1  mycroft 		prev_g->gt_next = g->gt_next;
    972  1.1  mycroft 
    973  1.1  mycroft 		free(g);
    974  1.1  mycroft 		g = prev_g;
    975  1.1  mycroft 	    } else {
    976  1.1  mycroft 		prev_g = g;
    977  1.1  mycroft 	    }
    978  1.1  mycroft 	}
    979  1.1  mycroft     }
    980  1.1  mycroft }
    981  1.1  mycroft 
    982  1.1  mycroft /*
    983  1.1  mycroft  * update kernel table entry when a route entry changes
    984  1.1  mycroft  */
    985  1.1  mycroft void
    986  1.1  mycroft update_table_entry(r)
    987  1.1  mycroft     struct rtentry *r;
    988  1.1  mycroft {
    989  1.1  mycroft     struct gtable *g;
    990  1.1  mycroft     struct ptable *pt, *prev_pt;
    991  1.1  mycroft     int i;
    992  1.1  mycroft 
    993  1.1  mycroft     for (g = r->rt_groups; g; g = g->gt_next) {
    994  1.1  mycroft 	pt = g->gt_pruntbl;
    995  1.1  mycroft 	while (pt) {
    996  1.1  mycroft 	    prev_pt = pt->pt_next;
    997  1.1  mycroft 	    free(pt);
    998  1.1  mycroft 	    pt = prev_pt;
    999  1.1  mycroft 	}
   1000  1.1  mycroft 	g->gt_pruntbl = NULL;
   1001  1.1  mycroft 
   1002  1.1  mycroft 	g->gt_grpmems = 0;
   1003  1.1  mycroft 
   1004  1.1  mycroft 	/* obtain the multicast group membership list */
   1005  1.1  mycroft 	for (i = 0; i < numvifs; i++) {
   1006  1.1  mycroft 	    if (VIFM_ISSET(i, r->rt_children) &&
   1007  1.1  mycroft 		!(VIFM_ISSET(i, r->rt_leaves)))
   1008  1.1  mycroft 		VIFM_SET(i, g->gt_grpmems);
   1009  1.1  mycroft 
   1010  1.1  mycroft 	    if (VIFM_ISSET(i, r->rt_leaves) && grplst_mem(i, g->gt_mcastgrp))
   1011  1.1  mycroft 		VIFM_SET(i, g->gt_grpmems);
   1012  1.1  mycroft 	}
   1013  1.1  mycroft 	if (VIFM_ISSET(r->rt_parent, g->gt_scope))
   1014  1.1  mycroft 	    g->gt_scope = -1;
   1015  1.1  mycroft 	g->gt_grpmems &= ~g->gt_scope;
   1016  1.1  mycroft 
   1017  1.1  mycroft 	log(LOG_DEBUG, 0, "updating cache entries (%s %s) gm:%x",
   1018  1.1  mycroft 	    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1019  1.1  mycroft 	    inet_fmt(g->gt_mcastgrp, s2),
   1020  1.1  mycroft 	    g->gt_grpmems);
   1021  1.1  mycroft 
   1022  1.1  mycroft 	if (g->gt_grpmems && g->gt_prsent_timer) {
   1023  1.1  mycroft 	    g->gt_grftsnt = 1;
   1024  1.1  mycroft 	    send_graft(g);
   1025  1.1  mycroft 	    g->gt_prsent_timer = 0;
   1026  1.1  mycroft 	}
   1027  1.1  mycroft 
   1028  1.1  mycroft 	/* update ttls and add entry into kernel */
   1029  1.1  mycroft 	prun_add_ttls(g);
   1030  1.1  mycroft 	update_kernel(g);
   1031  1.1  mycroft 
   1032  1.1  mycroft 	/* Check if we want to prune this group */
   1033  1.1  mycroft 	if (!g->gt_prsent_timer && g->gt_grpmems == 0 && r->rt_gateway) {
   1034  1.1  mycroft 	    g->gt_timer = CACHE_LIFETIME(cache_lifetime);
   1035  1.1  mycroft 	    send_prune(g);
   1036  1.1  mycroft 	}
   1037  1.1  mycroft     }
   1038  1.1  mycroft }
   1039  1.1  mycroft 
   1040  1.1  mycroft /*
   1041  1.1  mycroft  * set the forwarding flag for all mcastgrps on this vifi
   1042  1.1  mycroft  */
   1043  1.1  mycroft void
   1044  1.1  mycroft update_lclgrp(vifi, mcastgrp)
   1045  1.1  mycroft     vifi_t vifi;
   1046  1.1  mycroft     u_int32_t mcastgrp;
   1047  1.1  mycroft {
   1048  1.1  mycroft     struct rtentry *r;
   1049  1.1  mycroft     struct gtable *g;
   1050  1.1  mycroft 
   1051  1.1  mycroft     log(LOG_DEBUG, 0, "group %s joined on vif %d",
   1052  1.1  mycroft 	inet_fmt(mcastgrp, s1), vifi);
   1053  1.1  mycroft 
   1054  1.1  mycroft     for (g = kernel_table; g; g = g->gt_gnext) {
   1055  1.1  mycroft 	if (ntohl(mcastgrp) < ntohl(g->gt_mcastgrp))
   1056  1.1  mycroft 	    break;
   1057  1.1  mycroft 
   1058  1.1  mycroft 	r = g->gt_route;
   1059  1.1  mycroft 	if (g->gt_mcastgrp == mcastgrp &&
   1060  1.1  mycroft 	    VIFM_ISSET(vifi, r->rt_children)) {
   1061  1.1  mycroft 
   1062  1.1  mycroft 	    VIFM_SET(vifi, g->gt_grpmems);
   1063  1.1  mycroft 	    g->gt_grpmems &= ~g->gt_scope;
   1064  1.1  mycroft 	    if (g->gt_grpmems == 0)
   1065  1.1  mycroft 		continue;
   1066  1.1  mycroft 
   1067  1.1  mycroft 	    prun_add_ttls(g);
   1068  1.1  mycroft 	    log(LOG_DEBUG, 0, "update lclgrp (%s %s) gm:%x",
   1069  1.1  mycroft 		inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1070  1.1  mycroft 		inet_fmt(g->gt_mcastgrp, s2), g->gt_grpmems);
   1071  1.1  mycroft 
   1072  1.1  mycroft 	    update_kernel(g);
   1073  1.1  mycroft 	}
   1074  1.1  mycroft     }
   1075  1.1  mycroft }
   1076  1.1  mycroft 
   1077  1.1  mycroft /*
   1078  1.1  mycroft  * reset forwarding flag for all mcastgrps on this vifi
   1079  1.1  mycroft  */
   1080  1.1  mycroft void
   1081  1.1  mycroft delete_lclgrp(vifi, mcastgrp)
   1082  1.1  mycroft     vifi_t vifi;
   1083  1.1  mycroft     u_int32_t mcastgrp;
   1084  1.1  mycroft {
   1085  1.1  mycroft     struct rtentry *r;
   1086  1.1  mycroft     struct gtable *g;
   1087  1.1  mycroft 
   1088  1.1  mycroft     log(LOG_DEBUG, 0, "group %s left on vif %d",
   1089  1.1  mycroft 	inet_fmt(mcastgrp, s1), vifi);
   1090  1.1  mycroft 
   1091  1.1  mycroft     for (g = kernel_table; g; g = g->gt_gnext) {
   1092  1.1  mycroft 	if (ntohl(mcastgrp) < ntohl(g->gt_mcastgrp))
   1093  1.1  mycroft 	    break;
   1094  1.1  mycroft 
   1095  1.1  mycroft 	if (g->gt_mcastgrp == mcastgrp) {
   1096  1.1  mycroft 	    int stop_sending = 1;
   1097  1.1  mycroft 
   1098  1.1  mycroft 	    r = g->gt_route;
   1099  1.1  mycroft 	    /*
   1100  1.1  mycroft 	     * If this is not a leaf, then we have router neighbors on this
   1101  1.1  mycroft 	     * vif.  Only turn off forwarding if they have all pruned.
   1102  1.1  mycroft 	     */
   1103  1.1  mycroft 	    if (!VIFM_ISSET(vifi, r->rt_leaves)) {
   1104  1.1  mycroft 		struct listaddr *vr;
   1105  1.1  mycroft 
   1106  1.1  mycroft 		for (vr = uvifs[vifi].uv_neighbors; vr; vr = vr->al_next)
   1107  1.1  mycroft 		  if (find_prune_entry(vr->al_addr, g->gt_pruntbl) == NULL) {
   1108  1.1  mycroft 		      stop_sending = 0;
   1109  1.1  mycroft 		      break;
   1110  1.1  mycroft 		  }
   1111  1.1  mycroft 	    }
   1112  1.1  mycroft 
   1113  1.1  mycroft 	    if (stop_sending) {
   1114  1.1  mycroft 		VIFM_CLR(vifi, g->gt_grpmems);
   1115  1.1  mycroft 		log(LOG_DEBUG, 0, "delete lclgrp (%s %s) gm:%x",
   1116  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1117  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2), g->gt_grpmems);
   1118  1.1  mycroft 
   1119  1.1  mycroft 		prun_add_ttls(g);
   1120  1.1  mycroft 		update_kernel(g);
   1121  1.1  mycroft 
   1122  1.1  mycroft 		/*
   1123  1.1  mycroft 		 * If there are no more members of this particular group,
   1124  1.1  mycroft 		 *  send prune upstream
   1125  1.1  mycroft 		 */
   1126  1.1  mycroft 		if (!g->gt_prsent_timer && g->gt_grpmems == 0 && r->rt_gateway)
   1127  1.1  mycroft 		    send_prune(g);
   1128  1.1  mycroft 	    }
   1129  1.1  mycroft 	}
   1130  1.1  mycroft     }
   1131  1.1  mycroft }
   1132  1.1  mycroft 
   1133  1.1  mycroft /*
   1134  1.1  mycroft  * Takes the prune message received and then strips it to
   1135  1.1  mycroft  * determine the (src, grp) pair to be pruned.
   1136  1.1  mycroft  *
   1137  1.1  mycroft  * Adds the router to the (src, grp) entry then.
   1138  1.1  mycroft  *
   1139  1.1  mycroft  * Determines if further packets have to be sent down that vif
   1140  1.1  mycroft  *
   1141  1.1  mycroft  * Determines if a corresponding prune message has to be generated
   1142  1.1  mycroft  */
   1143  1.1  mycroft void
   1144  1.1  mycroft accept_prune(src, dst, p, datalen)
   1145  1.1  mycroft     u_int32_t src;
   1146  1.1  mycroft     u_int32_t dst;
   1147  1.1  mycroft     char *p;
   1148  1.1  mycroft     int datalen;
   1149  1.1  mycroft {
   1150  1.1  mycroft     u_int32_t prun_src;
   1151  1.1  mycroft     u_int32_t prun_grp;
   1152  1.1  mycroft     u_int32_t prun_tmr;
   1153  1.1  mycroft     vifi_t vifi;
   1154  1.1  mycroft     int i;
   1155  1.1  mycroft     int stop_sending;
   1156  1.1  mycroft     struct rtentry *r;
   1157  1.1  mycroft     struct gtable *g;
   1158  1.1  mycroft     struct ptable *pt;
   1159  1.1  mycroft     struct listaddr *vr;
   1160  1.1  mycroft 
   1161  1.1  mycroft     /* Don't process any prunes if router is not pruning */
   1162  1.1  mycroft     if (pruning == 0)
   1163  1.1  mycroft 	return;
   1164  1.1  mycroft 
   1165  1.1  mycroft     if ((vifi = find_vif(src, dst)) == NO_VIF) {
   1166  1.1  mycroft 	log(LOG_INFO, 0,
   1167  1.1  mycroft     	    "ignoring prune report from non-neighbor %s",
   1168  1.1  mycroft 	    inet_fmt(src, s1));
   1169  1.1  mycroft 	return;
   1170  1.1  mycroft     }
   1171  1.1  mycroft 
   1172  1.1  mycroft     /* Check if enough data is present */
   1173  1.1  mycroft     if (datalen < 12)
   1174  1.1  mycroft 	{
   1175  1.1  mycroft 	    log(LOG_WARNING, 0,
   1176  1.1  mycroft 		"non-decipherable prune from %s",
   1177  1.1  mycroft 		inet_fmt(src, s1));
   1178  1.1  mycroft 	    return;
   1179  1.1  mycroft 	}
   1180  1.1  mycroft 
   1181  1.1  mycroft     for (i = 0; i< 4; i++)
   1182  1.1  mycroft 	((char *)&prun_src)[i] = *p++;
   1183  1.1  mycroft     for (i = 0; i< 4; i++)
   1184  1.1  mycroft 	((char *)&prun_grp)[i] = *p++;
   1185  1.1  mycroft     for (i = 0; i< 4; i++)
   1186  1.1  mycroft 	((char *)&prun_tmr)[i] = *p++;
   1187  1.1  mycroft 
   1188  1.1  mycroft     log(LOG_DEBUG, 0, "%s on vif %d prunes (%s %s)/%d",
   1189  1.1  mycroft 	inet_fmt(src, s1), vifi,
   1190  1.1  mycroft 	inet_fmt(prun_src, s2), inet_fmt(prun_grp, s3), prun_tmr);
   1191  1.1  mycroft 
   1192  1.1  mycroft     /*
   1193  1.1  mycroft      * Find the subnet for the prune
   1194  1.1  mycroft      */
   1195  1.1  mycroft     if (find_src_grp(prun_src, 0, prun_grp)) {
   1196  1.1  mycroft 	g = gtp ? gtp->gt_gnext : kernel_table;
   1197  1.1  mycroft     	r = g->gt_route;
   1198  1.1  mycroft 
   1199  1.1  mycroft 	if (!VIFM_ISSET(vifi, r->rt_children)) {
   1200  1.1  mycroft 	    log(LOG_WARNING, 0, "prune received from non-child %s for (%s %s)",
   1201  1.1  mycroft 		inet_fmt(src, s1), inet_fmt(prun_src, s2),
   1202  1.1  mycroft 		inet_fmt(prun_grp, s3));
   1203  1.1  mycroft 	    return;
   1204  1.1  mycroft 	}
   1205  1.1  mycroft 	if (VIFM_ISSET(vifi, g->gt_scope)) {
   1206  1.1  mycroft 	    log(LOG_WARNING, 0, "prune received from %s on scoped grp (%s %s)",
   1207  1.1  mycroft 		inet_fmt(src, s1), inet_fmt(prun_src, s2),
   1208  1.1  mycroft 		inet_fmt(prun_grp, s3));
   1209  1.1  mycroft 	    return;
   1210  1.1  mycroft 	}
   1211  1.1  mycroft 	if ((pt = find_prune_entry(src, g->gt_pruntbl)) != NULL) {
   1212  1.1  mycroft 	    /*
   1213  1.1  mycroft 	     * If it's about to expire, then it's only still around because
   1214  1.1  mycroft 	     * of timer granularity, so don't warn about it.
   1215  1.1  mycroft 	     */
   1216  1.1  mycroft 	    if (pt->pt_timer > 10) {
   1217  1.1  mycroft 		log(LOG_WARNING, 0, "%s %d from %s for (%s %s)/%d %s %d %s %x",
   1218  1.1  mycroft 		    "duplicate prune received on vif",
   1219  1.1  mycroft 		    vifi, inet_fmt(src, s1), inet_fmt(prun_src, s2),
   1220  1.1  mycroft 		    inet_fmt(prun_grp, s3), prun_tmr,
   1221  1.1  mycroft 		    "old timer:", pt->pt_timer, "cur gm:", g->gt_grpmems);
   1222  1.1  mycroft 	    }
   1223  1.1  mycroft 	    pt->pt_timer = prun_tmr;
   1224  1.1  mycroft 	} else {
   1225  1.1  mycroft 	    /* allocate space for the prune structure */
   1226  1.1  mycroft 	    pt = (struct ptable *)(malloc(sizeof(struct ptable)));
   1227  1.1  mycroft 	    if (pt == NULL)
   1228  1.1  mycroft 	      log(LOG_ERR, 0, "pt: ran out of memory");
   1229  1.1  mycroft 
   1230  1.1  mycroft 	    pt->pt_vifi = vifi;
   1231  1.1  mycroft 	    pt->pt_router = src;
   1232  1.1  mycroft 	    pt->pt_timer = prun_tmr;
   1233  1.1  mycroft 
   1234  1.1  mycroft 	    pt->pt_next = g->gt_pruntbl;
   1235  1.1  mycroft 	    g->gt_pruntbl = pt;
   1236  1.1  mycroft 	}
   1237  1.1  mycroft 
   1238  1.1  mycroft 	/* Refresh the group's lifetime */
   1239  1.1  mycroft 	g->gt_timer = CACHE_LIFETIME(cache_lifetime);
   1240  1.1  mycroft 	if (g->gt_timer < prun_tmr)
   1241  1.1  mycroft 	    g->gt_timer = prun_tmr;
   1242  1.1  mycroft 
   1243  1.1  mycroft 	/*
   1244  1.1  mycroft 	 * check if any more packets need to be sent on the
   1245  1.1  mycroft 	 * vif which sent this message
   1246  1.1  mycroft 	 */
   1247  1.1  mycroft 	stop_sending = 1;
   1248  1.1  mycroft 	for (vr = uvifs[vifi].uv_neighbors; vr; vr = vr->al_next)
   1249  1.1  mycroft 	  if (find_prune_entry(vr->al_addr, g->gt_pruntbl) == NULL)  {
   1250  1.1  mycroft 	      stop_sending = 0;
   1251  1.1  mycroft 	      break;
   1252  1.1  mycroft 	  }
   1253  1.1  mycroft 
   1254  1.1  mycroft 	if (stop_sending && !grplst_mem(vifi, prun_grp)) {
   1255  1.1  mycroft 	    VIFM_CLR(vifi, g->gt_grpmems);
   1256  1.1  mycroft 	    log(LOG_DEBUG, 0, "prune (%s %s), stop sending on vif %d, gm:%x",
   1257  1.1  mycroft 		inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1258  1.1  mycroft 		inet_fmt(g->gt_mcastgrp, s2), vifi, g->gt_grpmems);
   1259  1.1  mycroft 
   1260  1.1  mycroft 	    prun_add_ttls(g);
   1261  1.1  mycroft 	    update_kernel(g);
   1262  1.1  mycroft 	}
   1263  1.1  mycroft 
   1264  1.1  mycroft 	/*
   1265  1.1  mycroft 	 * check if all the child routers have expressed no interest
   1266  1.1  mycroft 	 * in this group and if this group does not exist in the
   1267  1.1  mycroft 	 * interface
   1268  1.1  mycroft 	 * Send a prune message then upstream
   1269  1.1  mycroft 	 */
   1270  1.1  mycroft 	if (!g->gt_prsent_timer && g->gt_grpmems == 0 && r->rt_gateway) {
   1271  1.1  mycroft 	    send_prune(g);
   1272  1.1  mycroft 	}
   1273  1.1  mycroft     } else {
   1274  1.1  mycroft 	/*
   1275  1.1  mycroft 	 * There is no kernel entry for this group.  Therefore, we can
   1276  1.1  mycroft 	 * simply ignore the prune, as we are not forwarding this traffic
   1277  1.1  mycroft 	 * downstream.
   1278  1.1  mycroft 	 */
   1279  1.1  mycroft 	log(LOG_DEBUG, 0, "%s (%s %s)/%d from %s",
   1280  1.1  mycroft 	    "prune message received with no kernel entry for",
   1281  1.1  mycroft 	    inet_fmt(prun_src, s1), inet_fmt(prun_grp, s2),
   1282  1.1  mycroft 	    prun_tmr, inet_fmt(src, s3));
   1283  1.1  mycroft 	return;
   1284  1.1  mycroft     }
   1285  1.1  mycroft }
   1286  1.1  mycroft 
   1287  1.1  mycroft /*
   1288  1.1  mycroft  * Checks if this mcastgrp is present in the kernel table
   1289  1.1  mycroft  * If so and if a prune was sent, it sends a graft upwards
   1290  1.1  mycroft  */
   1291  1.1  mycroft void
   1292  1.1  mycroft chkgrp_graft(vifi, mcastgrp)
   1293  1.1  mycroft     vifi_t	vifi;
   1294  1.1  mycroft     u_int32_t	mcastgrp;
   1295  1.1  mycroft {
   1296  1.1  mycroft     struct rtentry *r;
   1297  1.1  mycroft     struct gtable *g;
   1298  1.1  mycroft 
   1299  1.1  mycroft     for (g = kernel_table; g; g = g->gt_gnext) {
   1300  1.1  mycroft 	if (ntohl(mcastgrp) < ntohl(g->gt_mcastgrp))
   1301  1.1  mycroft 	    break;
   1302  1.1  mycroft 
   1303  1.1  mycroft 	r = g->gt_route;
   1304  1.1  mycroft 	if (g->gt_mcastgrp == mcastgrp && VIFM_ISSET(vifi, r->rt_children))
   1305  1.1  mycroft 	    if (g->gt_prsent_timer) {
   1306  1.1  mycroft 		VIFM_SET(vifi, g->gt_grpmems);
   1307  1.1  mycroft 
   1308  1.1  mycroft 		/*
   1309  1.1  mycroft 		 * If the vif that was joined was a scoped vif,
   1310  1.1  mycroft 		 * ignore it ; don't graft back
   1311  1.1  mycroft 		 */
   1312  1.1  mycroft 		g->gt_grpmems &= ~g->gt_scope;
   1313  1.1  mycroft 		if (g->gt_grpmems == 0)
   1314  1.1  mycroft 		    continue;
   1315  1.1  mycroft 
   1316  1.1  mycroft 		/* set the flag for graft retransmission */
   1317  1.1  mycroft 		g->gt_grftsnt = 1;
   1318  1.1  mycroft 
   1319  1.1  mycroft 		/* send graft upwards */
   1320  1.1  mycroft 		send_graft(g);
   1321  1.1  mycroft 
   1322  1.1  mycroft 		/* reset the prune timer and update cache timer*/
   1323  1.1  mycroft 		g->gt_prsent_timer = 0;
   1324  1.1  mycroft 		g->gt_timer = max_prune_lifetime;
   1325  1.1  mycroft 
   1326  1.1  mycroft 		log(LOG_DEBUG, 0, "chkgrp graft (%s %s) gm:%x",
   1327  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1328  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2), g->gt_grpmems);
   1329  1.1  mycroft 
   1330  1.1  mycroft 		prun_add_ttls(g);
   1331  1.1  mycroft 		update_kernel(g);
   1332  1.1  mycroft 	    }
   1333  1.1  mycroft     }
   1334  1.1  mycroft }
   1335  1.1  mycroft 
   1336  1.1  mycroft /* determine the multicast group and src
   1337  1.1  mycroft  *
   1338  1.1  mycroft  * if it does, then determine if a prune was sent
   1339  1.1  mycroft  * upstream.
   1340  1.1  mycroft  * if prune sent upstream, send graft upstream and send
   1341  1.1  mycroft  * ack downstream.
   1342  1.1  mycroft  *
   1343  1.1  mycroft  * if no prune sent upstream, change the forwarding bit
   1344  1.1  mycroft  * for this interface and send ack downstream.
   1345  1.1  mycroft  *
   1346  1.1  mycroft  * if no entry exists for this group send ack downstream.
   1347  1.1  mycroft  */
   1348  1.1  mycroft void
   1349  1.1  mycroft accept_graft(src, dst, p, datalen)
   1350  1.1  mycroft     u_int32_t 	src;
   1351  1.1  mycroft     u_int32_t	dst;
   1352  1.1  mycroft     char	*p;
   1353  1.1  mycroft     int		datalen;
   1354  1.1  mycroft {
   1355  1.1  mycroft     vifi_t 	vifi;
   1356  1.1  mycroft     u_int32_t 	graft_src;
   1357  1.1  mycroft     u_int32_t	graft_grp;
   1358  1.1  mycroft     int 	i;
   1359  1.1  mycroft     struct rtentry *r;
   1360  1.1  mycroft     struct gtable *g;
   1361  1.1  mycroft     struct ptable *pt, **ptnp;
   1362  1.1  mycroft 
   1363  1.1  mycroft     if ((vifi = find_vif(src, dst)) == NO_VIF) {
   1364  1.1  mycroft 	log(LOG_INFO, 0,
   1365  1.1  mycroft     	    "ignoring graft from non-neighbor %s",
   1366  1.1  mycroft 	    inet_fmt(src, s1));
   1367  1.1  mycroft 	return;
   1368  1.1  mycroft     }
   1369  1.1  mycroft 
   1370  1.1  mycroft     if (datalen < 8) {
   1371  1.1  mycroft 	log(LOG_WARNING, 0,
   1372  1.1  mycroft 	    "received non-decipherable graft from %s",
   1373  1.1  mycroft 	    inet_fmt(src, s1));
   1374  1.1  mycroft 	return;
   1375  1.1  mycroft     }
   1376  1.1  mycroft 
   1377  1.1  mycroft     for (i = 0; i< 4; i++)
   1378  1.1  mycroft 	((char *)&graft_src)[i] = *p++;
   1379  1.1  mycroft     for (i = 0; i< 4; i++)
   1380  1.1  mycroft 	((char *)&graft_grp)[i] = *p++;
   1381  1.1  mycroft 
   1382  1.1  mycroft     log(LOG_DEBUG, 0, "%s on vif %d grafts (%s %s)",
   1383  1.1  mycroft 	inet_fmt(src, s1), vifi,
   1384  1.1  mycroft 	inet_fmt(graft_src, s2), inet_fmt(graft_grp, s3));
   1385  1.1  mycroft 
   1386  1.1  mycroft     /*
   1387  1.1  mycroft      * Find the subnet for the graft
   1388  1.1  mycroft      */
   1389  1.1  mycroft     if (find_src_grp(graft_src, 0, graft_grp)) {
   1390  1.1  mycroft 	g = gtp ? gtp->gt_gnext : kernel_table;
   1391  1.1  mycroft 	r = g->gt_route;
   1392  1.1  mycroft 
   1393  1.1  mycroft 	if (VIFM_ISSET(vifi, g->gt_scope)) {
   1394  1.1  mycroft 	    log(LOG_WARNING, 0, "graft received from %s on scoped grp (%s %s)",
   1395  1.1  mycroft 		inet_fmt(src, s1), inet_fmt(graft_src, s2),
   1396  1.1  mycroft 		inet_fmt(graft_grp, s3));
   1397  1.1  mycroft 	    return;
   1398  1.1  mycroft 	}
   1399  1.1  mycroft 
   1400  1.1  mycroft 	ptnp = &g->gt_pruntbl;
   1401  1.1  mycroft 	while ((pt = *ptnp) != NULL) {
   1402  1.1  mycroft 	    if ((pt->pt_vifi == vifi) && (pt->pt_router == src)) {
   1403  1.1  mycroft 		*ptnp = pt->pt_next;
   1404  1.1  mycroft 		free(pt);
   1405  1.1  mycroft 
   1406  1.1  mycroft 		VIFM_SET(vifi, g->gt_grpmems);
   1407  1.1  mycroft 		log(LOG_DEBUG, 0, "accept graft (%s %s) gm:%x",
   1408  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1409  1.1  mycroft 		    inet_fmt(g->gt_mcastgrp, s2), g->gt_grpmems);
   1410  1.1  mycroft 
   1411  1.1  mycroft 		prun_add_ttls(g);
   1412  1.1  mycroft 		update_kernel(g);
   1413  1.1  mycroft 		break;
   1414  1.1  mycroft 	    } else {
   1415  1.1  mycroft 		ptnp = &pt->pt_next;
   1416  1.1  mycroft 	    }
   1417  1.1  mycroft 	}
   1418  1.1  mycroft 
   1419  1.1  mycroft 	/* send ack downstream */
   1420  1.1  mycroft 	send_graft_ack(dst, src, graft_src, graft_grp);
   1421  1.1  mycroft 	g->gt_timer = max_prune_lifetime;
   1422  1.1  mycroft 
   1423  1.1  mycroft 	if (g->gt_prsent_timer) {
   1424  1.1  mycroft 	    /* set the flag for graft retransmission */
   1425  1.1  mycroft 	    g->gt_grftsnt = 1;
   1426  1.1  mycroft 
   1427  1.1  mycroft 	    /* send graft upwards */
   1428  1.1  mycroft 	    send_graft(g);
   1429  1.1  mycroft 
   1430  1.1  mycroft 	    /* reset the prune sent timer */
   1431  1.1  mycroft 	    g->gt_prsent_timer = 0;
   1432  1.1  mycroft 	}
   1433  1.1  mycroft     } else {
   1434  1.1  mycroft 	/*
   1435  1.1  mycroft 	 * We have no state for the source and group in question.
   1436  1.1  mycroft 	 * We can simply acknowledge the graft, since we know
   1437  1.1  mycroft 	 * that we have no prune state, and grafts are requests
   1438  1.1  mycroft 	 * to remove prune state.
   1439  1.1  mycroft 	 */
   1440  1.1  mycroft 	send_graft_ack(dst, src, graft_src, graft_grp);
   1441  1.1  mycroft 	log(LOG_DEBUG, 0, "%s (%s %s) from %s",
   1442  1.1  mycroft 	    "graft received with no kernel entry for",
   1443  1.1  mycroft 	    inet_fmt(graft_src, s1), inet_fmt(graft_grp, s2),
   1444  1.1  mycroft 	    inet_fmt(src, s3));
   1445  1.1  mycroft 	return;
   1446  1.1  mycroft     }
   1447  1.1  mycroft }
   1448  1.1  mycroft 
   1449  1.1  mycroft /*
   1450  1.1  mycroft  * find out which group is involved first of all
   1451  1.1  mycroft  * then determine if a graft was sent.
   1452  1.1  mycroft  * if no graft sent, ignore the message
   1453  1.1  mycroft  * if graft was sent and the ack is from the right
   1454  1.1  mycroft  * source, remove the graft timer so that we don't
   1455  1.1  mycroft  * have send a graft again
   1456  1.1  mycroft  */
   1457  1.1  mycroft void
   1458  1.1  mycroft accept_g_ack(src, dst, p, datalen)
   1459  1.1  mycroft     u_int32_t 	src;
   1460  1.1  mycroft     u_int32_t	dst;
   1461  1.1  mycroft     char	*p;
   1462  1.1  mycroft     int		datalen;
   1463  1.1  mycroft {
   1464  1.1  mycroft     struct gtable *g;
   1465  1.1  mycroft     vifi_t 	vifi;
   1466  1.1  mycroft     u_int32_t 	grft_src;
   1467  1.1  mycroft     u_int32_t	grft_grp;
   1468  1.1  mycroft     int 	i;
   1469  1.1  mycroft 
   1470  1.1  mycroft     if ((vifi = find_vif(src, dst)) == NO_VIF) {
   1471  1.1  mycroft 	log(LOG_INFO, 0,
   1472  1.1  mycroft     	    "ignoring graft ack from non-neighbor %s",
   1473  1.1  mycroft 	    inet_fmt(src, s1));
   1474  1.1  mycroft 	return;
   1475  1.1  mycroft     }
   1476  1.1  mycroft 
   1477  1.1  mycroft     if (datalen < 0  || datalen > 8) {
   1478  1.1  mycroft 	log(LOG_WARNING, 0,
   1479  1.1  mycroft 	    "received non-decipherable graft ack from %s",
   1480  1.1  mycroft 	    inet_fmt(src, s1));
   1481  1.1  mycroft 	return;
   1482  1.1  mycroft     }
   1483  1.1  mycroft 
   1484  1.1  mycroft     for (i = 0; i< 4; i++)
   1485  1.1  mycroft 	((char *)&grft_src)[i] = *p++;
   1486  1.1  mycroft     for (i = 0; i< 4; i++)
   1487  1.1  mycroft 	((char *)&grft_grp)[i] = *p++;
   1488  1.1  mycroft 
   1489  1.1  mycroft     log(LOG_DEBUG, 0, "%s on vif %d acks graft (%s, %s)",
   1490  1.1  mycroft 	inet_fmt(src, s1), vifi,
   1491  1.1  mycroft 	inet_fmt(grft_src, s2), inet_fmt(grft_grp, s3));
   1492  1.1  mycroft 
   1493  1.1  mycroft     /*
   1494  1.1  mycroft      * Find the subnet for the graft ack
   1495  1.1  mycroft      */
   1496  1.1  mycroft     if (find_src_grp(grft_src, 0, grft_grp)) {
   1497  1.1  mycroft 	g = gtp ? gtp->gt_gnext : kernel_table;
   1498  1.1  mycroft 	g->gt_grftsnt = 0;
   1499  1.1  mycroft     } else {
   1500  1.1  mycroft 	log(LOG_WARNING, 0, "%s (%s, %s) from %s",
   1501  1.1  mycroft 	    "rcvd graft ack with no kernel entry for",
   1502  1.1  mycroft 	    inet_fmt(grft_src, s1), inet_fmt(grft_grp, s2),
   1503  1.1  mycroft 	    inet_fmt(src, s3));
   1504  1.1  mycroft 	return;
   1505  1.1  mycroft     }
   1506  1.1  mycroft }
   1507  1.1  mycroft 
   1508  1.1  mycroft 
   1509  1.1  mycroft /*
   1510  1.1  mycroft  * free all prune entries and kernel routes
   1511  1.1  mycroft  * normally, this should inform the kernel that all of its routes
   1512  1.1  mycroft  * are going away, but this is only called by restart(), which is
   1513  1.1  mycroft  * about to call MRT_DONE which does that anyway.
   1514  1.1  mycroft  */
   1515  1.1  mycroft void
   1516  1.1  mycroft free_all_prunes()
   1517  1.1  mycroft {
   1518  1.1  mycroft     register struct rtentry *r;
   1519  1.1  mycroft     register struct gtable *g, *prev_g;
   1520  1.1  mycroft     register struct stable *s, *prev_s;
   1521  1.1  mycroft     register struct ptable *p, *prev_p;
   1522  1.1  mycroft 
   1523  1.1  mycroft     for (r = routing_table; r; r = r->rt_next) {
   1524  1.1  mycroft 	g = r->rt_groups;
   1525  1.1  mycroft 	while (g) {
   1526  1.1  mycroft 	    s = g->gt_srctbl;
   1527  1.1  mycroft 	    while (s) {
   1528  1.1  mycroft 		prev_s = s->st_next;
   1529  1.1  mycroft 		free(s);
   1530  1.1  mycroft 		s = prev_s;
   1531  1.1  mycroft 	    }
   1532  1.1  mycroft 
   1533  1.1  mycroft 	    p = g->gt_pruntbl;
   1534  1.1  mycroft 	    while (p) {
   1535  1.1  mycroft 		prev_p = p->pt_next;
   1536  1.1  mycroft 		free(p);
   1537  1.1  mycroft 		p = prev_p;
   1538  1.1  mycroft 	    }
   1539  1.1  mycroft 
   1540  1.1  mycroft 	    prev_g = g->gt_next;
   1541  1.1  mycroft 	    free(g);
   1542  1.1  mycroft 	    g = prev_g;
   1543  1.1  mycroft 	}
   1544  1.1  mycroft 	r->rt_groups = NULL;
   1545  1.1  mycroft     }
   1546  1.1  mycroft     kernel_table = NULL;
   1547  1.1  mycroft 
   1548  1.1  mycroft     g = kernel_no_route;
   1549  1.1  mycroft     while (g) {
   1550  1.1  mycroft 	if (g->gt_srctbl)
   1551  1.1  mycroft 	    free(g->gt_srctbl);
   1552  1.1  mycroft 
   1553  1.1  mycroft 	prev_g = g->gt_next;
   1554  1.1  mycroft 	free(g);
   1555  1.1  mycroft 	g = prev_g;
   1556  1.1  mycroft     }
   1557  1.1  mycroft     kernel_no_route = NULL;
   1558  1.1  mycroft }
   1559  1.1  mycroft 
   1560  1.1  mycroft /*
   1561  1.1  mycroft  * When a new route is created, search
   1562  1.1  mycroft  * a) The less-specific part of the routing table
   1563  1.1  mycroft  * b) The route-less kernel table
   1564  1.1  mycroft  * for sources that the new route might want to handle.
   1565  1.1  mycroft  *
   1566  1.1  mycroft  * "Inheriting" these sources might be cleanest, but simply deleting
   1567  1.1  mycroft  * them is easier, and letting the kernel re-request them.
   1568  1.1  mycroft  */
   1569  1.1  mycroft void
   1570  1.1  mycroft steal_sources(rt)
   1571  1.1  mycroft     struct rtentry *rt;
   1572  1.1  mycroft {
   1573  1.1  mycroft     register struct rtentry *rp;
   1574  1.1  mycroft     register struct gtable *gt, **gtnp;
   1575  1.1  mycroft     register struct stable *st, **stnp;
   1576  1.1  mycroft 
   1577  1.1  mycroft     for (rp = rt->rt_next; rp; rp = rp->rt_next) {
   1578  1.1  mycroft 	if ((rt->rt_origin & rp->rt_originmask) == rp->rt_origin) {
   1579  1.1  mycroft 	    log(LOG_DEBUG, 0, "Route for %s stealing sources from %s",
   1580  1.1  mycroft 		inet_fmts(rt->rt_origin, rt->rt_originmask, s1),
   1581  1.1  mycroft 		inet_fmts(rp->rt_origin, rp->rt_originmask, s2));
   1582  1.1  mycroft 	    for (gt = rp->rt_groups; gt; gt = gt->gt_next) {
   1583  1.1  mycroft 		stnp = >->gt_srctbl;
   1584  1.1  mycroft 		while ((st = *stnp) != NULL) {
   1585  1.1  mycroft 		    if ((st->st_origin & rt->rt_originmask) == rt->rt_origin) {
   1586  1.1  mycroft 			log(LOG_DEBUG, 0, "%s stealing (%s %s) from %s",
   1587  1.1  mycroft 			    inet_fmts(rt->rt_origin, rt->rt_originmask, s1),
   1588  1.1  mycroft 			    inet_fmt(st->st_origin, s3),
   1589  1.1  mycroft 			    inet_fmt(gt->gt_mcastgrp, s4),
   1590  1.1  mycroft 			    inet_fmts(rp->rt_origin, rp->rt_originmask, s2));
   1591  1.1  mycroft 			if (k_del_rg(st->st_origin, gt) < 0) {
   1592  1.1  mycroft 			    log(LOG_WARNING, errno, "%s (%s, %s)",
   1593  1.1  mycroft 				"steal_sources trying to delete",
   1594  1.1  mycroft 				inet_fmt(st->st_origin, s1),
   1595  1.1  mycroft 				inet_fmt(gt->gt_mcastgrp, s2));
   1596  1.1  mycroft 			}
   1597  1.1  mycroft 			*stnp = st->st_next;
   1598  1.1  mycroft 			kroutes--;
   1599  1.1  mycroft 			free(st);
   1600  1.1  mycroft 		    } else {
   1601  1.1  mycroft 			stnp = &st->st_next;
   1602  1.1  mycroft 		    }
   1603  1.1  mycroft 		}
   1604  1.1  mycroft 	    }
   1605  1.1  mycroft 	}
   1606  1.1  mycroft     }
   1607  1.1  mycroft 
   1608  1.1  mycroft     gtnp = &kernel_no_route;
   1609  1.1  mycroft     while ((gt = *gtnp) != NULL) {
   1610  1.1  mycroft 	if (gt->gt_srctbl && ((gt->gt_srctbl->st_origin & rt->rt_originmask)
   1611  1.1  mycroft 				    == rt->rt_origin)) {
   1612  1.1  mycroft 	    log(LOG_DEBUG, 0, "%s stealing (%s %s) from %s",
   1613  1.1  mycroft 		inet_fmts(rt->rt_origin, rt->rt_originmask, s1),
   1614  1.1  mycroft 		inet_fmt(gt->gt_srctbl->st_origin, s3),
   1615  1.1  mycroft 		inet_fmt(gt->gt_mcastgrp, s4),
   1616  1.1  mycroft 		"no_route table");
   1617  1.1  mycroft 	    if (k_del_rg(gt->gt_srctbl->st_origin, gt) < 0) {
   1618  1.1  mycroft 		log(LOG_WARNING, errno, "%s (%s %s)",
   1619  1.1  mycroft 		    "steal_sources trying to delete",
   1620  1.1  mycroft 		    inet_fmt(gt->gt_srctbl->st_origin, s1),
   1621  1.1  mycroft 		    inet_fmt(gt->gt_mcastgrp, s2));
   1622  1.1  mycroft 	    }
   1623  1.1  mycroft 	    kroutes--;
   1624  1.1  mycroft 	    free(gt->gt_srctbl);
   1625  1.1  mycroft 	    *gtnp = gt->gt_next;
   1626  1.1  mycroft 	    if (gt->gt_next)
   1627  1.1  mycroft 		gt->gt_next->gt_prev = gt->gt_prev;
   1628  1.1  mycroft 	    free(gt);
   1629  1.1  mycroft 	} else {
   1630  1.1  mycroft 	    gtnp = >->gt_next;
   1631  1.1  mycroft 	}
   1632  1.1  mycroft     }
   1633  1.1  mycroft }
   1634  1.1  mycroft 
   1635  1.1  mycroft /*
   1636  1.1  mycroft  * Advance the timers on all the cache entries.
   1637  1.1  mycroft  * If there are any entries whose timers have expired,
   1638  1.1  mycroft  * remove these entries from the kernel cache.
   1639  1.1  mycroft  */
   1640  1.1  mycroft void
   1641  1.1  mycroft age_table_entry()
   1642  1.1  mycroft {
   1643  1.1  mycroft     struct rtentry *r;
   1644  1.1  mycroft     struct gtable *gt, **gtnptr;
   1645  1.1  mycroft     struct stable *st, **stnp;
   1646  1.1  mycroft     struct ptable *pt, **ptnp;
   1647  1.1  mycroft     struct sioc_sg_req sg_req;
   1648  1.1  mycroft 
   1649  1.1  mycroft     log(LOG_DEBUG, 0, "ageing entries");
   1650  1.1  mycroft 
   1651  1.1  mycroft     gtnptr = &kernel_table;
   1652  1.1  mycroft     while ((gt = *gtnptr) != NULL) {
   1653  1.1  mycroft 	r = gt->gt_route;
   1654  1.1  mycroft 
   1655  1.1  mycroft 	/* advance the timer for the kernel entry */
   1656  1.1  mycroft 	gt->gt_timer -= ROUTE_MAX_REPORT_DELAY;
   1657  1.1  mycroft 
   1658  1.1  mycroft 	/* decrement prune timer if need be */
   1659  1.1  mycroft 	if (gt->gt_prsent_timer > 0) {
   1660  1.1  mycroft 	    gt->gt_prsent_timer -= ROUTE_MAX_REPORT_DELAY;
   1661  1.1  mycroft 	    if (gt->gt_prsent_timer <= 0) {
   1662  1.1  mycroft 		log(LOG_DEBUG, 0, "upstream prune tmo (%s %s)",
   1663  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1664  1.1  mycroft 		    inet_fmt(gt->gt_mcastgrp, s2));
   1665  1.1  mycroft 		gt->gt_prsent_timer = -1;
   1666  1.1  mycroft 	    }
   1667  1.1  mycroft 	}
   1668  1.1  mycroft 
   1669  1.1  mycroft 	/* retransmit graft if graft sent flag is still set */
   1670  1.1  mycroft 	if (gt->gt_grftsnt) {
   1671  1.1  mycroft 	    register int y;
   1672  1.1  mycroft 	    CHK_GS(gt->gt_grftsnt++, y);
   1673  1.1  mycroft 	    if (y)
   1674  1.1  mycroft 		send_graft(gt);
   1675  1.1  mycroft 	}
   1676  1.1  mycroft 
   1677  1.1  mycroft 	/*
   1678  1.1  mycroft 	 * Age prunes
   1679  1.1  mycroft 	 *
   1680  1.1  mycroft 	 * If a prune expires, forward again on that vif.
   1681  1.1  mycroft 	 */
   1682  1.1  mycroft 	ptnp = >->gt_pruntbl;
   1683  1.1  mycroft 	while ((pt = *ptnp) != NULL) {
   1684  1.1  mycroft 	    if ((pt->pt_timer -= ROUTE_MAX_REPORT_DELAY) <= 0) {
   1685  1.1  mycroft 		log(LOG_DEBUG, 0, "expire prune (%s %s) from %s on vif %d",
   1686  1.1  mycroft 		    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1687  1.1  mycroft 		    inet_fmt(gt->gt_mcastgrp, s2),
   1688  1.1  mycroft 		    inet_fmt(pt->pt_router, s3),
   1689  1.1  mycroft 		    pt->pt_vifi);
   1690  1.1  mycroft 
   1691  1.1  mycroft 		/*
   1692  1.1  mycroft 		 * No need to send a graft, any prunes that we sent
   1693  1.1  mycroft 		 * will expire before any prunes that we have received.
   1694  1.1  mycroft 		 */
   1695  1.1  mycroft 		if (gt->gt_prsent_timer > 0) {
   1696  1.1  mycroft 		    log(LOG_DEBUG, 0, "prune expired with %d left on %s",
   1697  1.1  mycroft 			gt->gt_prsent_timer, "prsent_timer");
   1698  1.1  mycroft 		    gt->gt_prsent_timer = 0;
   1699  1.1  mycroft 		}
   1700  1.1  mycroft 
   1701  1.1  mycroft 		/* modify the kernel entry to forward packets */
   1702  1.1  mycroft 		if (!VIFM_ISSET(pt->pt_vifi, gt->gt_grpmems)) {
   1703  1.1  mycroft 		    VIFM_SET(pt->pt_vifi, gt->gt_grpmems);
   1704  1.1  mycroft 		    log(LOG_DEBUG, 0, "forw again (%s %s) gm:%x vif:%d",
   1705  1.1  mycroft 			inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1706  1.1  mycroft 			inet_fmt(gt->gt_mcastgrp, s2), gt->gt_grpmems,
   1707  1.1  mycroft 			pt->pt_vifi);
   1708  1.1  mycroft 
   1709  1.1  mycroft 		    prun_add_ttls(gt);
   1710  1.1  mycroft 		    update_kernel(gt);
   1711  1.1  mycroft 		}
   1712  1.1  mycroft 
   1713  1.1  mycroft 		/* remove the router's prune entry and await new one */
   1714  1.1  mycroft 		*ptnp = pt->pt_next;
   1715  1.1  mycroft 		free(pt);
   1716  1.1  mycroft 	    } else {
   1717  1.1  mycroft 		ptnp = &pt->pt_next;
   1718  1.1  mycroft 	    }
   1719  1.1  mycroft 	}
   1720  1.1  mycroft 
   1721  1.1  mycroft 	/*
   1722  1.1  mycroft 	 * If the cache entry has expired, check for downstream prunes.
   1723  1.1  mycroft 	 *
   1724  1.1  mycroft 	 * If there are downstream prunes, refresh the cache entry's timer.
   1725  1.1  mycroft 	 * Otherwise, check for traffic.  If no traffic, delete this
   1726  1.1  mycroft 	 * entry.
   1727  1.1  mycroft 	 */
   1728  1.1  mycroft 	if (gt->gt_timer <= 0) {
   1729  1.1  mycroft 	    if (gt->gt_pruntbl) {
   1730  1.1  mycroft 		if (gt->gt_prsent_timer == -1)
   1731  1.1  mycroft 		    gt->gt_prsent_timer = 0;
   1732  1.1  mycroft 		gt->gt_timer = CACHE_LIFETIME(cache_lifetime);
   1733  1.1  mycroft 		gtnptr = >->gt_gnext;
   1734  1.1  mycroft 		continue;
   1735  1.1  mycroft 	    }
   1736  1.1  mycroft 
   1737  1.1  mycroft 	    /*
   1738  1.1  mycroft 	     * If this entry was pruned, but all downstream prunes
   1739  1.1  mycroft 	     * have expired, then it is safe to simply delete it.
   1740  1.1  mycroft 	     * Otherwise, check for traffic before deleting.
   1741  1.1  mycroft 	     */
   1742  1.1  mycroft 	    if (gt->gt_prsent_timer == 0) {
   1743  1.1  mycroft 		sg_req.grp.s_addr = gt->gt_mcastgrp;
   1744  1.1  mycroft 		stnp = >->gt_srctbl;
   1745  1.1  mycroft 		while ((st = *stnp) != NULL) {
   1746  1.1  mycroft 		    sg_req.src.s_addr = st->st_origin;
   1747  1.1  mycroft 		    if (ioctl(udp_socket, SIOCGETSGCNT, (char *)&sg_req)
   1748  1.1  mycroft 			    < 0) {
   1749  1.1  mycroft 			log(LOG_WARNING, errno, "%s (%s %s)",
   1750  1.1  mycroft 			    "age_table_entry: SIOCGETSGCNT failing for",
   1751  1.1  mycroft 			    inet_fmt(st->st_origin, s1),
   1752  1.1  mycroft 			    inet_fmt(gt->gt_mcastgrp, s2));
   1753  1.1  mycroft 			/* Make sure it gets deleted below */
   1754  1.1  mycroft 			sg_req.pktcnt = st->st_pktcnt;
   1755  1.1  mycroft 		    }
   1756  1.1  mycroft 		    if (sg_req.pktcnt == st->st_pktcnt) {
   1757  1.1  mycroft 			*stnp = st->st_next;
   1758  1.1  mycroft 			log(LOG_DEBUG, 0,
   1759  1.1  mycroft 			    "age_table_entry deleting (%s %s)",
   1760  1.1  mycroft 			    inet_fmt(st->st_origin, s1),
   1761  1.1  mycroft 			    inet_fmt(gt->gt_mcastgrp, s2));
   1762  1.1  mycroft 			if (k_del_rg(st->st_origin, gt) < 0) {
   1763  1.1  mycroft 			    log(LOG_WARNING, errno,
   1764  1.1  mycroft 				"age_table_entry trying to delete (%s %s)",
   1765  1.1  mycroft 				inet_fmt(st->st_origin, s1),
   1766  1.1  mycroft 				inet_fmt(gt->gt_mcastgrp, s2));
   1767  1.1  mycroft 			}
   1768  1.1  mycroft 			kroutes--;
   1769  1.1  mycroft 			free(st);
   1770  1.1  mycroft 		    } else {
   1771  1.1  mycroft 			stnp = &st->st_next;
   1772  1.1  mycroft 		    }
   1773  1.1  mycroft 		}
   1774  1.1  mycroft 
   1775  1.1  mycroft 		if (gt->gt_srctbl) {
   1776  1.1  mycroft 		    /* At least one source in the list still has traffic */
   1777  1.1  mycroft 		    gt->gt_timer = CACHE_LIFETIME(cache_lifetime);
   1778  1.1  mycroft 		    gtnptr = >->gt_gnext;
   1779  1.1  mycroft 		    continue;
   1780  1.1  mycroft 		}
   1781  1.1  mycroft 	    }
   1782  1.1  mycroft 
   1783  1.1  mycroft 	    log(LOG_DEBUG, 0, "timeout cache entry (%s, %s)",
   1784  1.1  mycroft 		inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1785  1.1  mycroft 		inet_fmt(gt->gt_mcastgrp, s2));
   1786  1.1  mycroft 
   1787  1.1  mycroft 	    /* free all the source entries */
   1788  1.1  mycroft 	    while (st = gt->gt_srctbl) {
   1789  1.1  mycroft 		log(LOG_DEBUG, 0,
   1790  1.1  mycroft 		    "age_table_entry (P) deleting (%s %s)",
   1791  1.1  mycroft 		    inet_fmt(st->st_origin, s1),
   1792  1.1  mycroft 		    inet_fmt(gt->gt_mcastgrp, s2));
   1793  1.1  mycroft 		if (k_del_rg(st->st_origin, gt) < 0) {
   1794  1.1  mycroft 		    log(LOG_WARNING, errno,
   1795  1.1  mycroft 			"age_table_entry (P) trying to delete (%s %s)",
   1796  1.1  mycroft 			inet_fmt(st->st_origin, s1),
   1797  1.1  mycroft 			inet_fmt(gt->gt_mcastgrp, s2));
   1798  1.1  mycroft 		}
   1799  1.1  mycroft 		kroutes--;
   1800  1.1  mycroft 		gt->gt_srctbl = st->st_next;
   1801  1.1  mycroft 		free(st);
   1802  1.1  mycroft 	    }
   1803  1.1  mycroft 
   1804  1.1  mycroft 	    /* free all the prune list entries */
   1805  1.1  mycroft 	    while (gt->gt_pruntbl) {
   1806  1.1  mycroft 		gt->gt_pruntbl = pt->pt_next;
   1807  1.1  mycroft 		free(pt);
   1808  1.1  mycroft 	    }
   1809  1.1  mycroft 
   1810  1.1  mycroft 	    if (gt->gt_prev)
   1811  1.1  mycroft 		gt->gt_prev->gt_next = gt->gt_next;
   1812  1.1  mycroft 	    else
   1813  1.1  mycroft 		gt->gt_route->rt_groups = gt->gt_next;
   1814  1.1  mycroft 	    if (gt->gt_next)
   1815  1.1  mycroft 		gt->gt_next->gt_prev = gt->gt_prev;
   1816  1.1  mycroft 
   1817  1.1  mycroft 	    if (gt->gt_gprev) {
   1818  1.1  mycroft 		gt->gt_gprev->gt_gnext = gt->gt_gnext;
   1819  1.1  mycroft 		gtnptr = >->gt_gprev->gt_gnext;
   1820  1.1  mycroft 	    } else {
   1821  1.1  mycroft 		kernel_table = gt->gt_gnext;
   1822  1.1  mycroft 		gtnptr = &kernel_table;
   1823  1.1  mycroft 	    }
   1824  1.1  mycroft 	    if (gt->gt_gnext)
   1825  1.1  mycroft 		gt->gt_gnext->gt_gprev = gt->gt_gprev;
   1826  1.1  mycroft 
   1827  1.1  mycroft 	    free((char *)gt);
   1828  1.1  mycroft 	} else {
   1829  1.1  mycroft 	    if (gt->gt_prsent_timer == -1)
   1830  1.1  mycroft 		gt->gt_prsent_timer = 0;
   1831  1.1  mycroft 	    gtnptr = >->gt_gnext;
   1832  1.1  mycroft 	}
   1833  1.1  mycroft     }
   1834  1.1  mycroft 
   1835  1.1  mycroft     /*
   1836  1.1  mycroft      * When traversing the no_route table, the decision is much easier.
   1837  1.1  mycroft      * Just delete it if it has timed out.
   1838  1.1  mycroft      */
   1839  1.1  mycroft     gtnptr = &kernel_no_route;
   1840  1.1  mycroft     while ((gt = *gtnptr) != NULL) {
   1841  1.1  mycroft 	/* advance the timer for the kernel entry */
   1842  1.1  mycroft 	gt->gt_timer -= ROUTE_MAX_REPORT_DELAY;
   1843  1.1  mycroft 
   1844  1.1  mycroft 	if (gt->gt_timer < 0) {
   1845  1.1  mycroft 	    if (gt->gt_srctbl) {
   1846  1.1  mycroft 		if (k_del_rg(gt->gt_srctbl->st_origin, gt) < 0) {
   1847  1.1  mycroft 		    log(LOG_WARNING, errno, "%s (%s %s)",
   1848  1.1  mycroft 			"age_table_entry trying to delete no-route",
   1849  1.1  mycroft 			inet_fmt(gt->gt_srctbl->st_origin, s1),
   1850  1.1  mycroft 			inet_fmt(gt->gt_mcastgrp, s2));
   1851  1.1  mycroft 		}
   1852  1.1  mycroft 		free(gt->gt_srctbl);
   1853  1.1  mycroft 	    }
   1854  1.1  mycroft 	    *gtnptr = gt->gt_next;
   1855  1.1  mycroft 	    if (gt->gt_next)
   1856  1.1  mycroft 		gt->gt_next->gt_prev = gt->gt_prev;
   1857  1.1  mycroft 
   1858  1.1  mycroft 	    free((char *)gt);
   1859  1.1  mycroft 	} else {
   1860  1.1  mycroft 	    gtnptr = >->gt_next;
   1861  1.1  mycroft 	}
   1862  1.1  mycroft     }
   1863  1.1  mycroft }
   1864  1.1  mycroft 
   1865  1.1  mycroft char *
   1866  1.1  mycroft scaletime(t)
   1867  1.1  mycroft     u_long t;
   1868  1.1  mycroft {
   1869  1.1  mycroft     static char buf1[5];
   1870  1.1  mycroft     static char buf2[5];
   1871  1.1  mycroft     static char *buf=buf1;
   1872  1.1  mycroft     char s;
   1873  1.1  mycroft     char *p;
   1874  1.1  mycroft 
   1875  1.1  mycroft     p = buf;
   1876  1.1  mycroft     if (buf == buf1)
   1877  1.1  mycroft 	buf = buf2;
   1878  1.1  mycroft     else
   1879  1.1  mycroft 	buf = buf1;
   1880  1.1  mycroft 
   1881  1.1  mycroft     if (t < 120) {
   1882  1.1  mycroft 	s = 's';
   1883  1.1  mycroft     } else if (t < 3600) {
   1884  1.1  mycroft 	t /= 60;
   1885  1.1  mycroft 	s = 'm';
   1886  1.1  mycroft     } else if (t < 86400) {
   1887  1.1  mycroft 	t /= 3600;
   1888  1.1  mycroft 	s = 'h';
   1889  1.1  mycroft     } else if (t < 864000) {
   1890  1.1  mycroft 	t /= 86400;
   1891  1.1  mycroft 	s = 'd';
   1892  1.1  mycroft     } else {
   1893  1.1  mycroft 	t /= 604800;
   1894  1.1  mycroft 	s = 'w';
   1895  1.1  mycroft     }
   1896  1.1  mycroft     if (t > 999)
   1897  1.1  mycroft 	return "*** ";
   1898  1.1  mycroft 
   1899  1.1  mycroft     sprintf(p,"%3d%c", t, s);
   1900  1.1  mycroft 
   1901  1.1  mycroft     return p;
   1902  1.1  mycroft }
   1903  1.1  mycroft 
   1904  1.1  mycroft /*
   1905  1.1  mycroft  * Print the contents of the cache table on file 'fp2'.
   1906  1.1  mycroft  */
   1907  1.1  mycroft void
   1908  1.1  mycroft dump_cache(fp2)
   1909  1.1  mycroft     FILE *fp2;
   1910  1.1  mycroft {
   1911  1.1  mycroft     register struct rtentry *r;
   1912  1.1  mycroft     register struct gtable *gt;
   1913  1.1  mycroft     register struct stable *st;
   1914  1.1  mycroft     register struct ptable *pt;
   1915  1.1  mycroft     register int i;
   1916  1.1  mycroft     register time_t thyme = time(0);
   1917  1.1  mycroft 
   1918  1.1  mycroft     fprintf(fp2,
   1919  1.1  mycroft 	    "Multicast Routing Cache Table (%d entries)\n%s", kroutes,
   1920  1.1  mycroft     " Origin             Mcast-group     CTmr  Age Ptmr IVif Forwvifs\n");
   1921  1.1  mycroft 
   1922  1.1  mycroft     for (gt = kernel_no_route; gt; gt = gt->gt_next) {
   1923  1.1  mycroft 	if (gt->gt_srctbl) {
   1924  1.1  mycroft 	    fprintf(fp2, " %-18s %-15s %-4s %-4s    - -1\n",
   1925  1.1  mycroft 		inet_fmts(gt->gt_srctbl->st_origin, 0xffffffff, s1),
   1926  1.1  mycroft 		inet_fmt(gt->gt_mcastgrp, s2), scaletime(gt->gt_timer),
   1927  1.1  mycroft 		scaletime(thyme - gt->gt_ctime));
   1928  1.1  mycroft 	    fprintf(fp2, ">%s\n", inet_fmt(gt->gt_srctbl->st_origin, s1));
   1929  1.1  mycroft 	}
   1930  1.1  mycroft     }
   1931  1.1  mycroft 
   1932  1.1  mycroft     for (gt = kernel_table; gt; gt = gt->gt_gnext) {
   1933  1.1  mycroft 	r = gt->gt_route;
   1934  1.1  mycroft 	fprintf(fp2, " %-18s %-15s",
   1935  1.1  mycroft 	    inet_fmts(r->rt_origin, r->rt_originmask, s1),
   1936  1.1  mycroft 	    inet_fmt(gt->gt_mcastgrp, s2));
   1937  1.1  mycroft 
   1938  1.1  mycroft 	fprintf(fp2, " %-4s", scaletime(gt->gt_timer));
   1939  1.1  mycroft 
   1940  1.1  mycroft 	fprintf(fp2, " %-4s %-4s ", scaletime(thyme - gt->gt_ctime),
   1941  1.1  mycroft 			gt->gt_prsent_timer ? scaletime(gt->gt_prsent_timer) :
   1942  1.1  mycroft 					      "   -");
   1943  1.1  mycroft 
   1944  1.1  mycroft 	fprintf(fp2, "%2u%c%c ", r->rt_parent,
   1945  1.1  mycroft 	    gt->gt_prsent_timer ? 'P' : ' ',
   1946  1.1  mycroft 	    VIFM_ISSET(r->rt_parent, gt->gt_scope) ? 'B' : ' ');
   1947  1.1  mycroft 
   1948  1.1  mycroft 	for (i = 0; i < numvifs; ++i) {
   1949  1.1  mycroft 	    if (VIFM_ISSET(i, gt->gt_grpmems))
   1950  1.1  mycroft 		fprintf(fp2, " %u ", i);
   1951  1.1  mycroft 	    else if (VIFM_ISSET(i, r->rt_children) &&
   1952  1.1  mycroft 		     !VIFM_ISSET(i, r->rt_leaves))
   1953  1.1  mycroft 		fprintf(fp2, " %u%c", i,
   1954  1.1  mycroft 			VIFM_ISSET(i, gt->gt_scope) ? 'b' : 'p');
   1955  1.1  mycroft 	}
   1956  1.1  mycroft 	fprintf(fp2, "\n");
   1957  1.1  mycroft 	for (st = gt->gt_srctbl; st; st = st->st_next) {
   1958  1.1  mycroft 	    fprintf(fp2, ">%s\n", inet_fmt(st->st_origin, s1));
   1959  1.1  mycroft 	}
   1960  1.1  mycroft #ifdef DEBUG_PRUNES
   1961  1.1  mycroft 	for (pt = gt->gt_pruntbl; pt; pt = pt->pt_next) {
   1962  1.1  mycroft 	    fprintf(fp2, "<r:%s v:%d t:%d\n", inet_fmt(pt->pt_router, s1),
   1963  1.1  mycroft 		pt->pt_vifi, pt->pt_timer);
   1964  1.1  mycroft 	}
   1965  1.1  mycroft #endif
   1966  1.1  mycroft     }
   1967  1.1  mycroft }
   1968  1.1  mycroft 
   1969  1.1  mycroft /*
   1970  1.1  mycroft  * Traceroute function which returns traceroute replies to the requesting
   1971  1.1  mycroft  * router. Also forwards the request to downstream routers.
   1972  1.1  mycroft  */
   1973  1.1  mycroft void
   1974  1.1  mycroft accept_mtrace(src, dst, group, data, no, datalen)
   1975  1.1  mycroft     u_int32_t src;
   1976  1.1  mycroft     u_int32_t dst;
   1977  1.1  mycroft     u_int32_t group;
   1978  1.1  mycroft     char *data;
   1979  1.1  mycroft     u_char no;
   1980  1.1  mycroft     int datalen;
   1981  1.1  mycroft {
   1982  1.1  mycroft     u_char type;
   1983  1.1  mycroft     struct rtentry *rt;
   1984  1.1  mycroft     struct gtable *gt;
   1985  1.1  mycroft     struct tr_query *qry;
   1986  1.1  mycroft     struct tr_resp  *resp;
   1987  1.1  mycroft     int vifi;
   1988  1.1  mycroft     char *p;
   1989  1.1  mycroft     int rcount;
   1990  1.1  mycroft     int errcode = TR_NO_ERR;
   1991  1.1  mycroft     int resptype;
   1992  1.1  mycroft     struct timeval tp;
   1993  1.1  mycroft     struct sioc_vif_req v_req;
   1994  1.1  mycroft     struct sioc_sg_req sg_req;
   1995  1.1  mycroft 
   1996  1.1  mycroft     /* Remember qid across invocations */
   1997  1.1  mycroft     static u_int32_t oqid = 0;
   1998  1.1  mycroft 
   1999  1.1  mycroft     /* timestamp the request/response */
   2000  1.1  mycroft     gettimeofday(&tp, 0);
   2001  1.1  mycroft 
   2002  1.1  mycroft     /*
   2003  1.1  mycroft      * Check if it is a query or a response
   2004  1.1  mycroft      */
   2005  1.1  mycroft     if (datalen == QLEN) {
   2006  1.1  mycroft 	type = QUERY;
   2007  1.1  mycroft 	log(LOG_DEBUG, 0, "Traceroute query rcvd from %s to %s",
   2008  1.1  mycroft 	    inet_fmt(src, s1), inet_fmt(dst, s2));
   2009  1.1  mycroft     }
   2010  1.1  mycroft     else if ((datalen - QLEN) % RLEN == 0) {
   2011  1.1  mycroft 	type = RESP;
   2012  1.1  mycroft 	log(LOG_DEBUG, 0, "Traceroute response rcvd from %s to %s",
   2013  1.1  mycroft 	    inet_fmt(src, s1), inet_fmt(dst, s2));
   2014  1.1  mycroft 	if IN_MULTICAST(ntohl(dst)) {
   2015  1.1  mycroft 	    log(LOG_DEBUG, 0, "Dropping multicast response");
   2016  1.1  mycroft 	    return;
   2017  1.1  mycroft 	}
   2018  1.1  mycroft     }
   2019  1.1  mycroft     else {
   2020  1.1  mycroft 	log(LOG_WARNING, 0, "%s from %s to %s",
   2021  1.1  mycroft 	    "Non decipherable tracer request recieved",
   2022  1.1  mycroft 	    inet_fmt(src, s1), inet_fmt(dst, s2));
   2023  1.1  mycroft 	return;
   2024  1.1  mycroft     }
   2025  1.1  mycroft 
   2026  1.1  mycroft     qry = (struct tr_query *)data;
   2027  1.1  mycroft 
   2028  1.1  mycroft     if (oqid == qry->tr_qid) {
   2029  1.1  mycroft 	/*
   2030  1.1  mycroft 	 * If the multicast router is a member of the group being
   2031  1.1  mycroft 	 * queried, and the query is multicasted, then the router can
   2032  1.1  mycroft 	 * recieve multiple copies of the same query.  If we have already
   2033  1.1  mycroft 	 * replied to this traceroute, just ignore it this time.
   2034  1.1  mycroft 	 *
   2035  1.1  mycroft 	 * This is not a total solution, but since if this fails you
   2036  1.1  mycroft 	 * only get N copies, N <= the number of interfaces on the router,
   2037  1.1  mycroft 	 * it is not fatal.
   2038  1.1  mycroft 	 */
   2039  1.1  mycroft 	log(LOG_DEBUG, 0, "ignoring duplicate traceroute packet");
   2040  1.1  mycroft 	return;
   2041  1.1  mycroft     } else
   2042  1.1  mycroft 	oqid = qry->tr_qid;
   2043  1.1  mycroft 
   2044  1.1  mycroft     /*
   2045  1.1  mycroft      * if it is a packet with all reports filled, drop it
   2046  1.1  mycroft      */
   2047  1.1  mycroft     if ((rcount = (datalen - QLEN)/RLEN) == no) {
   2048  1.1  mycroft 	log(LOG_DEBUG, 0, "packet with all reports filled in");
   2049  1.1  mycroft 	return;
   2050  1.1  mycroft     }
   2051  1.1  mycroft 
   2052  1.1  mycroft     log(LOG_DEBUG, 0, "s: %s g: %s d: %s ", inet_fmt(qry->tr_src, s1),
   2053  1.1  mycroft 	    inet_fmt(group, s2), inet_fmt(qry->tr_dst, s3));
   2054  1.1  mycroft     log(LOG_DEBUG, 0, "rttl: %d rd: %s", qry->tr_rttl,
   2055  1.1  mycroft 	    inet_fmt(qry->tr_raddr, s1));
   2056  1.1  mycroft     log(LOG_DEBUG, 0, "rcount:%d", rcount);
   2057  1.1  mycroft 
   2058  1.1  mycroft     /* determine the routing table entry for this traceroute */
   2059  1.1  mycroft     rt = determine_route(qry->tr_src);
   2060  1.1  mycroft     if (rt) {
   2061  1.1  mycroft 	log(LOG_DEBUG, 0, "rt parent vif: %d rtr: %s metric: %d",
   2062  1.1  mycroft 		rt->rt_parent, inet_fmt(rt->rt_gateway, s1), rt->rt_metric);
   2063  1.1  mycroft 	log(LOG_DEBUG, 0, "rt origin %s",
   2064  1.1  mycroft 		inet_fmts(rt->rt_origin, rt->rt_originmask, s1));
   2065  1.1  mycroft     } else
   2066  1.1  mycroft 	log(LOG_DEBUG, 0, "...no route");
   2067  1.1  mycroft 
   2068  1.1  mycroft     /*
   2069  1.1  mycroft      * Query type packet - check if rte exists
   2070  1.1  mycroft      * Check if the query destination is a vif connected to me.
   2071  1.1  mycroft      * and if so, whether I should start response back
   2072  1.1  mycroft      */
   2073  1.1  mycroft     if (type == QUERY) {
   2074  1.1  mycroft 	if (rt == NULL) {
   2075  1.1  mycroft 	    log(LOG_DEBUG, 0, "Mcast traceroute: no route entry %s",
   2076  1.1  mycroft 		   inet_fmt(qry->tr_src, s1));
   2077  1.1  mycroft 	    if (IN_MULTICAST(ntohl(dst)))
   2078  1.1  mycroft 		return;
   2079  1.1  mycroft 	}
   2080  1.1  mycroft 	vifi = find_vif(qry->tr_dst, 0);
   2081  1.1  mycroft 
   2082  1.1  mycroft 	if (vifi == NO_VIF) {
   2083  1.1  mycroft 	    /* The traceroute destination is not on one of my subnet vifs. */
   2084  1.1  mycroft 	    log(LOG_DEBUG, 0, "Destination %s not an interface",
   2085  1.1  mycroft 		   inet_fmt(qry->tr_dst, s1));
   2086  1.1  mycroft 	    if (IN_MULTICAST(ntohl(dst)))
   2087  1.1  mycroft 		return;
   2088  1.1  mycroft 	    errcode = TR_WRONG_IF;
   2089  1.1  mycroft 	} else if (rt != NULL && !VIFM_ISSET(vifi, rt->rt_children)) {
   2090  1.1  mycroft 	    log(LOG_DEBUG, 0, "Destination %s not on forwarding tree for src %s",
   2091  1.1  mycroft 		   inet_fmt(qry->tr_dst, s1), inet_fmt(qry->tr_src, s2));
   2092  1.1  mycroft 	    if (IN_MULTICAST(ntohl(dst)))
   2093  1.1  mycroft 		return;
   2094  1.1  mycroft 	    errcode = TR_WRONG_IF;
   2095  1.1  mycroft 	}
   2096  1.1  mycroft     }
   2097  1.1  mycroft     else {
   2098  1.1  mycroft 	/*
   2099  1.1  mycroft 	 * determine which interface the packet came in on
   2100  1.1  mycroft 	 * RESP packets travel hop-by-hop so this either traversed
   2101  1.1  mycroft 	 * a tunnel or came from a directly attached mrouter.
   2102  1.1  mycroft 	 */
   2103  1.1  mycroft 	if ((vifi = find_vif(src, dst)) == NO_VIF) {
   2104  1.1  mycroft 	    log(LOG_DEBUG, 0, "Wrong interface for packet");
   2105  1.1  mycroft 	    errcode = TR_WRONG_IF;
   2106  1.1  mycroft 	}
   2107  1.1  mycroft     }
   2108  1.1  mycroft 
   2109  1.1  mycroft     log(LOG_DEBUG, 0, "Sending traceroute response");
   2110  1.1  mycroft 
   2111  1.1  mycroft     /* copy the packet to the sending buffer */
   2112  1.1  mycroft     p = send_buf + MIN_IP_HEADER_LEN + IGMP_MINLEN;
   2113  1.1  mycroft 
   2114  1.1  mycroft     bcopy(data, p, datalen);
   2115  1.1  mycroft 
   2116  1.1  mycroft     p += datalen;
   2117  1.1  mycroft 
   2118  1.1  mycroft     /*
   2119  1.1  mycroft      * If there is no room to insert our reply, coopt the previous hop
   2120  1.1  mycroft      * error indication to relay this fact.
   2121  1.1  mycroft      */
   2122  1.1  mycroft     if (p + sizeof(struct tr_resp) > send_buf + RECV_BUF_SIZE) {
   2123  1.1  mycroft 	resp = (struct tr_resp *)p - 1;
   2124  1.1  mycroft 	resp->tr_rflags = TR_NO_SPACE;
   2125  1.1  mycroft 	rt = NULL;
   2126  1.1  mycroft 	goto sendit;
   2127  1.1  mycroft     }
   2128  1.1  mycroft 
   2129  1.1  mycroft     /*
   2130  1.1  mycroft      * fill in initial response fields
   2131  1.1  mycroft      */
   2132  1.1  mycroft     resp = (struct tr_resp *)p;
   2133  1.1  mycroft     bzero(resp, sizeof(struct tr_resp));
   2134  1.1  mycroft     datalen += RLEN;
   2135  1.1  mycroft 
   2136  1.1  mycroft     resp->tr_qarr    = ((tp.tv_sec + JAN_1970) << 16) +
   2137  1.1  mycroft 				((tp.tv_usec >> 4) & 0xffff);
   2138  1.1  mycroft 
   2139  1.1  mycroft     resp->tr_rproto  = PROTO_DVMRP;
   2140  1.1  mycroft     if (errcode != TR_NO_ERR) {
   2141  1.1  mycroft 	resp->tr_rflags	 = errcode;
   2142  1.1  mycroft 	rt = NULL;	/* hack to enforce send straight to requestor */
   2143  1.1  mycroft 	goto sendit;
   2144  1.1  mycroft     }
   2145  1.1  mycroft     resp->tr_outaddr = uvifs[vifi].uv_lcl_addr;
   2146  1.1  mycroft     resp->tr_fttl    = uvifs[vifi].uv_threshold;
   2147  1.1  mycroft     resp->tr_rflags  = TR_NO_ERR;
   2148  1.1  mycroft 
   2149  1.1  mycroft     /*
   2150  1.1  mycroft      * obtain # of packets out on interface
   2151  1.1  mycroft      */
   2152  1.1  mycroft     v_req.vifi = vifi;
   2153  1.1  mycroft     if (ioctl(udp_socket, SIOCGETVIFCNT, (char *)&v_req) >= 0)
   2154  1.1  mycroft 	resp->tr_vifout  =  v_req.ocount;
   2155  1.1  mycroft 
   2156  1.1  mycroft     /*
   2157  1.1  mycroft      * fill in scoping & pruning information
   2158  1.1  mycroft      */
   2159  1.1  mycroft     if (rt)
   2160  1.1  mycroft 	for (gt = rt->rt_groups; gt; gt = gt->gt_next) {
   2161  1.1  mycroft 	    if (gt->gt_mcastgrp >= group)
   2162  1.1  mycroft 		break;
   2163  1.1  mycroft 	}
   2164  1.1  mycroft     else
   2165  1.1  mycroft 	gt = NULL;
   2166  1.1  mycroft 
   2167  1.1  mycroft     if (gt && gt->gt_mcastgrp == group) {
   2168  1.1  mycroft 	sg_req.src.s_addr = qry->tr_src;
   2169  1.1  mycroft 	sg_req.grp.s_addr = group;
   2170  1.1  mycroft 	if (ioctl(udp_socket, SIOCGETSGCNT, (char *)&sg_req) >= 0)
   2171  1.1  mycroft 	    resp->tr_pktcnt = sg_req.pktcnt;
   2172  1.1  mycroft 
   2173  1.1  mycroft 	if (VIFM_ISSET(vifi, gt->gt_scope))
   2174  1.1  mycroft 	    resp->tr_rflags = TR_SCOPED;
   2175  1.1  mycroft 	else if (gt->gt_prsent_timer)
   2176  1.1  mycroft 	    resp->tr_rflags = TR_PRUNED;
   2177  1.1  mycroft 	else if (!VIFM_ISSET(vifi, gt->gt_grpmems))
   2178  1.1  mycroft 	    if (VIFM_ISSET(vifi, rt->rt_children) &&
   2179  1.1  mycroft 		!VIFM_ISSET(vifi, rt->rt_leaves))
   2180  1.1  mycroft 		resp->tr_rflags = TR_OPRUNED;
   2181  1.1  mycroft 	    else
   2182  1.1  mycroft 		resp->tr_rflags = TR_NO_FWD;
   2183  1.1  mycroft     } else {
   2184  1.1  mycroft 	if (scoped_addr(vifi, group))
   2185  1.1  mycroft 	    resp->tr_rflags = TR_SCOPED;
   2186  1.1  mycroft 	else if (!VIFM_ISSET(vifi, rt->rt_children))
   2187  1.1  mycroft 	    resp->tr_rflags = TR_NO_FWD;
   2188  1.1  mycroft     }
   2189  1.1  mycroft 
   2190  1.1  mycroft     /*
   2191  1.1  mycroft      *  if no rte exists, set NO_RTE error
   2192  1.1  mycroft      */
   2193  1.1  mycroft     if (rt == NULL) {
   2194  1.1  mycroft 	src = dst;		/* the dst address of resp. pkt */
   2195  1.1  mycroft 	resp->tr_inaddr   = 0;
   2196  1.1  mycroft 	resp->tr_rflags   = TR_NO_RTE;
   2197  1.1  mycroft 	resp->tr_rmtaddr  = 0;
   2198  1.1  mycroft     } else {
   2199  1.1  mycroft 	/* get # of packets in on interface */
   2200  1.1  mycroft 	v_req.vifi = rt->rt_parent;
   2201  1.1  mycroft 	if (ioctl(udp_socket, SIOCGETVIFCNT, (char *)&v_req) >= 0)
   2202  1.1  mycroft 	    resp->tr_vifin = v_req.icount;
   2203  1.1  mycroft 
   2204  1.1  mycroft 	MASK_TO_VAL(rt->rt_originmask, resp->tr_smask);
   2205  1.1  mycroft 	src = uvifs[rt->rt_parent].uv_lcl_addr;
   2206  1.1  mycroft 	resp->tr_inaddr = src;
   2207  1.1  mycroft 	resp->tr_rmtaddr = rt->rt_gateway;
   2208  1.1  mycroft 	if (!VIFM_ISSET(vifi, rt->rt_children)) {
   2209  1.1  mycroft 	    log(LOG_DEBUG, 0, "Destination %s not on forwarding tree for src %s",
   2210  1.1  mycroft 		   inet_fmt(qry->tr_dst, s1), inet_fmt(qry->tr_src, s2));
   2211  1.1  mycroft 	    resp->tr_rflags = TR_WRONG_IF;
   2212  1.1  mycroft 	}
   2213  1.1  mycroft 	if (rt->rt_metric >= UNREACHABLE) {
   2214  1.1  mycroft 	    resp->tr_rflags = TR_NO_RTE;
   2215  1.1  mycroft 	    /* Hack to send reply directly */
   2216  1.1  mycroft 	    rt = NULL;
   2217  1.1  mycroft 	}
   2218  1.1  mycroft     }
   2219  1.1  mycroft 
   2220  1.1  mycroft sendit:
   2221  1.1  mycroft     /*
   2222  1.1  mycroft      * if metric is 1 or no. of reports is 1, send response to requestor
   2223  1.1  mycroft      * else send to upstream router.  If the upstream router can't handle
   2224  1.1  mycroft      * mtrace, set an error code and send to requestor anyway.
   2225  1.1  mycroft      */
   2226  1.1  mycroft     log(LOG_DEBUG, 0, "rcount:%d, no:%d", rcount, no);
   2227  1.1  mycroft 
   2228  1.1  mycroft     if ((rcount + 1 == no) || (rt == NULL) || (rt->rt_metric == 1)) {
   2229  1.1  mycroft 	resptype = IGMP_MTRACE_REPLY;
   2230  1.1  mycroft 	dst = qry->tr_raddr;
   2231  1.1  mycroft     } else
   2232  1.1  mycroft 	if (!can_mtrace(rt->rt_parent, rt->rt_gateway)) {
   2233  1.1  mycroft 	    dst = qry->tr_raddr;
   2234  1.1  mycroft 	    resp->tr_rflags = TR_OLD_ROUTER;
   2235  1.1  mycroft 	    resptype = IGMP_MTRACE_REPLY;
   2236  1.1  mycroft 	} else {
   2237  1.1  mycroft 	    dst = rt->rt_gateway;
   2238  1.1  mycroft 	    resptype = IGMP_MTRACE_QUERY;
   2239  1.1  mycroft 	}
   2240  1.1  mycroft 
   2241  1.1  mycroft     log(LOG_DEBUG, 0, "Sending %s to %s from %s",
   2242  1.1  mycroft 	resptype == IGMP_MTRACE_REPLY ?  "response" : "request on",
   2243  1.1  mycroft 	inet_fmt(dst, s1), inet_fmt(src, s2));
   2244  1.1  mycroft 
   2245  1.1  mycroft     if (IN_MULTICAST(ntohl(dst))) {
   2246  1.1  mycroft 	k_set_ttl(qry->tr_rttl);
   2247  1.1  mycroft 	/* Let the kernel pick the source address, since we might have picked
   2248  1.1  mycroft 	 * a disabled phyint to multicast on.
   2249  1.1  mycroft 	 */
   2250  1.1  mycroft 	send_igmp(INADDR_ANY, dst,
   2251  1.1  mycroft 		  resptype, no, group,
   2252  1.1  mycroft 		  datalen);
   2253  1.1  mycroft 	k_set_ttl(1);
   2254  1.1  mycroft     } else
   2255  1.1  mycroft 	send_igmp(src, dst,
   2256  1.1  mycroft 		  resptype, no, group,
   2257  1.1  mycroft 		  datalen);
   2258  1.1  mycroft     return;
   2259  1.1  mycroft }
   2260