[BACK]Return to igmp.c CVS log [TXT][DIR] Up to [local] / sys / netinet

Annotation of sys/netinet/igmp.c, Revision 1.1.1.1

1.1       nbrk        1: /*     $OpenBSD: igmp.c,v 1.24 2007/07/20 19:00:35 claudio Exp $       */
                      2: /*     $NetBSD: igmp.c,v 1.15 1996/02/13 23:41:25 christos Exp $       */
                      3:
                      4: /*
                      5:  * Internet Group Management Protocol (IGMP) routines.
                      6:  *
                      7:  * Written by Steve Deering, Stanford, May 1988.
                      8:  * Modified by Rosen Sharma, Stanford, Aug 1994.
                      9:  * Modified by Bill Fenner, Xerox PARC, Feb 1995.
                     10:  *
                     11:  * MULTICAST Revision: 1.3
                     12:  */
                     13:
                     14: #include <sys/param.h>
                     15: #include <sys/mbuf.h>
                     16: #include <sys/socket.h>
                     17: #include <sys/protosw.h>
                     18:
                     19: #include <net/if.h>
                     20: #include <net/route.h>
                     21:
                     22: #include <netinet/in.h>
                     23: #include <netinet/in_var.h>
                     24: #include <netinet/in_systm.h>
                     25: #include <netinet/ip.h>
                     26: #include <netinet/ip_var.h>
                     27: #include <netinet/igmp.h>
                     28: #include <netinet/igmp_var.h>
                     29: #include <dev/rndvar.h>
                     30:
                     31: #include <sys/stdarg.h>
                     32:
                     33: #define IP_MULTICASTOPTS       0
                     34:
                     35: int            igmp_timers_are_running;
                     36: static struct router_info *rti_head;
                     37: struct igmpstat igmpstat;
                     38:
                     39: void igmp_sendpkt(struct in_multi *, int, in_addr_t);
                     40: int rti_fill(struct in_multi *);
                     41: struct router_info * rti_find(struct ifnet *);
                     42:
                     43: void
                     44: igmp_init()
                     45: {
                     46:
                     47:        /*
                     48:         * To avoid byte-swapping the same value over and over again.
                     49:         */
                     50:        igmp_timers_are_running = 0;
                     51:        rti_head = 0;
                     52: }
                     53:
                     54: /* Return -1 for error. */
                     55: int
                     56: rti_fill(inm)
                     57:        struct in_multi *inm;
                     58: {
                     59:        struct router_info *rti;
                     60:
                     61:        for (rti = rti_head; rti != 0; rti = rti->rti_next) {
                     62:                if (rti->rti_ifp == inm->inm_ia->ia_ifp) {
                     63:                        inm->inm_rti = rti;
                     64:                        if (rti->rti_type == IGMP_v1_ROUTER)
                     65:                                return (IGMP_v1_HOST_MEMBERSHIP_REPORT);
                     66:                        else
                     67:                                return (IGMP_v2_HOST_MEMBERSHIP_REPORT);
                     68:                }
                     69:        }
                     70:
                     71:        rti = (struct router_info *)malloc(sizeof(struct router_info),
                     72:                                           M_MRTABLE, M_NOWAIT);
                     73:        if (rti == NULL)
                     74:                return (-1);
                     75:        rti->rti_ifp = inm->inm_ia->ia_ifp;
                     76:        rti->rti_type = IGMP_v2_ROUTER;
                     77:        rti->rti_next = rti_head;
                     78:        rti_head = rti;
                     79:        inm->inm_rti = rti;
                     80:        return (IGMP_v2_HOST_MEMBERSHIP_REPORT);
                     81: }
                     82:
                     83: struct router_info *
                     84: rti_find(ifp)
                     85:        struct ifnet *ifp;
                     86: {
                     87:        struct router_info *rti;
                     88:
                     89:        for (rti = rti_head; rti != 0; rti = rti->rti_next) {
                     90:                if (rti->rti_ifp == ifp)
                     91:                        return (rti);
                     92:        }
                     93:
                     94:        rti = (struct router_info *)malloc(sizeof(struct router_info),
                     95:                                           M_MRTABLE, M_NOWAIT);
                     96:        if (rti == NULL)
                     97:                return (NULL);
                     98:        rti->rti_ifp = ifp;
                     99:        rti->rti_type = IGMP_v2_ROUTER;
                    100:        rti->rti_next = rti_head;
                    101:        rti_head = rti;
                    102:        return (rti);
                    103: }
                    104:
                    105: void
                    106: rti_delete(ifp)
                    107:        struct ifnet *ifp;
                    108: {
                    109:        struct router_info *rti, **prti = &rti_head;
                    110:
                    111:        for (rti = rti_head; rti != 0; rti = rti->rti_next) {
                    112:                if (rti->rti_ifp == ifp) {
                    113:                        *prti = rti->rti_next;
                    114:                        free(rti, M_MRTABLE);
                    115:                        break;
                    116:                }
                    117:                prti = &rti->rti_next;
                    118:        }
                    119: }
                    120:
                    121: void
                    122: igmp_input(struct mbuf *m, ...)
                    123: {
                    124:        int iphlen;
                    125:        struct ifnet *ifp = m->m_pkthdr.rcvif;
                    126:        struct ip *ip = mtod(m, struct ip *);
                    127:        struct igmp *igmp;
                    128:        int igmplen;
                    129:        int minlen;
                    130:        struct in_multi *inm;
                    131:        struct in_multistep step;
                    132:        struct router_info *rti;
                    133:        struct in_ifaddr *ia;
                    134:        int timer;
                    135:        va_list ap;
                    136:
                    137:        va_start(ap, m);
                    138:        iphlen = va_arg(ap, int);
                    139:        va_end(ap);
                    140:
                    141:        ++igmpstat.igps_rcv_total;
                    142:
                    143:        igmplen = ntohs(ip->ip_len) - iphlen;
                    144:
                    145:        /*
                    146:         * Validate lengths
                    147:         */
                    148:        if (igmplen < IGMP_MINLEN) {
                    149:                ++igmpstat.igps_rcv_tooshort;
                    150:                m_freem(m);
                    151:                return;
                    152:        }
                    153:        minlen = iphlen + IGMP_MINLEN;
                    154:        if ((m->m_flags & M_EXT || m->m_len < minlen) &&
                    155:            (m = m_pullup(m, minlen)) == NULL) {
                    156:                ++igmpstat.igps_rcv_tooshort;
                    157:                return;
                    158:        }
                    159:
                    160:        /*
                    161:         * Validate checksum
                    162:         */
                    163:        m->m_data += iphlen;
                    164:        m->m_len -= iphlen;
                    165:        igmp = mtod(m, struct igmp *);
                    166:        if (in_cksum(m, igmplen)) {
                    167:                ++igmpstat.igps_rcv_badsum;
                    168:                m_freem(m);
                    169:                return;
                    170:        }
                    171:        m->m_data -= iphlen;
                    172:        m->m_len += iphlen;
                    173:        ip = mtod(m, struct ip *);
                    174:
                    175:        switch (igmp->igmp_type) {
                    176:
                    177:        case IGMP_HOST_MEMBERSHIP_QUERY:
                    178:                ++igmpstat.igps_rcv_queries;
                    179:
                    180:                if (ifp->if_flags & IFF_LOOPBACK)
                    181:                        break;
                    182:
                    183:                if (igmp->igmp_code == 0) {
                    184:                        rti = rti_find(ifp);
                    185:                        if (rti == NULL) {
                    186:                                m_freem(m);
                    187:                                return;
                    188:                        }
                    189:                        rti->rti_type = IGMP_v1_ROUTER;
                    190:                        rti->rti_age = 0;
                    191:
                    192:                        if (ip->ip_dst.s_addr != INADDR_ALLHOSTS_GROUP) {
                    193:                                ++igmpstat.igps_rcv_badqueries;
                    194:                                m_freem(m);
                    195:                                return;
                    196:                        }
                    197:
                    198:                        /*
                    199:                         * Start the timers in all of our membership records
                    200:                         * for the interface on which the query arrived,
                    201:                         * except those that are already running and those
                    202:                         * that belong to a "local" group (224.0.0.X).
                    203:                         */
                    204:                        IN_FIRST_MULTI(step, inm);
                    205:                        while (inm != NULL) {
                    206:                                if (inm->inm_ia->ia_ifp == ifp &&
                    207:                                    inm->inm_timer == 0 &&
                    208:                                    !IN_LOCAL_GROUP(inm->inm_addr.s_addr)) {
                    209:                                        inm->inm_state = IGMP_DELAYING_MEMBER;
                    210:                                        inm->inm_timer = IGMP_RANDOM_DELAY(
                    211:                                            IGMP_MAX_HOST_REPORT_DELAY * PR_FASTHZ);
                    212:                                        igmp_timers_are_running = 1;
                    213:                                }
                    214:                                IN_NEXT_MULTI(step, inm);
                    215:                        }
                    216:                } else {
                    217:                        if (!IN_MULTICAST(ip->ip_dst.s_addr)) {
                    218:                                ++igmpstat.igps_rcv_badqueries;
                    219:                                m_freem(m);
                    220:                                return;
                    221:                        }
                    222:
                    223:                        timer = igmp->igmp_code * PR_FASTHZ / IGMP_TIMER_SCALE;
                    224:                        if (timer == 0)
                    225:                                timer = 1;
                    226:
                    227:                        /*
                    228:                         * Start the timers in all of our membership records
                    229:                         * for the interface on which the query arrived,
                    230:                         * except those that are already running and those
                    231:                         * that belong to a "local" group (224.0.0.X).  For
                    232:                         * timers already running, check if they need to be
                    233:                         * reset.
                    234:                         */
                    235:                        IN_FIRST_MULTI(step, inm);
                    236:                        while (inm != NULL) {
                    237:                                if (inm->inm_ia->ia_ifp == ifp &&
                    238:                                    !IN_LOCAL_GROUP(inm->inm_addr.s_addr) &&
                    239:                                    (ip->ip_dst.s_addr == INADDR_ALLHOSTS_GROUP ||
                    240:                                     ip->ip_dst.s_addr == inm->inm_addr.s_addr)) {
                    241:                                        switch (inm->inm_state) {
                    242:                                        case IGMP_DELAYING_MEMBER:
                    243:                                                if (inm->inm_timer <= timer)
                    244:                                                        break;
                    245:                                                /* FALLTHROUGH */
                    246:                                        case IGMP_IDLE_MEMBER:
                    247:                                        case IGMP_LAZY_MEMBER:
                    248:                                        case IGMP_AWAKENING_MEMBER:
                    249:                                                inm->inm_state =
                    250:                                                    IGMP_DELAYING_MEMBER;
                    251:                                                inm->inm_timer =
                    252:                                                    IGMP_RANDOM_DELAY(timer);
                    253:                                                igmp_timers_are_running = 1;
                    254:                                                break;
                    255:                                        case IGMP_SLEEPING_MEMBER:
                    256:                                                inm->inm_state =
                    257:                                                    IGMP_AWAKENING_MEMBER;
                    258:                                                break;
                    259:                                        }
                    260:                                }
                    261:                                IN_NEXT_MULTI(step, inm);
                    262:                        }
                    263:                }
                    264:
                    265:                break;
                    266:
                    267:        case IGMP_v1_HOST_MEMBERSHIP_REPORT:
                    268:                ++igmpstat.igps_rcv_reports;
                    269:
                    270:                if (ifp->if_flags & IFF_LOOPBACK)
                    271:                        break;
                    272:
                    273:                if (!IN_MULTICAST(igmp->igmp_group.s_addr) ||
                    274:                    igmp->igmp_group.s_addr != ip->ip_dst.s_addr) {
                    275:                        ++igmpstat.igps_rcv_badreports;
                    276:                        m_freem(m);
                    277:                        return;
                    278:                }
                    279:
                    280:                /*
                    281:                 * KLUDGE: if the IP source address of the report has an
                    282:                 * unspecified (i.e., zero) subnet number, as is allowed for
                    283:                 * a booting host, replace it with the correct subnet number
                    284:                 * so that a process-level multicast routing daemon can
                    285:                 * determine which subnet it arrived from.  This is necessary
                    286:                 * to compensate for the lack of any way for a process to
                    287:                 * determine the arrival interface of an incoming packet.
                    288:                 */
                    289:                if ((ip->ip_src.s_addr & IN_CLASSA_NET) == 0) {
                    290:                        IFP_TO_IA(ifp, ia);
                    291:                        if (ia)
                    292:                                ip->ip_src.s_addr = ia->ia_subnet;
                    293:                }
                    294:
                    295:                /*
                    296:                 * If we belong to the group being reported, stop
                    297:                 * our timer for that group.
                    298:                 */
                    299:                IN_LOOKUP_MULTI(igmp->igmp_group, ifp, inm);
                    300:                if (inm != NULL) {
                    301:                        inm->inm_timer = 0;
                    302:                        ++igmpstat.igps_rcv_ourreports;
                    303:
                    304:                        switch (inm->inm_state) {
                    305:                        case IGMP_IDLE_MEMBER:
                    306:                        case IGMP_LAZY_MEMBER:
                    307:                        case IGMP_AWAKENING_MEMBER:
                    308:                        case IGMP_SLEEPING_MEMBER:
                    309:                                inm->inm_state = IGMP_SLEEPING_MEMBER;
                    310:                                break;
                    311:                        case IGMP_DELAYING_MEMBER:
                    312:                                if (inm->inm_rti->rti_type == IGMP_v1_ROUTER)
                    313:                                        inm->inm_state = IGMP_LAZY_MEMBER;
                    314:                                else
                    315:                                        inm->inm_state = IGMP_SLEEPING_MEMBER;
                    316:                                break;
                    317:                        }
                    318:                }
                    319:
                    320:                break;
                    321:
                    322:        case IGMP_v2_HOST_MEMBERSHIP_REPORT:
                    323: #ifdef MROUTING
                    324:                /*
                    325:                 * Make sure we don't hear our own membership report.  Fast
                    326:                 * leave requires knowing that we are the only member of a
                    327:                 * group.
                    328:                 */
                    329:                IFP_TO_IA(ifp, ia);
                    330:                if (ia && ip->ip_src.s_addr == ia->ia_addr.sin_addr.s_addr)
                    331:                        break;
                    332: #endif
                    333:
                    334:                ++igmpstat.igps_rcv_reports;
                    335:
                    336:                if (ifp->if_flags & IFF_LOOPBACK)
                    337:                        break;
                    338:
                    339:                if (!IN_MULTICAST(igmp->igmp_group.s_addr) ||
                    340:                    igmp->igmp_group.s_addr != ip->ip_dst.s_addr) {
                    341:                        ++igmpstat.igps_rcv_badreports;
                    342:                        m_freem(m);
                    343:                        return;
                    344:                }
                    345:
                    346:                /*
                    347:                 * KLUDGE: if the IP source address of the report has an
                    348:                 * unspecified (i.e., zero) subnet number, as is allowed for
                    349:                 * a booting host, replace it with the correct subnet number
                    350:                 * so that a process-level multicast routing daemon can
                    351:                 * determine which subnet it arrived from.  This is necessary
                    352:                 * to compensate for the lack of any way for a process to
                    353:                 * determine the arrival interface of an incoming packet.
                    354:                 */
                    355:                if ((ip->ip_src.s_addr & IN_CLASSA_NET) == 0) {
                    356: #ifndef MROUTING
                    357:                        IFP_TO_IA(ifp, ia);
                    358: #endif
                    359:                        if (ia)
                    360:                                ip->ip_src.s_addr = ia->ia_subnet;
                    361:                }
                    362:
                    363:                /*
                    364:                 * If we belong to the group being reported, stop
                    365:                 * our timer for that group.
                    366:                 */
                    367:                IN_LOOKUP_MULTI(igmp->igmp_group, ifp, inm);
                    368:                if (inm != NULL) {
                    369:                        inm->inm_timer = 0;
                    370:                        ++igmpstat.igps_rcv_ourreports;
                    371:
                    372:                        switch (inm->inm_state) {
                    373:                        case IGMP_DELAYING_MEMBER:
                    374:                        case IGMP_IDLE_MEMBER:
                    375:                        case IGMP_AWAKENING_MEMBER:
                    376:                                inm->inm_state = IGMP_LAZY_MEMBER;
                    377:                                break;
                    378:                        case IGMP_LAZY_MEMBER:
                    379:                        case IGMP_SLEEPING_MEMBER:
                    380:                                break;
                    381:                        }
                    382:                }
                    383:
                    384:                break;
                    385:
                    386:        }
                    387:
                    388:        /*
                    389:         * Pass all valid IGMP packets up to any process(es) listening
                    390:         * on a raw IGMP socket.
                    391:         */
                    392:        rip_input(m);
                    393: }
                    394:
                    395: void
                    396: igmp_joingroup(inm)
                    397:        struct in_multi *inm;
                    398: {
                    399:        int i, s = splsoftnet();
                    400:
                    401:        inm->inm_state = IGMP_IDLE_MEMBER;
                    402:
                    403:        if (!IN_LOCAL_GROUP(inm->inm_addr.s_addr) &&
                    404:            (inm->inm_ia->ia_ifp->if_flags & IFF_LOOPBACK) == 0) {
                    405:                if ((i = rti_fill(inm)) == -1) {
                    406:                        splx(s);
                    407:                        return;
                    408:                }
                    409:                igmp_sendpkt(inm, i, 0);
                    410:                inm->inm_state = IGMP_DELAYING_MEMBER;
                    411:                inm->inm_timer = IGMP_RANDOM_DELAY(
                    412:                    IGMP_MAX_HOST_REPORT_DELAY * PR_FASTHZ);
                    413:                igmp_timers_are_running = 1;
                    414:        } else
                    415:                inm->inm_timer = 0;
                    416:        splx(s);
                    417: }
                    418:
                    419: void
                    420: igmp_leavegroup(inm)
                    421:        struct in_multi *inm;
                    422: {
                    423:
                    424:        switch (inm->inm_state) {
                    425:        case IGMP_DELAYING_MEMBER:
                    426:        case IGMP_IDLE_MEMBER:
                    427:                if (!IN_LOCAL_GROUP(inm->inm_addr.s_addr) &&
                    428:                    (inm->inm_ia->ia_ifp->if_flags & IFF_LOOPBACK) == 0)
                    429:                        if (inm->inm_rti->rti_type != IGMP_v1_ROUTER)
                    430:                                igmp_sendpkt(inm, IGMP_HOST_LEAVE_MESSAGE,
                    431:                                    INADDR_ALLROUTERS_GROUP);
                    432:                break;
                    433:        case IGMP_LAZY_MEMBER:
                    434:        case IGMP_AWAKENING_MEMBER:
                    435:        case IGMP_SLEEPING_MEMBER:
                    436:                break;
                    437:        }
                    438: }
                    439:
                    440: void
                    441: igmp_fasttimo()
                    442: {
                    443:        struct in_multi *inm;
                    444:        struct in_multistep step;
                    445:        int s;
                    446:
                    447:        /*
                    448:         * Quick check to see if any work needs to be done, in order
                    449:         * to minimize the overhead of fasttimo processing.
                    450:         */
                    451:        if (!igmp_timers_are_running)
                    452:                return;
                    453:
                    454:        s = splsoftnet();
                    455:        igmp_timers_are_running = 0;
                    456:        IN_FIRST_MULTI(step, inm);
                    457:        while (inm != NULL) {
                    458:                if (inm->inm_timer == 0) {
                    459:                        /* do nothing */
                    460:                } else if (--inm->inm_timer == 0) {
                    461:                        if (inm->inm_state == IGMP_DELAYING_MEMBER) {
                    462:                                if (inm->inm_rti->rti_type == IGMP_v1_ROUTER)
                    463:                                        igmp_sendpkt(inm,
                    464:                                            IGMP_v1_HOST_MEMBERSHIP_REPORT, 0);
                    465:                                else
                    466:                                        igmp_sendpkt(inm,
                    467:                                            IGMP_v2_HOST_MEMBERSHIP_REPORT, 0);
                    468:                                inm->inm_state = IGMP_IDLE_MEMBER;
                    469:                        }
                    470:                } else {
                    471:                        igmp_timers_are_running = 1;
                    472:                }
                    473:                IN_NEXT_MULTI(step, inm);
                    474:        }
                    475:        splx(s);
                    476: }
                    477:
                    478: void
                    479: igmp_slowtimo()
                    480: {
                    481:        struct router_info *rti;
                    482:        int s;
                    483:
                    484:        s = splsoftnet();
                    485:        for (rti = rti_head; rti != 0; rti = rti->rti_next) {
                    486:                if (rti->rti_type == IGMP_v1_ROUTER &&
                    487:                    ++rti->rti_age >= IGMP_AGE_THRESHOLD) {
                    488:                        rti->rti_type = IGMP_v2_ROUTER;
                    489:                }
                    490:        }
                    491:        splx(s);
                    492: }
                    493:
                    494: void
                    495: igmp_sendpkt(inm, type, addr)
                    496:        struct in_multi *inm;
                    497:        int type;
                    498:        in_addr_t addr;
                    499: {
                    500:        struct mbuf *m;
                    501:        struct igmp *igmp;
                    502:        struct ip *ip;
                    503:        struct ip_moptions imo;
                    504: #ifdef MROUTING
                    505:        extern struct socket *ip_mrouter;
                    506: #endif /* MROUTING */
                    507:
                    508:        MGETHDR(m, M_DONTWAIT, MT_HEADER);
                    509:        if (m == NULL)
                    510:                return;
                    511:        /*
                    512:         * Assume max_linkhdr + sizeof(struct ip) + IGMP_MINLEN
                    513:         * is smaller than mbuf size returned by MGETHDR.
                    514:         */
                    515:        m->m_data += max_linkhdr;
                    516:        m->m_len = sizeof(struct ip) + IGMP_MINLEN;
                    517:        m->m_pkthdr.len = sizeof(struct ip) + IGMP_MINLEN;
                    518:
                    519:        ip = mtod(m, struct ip *);
                    520:        ip->ip_tos = 0;
                    521:        ip->ip_len = htons(sizeof(struct ip) + IGMP_MINLEN);
                    522:        ip->ip_off = 0;
                    523:        ip->ip_p = IPPROTO_IGMP;
                    524:        ip->ip_src.s_addr = INADDR_ANY;
                    525:        if (addr) {
                    526:                ip->ip_dst.s_addr = addr;
                    527:        } else {
                    528:                ip->ip_dst = inm->inm_addr;
                    529:        }
                    530:
                    531:        m->m_data += sizeof(struct ip);
                    532:        m->m_len -= sizeof(struct ip);
                    533:        igmp = mtod(m, struct igmp *);
                    534:        igmp->igmp_type = type;
                    535:        igmp->igmp_code = 0;
                    536:        igmp->igmp_group = inm->inm_addr;
                    537:        igmp->igmp_cksum = 0;
                    538:        igmp->igmp_cksum = in_cksum(m, IGMP_MINLEN);
                    539:        m->m_data -= sizeof(struct ip);
                    540:        m->m_len += sizeof(struct ip);
                    541:
                    542:        imo.imo_multicast_ifp = inm->inm_ia->ia_ifp;
                    543:        imo.imo_multicast_ttl = 1;
                    544: #ifdef RSVP_ISI
                    545:        imo.imo_multicast_vif = -1;
                    546: #endif
                    547:        /*
                    548:         * Request loopback of the report if we are acting as a multicast
                    549:         * router, so that the process-level routing daemon can hear it.
                    550:         */
                    551: #ifdef MROUTING
                    552:        imo.imo_multicast_loop = (ip_mrouter != NULL);
                    553: #else
                    554:        imo.imo_multicast_loop = 0;
                    555: #endif /* MROUTING */
                    556:
                    557:        ip_output(m, (struct mbuf *)0, (struct route *)0, IP_MULTICASTOPTS,
                    558:            &imo, (void *)NULL);
                    559:
                    560:        ++igmpstat.igps_snd_reports;
                    561: }

CVSweb