/* * Copyright (c) 1995 Melvin Tang-Richardson. * All rights reserved. * * Essential things to do * - BPF and multicast support * - Testing * - 16-bit dma's * - Buffer overflow handling (This must be done a certain way) [Nuts] * * Other things to do * - Pipelined transmitts * - Dynamic buffer allocation * - Shared RAM if I cubed give me info * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by RiscBSD. * 4. The name of the company nor the name of the author may be used to * endorse or promote products derived from this software without specific * prior written permission. * * THIS SOFTWARE IS PROVIDED BY RISCBSD ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL RISCBSD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * if_eh.c * * Ether H driver. * */ /* Some system includes *****************************************************/ #include #include #include #include #include #include #include #include #include #include /* Hardware related include chip/card/bus ***********************************/ #include #include #include #include #include "if_ehreg.h" /* Includes for interfacing with the network subsystem **********************/ #include #include #include #include #include #include #include #include #ifdef INET #include #include #include #include #include #endif #ifdef NS #include #include #endif /****************************************************************************/ /* Some useful definitions **************************************************/ /****************************************************************************/ #define MY_MANUFACTURER (0x46) #define MY_PODULE (0xec) #define TXBUF_SIZE (1522) #define NTXBUF (2) #define RXBUF_SIZE (256) #define NRXBUF (20) #define ETHER_MIN_LEN (64) #define ETHER_MAX_LEN (1522) #define PRINTF printf /****************************************************************************/ /* Prototype internal data structures ***************************************/ /****************************************************************************/ struct eh_softc { /* Kernel interface variables */ struct device sc_dev; int sc_podule_number; podule_t *sc_podule; irqhandler_t sc_ih; /* Network subsystem interface */ struct arpcom sc_arpcom; int promisc; /* Driver defined state variables */ int sc_flags; /* Hardware defined state */ int sc_base; int sc_reg; int sc_nxtpkt; /* Card memory map */ int sc_physstart; int sc_physend; int sc_tbs[NTXBUF]; /* Transmit buffer start */ int sc_rbs; int sc_rbd; /* Txrx parameters */ int sc_txcur; /* Fill this buffer next */ int sc_txlst; /* Buffer in use */ }; /* sc_flags */ #define MODE_MASK (3) #define MODE_DEAD (0) #define MODE_PIO (1) #define MODE_SHRAM (2) /****************************************************************************/ /* Function and data prototypes *********************************************/ /****************************************************************************/ struct cfdriver ehcd; int ehprobe __P(( struct device *parent, void *match, void *aux )); void ehattach __P(( struct device *parent, struct device *self, void *aux )); void ehstart __P(( struct ifnet *ifp )); int ehioctl __P(( struct ifnet *ifp, u_long cmd, caddr_t data )); void ehwatchdog __P(( int unit )); void eh_stop_controller __P(( struct eh_softc *sc )); void eh_start_controller __P(( struct eh_softc *sc )); void eh_transmit_command __P(( struct eh_softc *sc )); int eh_copyin __P(( struct eh_softc *sc, int src, char *dest, int len )); int eh_copyout __P(( struct eh_softc *sc, char *src, int dest, int len )); void ehinit __P(( struct eh_softc *sc )); int ehstop __P(( struct eh_softc *sc )); int ehintr __P(( void *arg )); void eh_shutdown __P(( void *arg )); /****************************************************************************/ /* Some useful macros *******************************************************/ /****************************************************************************/ #define SetReg(r,v) WriteWord ( sc->sc_reg+(r), (v)|((v)<<16) ) #define GetReg(r) ReadByte ( sc->sc_reg+(r) ) #define PAGE(n) SetReg ( EH_COMMAND, \ ((GetReg(EH_COMMAND)&COM_MASK_PAGE)|(n<<6)) ) /* When initiating a remote dma, reset the TXP bit. This Does Not abort */ /* the transmit, but allows us overlap this remote dma */ #define REMOTE_DMA(n) SetReg ( EH_COMMAND, \ ((GetReg(EH_COMMAND)&(COM_MASK_DMA))|n) ) /****************************************************************************/ /* Bus attachment code ******************************************************/ /****************************************************************************/ int ehprobe ( struct device *parent, void *match, void *aux ) { struct eh_softc *sc = (void *) match; struct podule_attach_args *pa = (void *) aux; int counter; int temp; int podule; podule = findpodule ( MY_MANUFACTURER, MY_PODULE, pa->pa_podule_number ); if ( podule == -1 ) return 0; /* Start to fill in the softc with the trivial variables */ sc->sc_flags = MODE_PIO; /* Select pio mode until I get info from i^3 */ sc->sc_base = podules[podule].fast_base; sc->sc_reg = sc->sc_base + 0x800; /* The probe could be tidied up a little */ /* Reset the card */ SetReg ( EH_RESET, 0xff ); delay (10000); /* Explicitly abort remote DMA, mainly to set a legal state */ PAGE (0); eh_stop_controller ( sc ); SetReg ( EH_COMMAND, 0x20 ); /* This is wrong */ SetReg ( EH_ISR, 0xff ); /* Clear any previous work */ temp = GetReg ( EH_CONFIGA ); temp = GetReg ( EH_CONFIGA ); temp &= ~0x38; SetReg ( EH_CONFIGA, temp | 3<<3 ); temp = GetReg ( EH_CONFIGA ); temp = GetReg ( EH_CONFIGA ); /* Make the thing interrupt and test it */ eh_start_controller ( sc ); /* Temporary initialisation, to allow a transmit test */ SetReg ( EH_PSTART, 16384 >> 8 ); SetReg ( EH_PSTOP, 16896 >> 8 ); SetReg ( EH_BNRY, 16384 >> 8 ); SetReg ( EH_TCR, TCR_NORMALLOOP ); /* Erk!! No, put us in loopback */ SetReg ( EH_DCR, DCR_WTS|DCR_LS|DCR_LAS ); SetReg ( EH_RCR, 0 ); /* Transmit some garbage into the loopback */ SetReg ( EH_TPSR, 0x5000 ); SetReg ( EH_TBCR0, 0 ); SetReg ( EH_TBCR1, 3 ); eh_transmit_command ( sc ); /* The card shouldn't be this quick, so use it as a test */ if ( (GetReg(EH_TSR)&TSR_DONE) != 0 ) PRINTF ( "eh: Card responded rather quickly %02x\n", GetReg (EH_TSR) ); for ( counter=500; counter>0; counter-- ) { if ( (GetReg(EH_TSR)&TSR_DONE)!=0 ) break; delay ( 500 ); } if ( counter==0 ) { PRINTF ( "eh: card did not respond\n" ); return 0; } /* Let it sleep since we're not doing anything with it for now */ SetReg ( EH_ISR, 0 ); SetReg ( EH_DCR, DCR_WTS|DCR_LS|DCR_LAS ); /* Test the board ram. This code will change */ #define NLEN (0x2000) #define NBASE (0x4000) { char *test_data; char *read_buffer; MALLOC(test_data, char *, NLEN, M_DEVBUF, M_NOWAIT ); MALLOC(read_buffer, char *, NLEN, M_DEVBUF, M_NOWAIT ); /* Fill the test_data block */ for ( counter=0; countersc_physstart = 0x4000; sc->sc_physend = 0x6000; /* Get out ethernet address */ sc->sc_arpcom.ac_enaddr[0] = 0x00; sc->sc_arpcom.ac_enaddr[1] = 0x00; sc->sc_arpcom.ac_enaddr[2] = 0xc0; sc->sc_arpcom.ac_enaddr[3] = 0x41; sc->sc_arpcom.ac_enaddr[4] = bootconfig.machine_id[1]; sc->sc_arpcom.ac_enaddr[5] = bootconfig.machine_id[0]; /* Copy the ether addr cards filter */ PAGE(1); for ( counter=0; counter<6; counter++ ) SetReg ( ((counter+1)<<2), sc->sc_arpcom.ac_enaddr[counter] ); PAGE(0); /* Tell the podule system about our successful probe */ pa->pa_podule_number = podule; pa->pa_podule = &podules[podule]; return 1; } void ehattach ( struct device *parent, struct device *self, void *aux ) { struct eh_softc *sc = (void *) self; struct podule_attach_args *pa = (void *)aux; struct ifnet *ifp = &sc->sc_arpcom.ac_if; int irq; int temp; /* Check a few things about the attach args */ sc->sc_podule_number = pa->pa_podule_number; if ( sc->sc_podule_number == -1 ) panic ( "Podule has disappeared !" ); sc->sc_podule = &podules[sc->sc_podule_number]; podules[sc->sc_podule_number].attached = 1; /* Fill in my application form to attach to the networking system */ ifp->if_unit = sc->sc_dev.dv_unit; ifp->if_name = ehcd.cd_name; ifp->if_start = ehstart; ifp->if_ioctl = ehioctl; ifp->if_watchdog = ehwatchdog; ifp->if_flags = IFF_BROADCAST | IFF_NOTRAILERS; /* Signed, dated then sent */ if_attach ( ifp ); ether_ifattach ( ifp ); /* Not print some stuff for the attach line, starting with the address */ PRINTF ( " address %s", ether_sprintf(sc->sc_arpcom.ac_enaddr) ); /* Now the transfer mode we're operating in */ switch ( sc->sc_flags & MODE_MASK ) { case MODE_PIO: PRINTF ( " pio mode" ); break; case MODE_SHRAM: PRINTF ( " ram mode" ); break; default: panic ( "Ether H probe set an unknown mode" ); } /* Get an irq handler installed */ SetReg ( EH_IMR, 0 ); SetReg ( EH_ISR, 0xff ); sc->sc_ih.ih_func = ehintr; sc->sc_ih.ih_arg = sc; sc->sc_ih.ih_level = IPL_NET; irq = (sc->sc_podule_number>=MAX_PODULES) ? IRQ_NETSLOT : IRQ_PODULE; if ( irq_claim(irq, &sc->sc_ih) ) panic ( "Cannot install IRQ handler" ); /* This loopbacks the card on reset, and stops riscos locking */ if ( shutdownhook_establish ( eh_shutdown, (void *)sc )==NULL) panic ( "Cannot install shutdown handler" ); /* Reprogram CONFIGA. I dont think we should do this any more */ temp = GetReg ( EH_CONFIGA ); temp = GetReg ( EH_CONFIGA ); temp &= ~0x38; SetReg ( EH_CONFIGA, temp | (3<<3) ); /* And that's it */ PRINTF ( "\n" ); } /****************************************************************************/ /* Net subsystem interface **************************************************/ /****************************************************************************/ #define NEXT_TXBUF (((sc->sc_txcur+1)>=NTXBUF) ? 0 : (sc->sc_txcur+1)) void ehstart ( struct ifnet *ifp ) { struct eh_softc *sc = ehcd.cd_devs[ifp->if_unit]; int cur; struct mbuf *m0, *m; int nextbuf; char *buffer; char txbuf[TXBUF_SIZE]; int len = 0; if ((ifp->if_flags & IFF_OACTIVE) != 0 ) return; /* This is for pipelined transmit. At present just use 1 buffer */ for (;;) { #ifdef PIPELINE_TRANSMIT nextbuf = NEXT_TXBUF; if ( nextbuf == sc->sc_txlst ) ifp->if_flags |= IFF_OACTIVE; break; } #else nextbuf = 0; #endif /*** Speed up - In future, copy the bufs direct to the card ***/ /* Copy a frame out of the mbufs */ IF_DEQUEUE(&sc->sc_arpcom.ac_if.if_snd, m); if ( !m ) break; buffer = txbuf; len = 0; for ( m0=m; m && (len + m->m_len) < TXBUF_SIZE; m=m->m_next ) { bcopy(mtod(m,caddr_t), buffer, m->m_len); buffer+=m->m_len; len+=m->m_len; } m_freem(m0); len = max ( len, ETHER_MIN_LEN ); /* And blat it to the card */ eh_copyout ( sc, txbuf, sc->sc_tbs[nextbuf], len ); #ifdef PIPELINE_TRANSMIT sc->sc_txlst = nextbuf; #endif PAGE(0); SetReg ( EH_TPSR, sc->sc_tbs[nextbuf]>>8 ); SetReg ( EH_TBCR0, len&0xff ); SetReg ( EH_TBCR1, (len>>8) & 0xff ); cur = GetReg(EH_COMMAND); eh_transmit_command (sc ); ifp->if_flags |= IFF_OACTIVE; ifp->if_timer = 30; #ifndef PIPELINE_TRANSMIT break; #endif } } #define IZSET(a,b) ((a->if_flags&b)!=0) #define IZCLR(a,b) ((a->if_flags&b)==0) #define DOSET(a,b) (a->if_flags|=b) #define DOCLR(a,b) (a->if_flags&=~b) int ehioctl ( struct ifnet *ifp, u_long cmd, caddr_t data ) { struct eh_softc *sc = ehcd.cd_devs[ifp->if_unit]; struct ifaddr *ifa = (struct ifaddr *)data; int s = splimp (); int error = 0; switch ( cmd ) { case SIOCSIFADDR: ifp->if_flags |= IFF_UP; switch ( ifa->ifa_addr->sa_family ) { #ifdef INET case AF_INET: ehinit(sc); arp_ifinit(&sc->sc_arpcom, ifa); break; #endif default: ehinit(sc); break; } break; case SIOCSIFFLAGS: sc->promisc = ifp->if_flags & ( IFF_PROMISC | IFF_ALLMULTI ); if ( IZCLR(ifp,IFF_UP) && IZSET(ifp,IFF_RUNNING) ) { /* Interface marked down, but still running */ ehstop(sc); DOCLR(ifp,IFF_RUNNING); } else if ( IZSET(ifp,IFF_UP) && IZCLR(ifp,IFF_RUNNING) ) { /* Just marked up, not running yet */ ehinit(sc); } else { /* Reset to invoke changes in other registers */ ehstop(sc); ehinit(sc); } default: error = EINVAL; } splx (s); return error; } void ehwatchdog ( int unit ) { struct eh_softc *sc = ehcd.cd_devs[unit]; log (LOG_ERR, "%s: device timeout\n", sc->sc_dev.dv_xname ); sc->sc_arpcom.ac_if.if_oerrors++; ehinit (sc); } /* * void eh_stop_controller ( struct eh_softc *sc ) * * Software reset command, takes the controller offline, no packets will be * received or transmitted. Any reception or transmission inprogres will * continue to completion before entering the reset state. */ void eh_stop_controller ( struct eh_softc *sc ) { SetReg ( EH_COMMAND, ((GetReg(EH_COMMAND)&0xfc)|1) ); delay ( 10000 ); /* Change this to wait on the ISR bit */ } /* * void eh_start_controller ( struct eh_softc *sc ) * * Activte the NIC core after either power up or when the NIC Core has * been placed in a reset mode by eh_stop_controller or an error. */ void eh_start_controller ( struct eh_softc *sc ) { int temp = GetReg(EH_COMMAND); SetReg ( EH_COMMAND, ( (temp&0xfc)|2) ); } void eh_transmit_command ( struct eh_softc *sc ) { /* Explicit set here. There is no other reasonable combination */ SetReg ( EH_COMMAND, COM_ABORT|COM_STA|COM_TXP); } /****************************************************************************/ /* Remote DMA handling ******************************************************/ /****************************************************************************/ /* Remote DMA handling is going to undergo a major overhaul real soon now */ /* We musn't fail */ inline void eh_ensure_dma_ok ( struct eh_softc *sc ) { register int isr = GetReg ( EH_ISR ); register int status = GetReg ( EH_COMMAND ); if ((status&COM_ABORT)==0) { SetReg ( EH_COMMAND, status|COM_ABORT ); while ((isr&ISR_RDC)==0) { int dummy; dummy = GetReg ( EH_DATA_PORT ); delay(50); } } /* Reset the COM_TXP bit. This does not abort transmission */ SetReg ( EH_COMMAND, 0x22 ); } inline int eh_ensure_dma_competed ( struct eh_softc *sc, int type ) { register int status = GetReg ( EH_COMMAND ); if ( (status&COM_ABORT)==0 ) { int dummy=0; printf ( "EHDEBUG: Remote %d wasn't finished\n", type ); SetReg ( EH_COMMAND, status|COM_ABORT ); switch ( type ) { case COM_READ: dummy = GetReg ( EH_DATA_PORT ); break; case COM_WRITE: SetReg ( EH_DATA_PORT, dummy ); break; } ehinit (sc); return 1; } return 0; } /* Do an eh_copyin, but take into consideration ring wrap */ int eh_copyring ( struct eh_softc *sc, int src, char *dest, int len ) { if ( (src+len)>sc->sc_rbd ) { register int in = sc->sc_rbd - src; if ( eh_copyin ( sc, src, dest, in )==0 ) { printf ( "copyring failed pass 1\n" ); return 0; } if ( eh_copyin ( sc, sc->sc_rbs, dest+in, len-in )==0 ) { printf ( "copyring failed pass 2\n" ); return 0; } } else { if ( eh_copyin ( sc, src, dest, len )==0 ) { printf ( "copyring faild on 1 pass copy\n" ); return 0; } } return len; } int eh_copyin ( struct eh_softc *sc, int src, char *dest, int len ) { int counter; int s; short *dst = (short *)dest; s = splhigh(); /* Erm, maybe not needed now */ if ( len&1 ) { len++; } if ( src&1 ) { printf ( "EHDEBUG: Copying from odd address\n" ); src++; } /* Remote DMA to the DP83905 must be done with care. If we dont */ /* do it exactly right, it punishes us by locking the bus. */ eh_ensure_dma_ok (sc); /* Set up the DMA */ SetReg ( EH_RSAR0, src & 0xff ); SetReg ( EH_RSAR1, (src>>8) & 0xff ); SetReg ( EH_RBCR0, len & 0xff ); SetReg ( EH_RBCR1, (len>>8) & 0xff ); /* Set the DMA running */ /* REMOTE_DMA ( COM_READ ); */ /* SetReg ( EH_COMMAND, 0x0a ); */ SetReg ( EH_COMMAND, COM_READ|COM_STA ); for ( counter=0; countersc_reg + EH_DATA_PORT ); } splx(s); if ( eh_ensure_dma_competed ( sc, COM_READ ) ) return 0; return len; } int eh_copyout ( struct eh_softc *sc, char *src, int dest, int len ) { int counter; int s; short *sr = (short *)src; if ( len&1 ) { len++; } if ( dest&1 ) { printf ( "EHDEBUG: Copying to odd address\n" ); dest++; } s = splhigh(); /* Erm, maybe not needed now */ /* Remote DMA to the DP83905 must be done with care. If we dont */ /* do it exactly right, it punishes us by locking the bus. */ eh_ensure_dma_ok (sc); /* Set up the DMA */ SetReg ( EH_RSAR0, dest & 0xff ); SetReg ( EH_RSAR1, (dest>>8) & 0xff ); SetReg ( EH_RBCR0, len & 0xff ); SetReg ( EH_RBCR1, (len>>8) & 0xff ); /* Set the DMA running */ /*REMOTE_DMA ( COM_WRITE );*/ /* SetReg ( EH_COMMAND, 0x0a ); */ SetReg ( EH_COMMAND, COM_WRITE|COM_STA ); for ( counter=0; countersc_reg + EH_DATA_PORT, *(sr) ); sr++; } splx(s); if ( eh_ensure_dma_competed ( sc, COM_WRITE ) ) return 0; return len; } #define ALLOCBLK(v,s) (v)=card_freestart; \ card_freestart+=((s)+0xff)&(~0xff) void ehinit ( struct eh_softc *sc ) { int card_freestart; int counter; PAGE(0); /* Lay out the cards ram */ card_freestart = sc->sc_physstart; /* Allocate receive buffers */ ALLOCBLK ( sc->sc_rbs, (RXBUF_SIZE*NRXBUF) ); sc->sc_rbd = card_freestart; sc->sc_nxtpkt = sc->sc_rbs; /* Allocate transmit buffers */ for ( counter=0; countersc_tbs[counter], TXBUF_SIZE ); } if ( card_freestart > sc->sc_physend ) { printf ( "eh: card short of ram %02x required %02x present\n", card_freestart, sc->sc_physstart ); sc->sc_arpcom.ac_if.if_flags &= ~IFF_RUNNING; sc->sc_arpcom.ac_if.if_flags |= IFF_OACTIVE; return; } /* Bring the controller up */ /* The ordering here is critical. Dont change unless you know what */ /* you're doing */ /* Reset the card */ SetReg ( EH_RESET, 0xff ); delay (50000); SetReg ( EH_RESET, 0x00 ); delay (50000); /* 1. Program the command register for page 0 */ SetReg ( EH_COMMAND, 0x21 ); /* 2. Initialize the Data configuration register */ SetReg ( EH_DCR, DCR_WTS|DCR_LS|(0<<5) ); /* 3. Clear RBCR if using remote DMA */ SetReg ( EH_RBCR0, 0 ); SetReg ( EH_RBCR1, 0 ); /* 4. Initialise RCR */ SetReg ( EH_RCR, 0x04 ); /* 5. Place the controller in loopback mode 1 or 2 TCR=02H or 04H */ SetReg ( EH_TCR, 0x02 ); /* 6. Initiliase Receive ring buffer BNDRY PSTART PSTOP */ SetReg ( EH_BNRY, sc->sc_rbs >> 8 ); SetReg ( EH_PSTART, sc->sc_rbs >> 8 ); SetReg ( EH_PSTOP, sc->sc_rbd >> 8 ); /* 7. Clear ISR */ SetReg ( EH_ISR, 0xff ); /* 8. Initialise IMR */ SetReg ( EH_IMR, ISR_PRX|ISR_PTX|ISR_RXE|ISR_TXE|ISR_OVW|ISR_CNT ); /* 9. Program command register for page 1 COM=0x61 */ /* Initialise PAR MAR and CURR */ PAGE(1); for ( counter=0; counter<6; counter++ ) SetReg ( ((counter+1)<<2), sc->sc_arpcom.ac_enaddr[counter] ); SetReg ( EH_CURR, sc->sc_rbs >> 8 ); /* Put controller into start mode COM = 0x22 */ SetReg ( EH_COMMAND, 0x22 ); /* Initialise the TCR */ SetReg ( EH_TCR, TCR_NORMALLOOP ); /* Interface is up */ sc->sc_arpcom.ac_if.if_flags |= IFF_RUNNING; sc->sc_arpcom.ac_if.if_flags &= ~IFF_OACTIVE; ehstart(&sc->sc_arpcom.ac_if); } int ehstop ( struct eh_softc *sc ) { int s = splimp (); eh_stop_controller (sc); SetReg ( EH_DCR, DCR_LS|DCR_LAS ); splx(s); return 0; } #define INTR_ACK(s) SetReg ( EH_ISR, s ) /* In need of tidying */ #undef MYTEST void eh_rint ( struct eh_softc *sc ) { struct ifnet *ifp = &sc->sc_arpcom.ac_if; struct mbuf *top, **mp, *m; struct ether_header eh; struct eh_rxhdr hdr; int status = GetReg(EH_RSR); int rxstatus=0; int totlen, len; int ptr; int rbend; int thispacket; int bnry; #ifdef MYTEST int old; #endif /* Get the end of the ring buffer */ PAGE(1); rbend = GetReg ( EH_CURR ) << 8; PAGE(0); /* If we have something to receive then receive it */ if ( status&RSR_PRX ) { /* Remove all the packets from the ring buffer */ for ( ptr=sc->sc_nxtpkt; ptr!=rbend ; ptr = sc->sc_nxtpkt ) { thispacket = ptr; /* Copy the ether h receive header */ if ( eh_copyring ( sc, ptr, (char *)&hdr, 4 ) != 4 ) { printf ( "eh: Failed copying in header\n" ); printf ( "eh: thispacket %02x\n", thispacket ); ehinit (sc); return; } rxstatus = hdr.rx_status; /* Apparently, older NE2000 clones read the MSB of len wrong */ /* This code recalculates it. It seems as if the DP83905 */ /* does not get it wrong */ #ifdef MYTEST old=hdr.rx_rbc1; totlen = hdr.rx_rbc0; if ( (hdr.rx_nxtpkt<<8) > ptr ) hdr.rx_rbc1 = (hdr.rx_nxtpkt<<8)-ptr; else hdr.rx_rbc1 = (hdr.rx_nxtpkt<<8)+(sc->sc_rbd-sc->sc_rbs)-ptr; hdr.rx_rbc1=(hdr.rx_rbc1>>8)-1; if ( ((totlen&255)+4)>256 ) hdr.rx_rbc1--; #endif totlen = (hdr.rx_rbc1<<8) + hdr.rx_rbc0; #ifdef MYTEST if (old!=hdr.rx_rbc1) printf ( "CODE CAUGHT CHIP Actual %02x%02x CALC%02x%02x", old, (int)hdr.rx_rbc0, (int)hdr.rx_rbc1, (int)hdr.rx_rbc0 ); #endif ptr+=4; /* Check the packet's status */ if ( hdr.rx_status&(RSR_CRC|RSR_FAE|RSR_FO|RSR_MPA) ) { printf ( "eh: intact packet is corrupt %02x %04x\n", hdr.rx_status, ptr ); } if (((hdr.rx_nxtpkt<<8)>sc->sc_rbd) ||((hdr.rx_nxtpkt<<8)sc_rbs)) { printf ( "eh: ring buffer empty %04x %04x\n", hdr.rx_nxtpkt, ptr ); ehinit (sc); return; } sc->sc_nxtpkt = hdr.rx_nxtpkt << 8; if ( totlen>ETHER_MAX_LEN ) { printf ( "eh: giant packet received %04x\n", totlen ); totlen = ETHER_MAX_LEN; } /* Copy the ether header */ if ((eh_copyring ( sc, ptr, (char *)&eh, sizeof (eh)))!=sizeof(eh) ) { printf ( "eh: Failed copying in ethenet header\n" ); return; } ptr+=sizeof(eh); totlen-=sizeof(eh); /* Copy the packet into mbufs */ MGETHDR(m, M_DONTWAIT, MT_DATA); if ( m==0 ) return; m->m_pkthdr.rcvif = ifp; m->m_pkthdr.len = totlen; len = MHLEN; top = 0; mp = ⊤ while ( totlen > 0 ) { if (top) { MGET(m, M_DONTWAIT, MT_DATA); if ( m==0 ) { m_freem(top); return; } len = MLEN; } if ( totlen >= MINCLSIZE ) { MCLGET(m, M_DONTWAIT); if (m->m_flags & M_EXT) len = MCLBYTES; } m->m_len = len = min ( totlen, len ); if ((eh_copyring ( sc, ptr, mtod(m, caddr_t), len))!=len) printf ( "eh: failed copying in buffer %d\n", len ); ptr += len; totlen -= len; *mp = m; mp = &m->m_next; } /* Ok, I'm done with this packet */ /* Set the boundary pointer on the ring buffer */ bnry = (sc->sc_nxtpkt>>8)-1; if ( bnry < sc->sc_rbs ) bnry = sc->sc_rbd-1; SetReg ( EH_BNRY, bnry ); m = top; ifp->if_ipackets++; ether_input(ifp, &eh, m ); } } } int ehintr ( void *arg ) { struct eh_softc *sc = arg; register int isr = GetReg ( EH_ISR ); /* Is char of int faster ? */ register int times = 0; arg=arg; PAGE(0); if ( isr & ISR_RXE ) { int status = GetReg ( EH_RSR ); PRINTF ( "eh: Receive Error:" ); if ( status&RSR_PRX ) PRINTF ( " packet received intact (this should happen)" ); if ( status&RSR_CRC ) PRINTF ( " crc error" ); if ( status&RSR_FAE ) PRINTF ( " frame alignment error" ); if ( status&RSR_FO ) PRINTF ( " fifo overrun" ); if ( status&RSR_MPA ) PRINTF ( " missed packet" ); if ( status&RSR_DIS ) PRINTF ( " receiver disabled" ); printf ( "\n" ); INTR_ACK ( ISR_RXE ); ehinit (sc); return 0; } if ( isr & ISR_PRX ) { eh_rint (sc); INTR_ACK ( ISR_PRX ); } if ( isr & ISR_PTX ) { struct ifnet *ifp = &sc->sc_arpcom.ac_if; int status; INTR_ACK ( ISR_PTX ); ifp->if_timer=0; ifp->if_flags &= ~IFF_OACTIVE; status = GetReg ( EH_TSR ); if ( status&TSR_PTX ) ifp->if_opackets++; if ( status&TSR_COL ) ifp->if_collisions+=GetReg(EH_NCR); if ( status&TSR_ABT ) PRINTF ( "eh: too many collisions\n" ); if ( status&TSR_CRS ) PRINTF ( "eh: no carrier\n" ); if ( status&TSR_CDH ) PRINTF ( "eh: tranceiver failure, no heartbeat \n" ); if ( status&TSR_OWC ) PRINTF ( "eh: out of window \n" ); if ( status&(TSR_ABT|TSR_CRS|TSR_FU|TSR_CDH|TSR_OWC) ) ifp->if_oerrors++; ehstart(ifp); } if ( isr & ISR_TXE ) { INTR_ACK ( ISR_TXE ); PRINTF ( "ehintr: Transmit error\n" ); } if ( isr & ISR_OVW ) { INTR_ACK ( ISR_OVW ); PRINTF ( "ehintr: Counter overflow\n" ); } if ( isr & ISR_RST ) { INTR_ACK ( ISR_RST ); PRINTF ( "ehintr: Reset status\n" ); } if ((times++)>16) { PRINTF ( "eh: possible interrupt jammed on." ); SetReg ( EH_IMR, 0x0 ); } /* Dont do this for the mo until I'm sure. isr = GetReg ( EH_ISR ); if ( (isr&GetReg(EH_IMR))!=0 ) goto more; */ if ( (isr&GetReg(EH_IMR))!=0 ) printf ( "eh: Interrupt not serviced\n" ); return 0; } /****************************************************************************/ /* Auxilary routines ********************************************************/ /****************************************************************************/ void eh_shutdown ( void *arg ) { struct eh_softc *sc = (struct eh_softc *)arg; /* On shutdown put us in loopback */ SetReg ( EH_DCR, DCR_LAS ); SetReg ( EH_TCR, TCR_NICMOD ); /* and program remote dma so riscos doesnt lock */ SetReg ( EH_RSAR0, 0 ); SetReg ( EH_RSAR1, 1 ); SetReg ( EH_RBCR0, 0 ); SetReg ( EH_RBCR1, 1 ); } struct cfdriver ehcd = { NULL, "eh", ehprobe, ehattach, DV_DULL, sizeof(struct eh_softc), NULL, 0 };