Annotation of 43BSDReno/sys/hpdev/if_le.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1982, 1990 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * Redistribution is only permitted until one year after the first shipment
        !             6:  * of 4.4BSD by the Regents.  Otherwise, redistribution and use in source and
        !             7:  * binary forms are permitted provided that: (1) source distributions retain
        !             8:  * this entire copyright notice and comment, and (2) distributions including
        !             9:  * binaries display the following acknowledgement:  This product includes
        !            10:  * software developed by the University of California, Berkeley and its
        !            11:  * contributors'' in the documentation or other materials provided with the
        !            12:  * distribution and in all advertising materials mentioning features or use
        !            13:  * of this software.  Neither the name of the University nor the names of
        !            14:  * its contributors may be used to endorse or promote products derived from
        !            15:  * this software without specific prior written permission.
        !            16:  * THIS SOFTWARE IS PROVIDED AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
        !            17:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
        !            18:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
        !            19:  *
        !            20:  *     @(#)if_le.c     7.1 (Berkeley) 5/8/90
        !            21:  */
        !            22: 
        !            23: #include "le.h"
        !            24: #if NLE > 0
        !            25: 
        !            26: /*
        !            27:  * AMD 7990 LANCE
        !            28:  *
        !            29:  * This driver will generate and accept tailer encapsulated packets even
        !            30:  * though it buys us nothing.  The motivation was to avoid incompatibilities
        !            31:  * with VAXen, SUNs, and others that handle and benefit from them.
        !            32:  * This reasoning is dubious.
        !            33:  */
        !            34: #include "param.h"
        !            35: #include "systm.h"
        !            36: #include "mbuf.h"
        !            37: #include "buf.h"
        !            38: #include "protosw.h"
        !            39: #include "socket.h"
        !            40: #include "syslog.h"
        !            41: #include "ioctl.h"
        !            42: #include "errno.h"
        !            43: 
        !            44: #include "../net/if.h"
        !            45: #include "../net/netisr.h"
        !            46: #include "../net/route.h"
        !            47: 
        !            48: #ifdef INET
        !            49: #include "../netinet/in.h"
        !            50: #include "../netinet/in_systm.h"
        !            51: #include "../netinet/in_var.h"
        !            52: #include "../netinet/ip.h"
        !            53: #include "../netinet/if_ether.h"
        !            54: #endif
        !            55: 
        !            56: #ifdef NS
        !            57: #include "../netns/ns.h"
        !            58: #include "../netns/ns_if.h"
        !            59: #endif
        !            60: 
        !            61: #ifdef RMP
        !            62: #include "../netrmp/rmp.h"
        !            63: #include "../netrmp/rmp_var.h"
        !            64: #endif
        !            65: 
        !            66: #include "machine/cpu.h"
        !            67: #include "machine/isr.h"
        !            68: #include "machine/mtpr.h"
        !            69: #include "device.h"
        !            70: #include "if_lereg.h"
        !            71: 
        !            72: /* offsets for:           ID,   REGS,    MEM,  NVRAM */
        !            73: int    lestd[] = { 0, 0x4000, 0x8000, 0xC008 };
        !            74: 
        !            75: int    leattach();
        !            76: struct driver ledriver = {
        !            77:        leattach, "le",
        !            78: };
        !            79: 
        !            80: struct isr le_isr[NLE];
        !            81: int    ledebug = 0;            /* console error messages */
        !            82: 
        !            83: int    leintr(), leinit(), leioctl(), lestart(), ether_output();
        !            84: struct mbuf *leget();
        !            85: extern struct ifnet loif;
        !            86: 
        !            87: /*
        !            88:  * Ethernet software status per interface.
        !            89:  *
        !            90:  * Each interface is referenced by a network interface structure,
        !            91:  * le_if, which the routing code uses to locate the interface.
        !            92:  * This structure contains the output queue for the interface, its address, ...
        !            93:  */
        !            94: struct le_softc {
        !            95:        struct  arpcom sc_ac;   /* common Ethernet structures */
        !            96: #define        sc_if   sc_ac.ac_if     /* network-visible interface */
        !            97: #define        sc_addr sc_ac.ac_enaddr /* hardware Ethernet address */
        !            98:        struct  lereg0 *sc_r0;  /* DIO registers */
        !            99:        struct  lereg1 *sc_r1;  /* LANCE registers */
        !           100:        struct  lereg2 *sc_r2;  /* dual-port RAM */
        !           101:        int     sc_rmd;         /* predicted next rmd to process */
        !           102:        int     sc_runt;
        !           103:        int     sc_jab;
        !           104:        int     sc_merr;
        !           105:        int     sc_babl;
        !           106:        int     sc_cerr;
        !           107:        int     sc_miss;
        !           108:        int     sc_xint;
        !           109:        int     sc_xown;
        !           110:        int     sc_uflo;
        !           111:        int     sc_rxlen;
        !           112:        int     sc_rxoff;
        !           113:        int     sc_txoff;
        !           114:        int     sc_busy;
        !           115: } le_softc[NLE];
        !           116: 
        !           117: /* access LANCE registers */
        !           118: #define        LERDWR(cntl, src, dst) \
        !           119:        do { \
        !           120:                (dst) = (src); \
        !           121:        } while (((cntl)->ler0_status & LE_ACK) == 0);
        !           122: 
        !           123: /*
        !           124:  * Interface exists: make available by filling in network interface
        !           125:  * record.  System will initialize the interface when it is ready
        !           126:  * to accept packets.
        !           127:  */
        !           128: leattach(hd)
        !           129:        struct hp_device *hd;
        !           130: {
        !           131:        register struct lereg0 *ler0;
        !           132:        register struct lereg2 *ler2;
        !           133:        struct lereg2 *lemem = 0;
        !           134:        struct le_softc *le = &le_softc[hd->hp_unit];
        !           135:        struct ifnet *ifp = &le->sc_if;
        !           136:        char *cp;
        !           137:        int i;
        !           138: 
        !           139:        ler0 = le->sc_r0 = (struct lereg0 *)(lestd[0] + (int)hd->hp_addr);
        !           140:        le->sc_r1 = (struct lereg1 *)(lestd[1] + (int)hd->hp_addr);
        !           141:        ler2 = le->sc_r2 = (struct lereg2 *)(lestd[2] + (int)hd->hp_addr);
        !           142:        if (ler0->ler0_id != LEID)
        !           143:                return(0);
        !           144:        le_isr[hd->hp_unit].isr_intr = leintr;
        !           145:        hd->hp_ipl = le_isr[hd->hp_unit].isr_ipl = LE_IPL(ler0->ler0_status);
        !           146:        le_isr[hd->hp_unit].isr_arg = hd->hp_unit;
        !           147:        ler0->ler0_id = 0xFF;
        !           148:        DELAY(100);
        !           149: 
        !           150:        /*
        !           151:         * Read the ethernet address off the board, one nibble at a time.
        !           152:         */
        !           153:        cp = (char *)(lestd[3] + (int)hd->hp_addr);
        !           154:        for (i = 0; i < sizeof(le->sc_addr); i++) {
        !           155:                le->sc_addr[i] = (*++cp & 0xF) << 4;
        !           156:                cp++;
        !           157:                le->sc_addr[i] |= *++cp & 0xF;
        !           158:                cp++;
        !           159:        }
        !           160:        printf("le%d: hardware address %s\n", hd->hp_unit,
        !           161:                ether_sprintf(le->sc_addr));
        !           162: 
        !           163:        /*
        !           164:         * Setup for transmit/receive
        !           165:         */
        !           166:        ler2->ler2_mode = LE_MODE;
        !           167:        ler2->ler2_padr[0] = le->sc_addr[1];
        !           168:        ler2->ler2_padr[1] = le->sc_addr[0];
        !           169:        ler2->ler2_padr[2] = le->sc_addr[3];
        !           170:        ler2->ler2_padr[3] = le->sc_addr[2];
        !           171:        ler2->ler2_padr[4] = le->sc_addr[5];
        !           172:        ler2->ler2_padr[5] = le->sc_addr[4];
        !           173: #ifdef RMP
        !           174:        /*
        !           175:         * Set up logical addr filter to accept multicast 9:0:9:0:0:4
        !           176:         * This should be an ioctl() to the driver.  (XXX)
        !           177:         */
        !           178:        ler2->ler2_ladrf0 = 0x00100000;
        !           179:        ler2->ler2_ladrf1 = 0x0;
        !           180: #else
        !           181:        ler2->ler2_ladrf0 = 0;
        !           182:        ler2->ler2_ladrf1 = 0;
        !           183: #endif
        !           184:        ler2->ler2_rlen = LE_RLEN;
        !           185:        ler2->ler2_rdra = (int)lemem->ler2_rmd;
        !           186:        ler2->ler2_tlen = LE_TLEN;
        !           187:        ler2->ler2_tdra = (int)lemem->ler2_tmd;
        !           188:        isrlink(&le_isr[hd->hp_unit]);
        !           189:        ler0->ler0_status = LE_IE;
        !           190: 
        !           191:        ifp->if_unit = hd->hp_unit;
        !           192:        ifp->if_name = "le";
        !           193:        ifp->if_mtu = ETHERMTU;
        !           194:        ifp->if_init = leinit;
        !           195:        ifp->if_ioctl = leioctl;
        !           196:        ifp->if_output = ether_output;
        !           197:        ifp->if_start = lestart;
        !           198:        ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX;
        !           199:        if_attach(ifp);
        !           200:        return (1);
        !           201: }
        !           202: 
        !           203: ledrinit(ler2)
        !           204:        register struct lereg2 *ler2;
        !           205: {
        !           206:        register struct lereg2 *lemem = 0;
        !           207:        register int i;
        !           208: 
        !           209:        for (i = 0; i < LERBUF; i++) {
        !           210:                ler2->ler2_rmd[i].rmd0 = (int)lemem->ler2_rbuf[i];
        !           211:                ler2->ler2_rmd[i].rmd1 = LE_OWN;
        !           212:                ler2->ler2_rmd[i].rmd2 = -LEMTU;
        !           213:                ler2->ler2_rmd[i].rmd3 = 0;
        !           214:        }
        !           215:        for (i = 0; i < LETBUF; i++) {
        !           216:                ler2->ler2_tmd[i].tmd0 = (int)lemem->ler2_tbuf[i];
        !           217:                ler2->ler2_tmd[i].tmd1 = 0;
        !           218:                ler2->ler2_tmd[i].tmd2 = 0;
        !           219:                ler2->ler2_tmd[i].tmd3 = 0;
        !           220:        }
        !           221: }
        !           222: 
        !           223: lereset(unit)
        !           224:        register int unit;
        !           225: {
        !           226:        register struct le_softc *le = &le_softc[unit];
        !           227:        register struct lereg0 *ler0 = le->sc_r0;
        !           228:        register struct lereg1 *ler1 = le->sc_r1;
        !           229:        register struct lereg2 *lemem = 0;
        !           230:        register int timo = 100000;
        !           231:        register int stat;
        !           232: 
        !           233: #ifdef lint
        !           234:        stat = unit;
        !           235: #endif
        !           236:        LERDWR(ler0, LE_CSR0, ler1->ler1_rap);
        !           237:        LERDWR(ler0, LE_STOP, ler1->ler1_rdp);
        !           238:        ledrinit(le->sc_r2);
        !           239:        le->sc_rmd = 0;
        !           240:        LERDWR(ler0, LE_CSR1, ler1->ler1_rap);
        !           241:        LERDWR(ler0, (int)&lemem->ler2_mode, ler1->ler1_rdp);
        !           242:        LERDWR(ler0, LE_CSR2, ler1->ler1_rap);
        !           243:        LERDWR(ler0, 0, ler1->ler1_rdp);
        !           244:        LERDWR(ler0, LE_CSR0, ler1->ler1_rap);
        !           245:        LERDWR(ler0, LE_INIT, ler1->ler1_rdp);
        !           246:        do {
        !           247:                if (--timo == 0) {
        !           248:                        printf("le%d: init timeout, stat = 0x%x\n",
        !           249:                               unit, stat);
        !           250:                        break;
        !           251:                }
        !           252:                LERDWR(ler0, ler1->ler1_rdp, stat);
        !           253:        } while ((stat & LE_IDON) == 0);
        !           254:        LERDWR(ler0, LE_STOP, ler1->ler1_rdp);
        !           255:        LERDWR(ler0, LE_CSR3, ler1->ler1_rap);
        !           256:        LERDWR(ler0, LE_BSWP, ler1->ler1_rdp);
        !           257:        LERDWR(ler0, LE_CSR0, ler1->ler1_rap);
        !           258:        LERDWR(ler0, LE_STRT | LE_INEA, ler1->ler1_rdp);
        !           259:        le->sc_if.if_flags &= ~IFF_OACTIVE;
        !           260: }
        !           261: 
        !           262: /*
        !           263:  * Initialization of interface
        !           264:  */
        !           265: leinit(unit)
        !           266:        int unit;
        !           267: {
        !           268:        struct le_softc *le = &le_softc[unit];
        !           269:        register struct ifnet *ifp = &le->sc_if;
        !           270:        int s;
        !           271: 
        !           272:        /* not yet, if address still unknown */
        !           273:        if (ifp->if_addrlist == (struct ifaddr *)0)
        !           274:                return;
        !           275:        if ((ifp->if_flags & IFF_RUNNING) == 0) {
        !           276:                s = splimp();
        !           277:                ifp->if_flags |= IFF_RUNNING;
        !           278:                lereset(unit);
        !           279:                (void) lestart(ifp);
        !           280:                splx(s);
        !           281:        }
        !           282: }
        !           283: 
        !           284: /*
        !           285:  * Start output on interface.  Get another datagram to send
        !           286:  * off of the interface queue, and copy it to the interface
        !           287:  * before starting the output.
        !           288:  */
        !           289: lestart(ifp)
        !           290:        struct ifnet *ifp;
        !           291: {
        !           292:        register struct le_softc *le = &le_softc[ifp->if_unit];
        !           293:        register struct letmd *tmd;
        !           294:        register struct mbuf *m;
        !           295:        int len;
        !           296: 
        !           297:        if ((le->sc_if.if_flags & IFF_RUNNING) == 0)
        !           298:                return (0);
        !           299:        IF_DEQUEUE(&le->sc_if.if_snd, m);
        !           300:        if (m == 0)
        !           301:                return (0);
        !           302:        len = leput(le->sc_r2->ler2_tbuf[0], m);
        !           303:        tmd = le->sc_r2->ler2_tmd;
        !           304:        tmd->tmd3 = 0;
        !           305:        tmd->tmd2 = -len;
        !           306:        tmd->tmd1 = LE_OWN | LE_STP | LE_ENP;
        !           307:        le->sc_if.if_flags |= IFF_OACTIVE;
        !           308:        return (0);
        !           309: }
        !           310: 
        !           311: leintr(unit)
        !           312:        register int unit;
        !           313: {
        !           314:        register struct le_softc *le = &le_softc[unit];
        !           315:        register struct lereg0 *ler0 = le->sc_r0;
        !           316:        register struct lereg1 *ler1;
        !           317:        register int stat;
        !           318: 
        !           319:        if ((ler0->ler0_status & LE_IR) == 0)
        !           320:                return(0);
        !           321:        if (ler0->ler0_status & LE_JAB) {
        !           322:                le->sc_jab++;
        !           323:                lereset(unit);
        !           324:                return(1);
        !           325:        }
        !           326:        ler1 = le->sc_r1;
        !           327:        LERDWR(ler0, ler1->ler1_rdp, stat);
        !           328:        if (stat & LE_SERR) {
        !           329:                leerror(unit, stat);
        !           330:                if (stat & LE_MERR) {
        !           331:                        le->sc_merr++;
        !           332:                        lereset(unit);
        !           333:                        return(1);
        !           334:                }
        !           335:                if (stat & LE_BABL)
        !           336:                        le->sc_babl++;
        !           337:                if (stat & LE_CERR)
        !           338:                        le->sc_cerr++;
        !           339:                if (stat & LE_MISS)
        !           340:                        le->sc_miss++;
        !           341:                LERDWR(ler0, LE_BABL|LE_CERR|LE_MISS|LE_INEA, ler1->ler1_rdp);
        !           342:        }
        !           343:        if ((stat & LE_RXON) == 0) {
        !           344:                le->sc_rxoff++;
        !           345:                lereset(unit);
        !           346:                return(1);
        !           347:        }
        !           348:        if ((stat & LE_TXON) == 0) {
        !           349:                le->sc_txoff++;
        !           350:                lereset(unit);
        !           351:                return(1);
        !           352:        }
        !           353:        if (stat & LE_RINT) {
        !           354:                /* interrupt is cleared in lerint */
        !           355:                lerint(unit);
        !           356:        }
        !           357:        if (stat & LE_TINT) {
        !           358:                LERDWR(ler0, LE_TINT|LE_INEA, ler1->ler1_rdp);
        !           359:                lexint(unit);
        !           360:        }
        !           361:        return(1);
        !           362: }
        !           363: 
        !           364: /*
        !           365:  * Ethernet interface transmitter interrupt.
        !           366:  * Start another output if more data to send.
        !           367:  */
        !           368: lexint(unit)
        !           369:        register int unit;
        !           370: {
        !           371:        register struct le_softc *le = &le_softc[unit];
        !           372:        register struct letmd *tmd = le->sc_r2->ler2_tmd;
        !           373: 
        !           374:        if ((le->sc_if.if_flags & IFF_OACTIVE) == 0) {
        !           375:                le->sc_xint++;
        !           376:                return;
        !           377:        }
        !           378:        if (tmd->tmd1 & LE_OWN) {
        !           379:                le->sc_xown++;
        !           380:                return;
        !           381:        }
        !           382:        if (tmd->tmd1 & LE_ERR) {
        !           383: err:
        !           384:                lexerror(unit);
        !           385:                le->sc_if.if_oerrors++;
        !           386:                if (tmd->tmd3 & (LE_TBUFF|LE_UFLO)) {
        !           387:                        le->sc_uflo++;
        !           388:                        lereset(unit);
        !           389:                }
        !           390:                else if (tmd->tmd3 & LE_LCOL)
        !           391:                        le->sc_if.if_collisions++;
        !           392:                else if (tmd->tmd3 & LE_RTRY)
        !           393:                        le->sc_if.if_collisions += 16;
        !           394:        }
        !           395:        else if (tmd->tmd3 & LE_TBUFF)
        !           396:                /* XXX documentation says BUFF not included in ERR */
        !           397:                goto err;
        !           398:        else if (tmd->tmd1 & LE_ONE)
        !           399:                le->sc_if.if_collisions++;
        !           400:        else if (tmd->tmd1 & LE_MORE)
        !           401:                /* what is the real number? */
        !           402:                le->sc_if.if_collisions += 2;
        !           403:        else
        !           404:                le->sc_if.if_opackets++;
        !           405:        le->sc_if.if_flags &= ~IFF_OACTIVE;
        !           406:        (void) lestart(&le->sc_if);
        !           407: }
        !           408: 
        !           409: #define        LENEXTRMP \
        !           410:        if (++bix == LERBUF) bix = 0, rmd = le->sc_r2->ler2_rmd; else ++rmd
        !           411: 
        !           412: /*
        !           413:  * Ethernet interface receiver interrupt.
        !           414:  * If input error just drop packet.
        !           415:  * Decapsulate packet based on type and pass to type specific
        !           416:  * higher-level input routine.
        !           417:  */
        !           418: lerint(unit)
        !           419:        int unit;
        !           420: {
        !           421:        register struct le_softc *le = &le_softc[unit];
        !           422:        register int bix = le->sc_rmd;
        !           423:        register struct lermd *rmd = &le->sc_r2->ler2_rmd[bix];
        !           424: 
        !           425:        /*
        !           426:         * Out of sync with hardware, should never happen?
        !           427:         */
        !           428:        if (rmd->rmd1 & LE_OWN) {
        !           429:                LERDWR(le->sc_r0, LE_RINT|LE_INEA, le->sc_r1->ler1_rdp);
        !           430:                return;
        !           431:        }
        !           432: 
        !           433:        /*
        !           434:         * Process all buffers with valid data
        !           435:         */
        !           436:        while ((rmd->rmd1 & LE_OWN) == 0) {
        !           437:                int len = rmd->rmd3;
        !           438: 
        !           439:                /* Clear interrupt to avoid race condition */
        !           440:                LERDWR(le->sc_r0, LE_RINT|LE_INEA, le->sc_r1->ler1_rdp);
        !           441: 
        !           442:                if (rmd->rmd1 & LE_ERR) {
        !           443:                        le->sc_rmd = bix;
        !           444:                        lererror(unit, "bad packet");
        !           445:                        le->sc_if.if_ierrors++;
        !           446:                } else if ((rmd->rmd1 & (LE_STP|LE_ENP)) != (LE_STP|LE_ENP)) {
        !           447:                        /*
        !           448:                         * Find the end of the packet so we can see how long
        !           449:                         * it was.  We still throw it away.
        !           450:                         */
        !           451:                        do {
        !           452:                                LERDWR(le->sc_r0, LE_RINT|LE_INEA,
        !           453:                                       le->sc_r1->ler1_rdp);
        !           454:                                rmd->rmd3 = 0;
        !           455:                                rmd->rmd1 = LE_OWN;
        !           456:                                LENEXTRMP;
        !           457:                        } while (!(rmd->rmd1 & (LE_OWN|LE_ERR|LE_STP|LE_ENP)));
        !           458:                        le->sc_rmd = bix;
        !           459:                        lererror(unit, "chained buffer");
        !           460:                        le->sc_rxlen++;
        !           461:                        /*
        !           462:                         * If search terminated without successful completion
        !           463:                         * we reset the hardware (conservative).
        !           464:                         */
        !           465:                        if ((rmd->rmd1 & (LE_OWN|LE_ERR|LE_STP|LE_ENP)) !=
        !           466:                            LE_ENP) {
        !           467:                                lereset(unit);
        !           468:                                return;
        !           469:                        }
        !           470:                } else
        !           471:                        leread(unit, le->sc_r2->ler2_rbuf[bix], len);
        !           472:                rmd->rmd3 = 0;
        !           473:                rmd->rmd1 = LE_OWN;
        !           474:                LENEXTRMP;
        !           475:        }
        !           476:        le->sc_rmd = bix;
        !           477: }
        !           478: 
        !           479: leread(unit, buf, len)
        !           480:        int unit;
        !           481:        char *buf;
        !           482:        int len;
        !           483: {
        !           484:        register struct le_softc *le = &le_softc[unit];
        !           485:        register struct ether_header *et;
        !           486:        struct mbuf *m;
        !           487:        int off, resid;
        !           488: 
        !           489:        le->sc_if.if_ipackets++;
        !           490:        et = (struct ether_header *)buf;
        !           491:        et->ether_type = ntohs((u_short)et->ether_type);
        !           492:        /* adjust input length to account for header and CRC */
        !           493:        len = len - sizeof(struct ether_header) - 4;
        !           494: 
        !           495: #ifdef RMP
        !           496:        /*  (XXX)
        !           497:         *
        !           498:         *  If Ethernet Type field is < MaxPacketSize, we probably have
        !           499:         *  a IEEE802 packet here.  Make sure that the size is at least
        !           500:         *  that of the HP LLC.  Also do sanity checks on length of LLC
        !           501:         *  (old Ethernet Type field) and packet length.
        !           502:         *
        !           503:         *  Provided the above checks succeed, change `len' to reflect
        !           504:         *  the length of the LLC (i.e. et->ether_type) and change the
        !           505:         *  type field to ETHERTYPE_IEEE so we can switch() on it later.
        !           506:         *  Yes, this is a hack and will eventually be done "right".
        !           507:         */
        !           508:        if (et->ether_type <= IEEE802LEN_MAX && len >= sizeof(struct hp_llc) &&
        !           509:            len >= et->ether_type && len >= IEEE802LEN_MIN) {
        !           510:                len = et->ether_type;
        !           511:                et->ether_type = ETHERTYPE_IEEE;        /* hack! */
        !           512:        }
        !           513: #endif
        !           514: 
        !           515: #define        ledataaddr(et, off, type)       ((type)(((caddr_t)((et)+1)+(off))))
        !           516:        if (et->ether_type >= ETHERTYPE_TRAIL &&
        !           517:            et->ether_type < ETHERTYPE_TRAIL+ETHERTYPE_NTRAILER) {
        !           518:                off = (et->ether_type - ETHERTYPE_TRAIL) * 512;
        !           519:                if (off >= ETHERMTU)
        !           520:                        return;         /* sanity */
        !           521:                et->ether_type = ntohs(*ledataaddr(et, off, u_short *));
        !           522:                resid = ntohs(*(ledataaddr(et, off+2, u_short *)));
        !           523:                if (off + resid > len)
        !           524:                        return;         /* sanity */
        !           525:                len = off + resid;
        !           526:        } else
        !           527:                off = 0;
        !           528: 
        !           529:        if (len <= 0) {
        !           530:                if (ledebug)
        !           531:                        log(LOG_WARNING,
        !           532:                            "le%d: ierror(runt packet): from %s: len=%d\n",
        !           533:                            unit, ether_sprintf(et->ether_shost), len);
        !           534:                le->sc_runt++;
        !           535:                le->sc_if.if_ierrors++;
        !           536:                return;
        !           537:        }
        !           538: 
        !           539:        /*
        !           540:         * Pull packet off interface.  Off is nonzero if packet
        !           541:         * has trailing header; leget will then force this header
        !           542:         * information to be at the front, but we still have to drop
        !           543:         * the type and length which are at the front of any trailer data.
        !           544:         */
        !           545:        m = leget(buf, len, off, &le->sc_if);
        !           546:        if (m == 0)
        !           547:                return;
        !           548: #ifdef RMP
        !           549:        /*
        !           550:         * (XXX)
        !           551:         * This needs to be integrated with the ISO stuff in ether_input()
        !           552:         */
        !           553:        if (et->ether_type == ETHERTYPE_IEEE) {
        !           554:                /*
        !           555:                 *  Snag the Logical Link Control header (IEEE 802.2).
        !           556:                 */
        !           557:                struct hp_llc *llc = &(mtod(m, struct rmp_packet *)->hp_llc);
        !           558: 
        !           559:                /*
        !           560:                 *  If the DSAP (and HP's extended DXSAP) indicate this
        !           561:                 *  is an RMP packet, hand it to the raw input routine.
        !           562:                 */
        !           563:                if (llc->dsap == IEEE_DSAP_HP && llc->dxsap == HPEXT_DXSAP) {
        !           564:                        static struct sockproto rmp_sp = {AF_RMP,RMPPROTO_BOOT};
        !           565:                        static struct sockaddr rmp_src = {AF_RMP};
        !           566:                        static struct sockaddr rmp_dst = {AF_RMP};
        !           567: 
        !           568:                        bcopy(et->ether_shost, rmp_src.sa_data,
        !           569:                              sizeof(et->ether_shost));
        !           570:                        bcopy(et->ether_dhost, rmp_dst.sa_data,
        !           571:                              sizeof(et->ether_dhost));
        !           572: 
        !           573:                        raw_input(m, &rmp_sp, &rmp_src, &rmp_dst);
        !           574:                        return;
        !           575:                }
        !           576:        }
        !           577: #endif
        !           578:        ether_input(&le->sc_if, et, m);
        !           579: }
        !           580: 
        !           581: /*
        !           582:  * Routine to copy from mbuf chain to transmit
        !           583:  * buffer in board local memory.
        !           584:  */
        !           585: leput(lebuf, m)
        !           586:        register char *lebuf;
        !           587:        register struct mbuf *m;
        !           588: {
        !           589:        register struct mbuf *mp;
        !           590:        register int len, tlen = 0;
        !           591: 
        !           592:        for (mp = m; mp; mp = mp->m_next) {
        !           593:                len = mp->m_len;
        !           594:                if (len == 0)
        !           595:                        continue;
        !           596:                tlen += len;
        !           597:                bcopy(mtod(mp, char *), lebuf, len);
        !           598:                lebuf += len;
        !           599:        }
        !           600:        m_freem(m);
        !           601:        if (tlen < LEMINSIZE) {
        !           602:                bzero(lebuf, LEMINSIZE - tlen);
        !           603:                tlen = LEMINSIZE;
        !           604:        }
        !           605:        return(tlen);
        !           606: }
        !           607: 
        !           608: /*
        !           609:  * Routine to copy from board local memory into mbufs.
        !           610:  */
        !           611: struct mbuf *
        !           612: leget(lebuf, totlen, off0, ifp)
        !           613:        char *lebuf;
        !           614:        int totlen, off0;
        !           615:        struct ifnet *ifp;
        !           616: {
        !           617:        register struct mbuf *m;
        !           618:        struct mbuf *top = 0, **mp = &top;
        !           619:        register int off = off0, len;
        !           620:        register char *cp;
        !           621:        char *epkt;
        !           622: 
        !           623:        lebuf += sizeof (struct ether_header);
        !           624:        cp = lebuf;
        !           625:        epkt = cp + totlen;
        !           626:        if (off) {
        !           627:                cp += off + 2 * sizeof(u_short);
        !           628:                totlen -= 2 * sizeof(u_short);
        !           629:        }
        !           630: 
        !           631:        MGETHDR(m, M_DONTWAIT, MT_DATA);
        !           632:        if (m == 0)
        !           633:                return (0);
        !           634:        m->m_pkthdr.rcvif = ifp;
        !           635:        m->m_pkthdr.len = totlen;
        !           636:        m->m_len = MHLEN;
        !           637: 
        !           638:        while (totlen > 0) {
        !           639:                if (top) {
        !           640:                        MGET(m, M_DONTWAIT, MT_DATA);
        !           641:                        if (m == 0) {
        !           642:                                m_freem(top);
        !           643:                                return (0);
        !           644:                        }
        !           645:                        m->m_len = MLEN;
        !           646:                }
        !           647:                len = min(totlen, epkt - cp);
        !           648:                if (len >= MINCLSIZE) {
        !           649:                        MCLGET(m, M_DONTWAIT);
        !           650:                        if (m->m_flags & M_EXT)
        !           651:                                m->m_len = len = min(len, MCLBYTES);
        !           652:                        else
        !           653:                                len = m->m_len;
        !           654:                } else {
        !           655:                        /*
        !           656:                         * Place initial small packet/header at end of mbuf.
        !           657:                         */
        !           658:                        if (len < m->m_len) {
        !           659:                                if (top == 0 && len + max_linkhdr <= m->m_len)
        !           660:                                        m->m_data += max_linkhdr;
        !           661:                                m->m_len = len;
        !           662:                        } else
        !           663:                                len = m->m_len;
        !           664:                }
        !           665:                bcopy(cp, mtod(m, caddr_t), (unsigned)len);
        !           666:                cp += len;
        !           667:                *mp = m;
        !           668:                mp = &m->m_next;
        !           669:                totlen -= len;
        !           670:                if (cp == epkt)
        !           671:                        cp = lebuf;
        !           672:        }
        !           673:        return (top);
        !           674: }
        !           675: 
        !           676: /*
        !           677:  * Process an ioctl request.
        !           678:  */
        !           679: leioctl(ifp, cmd, data)
        !           680:        register struct ifnet *ifp;
        !           681:        int cmd;
        !           682:        caddr_t data;
        !           683: {
        !           684:        register struct ifaddr *ifa = (struct ifaddr *)data;
        !           685:        struct le_softc *le = &le_softc[ifp->if_unit];
        !           686:        struct lereg1 *ler1 = le->sc_r1;
        !           687:        int s = splimp(), error = 0;
        !           688: 
        !           689:        switch (cmd) {
        !           690: 
        !           691:        case SIOCSIFADDR:
        !           692:                ifp->if_flags |= IFF_UP;
        !           693:                switch (ifa->ifa_addr->sa_family) {
        !           694: #ifdef INET
        !           695:                case AF_INET:
        !           696:                        leinit(ifp->if_unit);   /* before arpwhohas */
        !           697:                        ((struct arpcom *)ifp)->ac_ipaddr =
        !           698:                                IA_SIN(ifa)->sin_addr;
        !           699:                        arpwhohas((struct arpcom *)ifp, &IA_SIN(ifa)->sin_addr);
        !           700:                        break;
        !           701: #endif
        !           702: #ifdef NS
        !           703:                case AF_NS:
        !           704:                    {
        !           705:                        register struct ns_addr *ina = &(IA_SNS(ifa)->sns_addr);
        !           706: 
        !           707:                        if (ns_nullhost(*ina))
        !           708:                                ina->x_host = *(union ns_host *)(le->sc_addr);
        !           709:                        else {
        !           710:                                /* 
        !           711:                                 * The manual says we can't change the address 
        !           712:                                 * while the receiver is armed,
        !           713:                                 * so reset everything
        !           714:                                 */
        !           715:                                ifp->if_flags &= ~IFF_RUNNING; 
        !           716:                                bcopy((caddr_t)ina->x_host.c_host,
        !           717:                                    (caddr_t)le->sc_addr, sizeof(le->sc_addr));
        !           718:                        }
        !           719:                        leinit(ifp->if_unit); /* does le_setaddr() */
        !           720:                        break;
        !           721:                    }
        !           722: #endif
        !           723:                default:
        !           724:                        leinit(ifp->if_unit);
        !           725:                        break;
        !           726:                }
        !           727:                break;
        !           728: 
        !           729:        case SIOCSIFFLAGS:
        !           730:                if ((ifp->if_flags & IFF_UP) == 0 &&
        !           731:                    ifp->if_flags & IFF_RUNNING) {
        !           732:                        LERDWR(le->sc_r0, LE_STOP, ler1->ler1_rdp);
        !           733:                        ifp->if_flags &= ~IFF_RUNNING;
        !           734:                } else if (ifp->if_flags & IFF_UP &&
        !           735:                    (ifp->if_flags & IFF_RUNNING) == 0)
        !           736:                        leinit(ifp->if_unit);
        !           737:                break;
        !           738: 
        !           739:        default:
        !           740:                error = EINVAL;
        !           741:        }
        !           742:        splx(s);
        !           743:        return (error);
        !           744: }
        !           745: 
        !           746: leerror(unit, stat)
        !           747:        int unit;
        !           748:        int stat;
        !           749: {
        !           750:        if (!ledebug)
        !           751:                return;
        !           752: 
        !           753:        /*
        !           754:         * Not all transceivers implement heartbeat
        !           755:         * so we only log CERR once.
        !           756:         */
        !           757:        if ((stat & LE_CERR) && le_softc[unit].sc_cerr)
        !           758:                return;
        !           759:        log(LOG_WARNING,
        !           760:            "le%d: error: stat=%b\n", unit,
        !           761:            stat,
        !           762:            "\20\20ERR\17BABL\16CERR\15MISS\14MERR\13RINT\12TINT\11IDON\10INTR\07INEA\06RXON\05TXON\04TDMD\03STOP\02STRT\01INIT");
        !           763: }
        !           764: 
        !           765: lererror(unit, msg)
        !           766:        int unit;
        !           767:        char *msg;
        !           768: {
        !           769:        register struct le_softc *le = &le_softc[unit];
        !           770:        register struct lermd *rmd;
        !           771:        int len;
        !           772: 
        !           773:        if (!ledebug)
        !           774:                return;
        !           775: 
        !           776:        rmd = &le->sc_r2->ler2_rmd[le->sc_rmd];
        !           777:        len = rmd->rmd3;
        !           778:        log(LOG_WARNING,
        !           779:            "le%d: ierror(%s): from %s: buf=%d, len=%d, rmd1=%b\n",
        !           780:            unit, msg,
        !           781:            len > 11 ? ether_sprintf(&le->sc_r2->ler2_rbuf[le->sc_rmd][6]) : "unknown",
        !           782:            le->sc_rmd, len,
        !           783:            rmd->rmd1,
        !           784:            "\20\20OWN\17ERR\16FRAM\15OFLO\14CRC\13RBUF\12STP\11ENP");
        !           785: }
        !           786: 
        !           787: lexerror(unit)
        !           788:        int unit;
        !           789: {
        !           790:        register struct le_softc *le = &le_softc[unit];
        !           791:        register struct letmd *tmd;
        !           792:        int len;
        !           793: 
        !           794:        if (!ledebug)
        !           795:                return;
        !           796: 
        !           797:        tmd = le->sc_r2->ler2_tmd;
        !           798:        len = -tmd->tmd2;
        !           799:        log(LOG_WARNING,
        !           800:            "le%d: oerror: to %s: buf=%d, len=%d, tmd1=%b, tmd3=%b\n",
        !           801:            unit,
        !           802:            len > 5 ? ether_sprintf(&le->sc_r2->ler2_tbuf[0][0]) : "unknown",
        !           803:            0, len,
        !           804:            tmd->tmd1,
        !           805:            "\20\20OWN\17ERR\16RES\15MORE\14ONE\13DEF\12STP\11ENP",
        !           806:            tmd->tmd3,
        !           807:            "\20\20BUFF\17UFLO\16RES\15LCOL\14LCAR\13RTRY");
        !           808: }
        !           809: #endif

unix.superglobalmegacorp.com

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