summaryrefslogtreecommitdiff
path: root/sys/arch
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch')
-rw-r--r--sys/arch/mac68k/dev/if_sn.c623
-rw-r--r--sys/arch/mac68k/dev/if_sn_nubus.c189
-rw-r--r--sys/arch/mac68k/dev/if_sn_obio.c107
-rw-r--r--sys/arch/mac68k/dev/if_snreg.h5
-rw-r--r--sys/arch/mac68k/dev/if_snvar.h116
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 = &top;
+ 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 = &top;
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));