Annotation of XNU/bsd/netiso/esis.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
        !             3:  *
        !             4:  * @APPLE_LICENSE_HEADER_START@
        !             5:  * 
        !             6:  * The contents of this file constitute Original Code as defined in and
        !             7:  * are subject to the Apple Public Source License Version 1.1 (the
        !             8:  * "License").  You may not use this file except in compliance with the
        !             9:  * License.  Please obtain a copy of the License at
        !            10:  * http://www.apple.com/publicsource and read it before using this file.
        !            11:  * 
        !            12:  * This Original Code and all software distributed under the License are
        !            13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
        !            14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
        !            15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
        !            16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
        !            17:  * License for the specific language governing rights and limitations
        !            18:  * under the License.
        !            19:  * 
        !            20:  * @APPLE_LICENSE_HEADER_END@
        !            21:  */
        !            22: /*-
        !            23:  * Copyright (c) 1991, 1993
        !            24:  *     The Regents of the University of California.  All rights reserved.
        !            25:  *
        !            26:  * Redistribution and use in source and binary forms, with or without
        !            27:  * modification, are permitted provided that the following conditions
        !            28:  * are met:
        !            29:  * 1. Redistributions of source code must retain the above copyright
        !            30:  *    notice, this list of conditions and the following disclaimer.
        !            31:  * 2. Redistributions in binary form must reproduce the above copyright
        !            32:  *    notice, this list of conditions and the following disclaimer in the
        !            33:  *    documentation and/or other materials provided with the distribution.
        !            34:  * 3. All advertising materials mentioning features or use of this software
        !            35:  *    must display the following acknowledgement:
        !            36:  *     This product includes software developed by the University of
        !            37:  *     California, Berkeley and its contributors.
        !            38:  * 4. Neither the name of the University nor the names of its contributors
        !            39:  *    may be used to endorse or promote products derived from this software
        !            40:  *    without specific prior written permission.
        !            41:  *
        !            42:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
        !            43:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
        !            44:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
        !            45:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
        !            46:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
        !            47:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
        !            48:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
        !            49:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
        !            50:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
        !            51:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
        !            52:  * SUCH DAMAGE.
        !            53:  *
        !            54:  *     @(#)esis.c      8.1 (Berkeley) 6/10/93
        !            55:  */
        !            56: 
        !            57: /***********************************************************
        !            58:                Copyright IBM Corporation 1987
        !            59: 
        !            60:                       All Rights Reserved
        !            61: 
        !            62: Permission to use, copy, modify, and distribute this software and its 
        !            63: documentation for any purpose and without fee is hereby granted, 
        !            64: provided that the above copyright notice appear in all copies and that
        !            65: both that copyright notice and this permission notice appear in 
        !            66: supporting documentation, and that the name of IBM not be
        !            67: used in advertising or publicity pertaining to distribution of the
        !            68: software without specific, written prior permission.  
        !            69: 
        !            70: IBM DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
        !            71: ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
        !            72: IBM BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
        !            73: ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
        !            74: WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
        !            75: ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
        !            76: SOFTWARE.
        !            77: 
        !            78: ******************************************************************/
        !            79: 
        !            80: /*
        !            81:  * ARGO Project, Computer Sciences Dept., University of Wisconsin - Madison
        !            82:  */
        !            83: 
        !            84: #if ISO
        !            85: 
        !            86: #include <sys/param.h>
        !            87: #include <sys/systm.h>
        !            88: #include <sys/mbuf.h>
        !            89: #include <sys/domain.h>
        !            90: #include <sys/protosw.h>
        !            91: #include <sys/socket.h>
        !            92: #include <sys/socketvar.h>
        !            93: #include <sys/errno.h>
        !            94: #include <sys/kernel.h>
        !            95: 
        !            96: #include <net/if.h>
        !            97: #include <net/if_dl.h>
        !            98: #include <net/route.h>
        !            99: #include <net/raw_cb.h>
        !           100: 
        !           101: #include <netiso/iso.h>
        !           102: #include <netiso/iso_pcb.h>
        !           103: #include <netiso/iso_var.h>
        !           104: #include <netiso/iso_snpac.h>
        !           105: #include <netiso/clnl.h>
        !           106: #include <netiso/clnp.h>
        !           107: #include <netiso/clnp_stat.h>
        !           108: #include <netiso/esis.h>
        !           109: #include <netiso/argo_debug.h>
        !           110: 
        !           111: /*
        !           112:  *     Global variables to esis implementation
        !           113:  *
        !           114:  *     esis_holding_time - the holding time (sec) parameter for outgoing pdus
        !           115:  *     esis_config_time  - the frequency (sec) that hellos are generated
        !           116:  *     esis_esconfig_time - suggested es configuration time placed in the
        !           117:  *                                             ish.
        !           118:  *
        !           119:  */
        !           120: struct rawcb   esis_pcb;
        !           121: void                           esis_config(), snpac_age();
        !           122: int                            esis_sendspace = 2048;
        !           123: int                            esis_recvspace = 2048;
        !           124: short                  esis_holding_time = ESIS_HT;
        !           125: short                  esis_config_time = ESIS_CONFIG;
        !           126: short                  esis_esconfig_time = ESIS_CONFIG;
        !           127: extern int             iso_systype;
        !           128: struct sockaddr_dl     esis_dl = { sizeof(esis_dl), AF_LINK };
        !           129: extern char            all_es_snpa[], all_is_snpa[];
        !           130: 
        !           131: #define EXTEND_PACKET(m, mhdr, cp)\
        !           132:        if (((m)->m_next = m_getclr(M_DONTWAIT, MT_HEADER)) == NULL) {\
        !           133:                esis_stat.es_nomem++;\
        !           134:                m_freem(mhdr);\
        !           135:                return;\
        !           136:        } else {\
        !           137:                (m) = (m)->m_next;\
        !           138:                (cp) = mtod((m), caddr_t);\
        !           139:        }
        !           140: /*
        !           141:  * FUNCTION:           esis_init
        !           142:  *
        !           143:  * PURPOSE:                    Initialize the kernel portion of esis protocol
        !           144:  *
        !           145:  * RETURNS:                    nothing
        !           146:  *
        !           147:  * SIDE EFFECTS:       
        !           148:  *
        !           149:  * NOTES:                      
        !           150:  */
        !           151: esis_init()
        !           152: {
        !           153:        extern struct clnl_protosw clnl_protox[256];
        !           154:        int     esis_input(), isis_input();
        !           155: #ifdef ISO_X25ESIS
        !           156:        int     x25esis_input();
        !           157: #endif /* ISO_X25ESIS */
        !           158: 
        !           159:        esis_pcb.rcb_next = esis_pcb.rcb_prev = &esis_pcb;
        !           160:        llinfo_llc.lc_next = llinfo_llc.lc_prev = &llinfo_llc;
        !           161: 
        !           162:        timeout(snpac_age, (caddr_t)0, hz);
        !           163:        timeout(esis_config, (caddr_t)0, hz);
        !           164: 
        !           165:        clnl_protox[ISO9542_ESIS].clnl_input = esis_input;
        !           166:        clnl_protox[ISO10589_ISIS].clnl_input = isis_input;
        !           167: #ifdef ISO_X25ESIS
        !           168:        clnl_protox[ISO9542X25_ESIS].clnl_input = x25esis_input;
        !           169: #endif /* ISO_X25ESIS */
        !           170: }
        !           171: 
        !           172: /*
        !           173:  * FUNCTION:           esis_usrreq
        !           174:  *
        !           175:  * PURPOSE:                    Handle user level esis requests
        !           176:  *
        !           177:  * RETURNS:                    0 or appropriate errno
        !           178:  *
        !           179:  * SIDE EFFECTS:       
        !           180:  *
        !           181:  */
        !           182: /*ARGSUSED*/
        !           183: esis_usrreq(so, req, m, nam, control)
        !           184: struct socket  *so;            /* socket: used only to get to this code */
        !           185: int                            req;            /* request */
        !           186: struct mbuf            *m;                     /* data for request */
        !           187: struct mbuf            *nam;           /* optional name */
        !           188: struct mbuf            *control;       /* optional control */
        !           189: {
        !           190:        struct rawcb *rp = sotorawcb(so);
        !           191:        int error = 0;
        !           192: 
        !           193:        if ((so->so_state & SS_PRIV) == 0) {
        !           194:                error = EACCES;
        !           195:                goto release;
        !           196:        }
        !           197:        if (rp == NULL && req != PRU_ATTACH) {
        !           198:                error = EINVAL;
        !           199:                goto release;
        !           200:        }
        !           201: 
        !           202:        switch (req) {
        !           203:        case PRU_ATTACH:
        !           204:                if (rp != NULL) {
        !           205:                        error = EINVAL;
        !           206:                        break;
        !           207:                }
        !           208:                MALLOC(rp, struct rawcb *, sizeof(*rp), M_PCB, M_WAITOK);
        !           209:                if (so->so_pcb = (caddr_t)rp) {
        !           210:                        bzero(so->so_pcb, sizeof(*rp));
        !           211:                        insque(rp, &esis_pcb);
        !           212:                        rp->rcb_socket = so;
        !           213:                        error = soreserve(so, esis_sendspace, esis_recvspace);
        !           214:                } else
        !           215:                        error = ENOBUFS;
        !           216:                break;
        !           217: 
        !           218:        case PRU_SEND:
        !           219:                if (nam == NULL) {
        !           220:                        error = EINVAL;
        !           221:                        break;
        !           222:                }
        !           223:                /* error checking here */
        !           224:                error = isis_output(mtod(nam,struct sockaddr_dl *), m);
        !           225:                m = NULL;
        !           226:                break;
        !           227: 
        !           228:        case PRU_DETACH:
        !           229:                raw_detach(rp);
        !           230:                break;
        !           231: 
        !           232:        case PRU_SHUTDOWN:
        !           233:                socantsendmore(so);
        !           234:                break;
        !           235: 
        !           236:        case PRU_ABORT:
        !           237:                soisdisconnected(so);
        !           238:                raw_detach(rp);
        !           239:                break;
        !           240: 
        !           241:        case PRU_SENSE:
        !           242:                return (0);
        !           243: 
        !           244:        default:
        !           245:                return (EOPNOTSUPP);
        !           246:        }
        !           247: release:
        !           248:        if (m != NULL)
        !           249:                m_freem(m);
        !           250: 
        !           251:        return (error);
        !           252: }
        !           253: 
        !           254: /*
        !           255:  * FUNCTION:           esis_input
        !           256:  *
        !           257:  * PURPOSE:                    Process an incoming esis packet
        !           258:  *
        !           259:  * RETURNS:                    nothing
        !           260:  *
        !           261:  * SIDE EFFECTS:       
        !           262:  *
        !           263:  * NOTES:                      
        !           264:  */
        !           265: esis_input(m0, shp)
        !           266: struct mbuf            *m0;            /* ptr to first mbuf of pkt */
        !           267: struct snpa_hdr        *shp;   /* subnetwork header */
        !           268: {
        !           269:        register struct esis_fixed      *pdu = mtod(m0, struct esis_fixed *);
        !           270:        register int type;
        !           271: 
        !           272:        /*
        !           273:         *      check checksum if necessary
        !           274:         */
        !           275:        if (ESIS_CKSUM_REQUIRED(pdu) && iso_check_csum(m0, (int)pdu->esis_hdr_len)) {
        !           276:                esis_stat.es_badcsum++;
        !           277:                goto bad;
        !           278:        }
        !           279: 
        !           280:        /* check version */
        !           281:        if (pdu->esis_vers != ESIS_VERSION) {
        !           282:                esis_stat.es_badvers++;
        !           283:                goto bad;
        !           284:        }
        !           285:        type = pdu->esis_type & 0x1f;
        !           286:        switch (type) {
        !           287:                case ESIS_ESH:
        !           288:                        esis_eshinput(m0, shp);
        !           289:                        break;
        !           290: 
        !           291:                case ESIS_ISH:
        !           292:                        esis_ishinput(m0, shp);
        !           293:                        break;
        !           294: 
        !           295:                case ESIS_RD:
        !           296:                        esis_rdinput(m0, shp);
        !           297:                        break;
        !           298: 
        !           299:                default:
        !           300:                        esis_stat.es_badtype++;
        !           301:        }
        !           302: 
        !           303: bad:
        !           304:        if (esis_pcb.rcb_next != &esis_pcb)
        !           305:                isis_input(m0, shp);
        !           306:        else
        !           307:                m_freem(m0);
        !           308: }
        !           309: 
        !           310: /*
        !           311:  * FUNCTION:           esis_rdoutput
        !           312:  *
        !           313:  * PURPOSE:                    Transmit a redirect pdu
        !           314:  *
        !           315:  * RETURNS:                    nothing
        !           316:  *
        !           317:  * SIDE EFFECTS:       
        !           318:  *
        !           319:  * NOTES:                      Assumes there is enough space for fixed part of header,
        !           320:  *                                     DA, BSNPA and NET in first mbuf.
        !           321:  */
        !           322: esis_rdoutput(inbound_shp, inbound_m, inbound_oidx, rd_dstnsap, rt)
        !           323: struct snpa_hdr                *inbound_shp;   /* snpa hdr from incoming packet */
        !           324: struct mbuf                    *inbound_m;             /* incoming pkt itself */
        !           325: struct clnp_optidx     *inbound_oidx;  /* clnp options assoc with incoming pkt */
        !           326: struct iso_addr                *rd_dstnsap;    /* ultimate destination of pkt */
        !           327: struct rtentry         *rt;                    /* snpa cache info regarding next hop of
        !           328:                                                                                pkt */
        !           329: {
        !           330:        struct mbuf                     *m, *m0;
        !           331:        caddr_t                         cp;
        !           332:        struct esis_fixed       *pdu;
        !           333:        int                                     len, total_len = 0;
        !           334:        struct sockaddr_iso     siso;
        !           335:        struct ifnet            *ifp = inbound_shp->snh_ifp;
        !           336:        struct sockaddr_dl *sdl;
        !           337:        struct iso_addr *rd_gwnsap;
        !           338: 
        !           339:        if (rt->rt_flags & RTF_GATEWAY) {
        !           340:                rd_gwnsap = &((struct sockaddr_iso *)rt->rt_gateway)->siso_addr;
        !           341:                rt = rtalloc1(rt->rt_gateway, 0);
        !           342:        } else
        !           343:                rd_gwnsap = &((struct sockaddr_iso *)rt_key(rt))->siso_addr;
        !           344:        if (rt == 0 || (sdl = (struct sockaddr_dl *)rt->rt_gateway) == 0 ||
        !           345:                sdl->sdl_family != AF_LINK) {
        !           346:                /* maybe we should have a function that you
        !           347:                   could put in the iso_ifaddr structure
        !           348:                   which could translate iso_addrs into snpa's
        !           349:                   where there is a known mapping for that address type */
        !           350:                esis_stat.es_badtype++;
        !           351:                return;
        !           352:        }
        !           353:        esis_stat.es_rdsent++;
        !           354:        IFDEBUG(D_ESISOUTPUT)
        !           355:                printf("esis_rdoutput: ifp x%x (%s%d), ht %d, m x%x, oidx x%x\n",
        !           356:                        ifp, ifp->if_name, ifp->if_unit, esis_holding_time, inbound_m,
        !           357:                        inbound_oidx);
        !           358:                printf("\tdestination: %s\n", clnp_iso_addrp(rd_dstnsap));
        !           359:                printf("\tredirected toward:%s\n", clnp_iso_addrp(rd_gwnsap));
        !           360:        ENDDEBUG
        !           361: 
        !           362:        if ((m0 = m = m_gethdr(M_DONTWAIT, MT_HEADER)) == NULL) {
        !           363:                esis_stat.es_nomem++;
        !           364:                return;
        !           365:        }
        !           366:        bzero(mtod(m, caddr_t), MHLEN);
        !           367: 
        !           368:        pdu = mtod(m, struct esis_fixed *);
        !           369:        cp = (caddr_t)(pdu + 1); /*pointer arith.; 1st byte after header */
        !           370:        len = sizeof(struct esis_fixed);
        !           371: 
        !           372:        /*
        !           373:         *      Build fixed part of header
        !           374:         */
        !           375:        pdu->esis_proto_id = ISO9542_ESIS;
        !           376:        pdu->esis_vers = ESIS_VERSION;
        !           377:        pdu->esis_type = ESIS_RD;
        !           378:        HTOC(pdu->esis_ht_msb, pdu->esis_ht_lsb, esis_holding_time);
        !           379: 
        !           380:        /* Insert destination address */
        !           381:        (void) esis_insert_addr(&cp, &len, rd_dstnsap, m, 0);
        !           382: 
        !           383:        /* Insert the snpa of better next hop */
        !           384:        *cp++ = sdl->sdl_alen;
        !           385:        bcopy(LLADDR(sdl), cp, sdl->sdl_alen);
        !           386:        cp += sdl->sdl_alen;
        !           387:        len += (sdl->sdl_alen + 1);
        !           388: 
        !           389:        /* 
        !           390:         *      If the next hop is not the destination, then it ought to be
        !           391:         *      an IS and it should be inserted next. Else, set the
        !           392:         *      NETL to 0
        !           393:         */
        !           394:        /* PHASE2 use mask from ifp of outgoing interface */
        !           395:        if (!iso_addrmatch1(rd_dstnsap, rd_gwnsap)) {
        !           396:                /* this should not happen: 
        !           397:                if ((nhop_sc->sc_flags & SNPA_IS) == 0) {
        !           398:                        printf("esis_rdoutput: next hop is not dst and not an IS\n");
        !           399:                        m_freem(m0);
        !           400:                        return;
        !           401:                } */
        !           402:                (void) esis_insert_addr(&cp, &len, rd_gwnsap, m, 0);
        !           403:        } else {
        !           404:                *cp++ = 0;      /* NETL */
        !           405:                len++;
        !           406:        }
        !           407:        m->m_len = len;
        !           408: 
        !           409:        /*
        !           410:         *      PHASE2
        !           411:         *      If redirect is to an IS, add an address mask. The mask to be
        !           412:         *      used should be the mask present in the routing entry used to
        !           413:         *      forward the original data packet.
        !           414:         */
        !           415:        
        !           416:        /*
        !           417:         *      Copy Qos, priority, or security options present in original npdu
        !           418:         */
        !           419:        if (inbound_oidx) {
        !           420:                /* THIS CODE IS CURRENTLY (mostly) UNTESTED */
        !           421:                int optlen = 0;
        !           422:                if (inbound_oidx->cni_qos_formatp)
        !           423:                        optlen += (inbound_oidx->cni_qos_len + 2);
        !           424:                if (inbound_oidx->cni_priorp)   /* priority option is 1 byte long */
        !           425:                        optlen += 3;
        !           426:                if (inbound_oidx->cni_securep)
        !           427:                        optlen += (inbound_oidx->cni_secure_len + 2);
        !           428:                if (M_TRAILINGSPACE(m) < optlen) {
        !           429:                        EXTEND_PACKET(m, m0, cp);
        !           430:                        m->m_len = 0;
        !           431:                        /* assumes MLEN > optlen */
        !           432:                }
        !           433:                /* assume MLEN-len > optlen */
        !           434:                /* 
        !           435:                 *      When copying options, copy from ptr - 2 in order to grab
        !           436:                 *      the option code and length
        !           437:                 */
        !           438:                if (inbound_oidx->cni_qos_formatp) {
        !           439:                        bcopy(mtod(inbound_m, caddr_t) + inbound_oidx->cni_qos_formatp - 2,
        !           440:                                cp, (unsigned)(inbound_oidx->cni_qos_len + 2));
        !           441:                        cp += inbound_oidx->cni_qos_len + 2;
        !           442:                }
        !           443:                if (inbound_oidx->cni_priorp) {
        !           444:                        bcopy(mtod(inbound_m, caddr_t) + inbound_oidx->cni_priorp - 2,
        !           445:                                        cp, 3);
        !           446:                        cp += 3;
        !           447:                }
        !           448:                if (inbound_oidx->cni_securep) {
        !           449:                        bcopy(mtod(inbound_m, caddr_t) + inbound_oidx->cni_securep - 2, cp, 
        !           450:                                (unsigned)(inbound_oidx->cni_secure_len + 2));
        !           451:                        cp += inbound_oidx->cni_secure_len + 2;
        !           452:                }
        !           453:                m->m_len += optlen;
        !           454:                len += optlen;
        !           455:        }
        !           456: 
        !           457:        pdu->esis_hdr_len = m0->m_pkthdr.len = len;
        !           458:        iso_gen_csum(m0, ESIS_CKSUM_OFF, (int)pdu->esis_hdr_len);
        !           459: 
        !           460:        bzero((caddr_t)&siso, sizeof(siso));
        !           461:        siso.siso_family = AF_ISO;
        !           462:        siso.siso_data[0] = AFI_SNA;
        !           463:        siso.siso_nlen = 6 + 1; /* should be taken from snpa_hdr */
        !           464:                                                                                /* +1 is for AFI */
        !           465:        bcopy(inbound_shp->snh_shost, siso.siso_data + 1, 6);
        !           466:        (ifp->if_output)(ifp, m0, (struct sockaddr *)&siso, 0);
        !           467: }
        !           468: 
        !           469: /*
        !           470:  * FUNCTION:           esis_insert_addr
        !           471:  *
        !           472:  * PURPOSE:                    Insert an iso_addr into a buffer
        !           473:  *
        !           474:  * RETURNS:                    true if buffer was big enough, else false
        !           475:  *
        !           476:  * SIDE EFFECTS:       Increment buf & len according to size of iso_addr
        !           477:  *
        !           478:  * NOTES:                      Plus 1 here is for length byte
        !           479:  */
        !           480: esis_insert_addr(buf, len, isoa, m, nsellen)
        !           481: register caddr_t                       *buf;           /* ptr to buffer to put address into */
        !           482: int                                                    *len;           /* ptr to length of buffer so far */
        !           483: register struct iso_addr       *isoa;          /* ptr to address */
        !           484: register struct mbuf           *m;                     /* determine if there remains space */
        !           485: int                                                    nsellen;
        !           486: {
        !           487:        register int newlen, result = 0;
        !           488: 
        !           489:        isoa->isoa_len -= nsellen;
        !           490:        newlen = isoa->isoa_len + 1;
        !           491:        if (newlen <=  M_TRAILINGSPACE(m)) {
        !           492:                bcopy((caddr_t)isoa, *buf, newlen);
        !           493:                *len += newlen;
        !           494:                *buf += newlen;
        !           495:                m->m_len += newlen;
        !           496:                result = 1;
        !           497:        }
        !           498:        isoa->isoa_len += nsellen;
        !           499:        return (result);
        !           500: }
        !           501: 
        !           502: #define ESIS_EXTRACT_ADDR(d, b) { d = (struct iso_addr *)(b); b += (1 + *b); \
        !           503:            if (b > buflim) {esis_stat.es_toosmall++; goto bad;}}
        !           504: #define ESIS_NEXT_OPTION(b)    { b += (2 + b[1]); \
        !           505:            if (b > buflim) {esis_stat.es_toosmall++; goto bad;}}
        !           506: int ESHonly = 0;
        !           507: /*
        !           508:  
        !           509: /*
        !           510:  * FUNCTION:           esis_eshinput
        !           511:  *
        !           512:  * PURPOSE:                    Process an incoming ESH pdu
        !           513:  *
        !           514:  * RETURNS:                    nothing
        !           515:  *
        !           516:  * SIDE EFFECTS:       
        !           517:  *
        !           518:  * NOTES:                      
        !           519:  */
        !           520: esis_eshinput(m, shp)
        !           521: struct mbuf            *m;     /* esh pdu */
        !           522: struct snpa_hdr        *shp;   /* subnetwork header */
        !           523: {
        !           524:        struct  esis_fixed      *pdu = mtod(m, struct esis_fixed *);
        !           525:        u_short                         ht;             /* holding time */
        !           526:        struct  iso_addr        *nsap;
        !           527:        int                                     naddr;
        !           528:        u_char                          *buf = (u_char *)(pdu + 1);
        !           529:        u_char                          *buflim = pdu->esis_hdr_len + (u_char *)pdu;
        !           530:        int                                     new_entry = 0;
        !           531: 
        !           532:        esis_stat.es_eshrcvd++;
        !           533: 
        !           534:        CTOH(pdu->esis_ht_msb, pdu->esis_ht_lsb, ht);
        !           535: 
        !           536:        naddr = *buf++;
        !           537:        if (buf >= buflim)
        !           538:                goto bad;
        !           539:        if (naddr == 1) {
        !           540:                ESIS_EXTRACT_ADDR(nsap, buf);
        !           541:                new_entry = snpac_add(shp->snh_ifp,
        !           542:                                                                 nsap, shp->snh_shost, SNPA_ES, ht, 0);
        !           543:        } else {
        !           544:                int nsellength = 0, nlen = 0;
        !           545:                {
        !           546:                /* See if we want to compress out multiple nsaps differing
        !           547:                   only by nsel */
        !           548:                        register struct ifaddr *ifa = shp->snh_ifp->if_addrlist;
        !           549:                        for (; ifa; ifa = ifa->ifa_next)
        !           550:                                if (ifa->ifa_addr->sa_family == AF_ISO) {
        !           551:                                        nsellength = ((struct iso_ifaddr *)ifa)->ia_addr.siso_tlen;
        !           552:                                        break;
        !           553:                        }
        !           554:                }
        !           555:                IFDEBUG(D_ESISINPUT)
        !           556:                        printf("esis_eshinput: esh: ht %d, naddr %d nsellength %d\n",
        !           557:                                        ht, naddr, nsellength);
        !           558:                ENDDEBUG
        !           559:                while (naddr-- > 0) {
        !           560:                        struct iso_addr *nsap2; u_char *buf2;
        !           561:                        ESIS_EXTRACT_ADDR(nsap, buf);
        !           562:                        /* see if there is at least one more nsap in ESH differing
        !           563:                           only by nsel */
        !           564:                        if (nsellength != 0) for (buf2 = buf; buf2 < buflim;) {
        !           565:                                ESIS_EXTRACT_ADDR(nsap2, buf2);
        !           566:                                IFDEBUG(D_ESISINPUT)
        !           567:                                        printf("esis_eshinput: comparing %s ", 
        !           568:                                                clnp_iso_addrp(nsap));
        !           569:                                        printf("and %s\n", clnp_iso_addrp(nsap2));
        !           570:                                ENDDEBUG
        !           571:                                if (Bcmp(nsap->isoa_genaddr, nsap2->isoa_genaddr,
        !           572:                                                 nsap->isoa_len - nsellength) == 0) {
        !           573:                                        nlen = nsellength;
        !           574:                                        break;
        !           575:                                }
        !           576:                        }
        !           577:                        new_entry |= snpac_add(shp->snh_ifp,
        !           578:                                                                        nsap, shp->snh_shost, SNPA_ES, ht, nlen);
        !           579:                        nlen = 0;
        !           580:                }
        !           581:        }
        !           582:        IFDEBUG(D_ESISINPUT)
        !           583:                printf("esis_eshinput: nsap %s is %s\n", 
        !           584:                        clnp_iso_addrp(nsap), new_entry ? "new" : "old");
        !           585:        ENDDEBUG
        !           586:        if (new_entry && (iso_systype & SNPA_IS))
        !           587:                esis_shoutput(shp->snh_ifp, ESIS_ISH, esis_holding_time,
        !           588:                                                shp->snh_shost, 6, (struct iso_addr *)0);
        !           589: bad:
        !           590:        return;
        !           591: }
        !           592: 
        !           593: /*
        !           594:  * FUNCTION:           esis_ishinput
        !           595:  *
        !           596:  * PURPOSE:                    process an incoming ISH pdu
        !           597:  *
        !           598:  * RETURNS:                    
        !           599:  *
        !           600:  * SIDE EFFECTS:       
        !           601:  *
        !           602:  * NOTES:                      
        !           603:  */
        !           604: esis_ishinput(m, shp)
        !           605: struct mbuf            *m;     /* esh pdu */
        !           606: struct snpa_hdr        *shp;   /* subnetwork header */
        !           607: {
        !           608:        struct esis_fixed       *pdu = mtod(m, struct esis_fixed *);
        !           609:        u_short                         ht, newct;                      /* holding time */
        !           610:        struct iso_addr         *nsap;                          /* Network Entity Title */
        !           611:        register u_char         *buf = (u_char *) (pdu + 1);
        !           612:        register u_char         *buflim = pdu->esis_hdr_len + (u_char *)pdu;
        !           613:        int                                     new_entry;
        !           614: 
        !           615:        esis_stat.es_ishrcvd++;
        !           616:        CTOH(pdu->esis_ht_msb, pdu->esis_ht_lsb, ht);
        !           617: 
        !           618:        IFDEBUG(D_ESISINPUT)
        !           619:                printf("esis_ishinput: ish: ht %d\n", ht);
        !           620:        ENDDEBUG
        !           621:        if (ESHonly)
        !           622:                goto bad;
        !           623: 
        !           624:        ESIS_EXTRACT_ADDR(nsap, buf);
        !           625: 
        !           626:        while (buf < buflim) {
        !           627:                switch (*buf) {
        !           628:                case ESISOVAL_ESCT:
        !           629:                        if (iso_systype & SNPA_IS)
        !           630:                                break;
        !           631:                        if (buf[1] != 2)
        !           632:                                goto bad;
        !           633:                        CTOH(buf[2], buf[3], newct);
        !           634:                        if (esis_config_time != newct) {
        !           635:                                untimeout(esis_config,0);
        !           636:                                esis_config_time = newct;
        !           637:                                esis_config();
        !           638:                        }
        !           639:                        break;
        !           640:                
        !           641:                default:
        !           642:                        printf("Unknown ISH option: %x\n", *buf);
        !           643:                }
        !           644:                ESIS_NEXT_OPTION(buf);
        !           645:        }
        !           646:        new_entry = snpac_add(shp->snh_ifp, nsap, shp->snh_shost, SNPA_IS, ht, 0);
        !           647:        IFDEBUG(D_ESISINPUT)
        !           648:                printf("esis_ishinput: nsap %s is %s\n", 
        !           649:                        clnp_iso_addrp(nsap), new_entry ? "new" : "old");
        !           650:        ENDDEBUG
        !           651: 
        !           652:        if (new_entry)
        !           653:                esis_shoutput(shp->snh_ifp, 
        !           654:                        iso_systype & SNPA_ES ? ESIS_ESH : ESIS_ISH,
        !           655:                        esis_holding_time, shp->snh_shost, 6, (struct iso_addr *)0);
        !           656: bad:
        !           657:        return;
        !           658: }
        !           659: 
        !           660: /*
        !           661:  * FUNCTION:           esis_rdinput
        !           662:  *
        !           663:  * PURPOSE:                    Process an incoming RD pdu
        !           664:  *
        !           665:  * RETURNS:                    
        !           666:  *
        !           667:  * SIDE EFFECTS:       
        !           668:  *
        !           669:  * NOTES:                      
        !           670:  */
        !           671: esis_rdinput(m0, shp)
        !           672: struct mbuf            *m0;    /* esh pdu */
        !           673: struct snpa_hdr        *shp;   /* subnetwork header */
        !           674: {
        !           675:        struct esis_fixed       *pdu = mtod(m0, struct esis_fixed *);
        !           676:        u_short                         ht;             /* holding time */
        !           677:        struct iso_addr         *da, *net = 0, *netmask = 0, *snpamask = 0;
        !           678:        register struct iso_addr *bsnpa;
        !           679:        register u_char         *buf = (u_char *)(pdu + 1);
        !           680:        register u_char         *buflim = pdu->esis_hdr_len + (u_char *)pdu;
        !           681: 
        !           682:        esis_stat.es_rdrcvd++;
        !           683: 
        !           684:        /* intermediate systems ignore redirects */
        !           685:        if (iso_systype & SNPA_IS)
        !           686:                return;
        !           687:        if (ESHonly)
        !           688:                return;
        !           689: 
        !           690:        CTOH(pdu->esis_ht_msb, pdu->esis_ht_lsb, ht);
        !           691:        if (buf >= buflim)
        !           692:                return;
        !           693: 
        !           694:        /* Extract DA */
        !           695:        ESIS_EXTRACT_ADDR(da, buf);
        !           696: 
        !           697:        /* Extract better snpa */
        !           698:        ESIS_EXTRACT_ADDR(bsnpa, buf);
        !           699: 
        !           700:        /* Extract NET if present */
        !           701:        if (buf < buflim) {
        !           702:                if (*buf == 0)
        !           703:                        buf++; /* no NET present, skip NETL anyway */
        !           704:                else
        !           705:                        ESIS_EXTRACT_ADDR(net, buf);
        !           706:        }
        !           707: 
        !           708:        /* process options */
        !           709:        while (buf < buflim) {
        !           710:                switch (*buf) {
        !           711:                case ESISOVAL_SNPAMASK:
        !           712:                        if (snpamask) /* duplicate */
        !           713:                                return;
        !           714:                        snpamask = (struct iso_addr *)(buf + 1);
        !           715:                        break;
        !           716: 
        !           717:                case ESISOVAL_NETMASK:
        !           718:                        if (netmask) /* duplicate */
        !           719:                                return;
        !           720:                        netmask = (struct iso_addr *)(buf + 1);
        !           721:                        break;
        !           722: 
        !           723:                default:
        !           724:                        printf("Unknown option in ESIS RD (0x%x)\n", buf[-1]);
        !           725:                }
        !           726:                ESIS_NEXT_OPTION(buf);
        !           727:        }
        !           728: 
        !           729:        IFDEBUG(D_ESISINPUT)
        !           730:                printf("esis_rdinput: rd: ht %d, da %s\n", ht, clnp_iso_addrp(da));
        !           731:                if (net)
        !           732:                        printf("\t: net %s\n", clnp_iso_addrp(net));
        !           733:        ENDDEBUG
        !           734:        /*
        !           735:         *      If netl is zero, then redirect is to an ES. We need to add an entry
        !           736:         *      to the snpa cache for (destination, better snpa).
        !           737:         *      If netl is not zero, then the redirect is to an IS. In this
        !           738:         *      case, add an snpa cache entry for (net, better snpa).
        !           739:         *
        !           740:         *      If the redirect is to an IS, add a route entry towards that
        !           741:         *      IS.
        !           742:         */
        !           743:        if (net == 0 || net->isoa_len == 0 || snpamask) {
        !           744:                /* redirect to an ES */
        !           745:                snpac_add(shp->snh_ifp, da,
        !           746:                                bsnpa->isoa_genaddr, SNPA_ES, ht, 0);
        !           747:        } else {
        !           748:                snpac_add(shp->snh_ifp, net,
        !           749:                                bsnpa->isoa_genaddr, SNPA_IS, ht, 0);
        !           750:                snpac_addrt(shp->snh_ifp, da, net, netmask);
        !           751:        }
        !           752: bad: ;    /* Needed by ESIS_NEXT_OPTION */
        !           753: }
        !           754: 
        !           755: /*
        !           756:  * FUNCTION:           esis_config
        !           757:  *
        !           758:  * PURPOSE:                    Report configuration
        !           759:  *
        !           760:  * RETURNS:                    
        !           761:  *
        !           762:  * SIDE EFFECTS:       
        !           763:  *
        !           764:  * NOTES:                      Called every esis_config_time seconds
        !           765:  */
        !           766: void
        !           767: esis_config()
        !           768: {
        !           769:        register struct ifnet   *ifp;
        !           770: 
        !           771:        timeout(esis_config, (caddr_t)0, hz * esis_config_time);
        !           772: 
        !           773:        /* 
        !           774:         *      Report configuration for each interface that 
        !           775:         *      - is UP
        !           776:         *      - has BROADCAST capability
        !           777:         *      - has an ISO address
        !           778:         */
        !           779:        /* Todo: a better way would be to construct the esh or ish
        !           780:         * once and copy it out for all devices, possibly calling
        !           781:         * a method in the iso_ifaddr structure to encapsulate and
        !           782:         * transmit it.  This could work to advantage for non-broadcast media
        !           783:         */
        !           784:        
        !           785:        for (ifp = ifnet; ifp; ifp = ifp->if_next) {
        !           786:                if ((ifp->if_flags & IFF_UP) &&
        !           787:                    (ifp->if_flags & IFF_BROADCAST)) {
        !           788:                        /* search for an ISO address family */
        !           789:                        struct ifaddr   *ia;
        !           790: 
        !           791:                        for (ia = ifp->if_addrlist; ia; ia = ia->ifa_next) {
        !           792:                                if (ia->ifa_addr->sa_family == AF_ISO) {
        !           793:                                        esis_shoutput(ifp, 
        !           794:                                                iso_systype & SNPA_ES ? ESIS_ESH : ESIS_ISH,
        !           795:                                                esis_holding_time,
        !           796:                                                (caddr_t)(iso_systype & SNPA_ES ? all_is_snpa : 
        !           797:                                                all_es_snpa), 6, (struct iso_addr *)0);
        !           798:                                        break;
        !           799:                                }
        !           800:                        }
        !           801:                }
        !           802:        }
        !           803: }
        !           804: 
        !           805: /*
        !           806:  * FUNCTION:           esis_shoutput
        !           807:  *
        !           808:  * PURPOSE:                    Transmit an esh or ish pdu
        !           809:  *
        !           810:  * RETURNS:                    nothing
        !           811:  *
        !           812:  * SIDE EFFECTS:       
        !           813:  *
        !           814:  * NOTES:                      
        !           815:  */
        !           816: esis_shoutput(ifp, type, ht, sn_addr, sn_len, isoa)
        !           817: struct ifnet   *ifp;
        !           818: int                            type;
        !           819: short                  ht;
        !           820: caddr_t                sn_addr;
        !           821: int                            sn_len;
        !           822: struct iso_addr *isoa;
        !           823: {
        !           824:        struct mbuf                     *m, *m0;
        !           825:        caddr_t                         cp, naddrp;
        !           826:        int                                     naddr = 0;
        !           827:        struct esis_fixed       *pdu;
        !           828:        struct iso_ifaddr       *ia;
        !           829:        int                                     len;
        !           830:        struct sockaddr_iso     siso;
        !           831: 
        !           832:        if (type == ESIS_ESH)
        !           833:                esis_stat.es_eshsent++;
        !           834:        else if (type == ESIS_ISH) 
        !           835:                esis_stat.es_ishsent++;
        !           836:        else {
        !           837:                printf("esis_shoutput: bad pdu type\n");
        !           838:                return;
        !           839:        }
        !           840: 
        !           841:        IFDEBUG(D_ESISOUTPUT)
        !           842:                int     i;
        !           843:                printf("esis_shoutput: ifp x%x (%s%d), %s, ht %d, to: [%d] ",
        !           844:                        ifp, ifp->if_name, ifp->if_unit, type == ESIS_ESH ? "esh" : "ish",
        !           845:                        ht, sn_len);
        !           846:                for (i=0; i<sn_len; i++)
        !           847:                        printf("%x%c", *(sn_addr+i), i < (sn_len-1) ? ':' : ' ');
        !           848:                printf("\n");
        !           849:        ENDDEBUG
        !           850: 
        !           851:        if ((m0 = m = m_gethdr(M_DONTWAIT, MT_HEADER)) == NULL) {
        !           852:                esis_stat.es_nomem++;
        !           853:                return;
        !           854:        }
        !           855:        bzero(mtod(m, caddr_t), MHLEN);
        !           856: 
        !           857:        pdu = mtod(m, struct esis_fixed *);
        !           858:        naddrp = cp = (caddr_t)(pdu + 1);
        !           859:        len = sizeof(struct esis_fixed);
        !           860: 
        !           861:        /*
        !           862:         *      Build fixed part of header
        !           863:         */
        !           864:        pdu->esis_proto_id = ISO9542_ESIS;
        !           865:        pdu->esis_vers = ESIS_VERSION;
        !           866:        pdu->esis_type = type;
        !           867:        HTOC(pdu->esis_ht_msb, pdu->esis_ht_lsb, ht);
        !           868: 
        !           869:        if (type == ESIS_ESH) {
        !           870:                cp++;
        !           871:                len++;
        !           872:        }
        !           873: 
        !           874:        m->m_len = len;
        !           875:        if (isoa) {
        !           876:                /*
        !           877:                 * Here we are responding to a clnp packet sent to an NSAP
        !           878:                 * that is ours which was sent to the MAC addr all_es's.
        !           879:                 * It is possible that we did not specifically advertise this
        !           880:                 * NSAP, even though it is ours, so we will respond
        !           881:                 * directly to the sender that we are here.  If we do have
        !           882:                 * multiple NSEL's we'll tack them on so he can compress them out.
        !           883:                 */
        !           884:                (void) esis_insert_addr(&cp, &len, isoa, m, 0);
        !           885:                naddr = 1;
        !           886:        }
        !           887:        for (ia = iso_ifaddr; ia; ia = ia->ia_next) {
        !           888:                int nsellen = (type == ESIS_ISH ? ia->ia_addr.siso_tlen : 0); 
        !           889:                int n = ia->ia_addr.siso_nlen;
        !           890:                register struct iso_ifaddr *ia2;
        !           891: 
        !           892:                if (type == ESIS_ISH && naddr > 0)
        !           893:                        break;
        !           894:                for (ia2 = iso_ifaddr; ia2 != ia; ia2 = ia2->ia_next)
        !           895:                        if (Bcmp(ia->ia_addr.siso_data, ia2->ia_addr.siso_data, n) == 0)
        !           896:                                        break;
        !           897:                if (ia2 != ia)
        !           898:                        continue;       /* Means we have previously copied this nsap */
        !           899:                if (isoa && Bcmp(ia->ia_addr.siso_data, isoa->isoa_genaddr, n) == 0) {
        !           900:                        isoa = 0;
        !           901:                        continue;       /* Ditto */
        !           902:                }
        !           903:                IFDEBUG(D_ESISOUTPUT)
        !           904:                        printf("esis_shoutput: adding NSAP %s\n", 
        !           905:                                clnp_iso_addrp(&ia->ia_addr.siso_addr));
        !           906:                ENDDEBUG
        !           907:                if (!esis_insert_addr(&cp, &len,
        !           908:                                                          &ia->ia_addr.siso_addr, m, nsellen)) {
        !           909:                        EXTEND_PACKET(m, m0, cp);
        !           910:                        (void) esis_insert_addr(&cp, &len, &ia->ia_addr.siso_addr, m,
        !           911:                                                                        nsellen);
        !           912:                }
        !           913:                naddr++;
        !           914:        }
        !           915: 
        !           916:        if (type == ESIS_ESH)
        !           917:                *naddrp = naddr;
        !           918:        else {
        !           919:                /* add suggested es config timer option to ISH */
        !           920:                if (M_TRAILINGSPACE(m) < 4) {
        !           921:                        printf("esis_shoutput: extending packet\n");
        !           922:                        EXTEND_PACKET(m, m0, cp);
        !           923:                }
        !           924:                *cp++ = ESISOVAL_ESCT;
        !           925:                *cp++ = 2;
        !           926:                HTOC(*cp, *(cp+1), esis_esconfig_time);
        !           927:                len += 4;
        !           928:                m->m_len += 4;
        !           929:                IFDEBUG(D_ESISOUTPUT)
        !           930:                        printf("m0 0x%x, m 0x%x, data 0x%x, len %d, cp 0x%x\n",
        !           931:                        m0, m, m->m_data, m->m_len, cp);
        !           932:                ENDDEBUG
        !           933:        }
        !           934: 
        !           935:        m0->m_pkthdr.len = len;
        !           936:        pdu->esis_hdr_len = len;
        !           937:        iso_gen_csum(m0, ESIS_CKSUM_OFF, (int)pdu->esis_hdr_len);
        !           938: 
        !           939:        bzero((caddr_t)&siso, sizeof(siso));
        !           940:        siso.siso_family = AF_ISO;
        !           941:        siso.siso_data[0] = AFI_SNA;
        !           942:        siso.siso_nlen = sn_len + 1;
        !           943:        bcopy(sn_addr, siso.siso_data + 1, (unsigned)sn_len);
        !           944:        (ifp->if_output)(ifp, m0, (struct sockaddr *)&siso, 0);
        !           945: }
        !           946: 
        !           947: /*
        !           948:  * FUNCTION:           isis_input
        !           949:  *
        !           950:  * PURPOSE:                    Process an incoming isis packet
        !           951:  *
        !           952:  * RETURNS:                    nothing
        !           953:  *
        !           954:  * SIDE EFFECTS:       
        !           955:  *
        !           956:  * NOTES:                      
        !           957:  */
        !           958: isis_input(m0, shp)
        !           959: struct mbuf            *m0;            /* ptr to first mbuf of pkt */
        !           960: struct snpa_hdr        *shp;   /* subnetwork header */
        !           961: {
        !           962:        register int type;
        !           963:        register struct rawcb *rp, *first_rp = 0;
        !           964:        struct ifnet *ifp = shp->snh_ifp;
        !           965:        char workbuf[16];
        !           966:        struct mbuf *mm;
        !           967: 
        !           968:        IFDEBUG(D_ISISINPUT)
        !           969:                int i;
        !           970: 
        !           971:                printf("isis_input: pkt on ifp x%x (%s%d): from:", ifp, 
        !           972:                        ifp->if_name, ifp->if_unit);
        !           973:                for (i=0; i<6; i++)
        !           974:                        printf("%x%c", shp->snh_shost[i]&0xff, (i<5) ? ':' : ' ');
        !           975:                printf(" to:");
        !           976:                for (i=0; i<6; i++)
        !           977:                        printf("%x%c", shp->snh_dhost[i]&0xff, (i<5) ? ':' : ' ');
        !           978:                printf("\n");
        !           979:        ENDDEBUG
        !           980:        esis_dl.sdl_alen = ifp->if_addrlen;
        !           981:        esis_dl.sdl_index = ifp->if_index;
        !           982:        bcopy(shp->snh_shost, (caddr_t)esis_dl.sdl_data, esis_dl.sdl_alen);
        !           983:        for (rp = esis_pcb.rcb_next; rp != &esis_pcb; rp = rp->rcb_next) {
        !           984:                if (first_rp == 0) {
        !           985:                        first_rp = rp;
        !           986:                        continue;
        !           987:                }
        !           988:                if (mm = m_copy(m0, 0, M_COPYALL)) { /*can't block at interrupt level */
        !           989:                        if (sbappendaddr(&rp->rcb_socket->so_rcv,
        !           990:                                                          &esis_dl, mm, (struct mbuf *)0) != 0) {
        !           991:                                sorwakeup(rp->rcb_socket);
        !           992:                         } else {
        !           993:                                IFDEBUG(D_ISISINPUT)
        !           994:                                        printf("Error in sbappenaddr, mm = 0x%x\n", mm);
        !           995:                                ENDDEBUG
        !           996:                                m_freem(mm);
        !           997:                        }
        !           998:                }
        !           999:        }
        !          1000:        if (first_rp && sbappendaddr(&first_rp->rcb_socket->so_rcv,
        !          1001:                                                          &esis_dl, m0, (struct mbuf *)0) != 0) {
        !          1002:                sorwakeup(first_rp->rcb_socket);
        !          1003:                return;
        !          1004:        }
        !          1005:        m_freem(m0);
        !          1006: }
        !          1007: 
        !          1008: isis_output(sdl, m)
        !          1009: register struct sockaddr_dl    *sdl;
        !          1010: struct mbuf *m;
        !          1011: {
        !          1012:        register struct ifnet *ifp;
        !          1013:        struct ifaddr *ifa, *ifa_ifwithnet();
        !          1014:        struct sockaddr_iso siso;
        !          1015:        int error = 0;
        !          1016:        unsigned sn_len;
        !          1017: 
        !          1018:        ifa = ifa_ifwithnet((struct sockaddr *)sdl);    /* get ifp from sdl */
        !          1019:        if (ifa == 0) {
        !          1020:                IFDEBUG(D_ISISOUTPUT)
        !          1021:                        printf("isis_output: interface not found\n");
        !          1022:                ENDDEBUG
        !          1023:                error = EINVAL;
        !          1024:                goto release;
        !          1025:        }
        !          1026:        ifp = ifa->ifa_ifp;
        !          1027:        sn_len = sdl->sdl_alen;
        !          1028:        IFDEBUG(D_ISISOUTPUT)
        !          1029:                u_char *cp = (u_char *)LLADDR(sdl), *cplim = cp + sn_len;
        !          1030:                printf("isis_output: ifp 0x%x (%s%d), to: ",
        !          1031:                        ifp, ifp->if_name, ifp->if_unit);
        !          1032:                while (cp < cplim) {
        !          1033:                        printf("%x", *cp++);
        !          1034:                        printf("%c", (cp < cplim) ? ':' : ' ');
        !          1035:                }
        !          1036:                printf("\n");
        !          1037:        ENDDEBUG
        !          1038:        bzero((caddr_t)&siso, sizeof(siso));
        !          1039:        siso.siso_family = AF_ISO; /* This convention may be useful for X.25 */
        !          1040:        siso.siso_data[0] = AFI_SNA;
        !          1041:        siso.siso_nlen = sn_len + 1;
        !          1042:        bcopy(LLADDR(sdl), siso.siso_data + 1, sn_len);
        !          1043:        error = (ifp->if_output)(ifp, m, (struct sockaddr *)&siso, 0);
        !          1044:        if (error) {
        !          1045:                IFDEBUG(D_ISISOUTPUT)
        !          1046:                        printf("isis_output: error from ether_output is %d\n", error);
        !          1047:                ENDDEBUG
        !          1048:        }
        !          1049:        return (error);
        !          1050: 
        !          1051: release:
        !          1052:        if (m != NULL)
        !          1053:                m_freem(m);
        !          1054:        return(error);
        !          1055: }
        !          1056: 
        !          1057: 
        !          1058: /*
        !          1059:  * FUNCTION:           esis_ctlinput
        !          1060:  *
        !          1061:  * PURPOSE:                    Handle the PRC_IFDOWN transition
        !          1062:  *
        !          1063:  * RETURNS:                    nothing
        !          1064:  *
        !          1065:  * SIDE EFFECTS:       
        !          1066:  *
        !          1067:  * NOTES:                      Calls snpac_flush for interface specified.
        !          1068:  *                                     The loop through iso_ifaddr is stupid because
        !          1069:  *                                     back in if_down, we knew the ifp...
        !          1070:  */
        !          1071: esis_ctlinput(req, siso)
        !          1072: int                                            req;            /* request: we handle only PRC_IFDOWN */
        !          1073: struct sockaddr_iso            *siso;          /* address of ifp */
        !          1074: {
        !          1075:        register struct iso_ifaddr *ia; /* scan through interface addresses */
        !          1076: 
        !          1077:        if (req == PRC_IFDOWN)
        !          1078:                for (ia = iso_ifaddr; ia; ia = ia->ia_next) {
        !          1079:                        if (iso_addrmatch(IA_SIS(ia), siso))
        !          1080:                                snpac_flushifp(ia->ia_ifp);
        !          1081:                }
        !          1082: }
        !          1083: 
        !          1084: #endif /* ISO */

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.