Home | History | Annotate | Line # | Download | only in mrouted
igmp.c revision 1.1
      1  1.1  brezak /*
      2  1.1  brezak  * The mrouted program is covered by the license in the accompanying file
      3  1.1  brezak  * named "LICENSE".  Use of the mrouted program represents acceptance of
      4  1.1  brezak  * the terms and conditions listed in that file.
      5  1.1  brezak  *
      6  1.1  brezak  * The mrouted program is COPYRIGHT 1989 by The Board of Trustees of
      7  1.1  brezak  * Leland Stanford Junior University.
      8  1.1  brezak  *
      9  1.1  brezak  *
     10  1.1  brezak  * from: Id: igmp.c,v 1.5 1993/06/23 18:47:17 pavel Exp
     11  1.1  brezak  *      $Id: igmp.c,v 1.1 1994/01/11 20:15:53 brezak Exp $
     12  1.1  brezak  */
     13  1.1  brezak 
     14  1.1  brezak #ifndef lint
     15  1.1  brezak static char rcsid[] = "$Id: igmp.c,v 1.1 1994/01/11 20:15:53 brezak Exp $";
     16  1.1  brezak #endif
     17  1.1  brezak 
     18  1.1  brezak #include "defs.h"
     19  1.1  brezak 
     20  1.1  brezak 
     21  1.1  brezak /*
     22  1.1  brezak  * Exported variables.
     23  1.1  brezak  */
     24  1.1  brezak char		recv_buf[MAX_IP_PACKET_LEN]; /* input packet buffer         */
     25  1.1  brezak char		send_buf[MAX_IP_PACKET_LEN]; /* output packet buffer        */
     26  1.1  brezak int		igmp_socket;		     /* socket for all network I/O  */
     27  1.1  brezak u_long		allhosts_group;		     /* allhosts  addr in net order */
     28  1.1  brezak u_long		dvmrp_group;		     /* DVMRP grp addr in net order */
     29  1.1  brezak 
     30  1.1  brezak 
     31  1.1  brezak /*
     32  1.1  brezak  * Open and initialize the igmp socket, and fill in the non-changing
     33  1.1  brezak  * IP header fields in the output packet buffer.
     34  1.1  brezak  */
     35  1.1  brezak void init_igmp()
     36  1.1  brezak {
     37  1.1  brezak     struct ip *ip;
     38  1.1  brezak 
     39  1.1  brezak     if ((igmp_socket = socket(AF_INET, SOCK_RAW, IPPROTO_IGMP)) < 0)
     40  1.1  brezak 	log(LOG_ERR, errno, "IGMP socket");
     41  1.1  brezak 
     42  1.1  brezak     k_hdr_include(TRUE);	/* include IP header when sending */
     43  1.1  brezak     k_set_rcvbuf(48*1024);	/* lots of input buffering        */
     44  1.1  brezak     k_set_ttl(1);		/* restrict multicasts to one hop */
     45  1.1  brezak     k_set_loop(FALSE);		/* disable multicast loopback     */
     46  1.1  brezak 
     47  1.1  brezak     ip         = (struct ip *)send_buf;
     48  1.1  brezak     ip->ip_tos = 0;
     49  1.1  brezak     ip->ip_off = 0;
     50  1.1  brezak     ip->ip_p   = IPPROTO_IGMP;
     51  1.1  brezak     ip->ip_ttl = MAXTTL;	/* applies to unicasts only */
     52  1.1  brezak 
     53  1.1  brezak     allhosts_group = htonl(INADDR_ALLHOSTS_GROUP);
     54  1.1  brezak     dvmrp_group    = htonl(INADDR_DVMRP_GROUP);
     55  1.1  brezak }
     56  1.1  brezak 
     57  1.1  brezak static char *packet_kind(type, code)
     58  1.1  brezak      u_char type, code;
     59  1.1  brezak {
     60  1.1  brezak     switch (type) {
     61  1.1  brezak 	case IGMP_HOST_MEMBERSHIP_QUERY:	return "membership query  ";
     62  1.1  brezak 	case IGMP_HOST_MEMBERSHIP_REPORT:	return "membership report ";
     63  1.1  brezak 	case IGMP_DVMRP:
     64  1.1  brezak 	  switch (code) {
     65  1.1  brezak 	    case DVMRP_PROBE:	    		return "neighbor probe    ";
     66  1.1  brezak 	    case DVMRP_REPORT:	    		return "route report      ";
     67  1.1  brezak 	    case DVMRP_ASK_NEIGHBORS:   	return "neighbor request  ";
     68  1.1  brezak 	    case DVMRP_NEIGHBORS:	    	return "neighbor list     ";
     69  1.1  brezak 	    case DVMRP_ASK_NEIGHBORS2:   	return "neighbor request 2";
     70  1.1  brezak 	    case DVMRP_NEIGHBORS2:	    	return "neighbor list 2   ";
     71  1.1  brezak 	    default:		    		return "unknown DVMRP msg ";
     72  1.1  brezak 	  }
     73  1.1  brezak 	default:			    	return "unknown IGMP msg  ";
     74  1.1  brezak     }
     75  1.1  brezak }
     76  1.1  brezak 
     77  1.1  brezak /*
     78  1.1  brezak  * Process a newly received IGMP packet that is sitting in the input
     79  1.1  brezak  * packet buffer.
     80  1.1  brezak  */
     81  1.1  brezak void accept_igmp(recvlen)
     82  1.1  brezak     int recvlen;
     83  1.1  brezak {
     84  1.1  brezak     register vifi_t vifi;
     85  1.1  brezak     register u_long src, dst, group;
     86  1.1  brezak     struct ip *ip;
     87  1.1  brezak     struct igmp *igmp;
     88  1.1  brezak     int ipdatalen, iphdrlen, igmpdatalen;
     89  1.1  brezak 
     90  1.1  brezak     if (recvlen < sizeof(struct ip)) {
     91  1.1  brezak 	log(LOG_WARNING, 0,
     92  1.1  brezak 	    "received packet too short (%u bytes) for IP header", recvlen);
     93  1.1  brezak 	return;
     94  1.1  brezak     }
     95  1.1  brezak 
     96  1.1  brezak     ip        = (struct ip *)recv_buf;
     97  1.1  brezak     src       = ip->ip_src.s_addr;
     98  1.1  brezak     dst       = ip->ip_dst.s_addr;
     99  1.1  brezak     iphdrlen  = ip->ip_hl << 2;
    100  1.1  brezak     ipdatalen = ip->ip_len;
    101  1.1  brezak     if (iphdrlen + ipdatalen != recvlen) {
    102  1.1  brezak 	log(LOG_WARNING, 0,
    103  1.1  brezak 	    "received packet shorter (%u bytes) than hdr+data length (%u+%u)",
    104  1.1  brezak 	    recvlen, iphdrlen, ipdatalen);
    105  1.1  brezak 	return;
    106  1.1  brezak     }
    107  1.1  brezak 
    108  1.1  brezak     igmp        = (struct igmp *)(recv_buf + iphdrlen);
    109  1.1  brezak     group       = igmp->igmp_group.s_addr;
    110  1.1  brezak     igmpdatalen = ipdatalen - IGMP_MINLEN;
    111  1.1  brezak     if (igmpdatalen < 0) {
    112  1.1  brezak 	log(LOG_WARNING, 0,
    113  1.1  brezak 	    "received IP data field too short (%u bytes) for IGMP, from %s",
    114  1.1  brezak 	    ipdatalen, inet_fmt(src, s1));
    115  1.1  brezak 	return;
    116  1.1  brezak     }
    117  1.1  brezak 
    118  1.1  brezak     log(LOG_DEBUG, 0, "RECV %s from %-15s to %s",
    119  1.1  brezak 	packet_kind(igmp->igmp_type, igmp->igmp_code),
    120  1.1  brezak 	inet_fmt(src, s1), inet_fmt(dst, s2));
    121  1.1  brezak 
    122  1.1  brezak     switch (igmp->igmp_type) {
    123  1.1  brezak 
    124  1.1  brezak 	case IGMP_HOST_MEMBERSHIP_QUERY:
    125  1.1  brezak 	    return;	/* Answered automatically by the kernel. */
    126  1.1  brezak 
    127  1.1  brezak 	case IGMP_HOST_MEMBERSHIP_REPORT:
    128  1.1  brezak 	    accept_group_report(src, dst, group);
    129  1.1  brezak 	    return;
    130  1.1  brezak 
    131  1.1  brezak 	case IGMP_DVMRP:
    132  1.1  brezak 	    switch (igmp->igmp_code) {
    133  1.1  brezak 
    134  1.1  brezak 		case DVMRP_PROBE:
    135  1.1  brezak 		    accept_probe(src, dst);
    136  1.1  brezak 		    return;
    137  1.1  brezak 
    138  1.1  brezak 		case DVMRP_REPORT:
    139  1.1  brezak 		    accept_report(src, dst,
    140  1.1  brezak 				  (char *)(igmp+1), igmpdatalen);
    141  1.1  brezak 		    return;
    142  1.1  brezak 
    143  1.1  brezak 		case DVMRP_ASK_NEIGHBORS:
    144  1.1  brezak 		    accept_neighbor_request(src, dst);
    145  1.1  brezak 		    return;
    146  1.1  brezak 
    147  1.1  brezak 		case DVMRP_ASK_NEIGHBORS2:
    148  1.1  brezak 		    accept_neighbor_request2(src, dst);
    149  1.1  brezak 		    return;
    150  1.1  brezak 
    151  1.1  brezak 		case DVMRP_NEIGHBORS:
    152  1.1  brezak 		    accept_neighbors(src, dst, (char *)(igmp+1), igmpdatalen,
    153  1.1  brezak 				     group);
    154  1.1  brezak 		    return;
    155  1.1  brezak 
    156  1.1  brezak 		case DVMRP_NEIGHBORS2:
    157  1.1  brezak 		    accept_neighbors2(src, dst, (char *)(igmp+1), igmpdatalen,
    158  1.1  brezak 				     group);
    159  1.1  brezak 		    return;
    160  1.1  brezak 
    161  1.1  brezak 		default:
    162  1.1  brezak 		    log(LOG_INFO, 0,
    163  1.1  brezak 		     "ignoring unknown DVMRP message code %u from %s to %s",
    164  1.1  brezak 		     igmp->igmp_code, inet_fmt(src, s1),
    165  1.1  brezak 		     inet_fmt(dst, s2));
    166  1.1  brezak 		    return;
    167  1.1  brezak 	    }
    168  1.1  brezak 
    169  1.1  brezak 	default:
    170  1.1  brezak 	    log(LOG_INFO, 0,
    171  1.1  brezak 		"ignoring unknown IGMP message type %u from %s to %s",
    172  1.1  brezak 		igmp->igmp_type, inet_fmt(src, s1),
    173  1.1  brezak 		inet_fmt(dst, s2));
    174  1.1  brezak 	    return;
    175  1.1  brezak     }
    176  1.1  brezak }
    177  1.1  brezak 
    178  1.1  brezak 
    179  1.1  brezak /*
    180  1.1  brezak  * Construct an IGMP message in the output packet buffer.  The caller may
    181  1.1  brezak  * have already placed data in that buffer, of length 'datalen'.  Then send
    182  1.1  brezak  * the message from the interface with IP address 'src' to destination 'dst'.
    183  1.1  brezak  */
    184  1.1  brezak void send_igmp(src, dst, type, code, group, datalen)
    185  1.1  brezak     u_long src, dst;
    186  1.1  brezak     int type, code;
    187  1.1  brezak     u_long group;
    188  1.1  brezak     int datalen;
    189  1.1  brezak {
    190  1.1  brezak     static struct sockaddr_in sdst = {AF_INET};
    191  1.1  brezak     struct ip *ip;
    192  1.1  brezak     struct igmp *igmp;
    193  1.1  brezak 
    194  1.1  brezak     ip                      = (struct ip *)send_buf;
    195  1.1  brezak     ip->ip_src.s_addr       = src;
    196  1.1  brezak     ip->ip_dst.s_addr       = dst;
    197  1.1  brezak     ip->ip_len              = MIN_IP_HEADER_LEN + IGMP_MINLEN + datalen;
    198  1.1  brezak 
    199  1.1  brezak     igmp                    = (struct igmp *)(send_buf + MIN_IP_HEADER_LEN);
    200  1.1  brezak     igmp->igmp_type         = type;
    201  1.1  brezak     igmp->igmp_code         = code;
    202  1.1  brezak     igmp->igmp_group.s_addr = group;
    203  1.1  brezak     igmp->igmp_cksum        = 0;
    204  1.1  brezak     igmp->igmp_cksum        = inet_cksum((u_short *)igmp,
    205  1.1  brezak 					 IGMP_MINLEN + datalen);
    206  1.1  brezak 
    207  1.1  brezak     if (IN_MULTICAST(ntohl(dst))) k_set_if(src);
    208  1.1  brezak     if (dst == allhosts_group) k_set_loop(TRUE);
    209  1.1  brezak 
    210  1.1  brezak     sdst.sin_addr.s_addr = dst;
    211  1.1  brezak     if (sendto(igmp_socket, send_buf, ip->ip_len, 0,
    212  1.1  brezak 			(struct sockaddr *)&sdst, sizeof(sdst)) < 0) {
    213  1.1  brezak 	if (errno == ENETDOWN) check_vif_state();
    214  1.1  brezak 	else log(LOG_WARNING, errno, "sendto on %s", inet_fmt(src, s1));
    215  1.1  brezak     }
    216  1.1  brezak 
    217  1.1  brezak     if (dst == allhosts_group) k_set_loop(FALSE);
    218  1.1  brezak 
    219  1.1  brezak     log(LOG_DEBUG, 0, "SENT %s from %-15s to %s",
    220  1.1  brezak 	packet_kind(type, code), inet_fmt(src, s1), inet_fmt(dst, s2));
    221  1.1  brezak }
    222