diff options
author | denny <denny@cvs.openbsd.org> | 1997-06-18 02:59:31 +0000 |
---|---|---|
committer | denny <denny@cvs.openbsd.org> | 1997-06-18 02:59:31 +0000 |
commit | 4873820d037822cb9e8e5d7f6c6e728e7b24bf7a (patch) | |
tree | 279b2a9d9e836e677aa7df665b606a42a11d8806 /sys/arch/mac68k/dev | |
parent | 3447cebe96ec85236b4f6971ada0a2d02a6e10f0 (diff) |
Make if_sn.c far more MI, eventually share with pica.
sonic_get is faster and simpler.
sn_ioctl works like other current ether drivers now.
Sync up with netbsd to ease maintenance; incorporate some "gratuitous" formatting changes so I won't have to do two totally seperate mods each time.
Diffstat (limited to 'sys/arch/mac68k/dev')
-rw-r--r-- | sys/arch/mac68k/dev/if_sn.c | 623 | ||||
-rw-r--r-- | sys/arch/mac68k/dev/if_sn_nubus.c | 189 | ||||
-rw-r--r-- | sys/arch/mac68k/dev/if_sn_obio.c | 107 | ||||
-rw-r--r-- | sys/arch/mac68k/dev/if_snreg.h | 5 | ||||
-rw-r--r-- | sys/arch/mac68k/dev/if_snvar.h | 116 |
5 files changed, 509 insertions, 531 deletions
diff --git a/sys/arch/mac68k/dev/if_sn.c b/sys/arch/mac68k/dev/if_sn.c index e65ad212d58..c9c11d9a523 100644 --- a/sys/arch/mac68k/dev/if_sn.c +++ b/sys/arch/mac68k/dev/if_sn.c @@ -1,8 +1,7 @@ -/* $NetBSD: if_sn.c,v 1.7 1997/03/20 17:47:51 scottr Exp $ */ -/* $OpenBSD: if_sn.c,v 1.22 1997/04/25 03:29:15 briggs Exp $ */ +/* $NetBSD: if_sn.c,v 1.13 1997/04/25 03:40:10 briggs Exp $ */ /* - * National Semiconductor SONIC Driver + * National Semiconductor DP8393X SONIC Driver * Copyright (c) 1991 Algorithmics Ltd (http://www.algor.co.uk) * You may use, copy, and modify this program so long as you retain the * copyright line. @@ -30,7 +29,6 @@ #include <net/if.h> #include <net/if_dl.h> #include <net/netisr.h> -#include <net/route.h> #ifdef INET #include <netinet/in.h> @@ -42,31 +40,43 @@ #include <vm/vm.h> -extern int kvtop(caddr_t addr); - #include "bpfilter.h" #if NBPFILTER > 0 #include <net/bpf.h> #include <net/bpfdesc.h> #endif -typedef unsigned char uchar; - #include <machine/bus.h> #include <machine/cpu.h> -#include <machine/viareg.h> +/* #include <machine/viareg.h> */ #include <mac68k/dev/if_snreg.h> #include <mac68k/dev/if_snvar.h> -static void snwatchdog __P((struct ifnet *)); -static int sninit __P((struct sn_softc *sc)); -static int snstop __P((struct sn_softc *sc)); -static inline int sonicput __P((struct sn_softc *sc, struct mbuf *m0)); -static int snioctl __P((struct ifnet *ifp, u_long cmd, caddr_t data)); -static void snstart __P((struct ifnet *ifp)); -static void snreset __P((struct sn_softc *sc)); +static void snwatchdog __P((struct ifnet *)); +static int sninit __P((struct sn_softc *sc)); +static int snstop __P((struct sn_softc *sc)); +static int snioctl __P((struct ifnet *ifp, u_long cmd, caddr_t data)); +static void snstart __P((struct ifnet *ifp)); +static void snreset __P((struct sn_softc *sc)); + +static void caminitialise __P((struct sn_softc *)); +static void camentry __P((struct sn_softc *, int, u_char *ea)); +static void camprogram __P((struct sn_softc *)); +static void initialise_tda __P((struct sn_softc *)); +static void initialise_rda __P((struct sn_softc *)); +static void initialise_rra __P((struct sn_softc *)); +#ifdef SNDEBUG +static void camdump __P((struct sn_softc *sc)); +#endif + +static void sonictxint __P((struct sn_softc *)); +static void sonicrxint __P((struct sn_softc *)); -void camdump __P((struct sn_softc *sc)); +static __inline__ int sonicput __P((struct sn_softc *sc, struct mbuf *m0, + int mtd_next)); +static __inline__ int sonic_read __P((struct sn_softc *, caddr_t, int)); +static __inline__ struct mbuf *sonic_get __P((struct sn_softc *, + struct ether_header *, int)); struct cfdriver sn_cd = { NULL, "sn", DV_IFNET @@ -87,7 +97,7 @@ struct cfdriver sn_cd = { #endif #endif -int ethdebug = 0; +int sndebug = 0; /* * SONIC buffers need to be aligned 16 or 32 bit aligned. @@ -106,14 +116,15 @@ int ethdebug = 0; * to accept packets. */ int -snsetup(sc) +snsetup(sc, lladdr) struct sn_softc *sc; + u_int8_t *lladdr; { - struct ifnet *ifp = &sc->sc_if; - unsigned char *p; - unsigned char *pp; - int i; - + struct ifnet *ifp = &sc->sc_if; + u_char *p; + u_char *pp; + int i; + int offset; /* * XXX if_sn.c is intended to be MI. Should it allocate memory @@ -146,7 +157,7 @@ snsetup(sc) * around problems near the end of 64k !! */ p = sc->space; - pp = (unsigned char *)ROUNDUP ((int)p, NBPG); + pp = (u_char *)ROUNDUP ((int)p, NBPG); p = pp; /* @@ -155,7 +166,7 @@ snsetup(sc) * each page individually. */ for (i = 0; i < SN_NPAGES; i++) { - physaccess (p, (caddr_t) kvtop(p), NBPG, + physaccess (p, (caddr_t)SONIC_GETDMA(p), NBPG, PG_V | PG_RW | PG_CI); p += NBPG; } @@ -163,88 +174,89 @@ snsetup(sc) for (i = 0; i < NRRA; i++) { sc->p_rra[i] = (void *)p; - sc->v_rra[i] = kvtop(p); + sc->v_rra[i] = SONIC_GETDMA(p); p += RXRSRC_SIZE(sc); } - sc->v_rea = kvtop(p); + sc->v_rea = SONIC_GETDMA(p); - p = (unsigned char *)SOALIGN(sc, p); + p = (u_char *)SOALIGN(sc, p); - sc->p_cda = (void *) (p); - sc->v_cda = kvtop(p); + sc->p_cda = (void *)(p); + sc->v_cda = SONIC_GETDMA(p); p += CDA_SIZE(sc); - p = (unsigned char *)SOALIGN(sc, p); + p = (u_char *)SOALIGN(sc, p); for (i = 0; i < NTDA; i++) { struct mtd *mtdp = &sc->mtda[i]; mtdp->mtd_txp = (void *)p; - mtdp->mtd_vtxp = kvtop(p); + mtdp->mtd_vtxp = SONIC_GETDMA(p); p += TXP_SIZE(sc); } - p = (unsigned char *)SOALIGN(sc, p); + p = (u_char *)SOALIGN(sc, p); if ((p - pp) > NBPG) { printf ("%s: sizeof RRA (%ld) + CDA (%ld) +" - "TDA (%ld) > NBPG (%d). Punt!\n", - sc->sc_dev.dv_xname, - (ulong)sc->p_cda - (ulong)sc->p_rra[0], - (ulong)sc->mtda[0].mtd_txp - (ulong)sc->p_cda, - (ulong)p - (ulong)sc->mtda[0].mtd_txp, - NBPG); + "TDA (%ld) > NBPG (%d). Punt!\n", + sc->sc_dev.dv_xname, + (ulong)sc->p_cda - (ulong)sc->p_rra[0], + (ulong)sc->mtda[0].mtd_txp - (ulong)sc->p_cda, + (ulong)p - (ulong)sc->mtda[0].mtd_txp, + NBPG); return(1); } p = pp + NBPG; pp = p; - if ((NRDA * RXPKT_SIZE(sc)) > NBPG) { - printf ("%s: sizeof NRDA (%d) > NBPG (%d). Punt!\n", - sc->sc_dev.dv_xname, - NRDA * RXPKT_SIZE(sc), NBPG); - return (1); - } - - for (i = 0; i < NRDA; i++) { - sc->p_rda[i] = (void *) p; - sc->v_rda[i] = kvtop(p); - p += RXPKT_SIZE(sc); - } - - p = (unsigned char *)SOALIGN(sc, p); + sc->sc_nrda = NBPG / RXPKT_SIZE(sc); + sc->p_rda = (caddr_t) p; + sc->v_rda = SONIC_GETDMA(p); p = pp + NBPG; for (i = 0; i < NRBA; i++) { - sc->rbuf[i] = (caddr_t) p; + sc->rbuf[i] = (caddr_t)p; p += NBPG; } - for (i = 0; i < NTXB; i+=2) { - sc->tbuf[i] = (caddr_t) p; - sc->tbuf[i+1] = (caddr_t)(p + (NBPG/2)); - sc->vtbuf[i] = kvtop(sc->tbuf[i]); - sc->vtbuf[i+1] = kvtop(sc->tbuf[i+1]); - p += NBPG; + pp = p; + offset = TXBSIZE; + for (i = 0; i < NTDA; i++) { + struct mtd *mtdp = &sc->mtda[i]; + + mtdp->mtd_buf = p; + mtdp->mtd_vbuf = SONIC_GETDMA(p); + offset += TXBSIZE; + if (offset < NBPG) { + p += TXBSIZE; + } else { + p = pp + NBPG; + pp = p; + offset = TXBSIZE; + } } -#if 0 +#ifdef SNDEBUG camdump(sc); #endif - printf(" address %s\n", ether_sprintf(sc->sc_enaddr)); + printf(" address %s\n", ether_sprintf(lladdr)); -#if 0 -printf("sonic buffers: rra=%p cda=0x%x rda=0x%x tda=0x%x\n", - sc->p_rra[0], sc->p_cda, sc->p_rda[0], sc->mtda[0].mtd_txp); +#ifdef SNDEBUG + printf("%s: buffers: rra=%p cda=%p rda=%p tda=%p\n", + sc->sc_dev.dv_xname, sc->p_rra[0], sc->p_cda, + sc->p_rda, sc->mtda[0].mtd_txp); #endif bcopy(sc->sc_dev.dv_xname, ifp->if_xname, IFNAMSIZ); + bcopy(lladdr, sc->sc_enaddr, ETHER_ADDR_LEN); + ifp->if_softc = sc; ifp->if_ioctl = snioctl; ifp->if_start = snstart; - ifp->if_flags = - IFF_BROADCAST | IFF_SIMPLEX | IFF_NOTRAILERS | IFF_MULTICAST; + ifp->if_flags = + IFF_BROADCAST | IFF_SIMPLEX | IFF_NOTRAILERS | IFF_MULTICAST; ifp->if_watchdog = snwatchdog; #if NBPFILTER > 0 bpfattach(&ifp->if_bpf, ifp, DLT_EN10MB, sizeof(struct ether_header)); @@ -261,10 +273,11 @@ snioctl(ifp, cmd, data) u_long cmd; caddr_t data; { - struct ifaddr *ifa; - struct sn_softc *sc = ifp->if_softc; - int s = splnet(), err = 0; - int temp; + struct ifaddr *ifa; + struct ifreq *ifr; + struct sn_softc *sc = ifp->if_softc; + int s = splnet(), err = 0; + int temp; switch (cmd) { @@ -274,56 +287,60 @@ snioctl(ifp, cmd, data) switch (ifa->ifa_addr->sa_family) { #ifdef INET case AF_INET: - (void)sninit(ifp->if_softc); + (void)sninit(sc); arp_ifinit(&sc->sc_arpcom, ifa); break; #endif default: - (void)sninit(ifp->if_softc); + (void)sninit(sc); break; } break; case SIOCSIFFLAGS: if ((ifp->if_flags & IFF_UP) == 0 && - ifp->if_flags & IFF_RUNNING) { - snstop(ifp->if_softc); + (ifp->if_flags & IFF_RUNNING) != 0) { + /* + * If interface is marked down and it is running, + * then stop it. + */ + snstop(sc); ifp->if_flags &= ~IFF_RUNNING; - } else if (ifp->if_flags & IFF_UP && - (ifp->if_flags & IFF_RUNNING) == 0) - (void)sninit(ifp->if_softc); - /* - * If the state of the promiscuous bit changes, the interface - * must be reset to effect the change. - */ - if (((ifp->if_flags ^ sc->sc_iflags) & IFF_PROMISC) && - (ifp->if_flags & IFF_RUNNING)) { - sc->sc_iflags = ifp->if_flags; - printf("change in flags\n"); - temp = sc->sc_if.if_flags & IFF_UP; + } else if ((ifp->if_flags & IFF_UP) != 0 && + (ifp->if_flags & IFF_RUNNING) == 0) { + /* + * If interface is marked up and it is stopped, + * then start it. + */ + (void)sninit(sc); + } else { + /* + * reset the interface to pick up any other changes + * in flags + */ + temp = ifp->if_flags & IFF_UP; snreset(sc); - sc->sc_if.if_flags |= temp; + ifp->if_flags |= temp; snstart(ifp); } break; case SIOCADDMULTI: case SIOCDELMULTI: - if(cmd == SIOCADDMULTI) - err = ether_addmulti((struct ifreq *)data, - &sc->sc_arpcom); + ifr = (struct ifreq *) data; + if (cmd == SIOCADDMULTI) + err = ether_addmulti(ifr, &sc->sc_arpcom); else - err = ether_delmulti((struct ifreq *)data, - &sc->sc_arpcom); + err = ether_delmulti(ifr, &sc->sc_arpcom); if (err == ENETRESET) { /* * Multicast list has changed; set the hardware * filter accordingly. But remember UP flag! */ - temp = sc->sc_if.if_flags & IFF_UP; + temp = ifp->if_flags & IFF_UP; snreset(sc); - sc->sc_if.if_flags |= temp; + ifp->if_flags |= temp; err = 0; } break; @@ -341,16 +358,19 @@ static void snstart(ifp) struct ifnet *ifp; { - struct sn_softc *sc = ifp->if_softc; - struct mbuf *m; - int len; + struct sn_softc *sc = ifp->if_softc; + struct mbuf *m; + int mtd_next; - if ((sc->sc_if.if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING) + if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING) return; outloop: /* Check for room in the xmit buffer. */ - if (sc->txb_inuse == sc->txb_cnt) { + if ((mtd_next = (sc->mtd_free + 1)) == NTDA) + mtd_next = 0; + + if (mtd_next == sc->mtd_hw) { ifp->if_flags |= IFF_OACTIVE; return; } @@ -361,7 +381,7 @@ outloop: /* We need the header for m_pkthdr.len. */ if ((m->m_flags & M_PKTHDR) == 0) - panic("snstart: no header mbuf"); + panic("%s: snstart: no header mbuf", sc->sc_dev.dv_xname); #if NBPFILTER > 0 /* @@ -377,40 +397,22 @@ outloop: * the Tx ring, then send the packet directly. Otherwise append * it to the o/p queue. */ - if ((len = sonicput(sc, m)) > 0) { - len = m->m_pkthdr.len; - m_freem(m); + if ((sonicput(sc, m, mtd_next)) > 0) { } else { IF_PREPEND(&ifp->if_snd, m); return; } - /* Point to next buffer slot and wrap if necessary. */ - if (++sc->txb_new == sc->txb_cnt) - sc->txb_new = 0; - - sc->txb_inuse++; + sc->mtd_prev = sc->mtd_free; + sc->mtd_free = mtd_next; - ifp->if_opackets++; /* # of pkts */ - sc->sc_sum.ls_opacks++; /* # of pkts */ + ifp->if_opackets++; /* # of pkts */ /* Jump back for possibly more punishment. */ goto outloop; } /* - * This is called from sonicioctl() when /etc/ifconfig is run to set - * the address or switch the i/f on. - */ -static void caminitialise __P((struct sn_softc *)); -static void camentry __P((struct sn_softc *, int, unsigned char *ea)); -static void camprogram __P((struct sn_softc *)); -static void initialise_tda __P((struct sn_softc *)); -static void initialise_rda __P((struct sn_softc *)); -static void initialise_rra __P((struct sn_softc *)); -static void initialise_tba __P((struct sn_softc *)); - -/* * reset and restart the SONIC. Called in case of fatal * hardware/software errors. */ @@ -426,8 +428,8 @@ static int sninit(sc) struct sn_softc *sc; { - int s; - unsigned long s_rcr; + u_long s_rcr; + int s; if (sc->sc_if.if_flags & IFF_RUNNING) /* already running */ @@ -438,7 +440,8 @@ sninit(sc) NIC_PUT(sc, SNR_CR, CR_RST); /* DCR only accessable in reset mode! */ /* config it */ - NIC_PUT(sc, SNR_DCR, sc->snr_dcr); + NIC_PUT(sc, SNR_DCR, (sc->snr_dcr | + (sc->bitmode ? DCR_DW32 : DCR_DW16))); NIC_PUT(sc, SNR_DCR2, sc->snr_dcr2); s_rcr = RCR_BRD | RCR_LBNONE; @@ -461,7 +464,6 @@ sninit(sc) initialise_tda(sc); initialise_rda(sc); initialise_rra(sc); - initialise_tba(sc); /* enable the chip */ NIC_PUT(sc, SNR_CR, 0); @@ -482,6 +484,7 @@ sninit(sc) /* flag interface as "running" */ sc->sc_if.if_flags |= IFF_RUNNING; + sc->sc_if.if_flags &= ~IFF_OACTIVE; splx(s); return (0); @@ -497,7 +500,7 @@ snstop(sc) struct sn_softc *sc; { struct mtd *mtd; - int s = splnet(); + int s = splnet(); /* stick chip in reset */ NIC_PUT(sc, SNR_CR, CR_RST); @@ -508,10 +511,10 @@ snstop(sc) /* free all pending transmit mbufs */ while (sc->mtd_hw != sc->mtd_free) { mtd = &sc->mtda[sc->mtd_hw]; - mtd->mtd_buf = 0; + if (mtd->mtd_mbuf) + m_freem(mtd->mtd_mbuf); if (++sc->mtd_hw == NTDA) sc->mtd_hw = 0; } - sc->txb_inuse = 0; sc->sc_if.if_timer = 0; sc->sc_if.if_flags &= ~(IFF_RUNNING | IFF_UP); @@ -530,18 +533,18 @@ snwatchdog(ifp) struct ifnet *ifp; { struct sn_softc *sc = ifp->if_softc; - struct mtd *mtd; - int temp; + struct mtd *mtd; + int temp; if (sc->mtd_hw != sc->mtd_free) { /* something still pending for transmit */ mtd = &sc->mtda[sc->mtd_hw]; if (SRO(sc->bitmode, mtd->mtd_txp, TXP_STATUS) == 0) log(LOG_ERR, "%s: Tx - timeout\n", - sc->sc_dev.dv_xname); + sc->sc_dev.dv_xname); else log(LOG_ERR, "%s: Tx - lost interrupt\n", - sc->sc_dev.dv_xname); + sc->sc_dev.dv_xname); temp = ifp->if_flags & IFF_UP; snreset(sc); ifp->if_flags |= temp; @@ -551,72 +554,67 @@ snwatchdog(ifp) /* * stuff packet into sonic (at splnet) */ -static inline int -sonicput(sc, m0) +static __inline__ int +sonicput(sc, m0, mtd_next) struct sn_softc *sc; struct mbuf *m0; + int mtd_next; { - unsigned char *buff, *buffer; - void *txp; - struct mtd *mtdp; - struct mbuf *m; - unsigned int len = 0; - unsigned int totlen = 0; - int mtd_free = sc->mtd_free; - int mtd_next; - int txb_new = sc->txb_new; - - if (NIC_GET(sc, SNR_CR) & CR_TXP) { - return (0); - } + struct mtd *mtdp; + struct mbuf *m; + u_char *buff; + void *txp; + u_int len = 0; + u_int totlen = 0; /* grab the replacement mtd */ - mtdp = &sc->mtda[mtd_free]; - - if ((mtd_next = mtd_free + 1) == NTDA) - mtd_next = 0; + mtdp = &sc->mtda[sc->mtd_free]; - if (mtd_next == sc->mtd_hw) { - return (0); - } - - /* We are guaranteed, if we get here, that the xmit buffer is free. */ - buff = buffer = sc->tbuf[txb_new]; + buff = mtdp->mtd_buf; /* this packet goes to mtdnext fill in the TDA */ - mtdp->mtd_buf = buffer; + mtdp->mtd_mbuf = m0; txp = mtdp->mtd_txp; - SWO(sc->bitmode, txp, TXP_CONFIG, 0); + + /* Write to the config word. Every (NTDA/2)+1 packets we set an intr */ + if (sc->mtd_pint == 0) { + sc->mtd_pint = NTDA/2; + SWO(sc->bitmode, txp, TXP_CONFIG, TCR_PINT); + } else { + sc->mtd_pint--; + SWO(sc->bitmode, txp, TXP_CONFIG, 0); + } for (m = m0; m; m = m->m_next) { - unsigned char *data = mtod(m, u_char *); + u_char *data = mtod(m, u_char *); len = m->m_len; totlen += len; bcopy(data, buff, len); buff += len; } if (totlen >= TXBSIZE) { - panic("packet overflow in sonicput."); + panic("%s: sonicput: packet overflow", sc->sc_dev.dv_xname); } - SWO(sc->bitmode, txp, TXP_FRAGOFF+(0*TXP_FRAGSIZE)+TXP_FPTRLO, - LOWER(sc->vtbuf[txb_new])); - SWO(sc->bitmode, txp, TXP_FRAGOFF+(0*TXP_FRAGSIZE)+TXP_FPTRHI, - UPPER(sc->vtbuf[txb_new])); + + SWO(sc->bitmode, txp, TXP_FRAGOFF + (0 * TXP_FRAGSIZE) + TXP_FPTRLO, + LOWER(mtdp->mtd_vbuf)); + SWO(sc->bitmode, txp, TXP_FRAGOFF + (0 * TXP_FRAGSIZE) + TXP_FPTRHI, + UPPER(mtdp->mtd_vbuf)); if (totlen < ETHERMIN + sizeof(struct ether_header)) { int pad = ETHERMIN + sizeof(struct ether_header) - totlen; - bzero(buffer + totlen, pad); + bzero(mtdp->mtd_buf + totlen, pad); totlen = ETHERMIN + sizeof(struct ether_header); } - SWO(sc->bitmode, txp, TXP_FRAGOFF+(0*TXP_FRAGSIZE)+TXP_FSIZE, - totlen); + SWO(sc->bitmode, txp, TXP_FRAGOFF + (0 * TXP_FRAGSIZE) + TXP_FSIZE, + totlen); SWO(sc->bitmode, txp, TXP_FRAGCNT, 1); SWO(sc->bitmode, txp, TXP_PKTSIZE, totlen); /* link onto the next mtd that will be used */ - SWO(sc->bitmode, txp, TXP_FRAGOFF+(1*TXP_FRAGSIZE)+TXP_FPTRLO, - LOWER(sc->mtda[mtd_next].mtd_vtxp) | EOL); + SWO(sc->bitmode, txp, TXP_FRAGOFF + (1 * TXP_FRAGSIZE) + TXP_FPTRLO, + LOWER(sc->mtda[mtd_next].mtd_vtxp) | EOL); /* * The previous txp.tlink currently contains a pointer to @@ -624,10 +622,7 @@ sonicput(sc, m0) * pointer to the previous txp. */ SWO(sc->bitmode, sc->mtda[sc->mtd_prev].mtd_txp, sc->mtd_tlinko, - LOWER(mtdp->mtd_vtxp)); - - sc->mtd_prev = mtd_free; - sc->mtd_free = mtd_next; + LOWER(mtdp->mtd_vtxp)); /* make sure chip is running */ wbflush(); @@ -638,12 +633,10 @@ sonicput(sc, m0) return (totlen); } -static void sonictxint __P((struct sn_softc *)); -static void sonicrxint __P((struct sn_softc *)); - -static inline int sonic_read __P((struct sn_softc *, caddr_t, int)); -static inline struct mbuf *sonic_get __P((struct sn_softc *, struct ether_header *, int)); - +/* + * These are called from sonicioctl() when /etc/ifconfig is run to set + * the address or switch the i/f on. + */ /* * CAM support */ @@ -651,8 +644,8 @@ static void caminitialise(sc) struct sn_softc *sc; { - int i; void *p_cda = sc->p_cda; + int i; int bitmode = sc->bitmode; int camoffset; @@ -669,11 +662,11 @@ caminitialise(sc) static void camentry(sc, entry, ea) int entry; - unsigned char *ea; + u_char *ea; struct sn_softc *sc; { - int bitmode = sc->bitmode; void *p_cda = sc->p_cda; + int bitmode = sc->bitmode; int camoffset = entry * CDA_CAMDESC; SWO(bitmode, p_cda, camoffset + CDA_CAMEP, entry); @@ -681,25 +674,25 @@ camentry(sc, entry, ea) SWO(bitmode, p_cda, camoffset + CDA_CAMAP1, (ea[3] << 8) | ea[2]); SWO(bitmode, p_cda, camoffset + CDA_CAMAP0, (ea[1] << 8) | ea[0]); SWO(bitmode, p_cda, CDA_ENABLE, - (SRO(bitmode, p_cda, CDA_ENABLE) | (1 << entry))); + (SRO(bitmode, p_cda, CDA_ENABLE) | (1 << entry))); } static void camprogram(sc) struct sn_softc *sc; { - int timeout; - int mcount = 0; - struct ether_multi *enm; - struct ether_multistep step; - struct ifnet *ifp; + struct ether_multistep step; + struct ether_multi *enm; + struct ifnet *ifp; + int timeout; + int mcount = 0; caminitialise(sc); ifp = &sc->sc_if; /* Always load our own address first. */ - camentry (sc, 0, sc->sc_enaddr); + camentry (sc, mcount, sc->sc_enaddr); mcount++; /* Assume we won't need allmulti bit. */ @@ -714,7 +707,7 @@ camprogram(sc) } if (bcmp(enm->enm_addrlo, enm->enm_addrhi, - sizeof(enm->enm_addrlo)) != 0) { + sizeof(enm->enm_addrlo)) != 0) { /* * SONIC's CAM is programmed with specific * addresses. It has no way to specify a range. @@ -743,8 +736,7 @@ camprogram(sc) continue; if (timeout == 0) { /* XXX */ - panic("%s: CAM initialisation failed\n", - sc->sc_dev.dv_xname); + panic("%s: CAM initialisation failed\n", sc->sc_dev.dv_xname); } timeout = 10000; while (((NIC_GET(sc, SNR_ISR) & ISR_LCD) == 0) && timeout--) @@ -757,12 +749,12 @@ camprogram(sc) sc->sc_dev.dv_xname); } -#if 0 +#ifdef SNDEBUG static void camdump(sc) struct sn_softc *sc; { - int i; + int i; printf("CAM entries:\n"); NIC_PUT(sc, SNR_CR, CR_RST); @@ -777,7 +769,7 @@ camdump(sc) ap0 = NIC_GET(sc, SNR_CAP0); printf("%d: ap2=0x%x ap1=0x%x ap0=0x%x\n", i, ap2, ap1, ap0); } - printf("CAM enable 0x%x\n", NIC_GET(sc, SNR_CE)); + printf("CAM enable 0x%x\n", NIC_GET(sc, SNR_CEP)); NIC_PUT(sc, SNR_CR, 0); wbflush(); @@ -789,17 +781,18 @@ initialise_tda(sc) struct sn_softc *sc; { struct mtd *mtd; - int i; + int i; for (i = 0; i < NTDA; i++) { mtd = &sc->mtda[i]; - mtd->mtd_buf = 0; + mtd->mtd_mbuf = 0; } sc->mtd_hw = 0; - sc->mtd_prev = NTDA-1; + sc->mtd_prev = NTDA - 1; sc->mtd_free = 0; sc->mtd_tlinko = TXP_FRAGOFF + 1*TXP_FRAGSIZE + TXP_FPTRLO; + sc->mtd_pint = NTDA/2; NIC_PUT(sc, SNR_UTDA, UPPER(sc->mtda[0].mtd_vtxp)); NIC_PUT(sc, SNR_CTDA, LOWER(sc->mtda[0].mtd_vtxp)); @@ -809,24 +802,29 @@ static void initialise_rda(sc) struct sn_softc *sc; { - int bitmode = sc->bitmode; - int i; + int bitmode = sc->bitmode; + int i; + caddr_t p_rda = 0; + u_int32_t v_rda = 0; /* link the RDA's together into a circular list */ - for (i = 0; i < (NRDA - 1); i++) { - SWO(bitmode, sc->p_rda[i], RXPKT_RLINK, LOWER(sc->v_rda[i+1])); - SWO(bitmode, sc->p_rda[i], RXPKT_INUSE, 1); + for (i = 0; i < (sc->sc_nrda - 1); i++) { + p_rda = sc->p_rda + (i * RXPKT_SIZE(sc)); + v_rda = sc->v_rda + ((i+1) * RXPKT_SIZE(sc)); + SWO(bitmode, p_rda, RXPKT_RLINK, LOWER(v_rda)); + SWO(bitmode, p_rda, RXPKT_INUSE, 1); } - SWO(bitmode, sc->p_rda[NRDA - 1], RXPKT_RLINK, LOWER(sc->v_rda[0]) | EOL); - SWO(bitmode, sc->p_rda[NRDA - 1], RXPKT_INUSE, 1); + p_rda = sc->p_rda + ((sc->sc_nrda - 1) * RXPKT_SIZE(sc)); + SWO(bitmode, p_rda, RXPKT_RLINK, LOWER(sc->v_rda) | EOL); + SWO(bitmode, p_rda, RXPKT_INUSE, 1); /* mark end of receive descriptor list */ - sc->sc_rdamark = NRDA - 1; + sc->sc_rdamark = sc->sc_nrda - 1; sc->sc_rxmark = 0; - NIC_PUT(sc, SNR_URDA, UPPER(sc->v_rda[0])); - NIC_PUT(sc, SNR_CRDA, LOWER(sc->v_rda[0])); + NIC_PUT(sc, SNR_URDA, UPPER(sc->v_rda)); + NIC_PUT(sc, SNR_CRDA, LOWER(sc->v_rda)); wbflush(); } @@ -834,9 +832,9 @@ static void initialise_rra(sc) struct sn_softc *sc; { - int i; - unsigned int v; - int bitmode = sc->bitmode; + int i; + u_int v; + int bitmode = sc->bitmode; if (bitmode) NIC_PUT(sc, SNR_EOBC, RBASIZE(sc) / 2 - 2); @@ -852,7 +850,7 @@ initialise_rra(sc) /* fill up SOME of the rra with buffers */ for (i = 0; i < NRBA; i++) { - v = kvtop(sc->rbuf[i]); + v = SONIC_GETDMA(sc->rbuf[i]); SWO(bitmode, sc->p_rra[i], RXRSRC_PTRHI, UPPER(v)); SWO(bitmode, sc->p_rra[i], RXRSRC_PTRLO, LOWER(v)); SWO(bitmode, sc->p_rra[i], RXRSRC_WCHI, UPPER(NBPG/2)); @@ -863,21 +861,12 @@ initialise_rra(sc) wbflush(); } -static void -initialise_tba(sc) - struct sn_softc *sc; -{ - sc->txb_cnt = NTXB; - sc->txb_inuse = 0; - sc->txb_new = 0; -} - void snintr(arg, slot) void *arg; int slot; { - struct sn_softc *sc = (struct sn_softc *)arg; + struct sn_softc *sc = (struct sn_softc *)arg; int isr; while ((isr = (NIC_GET(sc, SNR_ISR) & ISR_ALL)) != 0) { @@ -941,56 +930,73 @@ static void sonictxint(sc) struct sn_softc *sc; { - void *txp; struct mtd *mtd; - /* XXX DG make mtd_hw a local var */ + void *txp; + unsigned short txp_status; + int mtd_hw; + struct ifnet *ifp = &sc->sc_if; - if (sc->mtd_hw == sc->mtd_free) + mtd_hw = sc->mtd_hw; + + if (mtd_hw == sc->mtd_free) return; - while (sc->mtd_hw != sc->mtd_free) { - mtd = &sc->mtda[sc->mtd_hw]; - if (mtd->mtd_buf == 0) - break; + while (mtd_hw != sc->mtd_free) { + mtd = &sc->mtda[mtd_hw]; txp = mtd->mtd_txp; - if (SRO(sc->bitmode, txp, TXP_STATUS) == 0) - return; /* it hasn't really gone yet */ + if (SRO(sc->bitmode, txp, TXP_STATUS) == 0) { + break; /* it hasn't really gone yet */ + } - if (ethdebug) { +#ifdef SNDEBUG + { struct ether_header *eh; eh = (struct ether_header *) mtd->mtd_buf; - printf("xmit status=0x%x len=%d type=0x%x from %s", + printf("%s: xmit status=0x%x len=%d type=0x%x from %s", + sc->sc_dev.dv_xname, SRO(sc->bitmode, txp, TXP_STATUS), SRO(sc->bitmode, txp, TXP_PKTSIZE), htons(eh->ether_type), ether_sprintf(eh->ether_shost)); printf(" (to %s)\n", ether_sprintf(eh->ether_dhost)); } - sc->txb_inuse--; - mtd->mtd_buf = 0; - if (++sc->mtd_hw == NTDA) sc->mtd_hw = 0; +#endif /* SNDEBUG */ - /* XXX - Do stats here. */ + ifp->if_flags &= ~IFF_OACTIVE; - if ((SRO(sc->bitmode, txp, TXP_STATUS) & TCR_PTX) == 0) { - printf("%s: Tx packet status=0x%x\n", - sc->sc_dev.dv_xname, - SRO(sc->bitmode, txp, TXP_STATUS)); + if (mtd->mtd_mbuf != 0) { + m_freem(mtd->mtd_mbuf); + mtd->mtd_mbuf = 0; + } + if (++mtd_hw == NTDA) mtd_hw = 0; + txp_status = SRO(sc->bitmode, txp, TXP_STATUS); + + ifp->if_collisions += (txp_status & TCR_EXC) ? 16 : + ((txp_status & TCR_NC) >> 12); + + if ((txp_status & TCR_PTX) == 0) { + ifp->if_oerrors++; + printf("%s: Tx packet status=0x%x\n", + sc->sc_dev.dv_xname, txp_status); + /* XXX - DG This looks bogus */ - if (sc->mtd_hw != sc->mtd_free) { + if (mtd_hw != sc->mtd_free) { printf("resubmitting remaining packets\n"); - mtd = &sc->mtda[sc->mtd_hw]; + mtd = &sc->mtda[mtd_hw]; NIC_PUT(sc, SNR_CTDA, LOWER(mtd->mtd_vtxp)); NIC_PUT(sc, SNR_CR, CR_TXP); wbflush(); - return; + break; } } } + + sc->mtd_hw = mtd_hw; + return; } /* @@ -1000,30 +1006,30 @@ static void sonicrxint(sc) struct sn_softc *sc; { - void *rda; - int orra; - int len; - int rramark; - int rdamark; - int bitmode = sc->bitmode; - u_int16_t rxpkt_ptr; + caddr_t rda; + int orra; + int len; + int rramark; + int rdamark; + int bitmode = sc->bitmode; + u_int16_t rxpkt_ptr; - rda = sc->p_rda[sc->sc_rxmark]; + rda = sc->p_rda + (sc->sc_rxmark * RXPKT_SIZE(sc)); while (SRO(bitmode, rda, RXPKT_INUSE) == 0) { - unsigned status = SRO(bitmode, rda, RXPKT_STATUS); + u_int status = SRO(bitmode, rda, RXPKT_STATUS); orra = RBASEQ(SRO(bitmode, rda, RXPKT_SEQNO)) & RRAMASK; rxpkt_ptr = SRO(bitmode, rda, RXPKT_PTRLO); len = SRO(bitmode, rda, RXPKT_BYTEC) - sizeof(struct ether_header) - FCSSIZE; if (status & RCR_PRX) { - caddr_t pkt = sc->rbuf[orra & RBAMASK] + (rxpkt_ptr & PGOFSET); - if (sonic_read(sc, pkt, len)) { + caddr_t pkt = + sc->rbuf[orra & RBAMASK] + (rxpkt_ptr & PGOFSET); + if (sonic_read(sc, pkt, len)) sc->sc_if.if_ipackets++; - sc->sc_sum.ls_ipacks++; - sc->sc_missed = 0; - } + else + sc->sc_if.if_ierrors++; } else sc->sc_if.if_ierrors++; @@ -1072,13 +1078,14 @@ sonicrxint(sc) SWO(bitmode, rda, RXPKT_INUSE, 1); SWO(bitmode, rda, RXPKT_RLINK, SRO(bitmode, rda, RXPKT_RLINK) | EOL); - SWO(bitmode, sc->p_rda[rdamark], RXPKT_RLINK, - SRO(bitmode, sc->p_rda[rdamark], RXPKT_RLINK) & ~EOL); + SWO(bitmode, (sc->p_rda + (rdamark * RXPKT_SIZE(sc))), RXPKT_RLINK, + SRO(bitmode, (sc->p_rda + (rdamark * RXPKT_SIZE(sc))), + RXPKT_RLINK) & ~EOL); sc->sc_rdamark = sc->sc_rxmark; - if (++sc->sc_rxmark >= NRDA) + if (++sc->sc_rxmark >= sc->sc_nrda) sc->sc_rxmark = 0; - rda = sc->p_rda[sc->sc_rxmark]; + rda = sc->p_rda + (sc->sc_rxmark * RXPKT_SIZE(sc)); } } @@ -1086,7 +1093,7 @@ sonicrxint(sc) * sonic_read -- pull packet off interface and forward to * appropriate protocol handler */ -static inline int +static __inline__ int sonic_read(sc, pkt, len) struct sn_softc *sc; caddr_t pkt; @@ -1097,16 +1104,19 @@ sonic_read(sc, pkt, len) struct mbuf *m; /* - * Get pointer to ethernet header (in input buffer). - */ + * Get pointer to ethernet header (in input buffer). + */ et = (struct ether_header *)pkt; - if (ethdebug) { - printf("rcvd 0x%p len=%d type=0x%x from %s", - et, len, htons(et->ether_type), +#ifdef SNDEBUG + { + printf("%s: rcvd 0x%p len=%d type=0x%x from %s", + sc->sc_dev.dv_xname, et, len, htons(et->ether_type), ether_sprintf(et->ether_shost)); printf(" (to %s)\n", ether_sprintf(et->ether_dhost)); } +#endif /* SNDEBUG */ + if (len < ETHERMIN || len > ETHERMTU) { printf("%s: invalid packet length %d bytes\n", sc->sc_dev.dv_xname, len); @@ -1124,8 +1134,8 @@ sonic_read(sc, pkt, len) len + sizeof(struct ether_header)); if ((ifp->if_flags & IFF_PROMISC) != 0 && (et->ether_dhost[0] & 1) == 0 && /* !mcast and !bcast */ - bcmp(et->ether_dhost, sc->sc_enaddr, - sizeof(et->ether_dhost)) != 0) + bcmp(et->ether_dhost, &sc->sc_enaddr, + sizeof(et->ether_dhost)) != 0) return (0); } #endif @@ -1133,36 +1143,32 @@ sonic_read(sc, pkt, len) if (m == NULL) return (0); ether_input(ifp, et, m); - return(1); + return (1); } -#define sonicdataaddr(eh, off, type) ((type)(((caddr_t)((eh)+1)+(off)))) +#define sonicdataaddr(eh, off, type) ((type)(((caddr_t)((eh) + 1) + (off)))) /* * munge the received packet into an mbuf chain - * because we are using stupid buffer management this - * is slow. */ -static inline struct mbuf * +static __inline__ struct mbuf * sonic_get(sc, eh, datalen) struct sn_softc *sc; struct ether_header *eh; int datalen; { - struct mbuf *m; - struct mbuf *top = 0, **mp = ⊤ + struct mbuf *m, *top, **mp; int len; - char *spkt = sonicdataaddr(eh, 0, caddr_t); - char *epkt = spkt + datalen; - char *cp = spkt; + caddr_t pkt = sonicdataaddr(eh, 0, caddr_t); - epkt = cp + datalen; MGETHDR(m, M_DONTWAIT, MT_DATA); if (m == 0) return (0); m->m_pkthdr.rcvif = &sc->sc_if; m->m_pkthdr.len = datalen; - m->m_len = MHLEN; + len = MHLEN; + top = 0; + mp = ⊤ while (datalen > 0) { if (top) { @@ -1171,34 +1177,25 @@ sonic_get(sc, eh, datalen) m_freem(top); return (0); } - m->m_len = MLEN; + len = MLEN; } - len = min(datalen, epkt - cp); - if (len >= MINCLSIZE) { + if (datalen >= MINCLSIZE) { MCLGET(m, M_DONTWAIT); - if (m->m_flags & M_EXT) - m->m_len = len = min(len, MCLBYTES); - else - len = m->m_len; - } else { - /* - * Place initial small packet/header at end of mbuf. - */ - if (len < m->m_len) { - if (top == 0 && len + max_linkhdr <= m->m_len) - m->m_data += max_linkhdr; - m->m_len = len; - } else - len = m->m_len; + if ((m->m_flags & M_EXT) == 0) { + if (top) m_freem(top); + return (0); + } + len = MLEN; } - bcopy(cp, mtod(m, caddr_t), (unsigned) len); - cp += len; + m->m_len = len = min(datalen, len); + + bcopy(pkt, mtod(m, caddr_t), (unsigned) len); + pkt += len; + datalen -= len; *mp = m; mp = &m->m_next; - datalen -= len; - if (cp == epkt) - cp = spkt; } + return (top); } @@ -1207,13 +1204,13 @@ static u_char bbr4[] = {0,8,4,12,2,10,6,14,1,9,5,13,3,11,7,15}; void sn_get_enaddr(t, h, o, dst) - bus_space_tag_t t; + bus_space_tag_t t; bus_space_handle_t h; - vm_offset_t o; - u_char *dst; + vm_offset_t o; + u_char *dst; { - int i, do_bbr; - u_char b; + int i, do_bbr; + u_char b; /* * For reasons known only to Apple, MAC addresses in the ethernet diff --git a/sys/arch/mac68k/dev/if_sn_nubus.c b/sys/arch/mac68k/dev/if_sn_nubus.c index 42b8188b43b..f357d786656 100644 --- a/sys/arch/mac68k/dev/if_sn_nubus.c +++ b/sys/arch/mac68k/dev/if_sn_nubus.c @@ -1,6 +1,4 @@ -/* $OpenBSD: if_sn_nubus.c,v 1.11 1997/05/01 18:32:48 briggs Exp $ */ -/* $NetBSD: if_sn_nubus.c,v 1.12 1997/05/01 18:17:13 briggs Exp $ */ - +/* $NetBSD: if_sn_nubus.c,v 1.13 1997/05/11 19:11:34 scottr Exp $ */ /* * Copyright (C) 1997 Allen Briggs * All rights reserved. @@ -41,51 +39,54 @@ #include <net/if.h> +#ifdef INET #include <netinet/in.h> #include <netinet/if_ether.h> +#endif #include <machine/bus.h> #include <machine/viareg.h> #include "nubus.h" -#include "if_aereg.h" /* For AE_VENDOR values */ #include "if_snreg.h" #include "if_snvar.h" static int sn_nubus_match __P((struct device *, void *, void *)); static void sn_nubus_attach __P((struct device *, struct device *, void *)); -static int sn_nb_card_vendor __P((struct nubus_attach_args *)); +static int sn_nb_card_vendor __P((bus_space_tag_t, bus_space_handle_t, + struct nubus_attach_args *)); struct cfattach sn_nubus_ca = { sizeof(struct sn_softc), sn_nubus_match, sn_nubus_attach }; + static int -sn_nubus_match(parent, vcf, aux) +sn_nubus_match(parent, cf, aux) struct device *parent; - void *vcf; + void *cf; void *aux; { struct nubus_attach_args *na = (struct nubus_attach_args *) aux; bus_space_handle_t bsh; int rv; - if (bus_space_map(na->na_tag, NUBUS_SLOT2PA(na->slot), NBMEMSIZE, - 0, &bsh)) + if (bus_space_map(na->na_tag, + NUBUS_SLOT2PA(na->slot), NBMEMSIZE, 0, &bsh)) return (0); rv = 0; if (na->category == NUBUS_CATEGORY_NETWORK && na->type == NUBUS_TYPE_ETHERNET) { - switch (sn_nb_card_vendor(na)) { - - case AE_VENDOR_APPLE: - case AE_VENDOR_DAYNA: - rv = 1; + switch (sn_nb_card_vendor(na->na_tag, bsh, na)) { + default: break; - default: + case SN_VENDOR_APPLE: + case SN_VENDOR_DAYNA: + case SN_VENDOR_APPLE16: + rv = 1; break; } } @@ -103,11 +104,14 @@ sn_nubus_attach(parent, self, aux) struct device *parent, *self; void *aux; { - struct sn_softc *sc = (void *)self; - struct nubus_attach_args *na = (struct nubus_attach_args *)aux; - int i, success, offset; + struct sn_softc *sc = (void *)self; + struct nubus_attach_args *na = (struct nubus_attach_args *)aux; + int i, success, offset; bus_space_tag_t bst; - bus_space_handle_t bsh, tmp_bsh; + bus_space_handle_t bsh, tmp_bsh; + u_int8_t myaddr[ETHER_ADDR_LEN]; + + (void)(&offset); /* Work around lame gcc initialization bug */ bst = na->na_tag; if (bus_space_map(bst, NUBUS_SLOT2PA(na->slot), NBMEMSIZE, 0, &bsh)) { @@ -116,88 +120,110 @@ sn_nubus_attach(parent, self, aux) } sc->sc_regt = bst; - sc->bitmode = 1; success = 0; - sc->bitmode = 1; /* 32-bit card */ - sc->slotno = na->slot; + sc->slotno = na->slot; - switch (sn_nb_card_vendor(na)) { - case AE_VENDOR_DAYNA: - sc->snr_dcr = DCR_ASYNC | DCR_WAIT0 | DCR_DW32 | - DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; + switch (sn_nb_card_vendor(bst, bsh, na)) { + case SN_VENDOR_DAYNA: + sc->snr_dcr = DCR_ASYNC | DCR_WAIT0 | + DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; sc->snr_dcr2 = 0; + sc->bitmode = 1; /* 32 bit card */ - if (bus_space_subregion(bst, bsh, 0x00180000, SN_REGSIZE, - &sc->sc_regh)) { + if (bus_space_subregion(bst, bsh, + 0x00180000, SN_REGSIZE, &sc->sc_regh)) { printf(": failed to map register space.\n"); break; } - if (bus_space_subregion(bst, bsh, 0x00ffe004, ETHER_ADDR_LEN, - &tmp_bsh)) { + if (bus_space_subregion(bst, bsh, + 0x00ffe004, ETHER_ADDR_LEN, &tmp_bsh)) { printf(": failed to map ROM space.\n"); break; } - sn_get_enaddr(bst, tmp_bsh, 0, sc->sc_arpcom.ac_enaddr); + sn_get_enaddr(bst, tmp_bsh, 0, myaddr); offset = 2; success = 1; - break; + break; - case AE_VENDOR_APPLE: - sc->snr_dcr = DCR_ASYNC | DCR_WAIT0 | DCR_DW32 | - DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; + case SN_VENDOR_APPLE: + sc->snr_dcr = DCR_ASYNC | DCR_WAIT0 | + DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; sc->snr_dcr2 = 0; + sc->bitmode = 1; /* 32 bit card */ - if (bus_space_subregion(bst, bsh, 0x0, SN_REGSIZE, - &sc->sc_regh)) { + if (bus_space_subregion(bst, bsh, + 0x0, SN_REGSIZE, &sc->sc_regh)) { printf(": failed to map register space.\n"); break; } - if (bus_space_subregion(bst, bsh, 0x40000, ETHER_ADDR_LEN, - &tmp_bsh)) { + if (bus_space_subregion(bst, bsh, + 0x40000, ETHER_ADDR_LEN, &tmp_bsh)) { printf(": failed to map ROM space.\n"); break; } - sn_get_enaddr(bst, tmp_bsh, 0, sc->sc_arpcom.ac_enaddr); + sn_get_enaddr(bst, tmp_bsh, 0, myaddr); offset = 0; success = 1; - break; - - default: - /* - * You can't actually get this default, the snmatch - * will fail for unknown hardware. If you're adding support - * for a new card, the following defaults are a - * good starting point. - */ - sc->snr_dcr = DCR_SYNC | DCR_WAIT0 | DCR_DW32 | - DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; + break; + + case SN_VENDOR_APPLE16: + sc->snr_dcr = DCR_ASYNC | DCR_WAIT0 | DCR_EXBUS | + DCR_DMABLOCK | DCR_PO1 | DCR_RFT16 | DCR_TFT16; sc->snr_dcr2 = 0; + sc->bitmode = 0; /* 16 bit card */ + + if (bus_space_subregion(bst, bsh, + 0x0, SN_REGSIZE, &sc->sc_regh)) { + printf(": failed to map register space.\n"); + break; + } + + if (bus_space_subregion(bst, bsh, + 0x40000, ETHER_ADDR_LEN, &tmp_bsh)) { + printf(": failed to map ROM space.\n"); + break; + } + + sn_get_enaddr(bst, tmp_bsh, 0, myaddr); + offset = 0; + success = 1; + break; + + default: + /* + * You can't actually get this default, the snmatch + * will fail for unknown hardware. If you're adding support + * for a new card, the following defaults are a + * good starting point. + */ + sc->snr_dcr = DCR_SYNC | DCR_WAIT0 | DCR_DW32 | + DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; + sc->snr_dcr2 = 0; success = 0; printf(": unknown card: attachment incomplete.\n"); - } + } if (!success) { bus_space_unmap(bst, bsh, NBMEMSIZE); return; } - snsetup(sc); /* Regs are addressed as words, big endian. */ for (i = 0; i < SN_NREGS; i++) { sc->sc_reg_map[i] = (bus_size_t)((i * 4) + offset); } /* snsetup returns 1 if something fails */ - if (snsetup(sc)) { + if (snsetup(sc, myaddr)) { bus_space_unmap(bst, bsh, NBMEMSIZE); return; } @@ -208,59 +234,34 @@ sn_nubus_attach(parent, self, aux) } static int -sn_nb_card_vendor(na) +sn_nb_card_vendor(bst, bsh, na) + bus_space_tag_t bst; + bus_space_handle_t bsh; struct nubus_attach_args *na; { - int vendor; + int vendor = SN_VENDOR_UNKNOWN; switch (na->drsw) { case NUBUS_DRSW_3COM: - switch (na->drhw) { - case NUBUS_DRHW_APPLE_SN: - case NUBUS_DRHW_APPLE_SNT: - vendor = AE_VENDOR_APPLE; - break; - default: - vendor = AE_VENDOR_UNKNOWN; - break; - } + if (na->drhw == NUBUS_DRHW_APPLE_SN) + vendor = SN_VENDOR_APPLE; + else if (na->drhw == NUBUS_DRHW_APPLE_SNT) + vendor = SN_VENDOR_APPLE16; break; case NUBUS_DRSW_APPLE: case NUBUS_DRSW_TECHWORKS: - vendor = AE_VENDOR_APPLE; - break; - case NUBUS_DRSW_ASANTE: - vendor = AE_VENDOR_ASANTE; - break; - case NUBUS_DRSW_FARALLON: - vendor = AE_VENDOR_FARALLON; - break; - case NUBUS_DRSW_FOCUS: - vendor = AE_VENDOR_FOCUS; + vendor = SN_VENDOR_APPLE; break; case NUBUS_DRSW_GATOR: - switch (na->drhw) { - default: - case NUBUS_DRHW_INTERLAN: - vendor = AE_VENDOR_INTERLAN; - break; - case NUBUS_DRHW_KINETICS: - if (strncmp( - nubus_get_card_name(na->fmt), "EtherPort", 9) == 0) - vendor = AE_VENDOR_KINETICS; - else - vendor = AE_VENDOR_DAYNA; - break; - } + if (na->drhw == NUBUS_DRHW_KINETICS && + strncmp(nubus_get_card_name(na->fmt), + "EtherPort", 9) != 0) + vendor = SN_VENDOR_DAYNA; break; case NUBUS_DRSW_DAYNA: - vendor = AE_VENDOR_DAYNA; + vendor = SN_VENDOR_DAYNA; break; - default: -#ifdef DIAGNOSTIC - printf("Unknown ethernet drsw: %x\n", na->drsw); -#endif - vendor = AE_VENDOR_UNKNOWN; } + return vendor; } diff --git a/sys/arch/mac68k/dev/if_sn_obio.c b/sys/arch/mac68k/dev/if_sn_obio.c index 09df12eebe5..14da748eda5 100644 --- a/sys/arch/mac68k/dev/if_sn_obio.c +++ b/sys/arch/mac68k/dev/if_sn_obio.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_sn_obio.c,v 1.13 1997/04/22 13:37:56 briggs Exp $ */ +/* $NetBSD: if_sn_obio.c,v 1.9 1997/04/22 20:56:15 scottr Exp $ */ /* * Copyright (C) 1997 Allen Briggs @@ -39,7 +39,6 @@ #include <sys/systm.h> #include <net/if.h> - #include <netinet/in.h> #include <netinet/if_ether.h> @@ -56,17 +55,17 @@ static int sn_obio_match __P((struct device *, void *, void *)); static void sn_obio_attach __P((struct device *, struct device *, void *)); -static int sn_obio_getaddr __P((struct sn_softc *)); -static int sn_obio_getaddr_kludge __P((struct sn_softc *)); +static int sn_obio_getaddr __P((struct sn_softc *, u_int8_t *)); +static int sn_obio_getaddr_kludge __P((struct sn_softc *, u_int8_t *)); struct cfattach sn_obio_ca = { sizeof(struct sn_softc), sn_obio_match, sn_obio_attach }; static int -sn_obio_match(parent, vcf, aux) +sn_obio_match(parent, cf, aux) struct device *parent; - void *vcf; + void *cf; void *aux; { if (mac68k_machine.sonic) @@ -83,33 +82,43 @@ sn_obio_attach(parent, self, aux) struct device *parent, *self; void *aux; { - struct obio_attach_args *oa = (struct obio_attach_args *) aux; - struct sn_softc *sc = (void *)self; + struct obio_attach_args *oa = (struct obio_attach_args *)aux; + struct sn_softc *sc = (void *)self; + u_int8_t myaddr[ETHER_ADDR_LEN]; int i; - sc->snr_dcr = DCR_WAIT0 | DCR_DMABLOCK | - DCR_RFT16 | DCR_TFT16; + sc->snr_dcr = DCR_WAIT0 | DCR_DMABLOCK | DCR_RFT16 | DCR_TFT16; sc->snr_dcr2 = 0; switch (current_mac_model->machineid) { - case MACH_MACQ700: case MACH_MACQ900: case MACH_MACQ950: - case MACH_MACQ800: case MACH_MACQ650: case MACH_MACC650: - case MACH_MACC610: case MACH_MACQ610: - sc->snr_dcr |= DCR_EXBUS | DCR_DW32; + case MACH_MACC610: + case MACH_MACC650: + case MACH_MACQ610: + case MACH_MACQ650: + case MACH_MACQ700: + case MACH_MACQ800: + case MACH_MACQ900: + case MACH_MACQ950: + sc->snr_dcr |= DCR_EXBUS; sc->bitmode = 1; break; case MACH_MACPB500: - sc->snr_dcr |= DCR_SYNC | DCR_LBR | DCR_DW16; - sc->bitmode = 0; + sc->snr_dcr |= DCR_SYNC | DCR_LBR; + sc->bitmode = 0; /* 16 bit interface */ break; - } + + default: + printf(": unsupported machine type\n"); + return; + } sc->sc_regt = oa->oa_tag; - if (bus_space_map(sc->sc_regt, SONIC_REG_BASE, SN_REGSIZE, - 0, &sc->sc_regh)) { - panic("failed to map space for SONIC regs.\n"); + if (bus_space_map(sc->sc_regt, + SONIC_REG_BASE, SN_REGSIZE, 0, &sc->sc_regh)) { + printf(": failed to map space for SONIC regs.\n"); + return; } sc->slotno = 9; @@ -119,17 +128,15 @@ sn_obio_attach(parent, self, aux) sc->sc_reg_map[i] = (bus_size_t)((i * 4) + 2); } - if (sn_obio_getaddr(sc)) { - printf(": Failed to get MAC address. Trying kludge.\n"); - if (sn_obio_getaddr_kludge(sc)) { - printf("Kludge failed, too. Attachment failed.\n"); - bus_space_unmap(sc->sc_regt, sc->sc_regh, SN_REGSIZE); - return; - } + if (sn_obio_getaddr(sc, myaddr) && + sn_obio_getaddr_kludge(sc, myaddr)) { /* XXX kludge for PB */ + printf(": failed to get MAC address.\n"); + bus_space_unmap(sc->sc_regt, sc->sc_regh, SN_REGSIZE); + return; } /* snsetup returns 1 if something fails */ - if (snsetup(sc)) { + if (snsetup(sc, myaddr)) { bus_space_unmap(sc->sc_regt, sc->sc_regh, SN_REGSIZE); return; } @@ -138,21 +145,24 @@ sn_obio_attach(parent, self, aux) } static int -sn_obio_getaddr(sc) +sn_obio_getaddr(sc, lladdr) struct sn_softc *sc; + u_int8_t *lladdr; { - bus_space_handle_t bsh; + bus_space_handle_t bsh; if (bus_space_map(sc->sc_regt, SONIC_PROM_BASE, NBPG, 0, &bsh)) { - panic("failed to map space to read SONIC address.\n"); + printf(": failed to map space to read SONIC address.\n%s", + sc->sc_dev.dv_xname); + return (-1); } if (!bus_probe(sc->sc_regt, bsh, 0, 1)) { bus_space_unmap(sc->sc_regt, bsh, NBPG); - return -1; + return (-1); } - sn_get_enaddr(sc->sc_regt, bsh, 0, sc->sc_arpcom.ac_enaddr); + sn_get_enaddr(sc->sc_regt, bsh, 0, lladdr); bus_space_unmap(sc->sc_regt, bsh, NBPG); @@ -164,49 +174,44 @@ sn_obio_getaddr(sc) * when we can properly get the MAC address on the PBs. */ static int -sn_obio_getaddr_kludge(sc) +sn_obio_getaddr_kludge(sc, lladdr) struct sn_softc *sc; + u_int8_t *lladdr; { - int i, ors=0; + int i, ors = 0; /* Shut down NIC */ - NIC_PUT(sc, SNR_CR, CR_STP); - wbflush(); NIC_PUT(sc, SNR_CR, CR_RST); wbflush(); - NIC_PUT(sc, SNR_IMR, 0); - wbflush(); - NIC_PUT(sc, SNR_ISR, ISR_ALL); - wbflush(); - NIC_PUT(sc, SNR_CR, CR_RST); - wbflush(); NIC_PUT(sc, SNR_CEP, 15); /* For some reason, Apple fills top first. */ wbflush(); i = NIC_GET(sc, SNR_CAP2); wbflush(); ors |= i; - sc->sc_enaddr[5] = i >> 8; - sc->sc_enaddr[4] = i; + lladdr[5] = i >> 8; + lladdr[4] = i; i = NIC_GET(sc, SNR_CAP1); wbflush(); ors |= i; - sc->sc_enaddr[3] = i >> 8; - sc->sc_enaddr[2] = i; + lladdr[3] = i >> 8; + lladdr[2] = i; i = NIC_GET(sc, SNR_CAP0); wbflush(); ors |= i; - sc->sc_enaddr[1] = i >> 8; - sc->sc_enaddr[0] = i; + lladdr[1] = i >> 8; + lladdr[0] = i; NIC_PUT(sc, SNR_CR, 0); wbflush(); - if (ors == 0) return -1; - return (0); + if (ors == 0) + return -1; + + return 0; } diff --git a/sys/arch/mac68k/dev/if_snreg.h b/sys/arch/mac68k/dev/if_snreg.h index 8fd0a819e02..9d1f138180f 100644 --- a/sys/arch/mac68k/dev/if_snreg.h +++ b/sys/arch/mac68k/dev/if_snreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_snreg.h,v 1.2 1997/03/29 23:26:50 briggs Exp $ */ +/* $NetBSD: if_snreg.h,v 1.3 1997/04/30 19:47:12 scottr Exp $ */ /* * Copyright (c) 1991 Algorithmics Ltd (http://www.algor.co.uk) @@ -7,7 +7,7 @@ */ /* - * if_sonic.h -- National Semiconductor DP83932BVF (SONIC) + * if_snreg.h -- National Semiconductor DP8393X (SONIC) register defs */ /* @@ -197,6 +197,7 @@ #define TCR_BCM 0x0002 /* byte count mismatch (TXpkt.pkt_size * != sum(TXpkt.frag_size) */ #define TCR_PTX 0x0001 /* packet transmitted without errors */ +#define TCR_NC 0xf000 /* after transmission, # of colls */ /* transmit control register aliases */ #define TCR_OWCSFD 0 /* start after start of frame delimiter */ diff --git a/sys/arch/mac68k/dev/if_snvar.h b/sys/arch/mac68k/dev/if_snvar.h index 9d0abe3bd83..266ac55e3bb 100644 --- a/sys/arch/mac68k/dev/if_snvar.h +++ b/sys/arch/mac68k/dev/if_snvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_snvar.h,v 1.7 1997/04/25 03:29:14 briggs Exp $ */ +/* $NetBSD: if_snvar.h,v 1.8 1997/04/25 03:40:09 briggs Exp $ */ /* * Copyright (c) 1991 Algorithmics Ltd (http://www.algor.co.uk) @@ -7,19 +7,27 @@ */ /* - * if_snvar.h -- National Semiconductor DP83932BVF (SONIC) OpenBSD/mac68k vars + * if_snvar.h -- National Semiconductor DP8393X (SONIC) NetBSD/mac68k vars */ /* + * Vendor types + */ +#define SN_VENDOR_UNKNOWN 0xff /* Unknown */ +#define SN_VENDOR_APPLE 0x00 /* Apple Computer/compatible */ +#define SN_VENDOR_DAYNA 0x01 /* Dayna/Kinetics EtherPort */ +#define SN_VENDOR_APPLE16 0x02 /* Apple Twisted Pair Ether NB */ + +/* * Memory access macros. Since we handle SONIC in 16 bit mode (PB5X0) * and 32 bit mode (everything else) using a single GENERIC kernel * binary, all structures have to be accessed using macros which can * adjust the offsets appropriately. */ -#define SWO(m, a, o, x) (m ? (*(u_int32_t *)((u_int32_t *)a + o) = (x)) : \ - (*(u_int16_t *)((u_int16_t *)a + o) = (x))) -#define SRO(m, a, o) (m ? (*(u_int32_t *)((u_int32_t *)a + o) & 0xffff) : \ - (*(u_int16_t *)((u_int16_t *)a + o) & 0xffff)) +#define SWO(m, a, o, x) (m ? (*(u_int32_t *)((u_int32_t *)(a) + (o)) = (x)) : \ + (*(u_int16_t *)((u_int16_t *)(a) + (o)) = (x))) +#define SRO(m, a, o) (m ? (*(u_int32_t *)((u_int32_t *)(a) + (o)) & 0xffff) : \ + (*(u_int16_t *)((u_int16_t *)(a) + (o)) & 0xffff)) /* * Register access macros. We use bus_space_* to talk to the Sonic @@ -28,11 +36,15 @@ */ #define NIC_GET(sc, reg) (bus_space_read_2((sc)->sc_regt, \ (sc)->sc_regh, \ - ((sc)->sc_reg_map[reg]))) + ((sc)->sc_reg_map[(reg)]))) #define NIC_PUT(sc, reg, val) (bus_space_write_2((sc)->sc_regt, \ (sc)->sc_regh, \ ((sc)->sc_reg_map[reg]), \ (val))) + +extern int kvtop(caddr_t addr); +#define SONIC_GETDMA(p) (u_int32_t)(kvtop((caddr_t)(p))) + #define SN_REGSIZE SN_NREGS*4 /* mac68k does not have any write buffers to flush... */ @@ -52,7 +64,6 @@ #define NRBA 8 /* # receive buffers < NRRA */ #define RBAMASK (NRBA-1) -#define NRDA 146 #define NTDA 8 /* # transmit descriptors */ #define NRRA 16 /* # receive resource descriptors */ #define RRAMASK (NRRA-1) /* the reason why NRRA must be power of two */ @@ -69,84 +80,49 @@ /* * transmit buffer area */ -#define NTXB NTDA /* Number of xmit buffers */ -#define TXBSIZE 1536 /* 6*2^8 -- the same size as the 8390 TXBUF */ +#define TXBSIZE 1536 /* 6*2^8 -- the same size as the 8390 TXBUF */ -#define SN_NPAGES 2 + 8 + 4 - -/* - * Statistics collected over time - */ -struct sn_stats { - int ls_opacks; /* packets transmitted */ - int ls_ipacks; /* packets received */ - int ls_tdr; /* contents of tdr after collision */ - int ls_tdef; /* packets where had to wait */ - int ls_tone; /* packets with one retry */ - int ls_tmore; /* packets with more than one retry */ - int ls_tbuff; /* transmit buff errors */ - int ls_tuflo; /* " uflo " */ - int ls_tlcol; - int ls_tlcar; - int ls_trtry; - int ls_rbuff; /* receive buff errors */ - int ls_rfram; /* framing */ - int ls_roflo; /* overflow */ - int ls_rcrc; - int ls_rrng; /* rx ring sequence error */ - int ls_babl; /* chip babl error */ - int ls_cerr; /* collision error */ - int ls_miss; /* missed packet */ - int ls_merr; /* memory error */ - int ls_copies; /* copies due to out of range mbufs */ - int ls_maxmbufs; /* max mbufs on transmit */ - int ls_maxslots; /* max ring slots on transmit */ -}; +#define SN_NPAGES 2 + NRBA + (NTDA/2) typedef struct mtd { void *mtd_txp; - int mtd_vtxp; - unsigned char *mtd_buf; + u_int32_t mtd_vtxp; + caddr_t mtd_buf; + u_int32_t mtd_vbuf; + struct mbuf *mtd_mbuf; } mtd_t; /* * The sn_softc for Mac68k if_sn. */ typedef struct sn_softc { - struct device sc_dev; - struct arpcom sc_arpcom; -#define sc_if sc_arpcom.ac_if /* network visible interface */ + struct device sc_dev; + struct arpcom sc_arpcom; +#define sc_if sc_arpcom.ac_if #define sc_enaddr sc_arpcom.ac_enaddr /* hardware ethernet address */ bus_space_tag_t sc_regt; bus_space_handle_t sc_regh; - struct sn_stats sc_sum; - short sc_iflags; - unsigned short bitmode; /* 32 bit mode == 1, 16 == 0 */ + int bitmode; /* 32 bit mode == 1, 16 == 0 */ bus_size_t sc_reg_map[SN_NREGS]; /* register offsets */ u_int16_t snr_dcr; /* DCR for this instance */ u_int16_t snr_dcr2; /* DCR2 for this instance */ - int slotno; /* Slot number */ - - int sc_rxmark; /* pos. in rx ring for reading buffs */ - int sc_rramark; /* index into p_rra of wp */ - void *p_rra[NRRA]; /* RX resource descs */ - int v_rra[NRRA]; /* DMA addresses of p_rra */ - int v_rea; /* ptr to the end of the rra space */ - - int sc_rdamark; - void *p_rda[NRDA]; - int v_rda[NRDA]; + int slotno; /* Slot number */ - caddr_t rbuf[NRBA]; + int sc_rramark; /* index into p_rra of wp */ + void *p_rra[NRRA]; /* RX resource descs */ + u_int32_t v_rra[NRRA]; /* DMA addresses of p_rra */ + u_int32_t v_rea; /* ptr to the end of the rra space */ - int sc_missed; /* missed packet counter */ + int sc_rxmark; /* current hw pos in rda ring */ + int sc_rdamark; /* current sw pos in rda ring */ + int sc_nrda; /* total number of RDAs */ + caddr_t p_rda; + u_int32_t v_rda; - int txb_cnt; /* total number of xmit buffers */ - int txb_inuse; /* number of active xmit buffers */ - int txb_new; /* index of next open slot. */ + caddr_t rbuf[NRBA]; struct mtd mtda[NTDA]; int mtd_hw; /* idx of first mtd given to hw */ @@ -157,12 +133,10 @@ typedef struct sn_softc { * to SONIC. Need to clear EOL on * this word to add a desc. */ - - caddr_t tbuf[NTXB]; - int vtbuf[NTXB]; /* DMA address of tbuf */ + int mtd_pint; /* Counter to set TXP_PINT */ void *p_cda; - int v_cda; + u_int32_t v_cda; unsigned char *space; } sn_softc_t; @@ -239,7 +213,7 @@ typedef struct sn_softc { #define CDA_ENABLE 64 /* mask enabling CAM entries */ #define CDA_SIZE(sc) ((4*16 + 1) * ((sc->bitmode) ? 4 : 2)) -int snsetup __P((struct sn_softc *sc)); -void sn_get_enaddr __P((bus_space_tag_t t, bus_space_handle_t h, - vm_offset_t o, u_char *dst)); +int snsetup __P((struct sn_softc *sc, u_int8_t *)); void snintr __P((void *, int)); +void sn_get_enaddr __P((bus_space_tag_t t, bus_space_handle_t h, + vm_offset_t o, u_char *dst)); |