Return to BSD News archive
Received: by minnie.vk1xwt.ampr.org with NNTP id AA5161 ; Tue, 22 Dec 92 07:00:31 EST Path: sserve!manuel.anu.edu.au!munnari.oz.au!sgiblab!spool.mu.edu!uunet!hoptoad!curt From: curt@hoptoad.uucp (Curt Mayer) Newsgroups: comp.unix.bsd Subject: Shitty NE2000 NFS performance problem fixed right (source code) Message-ID: <30089@hoptoad.uucp> Date: 18 Dec 92 21:20:25 GMT Distribution: world Organization: hardcore driver hackerz Lines: 854 Summary: buggy driver, RTFM after playing with if_ne for a day or two, i have made 8K NFS mounts solid. ftp performance is twice what the previous driver was. NFS is now filesystem limited, the way it should be. bad buffer management, coupled with not using a high performance mode the ns8390, made the old driver lose bad when you got a lot of back to back packets. CAVEATS: ne1000 mode is untested, as I do not have one. I put the ne2000 clone I have into 8 bit mode, and tested the 8 bit stuff. because of this, consider this driver beta. anyway, the new driver: # This is a shell archive. Save it in a file, remove anything before # this line, and then unpack it by entering "sh file". Note, it may # create directories; files and directories will be owned by you and # have default permissions. # # This archive contains: # # if_ne.c # echo x - if_ne.c sed 's/^X//' >if_ne.c << 'END-of-if_ne.c' X/*- X * Copyright (c) 1990, 1991 William F. Jolitz. X * Copyright (c) 1990 The Regents of the University of California. X * All rights reserved. X * X * Redistribution and use in source and binary forms, with or without X * modification, are permitted provided that the following conditions X * are met: X * 1. Redistributions of source code must retain the above copyright X * notice, this list of conditions and the following disclaimer. X * 2. Redistributions in binary form must reproduce the above copyright X * notice, this list of conditions and the following disclaimer in the X * documentation and/or other materials provided with the distribution. X * 3. All advertising materials mentioning features or use of this software X * must display the following acknowledgement: X * This product includes software developed by the University of X * California, Berkeley and its contributors. X * 4. Neither the name of the University nor the names of its contributors X * may be used to endorse or promote products derived from this software X * without specific prior written permission. X * X * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND X * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE X * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE X * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE X * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL X * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS X * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) X * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT X * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY X * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF X * SUCH DAMAGE. X * X * @(#)if_ne.c 7.4 (Berkeley) 5/21/91 X */ X X/* X * NE2000 Ethernet driver X * X * Parts inspired from Tim Tucker's if_wd driver for the wd8003, X * insight on the ne2000 gained from Robert Clements PC/FTP driver. X * X * receive bottom end totally rewritten by Curt Mayer, Dec 1992. X * no longer loses back to back packets. X * note to driver writers: RTFM! X */ X X#include "ne.h" X#if NNE > 0 X X#include "param.h" X#include "systm.h" X#include "mbuf.h" X#include "buf.h" X#include "protosw.h" X#include "socket.h" X#include "ioctl.h" X#include "errno.h" X#include "syslog.h" X X#include "net/if.h" X#include "net/netisr.h" X#include "net/route.h" X X#ifdef INET X#include "netinet/in.h" X#include "netinet/in_systm.h" X#include "netinet/in_var.h" X#include "netinet/ip.h" X#include "netinet/if_ether.h" X#endif X X#ifdef NS X#include "netns/ns.h" X#include "netns/ns_if.h" X#endif X X#include "i386/isa/isa_device.h" X#include "i386/isa/if_nereg.h" X#include "i386/isa/icu.h" X Xint neprobe(), neattach(), neintr(); Xint nestart(),neinit(), ether_output(), neioctl(); X Xstruct isa_driver nedriver = { X neprobe, neattach, "ne", X}; X Xstruct mbuf *neget(); X X#define ETHER_MIN_LEN 64 X#define ETHER_MAX_LEN 1536 X X/* X * Ethernet software status per interface. X * X * Each interface is referenced by a network interface structure, X * ns_if, which the routing code uses to locate the interface. X * This structure contains the output queue for the interface, its address, ... X */ Xstruct ne_softc { X struct arpcom ns_ac; /* Ethernet common part */ X#define ns_if ns_ac.ac_if /* network-visible interface */ X#define ns_addr ns_ac.ac_enaddr /* hardware Ethernet address */ X int ns_flags; X#define DSF_LOCK 1 /* block re-entering enstart */ X int ns_oactive; X int ns_mask; X struct prhdr ns_ph; /* hardware header of incoming packet*/ X u_char ns_pb[2048]; X u_char ns_txstart; /* transmitter buffer start */ X u_char ns_rxstart; /* receiver buffer start */ X u_char ns_rxend; /* receiver buffer end */ X short ns_port; /* i/o port base */ X short ns_mode; /* word/byte mode */ X} ne_softc[NNE] ; X#define ENBUFSIZE (sizeof(struct ether_header) + ETHERMTU + 2 + ETHER_MIN_LEN) X X#define PAT(n) (0xa55a + 37*(n)) X Xu_short boarddata[16]; X Xneprobe(dvp) X struct isa_device *dvp; X{ X int val, i, s, sum, pat; X register struct ne_softc *ns = &ne_softc[0]; X register nec; X X#ifdef lint X neintr(0); X#endif X X nec = ns->ns_port = dvp->id_iobase; X s = splimp(); X X /* Byte Transfers, Burst Mode Select, Fifo at 8 bytes */ X ns->ns_mode = DSDC_BMS|DSDC_FT1; X ns->ns_txstart = TBUF8 / DS_PGSIZE; X ns->ns_rxend = RBUFEND8 / DS_PGSIZE; X Xtryagain: X X ns->ns_rxstart = ns->ns_txstart + (PKTSZ / DS_PGSIZE); X X /* Reset the bastard */ X val = inb(nec + ne_reset); X DELAY(200); X outb(nec + ne_reset, val); X DELAY(200); X X outb(nec + ds_cmd, DSCM_STOP|DSCM_NODMA); X X i = 10000; X while ((inb(nec + ds0_isr) & DSIS_RESET) == 0 && i-- > 0); X if (i < 0) return (0); X X outb(nec + ds0_isr, 0xff); X outb(nec + ds0_dcr, ns->ns_mode); X outb(nec + ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_STOP); X DELAY(1000); X X /* Check cmd reg and fail if not right */ X if ((i = inb(nec + ds_cmd)) != (DSCM_NODMA|DSCM_PG0|DSCM_STOP)) X return(0); X X outb(nec + ds0_tcr, 0); X outb(nec + ds0_rcr, DSRC_MON); X outb(nec + ds0_pstart, ns->ns_rxstart); X outb(nec + ds0_pstop, ns->ns_rxend); X outb(nec + ds0_imr, 0); X outb(nec + ds0_isr, 0); X outb(nec + ds0_bnry, ns->ns_rxstart); X outb(nec + ds_cmd, DSCM_NODMA|DSCM_PG1|DSCM_STOP); X outb(nec + ds1_curr, ns->ns_rxstart); X outb(nec + ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_STOP); X X /* X * <groan> detect difference between units X * solely by where the RAM is decoded. X */ X pat = PAT(0); X neput(ns, &pat, ns->ns_txstart * DS_PGSIZE, 4); X nefetch(ns, &pat, ns->ns_txstart * DS_PGSIZE, 4); X if (pat != PAT(0)) { X if (ns->ns_mode & DSDC_WTS) X return (0); X X /* Word Transfers, Burst Mode Select, Fifo at 8 bytes */ X ns->ns_mode = DSDC_WTS|DSDC_BMS|DSDC_FT1; X ns->ns_txstart = TBUF16 / DS_PGSIZE; X ns->ns_rxend = RBUFEND16 / DS_PGSIZE; X goto tryagain; X } X X X /* Extract board address */ X nefetch (ns, (caddr_t)boarddata, 0, sizeof(boarddata)); X X for(i=0; i < 6; i++) X ns->ns_addr[i] = boarddata[i]; X splx(s); X return (1); X} X X/* X * Fetch from onboard ROM/RAM X */ Xnefetch (ns, up, ad, len) struct ne_softc *ns; caddr_t up; { X u_char cmd; X register nec = ns->ns_port; X int counter = 100000; X X cmd = inb (nec + ds_cmd); X outb (nec + ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_START); X X /* Setup remote dma */ X outb (nec + ds0_isr, DSIS_RDC); X X if ((ns->ns_mode & DSDC_WTS) && (len & 1)) X len++; /* roundup to words */ X X outb (nec+ds0_rbcr0, len); X outb (nec+ds0_rbcr1, len>>8); X outb (nec+ds0_rsar0, ad); X outb (nec+ds0_rsar1, ad>>8); X X /* Execute & extract from card */ X outb (nec+ds_cmd, DSCM_RREAD|DSCM_PG0|DSCM_START); X X if (ns->ns_mode & DSDC_WTS) X insw (nec+ne_data, up, len/2); X else X insb (nec+ne_data, up, len); X X /* Wait till done, then shutdown feature */ X while ((inb (nec+ds0_isr) & DSIS_RDC) == 0 && counter-- > 0) X ; X outb (nec+ds0_isr, DSIS_RDC); X outb (nec+ds_cmd, cmd); X} X X/* X * Put to onboard RAM X */ Xneput (ns, up, ad, len) struct ne_softc *ns; caddr_t up; { X u_char cmd; X register nec = ns->ns_port; X int counter = 100000; X X cmd = inb(nec+ds_cmd); X outb (nec+ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_START); X X /* Setup for remote dma */ X outb (nec+ds0_isr, DSIS_RDC); X X if ((ns->ns_mode & DSDC_WTS) && (len & 1)) X len++; /* roundup to words */ X X outb (nec+ds0_rbcr0, len); X outb (nec+ds0_rbcr1, len>>8); X outb (nec+ds0_rsar0, ad); X outb (nec+ds0_rsar1, ad>>8); X X /* Execute & stuff to card */ X outb (nec+ds_cmd, DSCM_RWRITE|DSCM_PG0|DSCM_START); X if (ns->ns_mode & DSDC_WTS) X outsw (nec+ne_data, up, len/2); X else X outsb (nec+ne_data, up, len); X X /* Wait till done, then shutdown feature */ X while ((inb (nec+ds0_isr) & DSIS_RDC) == 0 && counter-- > 0) X ; X outb (nec+ds0_isr, DSIS_RDC); X outb (nec+ds_cmd, cmd); X} X X/* X * Reset of interface. X */ Xnereset(unit, uban) X int unit, uban; X{ X if (unit >= NNE) X return; X printf("ne%d: reset\n", unit); X ne_softc[unit].ns_flags &= ~DSF_LOCK; X neinit(unit); X} X X/* X * Interface exists: make available by filling in network interface X * record. System will initialize the interface when it is ready X * to accept packets. We get the ethernet address here. X */ Xneattach(dvp) X struct isa_device *dvp; X{ X int unit = dvp->id_unit; X register struct ne_softc *ns = &ne_softc[unit]; X register struct ifnet *ifp = &ns->ns_if; X X ifp->if_unit = unit; X ifp->if_name = nedriver.name ; X ifp->if_mtu = ETHERMTU; X printf (" ne%d, address %s", X (ns->ns_mode & DSDC_WTS) ? 2000 : 1000, X ether_sprintf(ns->ns_addr)) ; X ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_NOTRAILERS; X ifp->if_init = neinit; X ifp->if_output = ether_output; X ifp->if_start = nestart; X ifp->if_ioctl = neioctl; X ifp->if_reset = nereset; X ifp->if_watchdog = 0; X if_attach(ifp); X} X X/* X * Initialization of interface; set up initialization block X * and transmit/receive descriptor rings. X */ Xneinit(unit) X int unit; X{ X register struct ne_softc *ns = &ne_softc[unit]; X struct ifnet *ifp = &ns->ns_if; X int s; X int i; char *cp; X register nec = ns->ns_port; X X if (ifp->if_addrlist == (struct ifaddr *)0) return; X if (ifp->if_flags & IFF_RUNNING) return; X X s = splimp(); X X /* set physical address on ethernet */ X outb (nec+ds_cmd, DSCM_NODMA|DSCM_PG1|DSCM_STOP); X for (i=0 ; i < 6 ; i++) outb(nec+ds1_par0+i,ns->ns_addr[i]); X X /* clr logical address hash filter for now */ X for (i=0 ; i < 8 ; i++) outb(nec+ds1_mar0+i,0xff); X X /* init regs */ X outb (nec+ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_STOP); X outb (nec+ds0_rbcr0, 0); X outb (nec+ds0_rbcr1, 0); X outb (nec+ds0_imr, 0); X outb (nec+ds0_isr, 0xff); X outb(nec+ds0_dcr, ns->ns_mode); X outb(nec+ds0_tcr, 0); X outb (nec+ds0_rcr, DSRC_MON); X outb (nec+ds0_tpsr, 0); X outb(nec+ds0_pstart, ns->ns_rxstart); X outb(nec+ds0_pstop, ns->ns_rxend); X outb(nec+ds0_bnry, ns->ns_rxstart); X outb (nec+ds_cmd, DSCM_NODMA|DSCM_PG1|DSCM_STOP); X outb(nec+ds1_curr, ns->ns_rxstart); X outb (nec+ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_START); X outb (nec+ds0_rcr, DSRC_AB); X outb(nec+ds0_dcr, ns->ns_mode); X outb (nec+ds0_imr, 0xff); X X ns->ns_if.if_flags |= IFF_RUNNING; X ns->ns_flags &= ~DSF_LOCK; X ns->ns_oactive = 0; ns->ns_mask = ~0; X nestart(ifp); X splx(s); X} X X/* X * Setup output on interface. X * Get another datagram to send off of the interface queue, X * and map it to the interface before starting the output. X * called only at splimp or interrupt level. X */ Xnestart(ifp) X struct ifnet *ifp; X{ X register struct ne_softc *ns = &ne_softc[ifp->if_unit]; X struct mbuf *m0, *m; X int buffer; X int len = 0, i, total,t; X register nec = ns->ns_port; X X /* X * The DS8390 has only one transmit buffer, if it is busy we X * must wait until the transmit interrupt completes. X */ X outb(nec+ds_cmd,DSCM_NODMA|DSCM_START); X X if (ns->ns_flags & DSF_LOCK) X return; X X if (inb(nec+ds_cmd) & DSCM_TRANS) X return; X X if ((ns->ns_if.if_flags & IFF_RUNNING) == 0) X return; X X IF_DEQUEUE(&ns->ns_if.if_snd, m); X X if (m == 0) X return; X X /* X * Copy the mbuf chain into the transmit buffer X */ X X ns->ns_flags |= DSF_LOCK; /* prevent entering nestart */ X buffer = ns->ns_txstart * DS_PGSIZE; X len = i = 0; X t = 0; X for (m0 = m; m != 0; m = m->m_next) X t += m->m_len; X X m = m0; X total = t; X for (m0 = m; m != 0; ) { X X if (m->m_len&1 && t > m->m_len) { X neput(ns, mtod(m, caddr_t), buffer, m->m_len - 1); X t -= m->m_len - 1; X buffer += m->m_len - 1; X m->m_data += m->m_len - 1; X m->m_len = 1; X m = m_pullup(m, 2); X } else { X neput(ns, mtod(m, caddr_t), buffer, m->m_len); X buffer += m->m_len; X t -= m->m_len; X MFREE(m, m0); X m = m0; X } X } X X /* X * Init transmit length registers, and set transmit start flag. X */ X X len = total; X if (len < ETHER_MIN_LEN) len = ETHER_MIN_LEN; X outb(nec+ds0_tbcr0,len&0xff); X outb(nec+ds0_tbcr1,(len>>8)&0xff); X outb(nec+ds0_tpsr, ns->ns_txstart); X outb(nec+ds_cmd, DSCM_TRANS|DSCM_NODMA|DSCM_START); X} X X/* X * Controller interrupt. X */ Xneintr(unit) X{ X register struct ne_softc *ns = &ne_softc[unit]; X u_char cmd,isr; X register nec = ns->ns_port; X u_char err; X X /* Save cmd, clear interrupt */ X cmd = inb (nec+ds_cmd); Xloop: X isr = inb (nec+ds0_isr); X outb(nec+ds_cmd,DSCM_NODMA|DSCM_START); X outb(nec+ds0_isr, isr); X X /* Receiver error */ X if (isr & DSIS_RXE) { X /* need to read these registers to clear status */ X err = inb(nec+ ds0_rsr); X (void)inb(nec+0xD); X (void)inb(nec+0xE); X (void)inb(nec+0xF); X ns->ns_if.if_ierrors++; X } X X /* We received something */ X if (isr & DSIS_RX) { X u_char bnry; X u_char curr; X int len; X int i; X unsigned char c; X X while (1) { X X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA|DSCM_PG0); X bnry = inb(nec+ds0_bnry); X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA|DSCM_PG1); X curr = inb(nec+ds1_curr); X X#ifdef NEDEBUG X printf("neintr: bnry %x curr %x\n", bnry, curr); X#endif X X /* if ring empty, done! */ X if (bnry == curr) { X break; X } X X /* send packet with auto packet release */ X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA|DSCM_PG0); X outb (nec+ds0_rbcr1, 0x0f); X outb(nec+ds0_dcr, ns->ns_mode | DSDC_AR); X outb (nec+ds_cmd, DSCM_SENDP|DSCM_PG0|DSCM_START); X X /* get length */ X if ((ns->ns_mode & DSDC_WTS)) X insw (nec+ne_data, (u_short *)&ns->ns_ph, 2); X else X insb (nec+ne_data, &ns->ns_ph, 4); X X#ifdef NEDEBUG X printf("neintr: sendp packet hdr: %x %x %x %x\n", X ns->ns_ph.pr_status, X ns->ns_ph.pr_nxtpg, X ns->ns_ph.pr_sz0, X ns->ns_ph.pr_sz1); X#endif X X ns->ns_if.if_ipackets++; X len = ns->ns_ph.pr_sz0 + (ns->ns_ph.pr_sz1<<8); X if (len < ETHER_MIN_LEN || len > ETHER_MAX_LEN) { X printf("neintr: bnry %x curr %x\n", bnry, curr); X printf("neintr: packet hdr: %x %x %x %x\n", X ns->ns_ph.pr_status, X ns->ns_ph.pr_nxtpg, X ns->ns_ph.pr_sz0, X ns->ns_ph.pr_sz1); X printf("isr = 0x%x reg_isr=0x%x\n", X isr, inb(nec+ds0_isr)); X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA|DSCM_PG0); X bnry = inb(nec+ds0_bnry); X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA|DSCM_PG1); X curr = inb(nec+ds1_curr); X printf("neintr: new bnry %x curr %x\n", bnry, curr); X printf("neintr: bad len %d\n-hanging-\n", X len); X while (1) ; X } X X#ifdef NEDEBUG X printf("neintr: snarfing %d bytes\n", len); X#endif X X /* read packet */ X if ((ns->ns_mode & DSDC_WTS)) { X insw (nec+ne_data, (u_short *)&ns->ns_pb, X len / 2); X if (len & 1) X ns->ns_pb[len-1] = inb(nec+ne_data); X } else { X insb (nec+ne_data, ns->ns_pb, len); X } X X if (!(inb (nec+ds0_isr) & DSIS_RDC)) { X printf("neintr: remote dma not done\nflushing...\n"); X len = 0; X while (1) { X printf("%d %x\n", len, inb(nec+ne_data)); X len++; X } X } X X /* clear RDC */ X outb (nec + ds0_isr, DSIS_RDC); X X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA|DSCM_PG0); X X /* adjust for ether header and checksum */ X len -= sizeof(struct ether_header) + sizeof(long); X X /* process packet */ X neread(ns,(caddr_t)(ns->ns_pb), len); X } X } X X /* Transmit error */ X if (isr & DSIS_TXE) { X ns->ns_flags &= ~DSF_LOCK; X /* Need to read these registers to clear status */ X ns->ns_if.if_collisions += inb(nec+ds0_tbcr0); X ns->ns_if.if_oerrors++; X } X X /* Packet Transmitted */ X if (isr & DSIS_TX) { X ns->ns_flags &= ~DSF_LOCK; X ++ns->ns_if.if_opackets; X ns->ns_if.if_collisions += inb(nec+ds0_tbcr0); X } X X /* Receiver ovverun? */ X if (isr & DSIS_ROVRN) { X log(LOG_ERR, "ne%d: error: isr %x\n", ns-ne_softc, isr X /*, DSIS_BITS*/); X outb(nec+ds0_rbcr0, 0); X outb(nec+ds0_rbcr1, 0); X outb(nec+ds0_tcr, DSTC_LB0); X outb(nec+ds0_rcr, DSRC_MON); X outb(nec+ds_cmd, DSCM_START|DSCM_NODMA); X outb(nec+ds0_rcr, DSRC_AB); X outb(nec+ds0_tcr, 0); X } X X /* Any more to send? */ X outb (nec+ds_cmd, DSCM_NODMA|DSCM_PG0|DSCM_START); X nestart(&ns->ns_if); X outb (nec+ds_cmd, cmd); X outb (nec+ds0_imr, 0xff); X X /* Still more to do? */ X isr = inb (nec+ds0_isr); X if(isr) goto loop; X} X X/* X * Pass a packet to the higher levels. X * We deal with the trailer protocol here. X */ Xneread(ns, buf, len) X register struct ne_softc *ns; X char *buf; X int len; X{ X register struct ether_header *eh; X struct mbuf *m; X int off, resid; X register struct ifqueue *inq; X X /* X * Deal with trailer protocol: if type is trailer type X * get true type from first 16-bit word past data. X * Remember that type was trailer by setting off. X */ X eh = (struct ether_header *)buf; X eh->ether_type = ntohs((u_short)eh->ether_type); X#define nedataaddr(eh, off, type) ((type)(((caddr_t)((eh)+1)+(off)))) X if (eh->ether_type >= ETHERTYPE_TRAIL && X eh->ether_type < ETHERTYPE_TRAIL+ETHERTYPE_NTRAILER) { X off = (eh->ether_type - ETHERTYPE_TRAIL) * 512; X if (off >= ETHERMTU) return; /* sanity */ X eh->ether_type = ntohs(*nedataaddr(eh, off, u_short *)); X resid = ntohs(*(nedataaddr(eh, off+2, u_short *))); X if (off + resid > len) return; /* sanity */ X len = off + resid; X } else off = 0; X X if (len == 0) return; X X /* X * Pull packet off interface. Off is nonzero if packet X * has trailing header; neget will then force this header X * information to be at the front, but we still have to drop X * the type and length which are at the front of any trailer data. X */ X m = neget(buf, len, off, &ns->ns_if); X if (m == 0) return; X X ether_input(&ns->ns_if, eh, m); X} X X/* X * Supporting routines X */ X X/* X * Pull read data off a interface. X * Len is length of data, with local net header stripped. X * Off is non-zero if a trailer protocol was used, and X * gives the offset of the trailer information. X * We copy the trailer information and then all the normal X * data into mbufs. When full cluster sized units are present X * we copy into clusters. X */ Xstruct mbuf * Xneget(buf, totlen, off0, ifp) X caddr_t buf; X int totlen, off0; X struct ifnet *ifp; X{ X struct mbuf *top, **mp, *m, *p; X int off = off0, len; X register caddr_t cp = buf; X char *epkt; X X buf += sizeof(struct ether_header); X cp = buf; X epkt = cp + totlen; X X X if (off) { X cp += off + 2 * sizeof(u_short); X totlen -= 2 * sizeof(u_short); X } X X MGETHDR(m, M_DONTWAIT, MT_DATA); X if (m == 0) X return (0); X m->m_pkthdr.rcvif = ifp; X m->m_pkthdr.len = totlen; X m->m_len = MHLEN; X X top = 0; X mp = ⊤ X while (totlen > 0) { X if (top) { X MGET(m, M_DONTWAIT, MT_DATA); X if (m == 0) { X m_freem(top); X return (0); X } X m->m_len = MLEN; X } X len = min(totlen, epkt - cp); X if (len >= MINCLSIZE) { X MCLGET(m, M_DONTWAIT); X if (m->m_flags & M_EXT) X m->m_len = len = min(len, MCLBYTES); X else X len = m->m_len; X } else { X /* X * Place initial small packet/header at end of mbuf. X */ X if (len < m->m_len) { X if (top == 0 && len + max_linkhdr <= m->m_len) X m->m_data += max_linkhdr; X m->m_len = len; X } else X len = m->m_len; X } X bcopy(cp, mtod(m, caddr_t), (unsigned)len); X cp += len; X *mp = m; X mp = &m->m_next; X totlen -= len; X if (cp == epkt) X cp = buf; X } X return (top); X} X X/* X * Process an ioctl request. X */ Xneioctl(ifp, cmd, data) X register struct ifnet *ifp; X int cmd; X caddr_t data; X{ X register struct ifaddr *ifa = (struct ifaddr *)data; X struct ne_softc *ns = &ne_softc[ifp->if_unit]; X struct ifreq *ifr = (struct ifreq *)data; X int s = splimp(), error = 0; X X X switch (cmd) { X X case SIOCSIFADDR: X ifp->if_flags |= IFF_UP; X X switch (ifa->ifa_addr->sa_family) { X#ifdef INET X case AF_INET: X neinit(ifp->if_unit); /* before arpwhohas */ X ((struct arpcom *)ifp)->ac_ipaddr = X IA_SIN(ifa)->sin_addr; X arpwhohas((struct arpcom *)ifp, &IA_SIN(ifa)->sin_addr); X break; X#endif X#ifdef NS X case AF_NS: X { X register struct ns_addr *ina = &(IA_SNS(ifa)->sns_addr); X X if (ns_nullhost(*ina)) X ina->x_host = *(union ns_host *)(ns->ns_addr); X else { X /* X * The manual says we can't change the address X * while the receiver is armed, X * so reset everything X */ X ifp->if_flags &= ~IFF_RUNNING; X bcopy((caddr_t)ina->x_host.c_host, X (caddr_t)ns->ns_addr, sizeof(ns->ns_addr)); X } X neinit(ifp->if_unit); /* does ne_setaddr() */ X break; X } X#endif X default: X neinit(ifp->if_unit); X break; X } X break; X X case SIOCSIFFLAGS: X if ((ifp->if_flags & IFF_UP) == 0 && X ifp->if_flags & IFF_RUNNING) { X ifp->if_flags &= ~IFF_RUNNING; X outb(ns->ns_port + ds_cmd, DSCM_STOP|DSCM_NODMA); X } else if (ifp->if_flags & IFF_UP && X (ifp->if_flags & IFF_RUNNING) == 0) X neinit(ifp->if_unit); X break; X X#ifdef notdef X case SIOCGHWADDR: X bcopy((caddr_t)ns->ns_addr, (caddr_t) &ifr->ifr_data, X sizeof(ns->ns_addr)); X break; X#endif X X default: X error = EINVAL; X } X splx(s); X return (error); X} X#endif END-of-if_ne.c exit -- curt mayer curt@toad.com 415-387-0217 home