summaryrefslogtreecommitdiff
path: root/sys/arch/sparc/dev/hme.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/sparc/dev/hme.c')
-rw-r--r--sys/arch/sparc/dev/hme.c374
1 files changed, 324 insertions, 50 deletions
diff --git a/sys/arch/sparc/dev/hme.c b/sys/arch/sparc/dev/hme.c
index 06583bf08e1..317aa69b0f1 100644
--- a/sys/arch/sparc/dev/hme.c
+++ b/sys/arch/sparc/dev/hme.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: hme.c,v 1.4 1998/07/17 21:33:07 jason Exp $ */
+/* $OpenBSD: hme.c,v 1.5 1998/09/01 17:36:59 jason Exp $ */
/*
* Copyright (c) 1998 Jason L. Wright (jason@thought.net)
@@ -72,7 +72,6 @@
#include <sparc/sparc/cpuvar.h>
#include <sparc/dev/sbusvar.h>
#include <sparc/dev/dmareg.h> /* for SBUS_BURST_* */
-#include <sparc/dev/stp2002var.h>
#include <sparc/dev/hmereg.h>
#include <sparc/dev/hmevar.h>
@@ -85,6 +84,7 @@ void hmereset __P((struct hme_softc *));
void hmestart __P((struct ifnet *));
void hmestop __P((struct hme_softc *));
void hmeinit __P((struct hme_softc *));
+void hme_meminit __P((struct hme_softc *));
static void hme_tcvr_write __P((struct hme_softc *, int reg,
u_short val));
@@ -99,6 +99,8 @@ static int hme_tcvr_reset __P((struct hme_softc *));
static void hme_poll_stop __P((struct hme_softc *sc));
+static int hme_rint __P((struct hme_softc *));
+static int hme_tint __P((struct hme_softc *));
static int hme_mint __P((struct hme_softc *, u_int32_t));
static int hme_eint __P((struct hme_softc *, u_int32_t));
@@ -110,7 +112,10 @@ static void hme_set_initial_advertisement __P((struct hme_softc *));
static void hme_reset_rx __P((struct hme_softc *));
static void hme_reset_tx __P((struct hme_softc *));
-static void hme_tx_dmawakeup __P((void *v));
+
+static struct mbuf * hme_get __P((struct hme_softc *, int, int));
+static void hme_read __P((struct hme_softc *, int, int));
+static int hme_put __P((struct hme_softc *, int, struct mbuf *));
static void hme_mcreset __P((struct hme_softc *));
@@ -145,13 +150,14 @@ hmeattach(parent, self, aux)
{
struct confargs *ca = aux;
struct hme_softc *sc = (struct hme_softc *)self;
- struct ifnet *ifp = &sc->sc_stp.stp_arpcom.ac_if;
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
int pri;
/* XXX the following declaration should be elsewhere */
extern void myetheraddr __P((u_char *));
if (ca->ca_ra.ra_nintr != 1) {
- printf(": expected 1 interrupt, got %d\n", ca->ca_ra.ra_nintr);
+ printf(": expected 1 interrupt, got %d\n",
+ ca->ca_ra.ra_nintr);
return;
}
pri = ca->ca_ra.ra_intr[0].int_pri;
@@ -184,8 +190,7 @@ hmeattach(parent, self, aux)
sc->sc_burst = ((struct sbus_softc *)parent)->sc_burst;
- sc->sc_stp.stp_tx_dmawakeup = hme_tx_dmawakeup;
- stp2002_meminit(&sc->sc_stp);
+ hme_meminit(sc);
hme_set_initial_advertisement(sc);
@@ -200,8 +205,8 @@ hmeattach(parent, self, aux)
* Otherwise, use the machine's builtin MAC.
*/
if (getprop(ca->ca_ra.ra_node, "local-mac-address",
- sc->sc_stp.stp_arpcom.ac_enaddr, ETHER_ADDR_LEN) <= 0) {
- myetheraddr(sc->sc_stp.stp_arpcom.ac_enaddr);
+ sc->sc_arpcom.ac_enaddr, ETHER_ADDR_LEN) <= 0) {
+ myetheraddr(sc->sc_arpcom.ac_enaddr);
}
bcopy(sc->sc_dev.dv_xname, ifp->if_xname, IFNAMSIZ);
@@ -220,7 +225,7 @@ hmeattach(parent, self, aux)
sc->sc_an_ticks = 0;
printf(" pri %d: address %s rev %d\n", pri,
- ether_sprintf(sc->sc_stp.stp_arpcom.ac_enaddr), sc->sc_rev);
+ ether_sprintf(sc->sc_arpcom.ac_enaddr), sc->sc_rev);
#if NBPFILTER > 0
bpfattach(&ifp->if_bpf, ifp, DLT_EN10MB, sizeof(struct ether_header));
@@ -240,18 +245,51 @@ void
hmestart(ifp)
struct ifnet *ifp;
{
- struct hme_softc *sc = (struct hme_softc *)ifp->if_softc;
+ struct hme_softc *sc = ifp->if_softc;
+ struct mbuf *m;
+ int bix, len;
- stp2002_start(&sc->sc_stp);
-}
+ if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING)
+ return;
-void
-hme_tx_dmawakeup(v)
- void *v;
-{
- struct hme_softc *sc = (struct hme_softc *)v;
+ bix = sc->sc_last_td;
+
+ for (;;) {
+ IF_DEQUEUE(&ifp->if_snd, m);
+ if (m == 0)
+ break;
+#if NBPFILTER > 0
+ /*
+ * If BPF is listening on this interface, let it see the
+ * packet before we commit it to the wire.
+ */
+ if (ifp->if_bpf)
+ bpf_mtap(ifp->if_bpf, m);
+#endif
+
+ /*
+ * Copy the mbuf chain into the transmit buffer.
+ */
+ len = hme_put(sc, bix, m);
+
+ /*
+ * Initialize transmit registers and start transmission.
+ */
+ sc->sc_desc->hme_txd[bix].tx_flags =
+ HME_TXD_OWN | HME_TXD_SOP | HME_TXD_EOP |
+ (len & HME_TXD_SIZE);
+ sc->sc_txr->tx_pnding = TXR_TP_DMAWAKEUP;
+
+ if (++bix == HME_TX_RING_SIZE)
+ bix = 0;
- sc->sc_txr->tx_pnding = TXR_TP_DMAWAKEUP;
+ if (++sc->sc_no_td == HME_TX_RING_SIZE) {
+ ifp->if_flags |= IFF_OACTIVE;
+ break;
+ }
+ }
+
+ sc->sc_last_td = bix;
}
#define MAX_STOP_TRIES 16
@@ -295,7 +333,7 @@ hmewatchdog(ifp)
struct hme_softc *sc = ifp->if_softc;
log(LOG_ERR, "%s: device timeout\n", sc->sc_dev.dv_xname);
- ++sc->sc_stp.stp_arpcom.ac_if.if_oerrors;
+ ++sc->sc_arpcom.ac_if.if_oerrors;
hmereset(sc);
}
@@ -313,7 +351,7 @@ hmeioctl(ifp, cmd, data)
s = splimp();
- if ((error = ether_ioctl(ifp, &sc->sc_stp.stp_arpcom, cmd, data)) > 0) {
+ if ((error = ether_ioctl(ifp, &sc->sc_arpcom, cmd, data)) > 0) {
splx(s);
return error;
}
@@ -325,7 +363,7 @@ hmeioctl(ifp, cmd, data)
#ifdef INET
case AF_INET:
hmeinit(sc);
- arp_ifinit(&sc->sc_stp.stp_arpcom, ifa);
+ arp_ifinit(&sc->sc_arpcom, ifa);
break;
#endif /* INET */
#ifdef NS
@@ -336,11 +374,11 @@ hmeioctl(ifp, cmd, data)
if (ns_nullhost(*ina))
ina->x_host =
- *(union ns_host *)(sc->sc_stp.stp_arpcom.ac_enaddr);
+ *(union ns_host *)(sc->sc_arpcom.ac_enaddr);
else
bcopy(ina->x_host.c_host,
- sc->sc_stp.stp_arpcom.ac_enaddr,
- sizeof(sc->sc_stp.stp_arpcom.ac_enaddr));
+ sc->sc_arpcom.ac_enaddr,
+ sizeof(sc->sc_arpcom.ac_enaddr));
/* Set new address. */
hmeinit(sc);
break;
@@ -388,8 +426,8 @@ hmeioctl(ifp, cmd, data)
case SIOCADDMULTI:
case SIOCDELMULTI:
error = (cmd == SIOCADDMULTI) ?
- ether_addmulti(ifr, &sc->sc_stp.stp_arpcom):
- ether_delmulti(ifr, &sc->sc_stp.stp_arpcom);
+ ether_addmulti(ifr, &sc->sc_arpcom):
+ ether_delmulti(ifr, &sc->sc_arpcom);
if (error == ENETRESET) {
/*
@@ -408,6 +446,44 @@ hmeioctl(ifp, cmd, data)
}
void
+hme_meminit(sc)
+ struct hme_softc *sc;
+{
+ struct hme_desc *desc;
+ int i;
+
+ if (sc->sc_desc_dva == NULL)
+ sc->sc_desc_dva = (struct hme_desc *) dvma_malloc(
+ sizeof(struct hme_desc), &sc->sc_desc, M_NOWAIT);
+ if (sc->sc_bufs_dva == NULL)
+ sc->sc_bufs_dva = (struct hme_bufs *) dvma_malloc(
+ sizeof(struct hme_bufs), &sc->sc_bufs, M_NOWAIT);
+
+ desc = sc->sc_desc;
+
+ /*
+ * Setup TX descriptors
+ */
+ sc->sc_first_td = sc->sc_last_td = sc->sc_no_td = 0;
+ for (i = 0; i < HME_TX_RING_SIZE; i++) {
+ desc->hme_txd[i].tx_addr =
+ (u_int32_t) &sc->sc_bufs_dva->tx_buf[i][0];
+ desc->hme_txd[i].tx_flags = 0;
+ }
+
+ /*
+ * Setup RX descriptors
+ */
+ sc->sc_last_rd = 0;
+ for (i = 0; i < HME_RX_RING_SIZE; i++) {
+ desc->hme_rxd[i].rx_addr =
+ (u_int32_t) &sc->sc_bufs_dva->rx_buf[i][0];
+ desc->hme_rxd[i].rx_flags = HME_RXD_OWN |
+ ((HME_RX_PKT_BUF_SZ - HME_RX_OFFSET) << 16);
+ }
+}
+
+void
hmeinit(sc)
struct hme_softc *sc;
{
@@ -421,7 +497,7 @@ hmeinit(sc)
hme_poll_stop(sc);
hmestop(sc);
- stp2002_meminit(&sc->sc_stp);
+ hme_meminit(sc);
tcvr->int_mask = 0xffff;
@@ -448,14 +524,14 @@ hmeinit(sc)
hme_reset_tx(sc);
hme_reset_rx(sc);
- cr->rand_seed = sc->sc_stp.stp_arpcom.ac_enaddr[5] |
- ((sc->sc_stp.stp_arpcom.ac_enaddr[4] << 8) & 0x3f00);
- cr->mac_addr0 = (sc->sc_stp.stp_arpcom.ac_enaddr[0] << 8) |
- sc->sc_stp.stp_arpcom.ac_enaddr[1];
- cr->mac_addr1 = (sc->sc_stp.stp_arpcom.ac_enaddr[2] << 8) |
- sc->sc_stp.stp_arpcom.ac_enaddr[3];
- cr->mac_addr2 = (sc->sc_stp.stp_arpcom.ac_enaddr[4] << 8) |
- sc->sc_stp.stp_arpcom.ac_enaddr[5];
+ cr->rand_seed = sc->sc_arpcom.ac_enaddr[5] |
+ ((sc->sc_arpcom.ac_enaddr[4] << 8) & 0x3f00);
+ cr->mac_addr0 = (sc->sc_arpcom.ac_enaddr[0] << 8) |
+ sc->sc_arpcom.ac_enaddr[1];
+ cr->mac_addr1 = (sc->sc_arpcom.ac_enaddr[2] << 8) |
+ sc->sc_arpcom.ac_enaddr[3];
+ cr->mac_addr2 = (sc->sc_arpcom.ac_enaddr[4] << 8) |
+ sc->sc_arpcom.ac_enaddr[5];
cr->jsize = HME_DEFAULT_JSIZE;
cr->ipkt_gap1 = HME_DEFAULT_IPKT_GAP1;
@@ -465,8 +541,8 @@ hmeinit(sc)
cr->htable1 = 0;
cr->htable0 = 0;
- rxr->rx_ring = sc->sc_stp.stp_rx_dvma;
- txr->tx_ring = sc->sc_stp.stp_tx_dvma;
+ rxr->rx_ring = (u_int32_t)&sc->sc_desc_dva->hme_rxd[0];
+ txr->tx_ring = (u_int32_t)&sc->sc_desc_dva->hme_txd[0];
if (sc->sc_burst & SBUS_BURST_64)
gr->cfg = GR_CFG_BURST64;
@@ -482,20 +558,20 @@ hmeinit(sc)
gr->imask = GR_IMASK_SENTFRAME | GR_IMASK_TXPERR |
GR_IMASK_GOTFRAME | GR_IMASK_RCNTEXP;
- txr->tx_rsize = (STP_TX_RING_SIZE >> TXR_RSIZE_SHIFT) - 1;
+ txr->tx_rsize = (HME_TX_RING_SIZE >> TXR_RSIZE_SHIFT) - 1;
txr->cfg |= TXR_CFG_DMAENABLE;
- c = RXR_CFG_DMAENABLE | (STP_RX_OFFSET << 3) | (STP_RX_CSUMLOC << 16);
-#if STP_RX_RING_SIZE == 32
+ c = RXR_CFG_DMAENABLE | (HME_RX_OFFSET << 3) | (HME_RX_CSUMLOC << 16);
+#if HME_RX_RING_SIZE == 32
c |= RXR_CFG_RINGSIZE32;
-#elif STP_RX_RING_SIZE == 64
+#elif HME_RX_RING_SIZE == 64
c |= RXR_CFG_RINGSIZE64;
-#elif STP_RX_RING_SIZE == 128
+#elif HME_RX_RING_SIZE == 128
c |= RXR_CFG_RINGSIZE128;
-#elif STP_RX_RING_SIZE == 256
+#elif HME_RX_RING_SIZE == 256
c |= RXR_CFG_RINGSIZE256;
#else
-#error "STP_RX_RING_SIZE must be 32, 64, 128, or 256."
+#error "HME_RX_RING_SIZE must be 32, 64, 128, or 256."
#endif
rxr->cfg = c;
DELAY(20);
@@ -983,7 +1059,7 @@ hme_negotiate_watchdog(arg)
void *arg;
{
struct hme_softc *sc = (struct hme_softc *)arg;
- struct ifnet *ifp = &sc->sc_stp.stp_arpcom.ac_if;
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
sc->sc_an_ticks++;
switch (sc->sc_an_state) {
@@ -1096,6 +1172,92 @@ hme_mint(sc, why)
}
/*
+ * transmit interrupt
+ */
+static int
+hme_tint(sc)
+ struct hme_softc *sc;
+{
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
+ struct hme_cr *cr = sc->sc_cr;
+ int bix;
+ struct hme_txd txd;
+
+ /*
+ * Get collision counters
+ */
+ ifp->if_collisions += cr->ex_ctr + cr->lt_ctr + cr->fc_ctr + cr->nc_ctr;
+ cr->ex_ctr = 0;
+ cr->lt_ctr = 0;
+ cr->fc_ctr = 0;
+ cr->nc_ctr = 0;
+
+ bix = sc->sc_first_td;
+
+ for (;;) {
+ if (sc->sc_no_td <= 0)
+ break;
+
+ bcopy(&sc->sc_desc->hme_txd[bix], &txd, sizeof(txd));
+
+ if (txd.tx_flags & HME_TXD_OWN)
+ break;
+
+ ifp->if_flags &= ~IFF_OACTIVE;
+ ifp->if_opackets++;
+
+ if (++bix == HME_TX_RING_SIZE)
+ bix = 0;
+
+ --sc->sc_no_td;
+ }
+
+ sc->sc_first_td = bix;
+
+ hmestart(ifp);
+
+ if (sc->sc_no_td == 0)
+ ifp->if_timer = 0;
+
+ return 1;
+}
+
+static int
+hme_rint(sc)
+ struct hme_softc *sc;
+{
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
+ int bix, len;
+ struct hme_rxd rxd;
+
+ bix = sc->sc_last_rd;
+
+ for (;;) {
+ bcopy(&sc->sc_desc->hme_rxd[bix], &rxd, sizeof(rxd));
+ len = rxd.rx_flags >> 16;
+
+ if (rxd.rx_flags & HME_RXD_OWN)
+ break;
+
+ if (rxd.rx_flags & HME_RXD_OVERFLOW)
+ ifp->if_ierrors++;
+ else
+ hme_read(sc, bix, len);
+
+ rxd.rx_flags = HME_RXD_OWN |
+ ((HME_RX_PKT_BUF_SZ - HME_RX_OFFSET) << 16);
+ bcopy(&rxd, &sc->sc_desc->hme_rxd[bix], sizeof(rxd));
+
+ if (++bix == HME_RX_RING_SIZE)
+ bix = 0;
+ }
+
+ sc->sc_last_rd = bix;
+
+ return 1;
+}
+
+/*
* error interrupt
*/
static int
@@ -1192,14 +1354,126 @@ hmeintr(v)
r |= hme_mint(sc, why);
if (why & (GR_STAT_TXALL | GR_STAT_HOSTTOTX))
- r |= stp2002_tint(&sc->sc_stp);
+ r |= hme_tint(sc);
if (why & GR_STAT_RXTOHOST)
- r |= stp2002_rint(&sc->sc_stp);
+ r |= hme_rint(sc);
return (r);
}
+static struct mbuf *
+hme_get(sc, idx, totlen)
+ struct hme_softc *sc;
+ int idx, totlen;
+{
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
+ struct mbuf *m;
+ struct mbuf *top, **mp;
+ int len, pad, boff = 0;
+
+ MGETHDR(m, M_DONTWAIT, MT_DATA);
+ if (m == NULL)
+ return NULL;
+ m->m_pkthdr.rcvif = ifp;
+ m->m_pkthdr.len = totlen;
+ pad = ALIGN(sizeof(struct ether_header)) - sizeof(struct ether_header);
+ m->m_data += pad;
+ len = MHLEN - pad;
+ top = NULL;
+ mp = &top;
+
+ while (totlen > 0) {
+ if (top) {
+ MGET(m, M_DONTWAIT, MT_DATA);
+ if (m == NULL) {
+ m_freem(top);
+ return NULL;
+ }
+ len = MLEN;
+ }
+ if (top && totlen >= MINCLSIZE) {
+ MCLGET(m, M_DONTWAIT);
+ if (m->m_flags & M_EXT)
+ len = MCLBYTES;
+ }
+ m->m_len = len = min(totlen, len);
+ bcopy(&sc->sc_bufs->rx_buf[idx][boff + HME_RX_OFFSET],
+ mtod(m, caddr_t), len);
+ boff += len;
+ totlen -= len;
+ *mp = m;
+ mp = &m->m_next;
+ }
+
+ return top;
+}
+
+static int
+hme_put(sc, idx, m)
+ struct hme_softc *sc;
+ int idx;
+ struct mbuf *m;
+{
+ struct mbuf *n;
+ int len, tlen = 0, boff = 0;
+
+ for (; m; m = n) {
+ len = m->m_len;
+ if (len == 0) {
+ MFREE(m, n);
+ continue;
+ }
+ bcopy(mtod(m, caddr_t), &sc->sc_bufs->tx_buf[idx][boff], len);
+ boff += len;
+ tlen += len;
+ MFREE(m, n);
+ }
+ return tlen;
+}
+
+static void
+hme_read(sc, idx, len)
+ struct hme_softc *sc;
+ int idx, len;
+{
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
+ struct ether_header *eh;
+ struct mbuf *m;
+
+ if (len <= sizeof(struct ether_header) ||
+ len > ETHERMTU + sizeof(struct ether_header)) {
+ printf("%s: invalid packet size %d; dropping\n",
+ ifp->if_xname, len);
+ ifp->if_ierrors++;
+ return;
+ }
+
+ /* Pull packet off interface. */
+ m = hme_get(sc, idx, len);
+ if (m == NULL) {
+ ifp->if_ierrors++;
+ return;
+ }
+
+ ifp->if_ipackets++;
+
+ /* We assume that the header fit entirely in one mbuf. */
+ eh = mtod(m, struct ether_header *);
+
+#if NBPFILTER > 0
+ /*
+ * Check if there's a BPF listener on this interface.
+ * If so, hand off the raw packet to BPF.
+ */
+ if (ifp->if_bpf)
+ bpf_mtap(ifp->if_bpf, m);
+#endif
+ /* Pass the packet up, with the ether header sort-of removed. */
+ m_adj(m, sizeof(struct ether_header));
+ ether_input(ifp, eh, m);
+}
+
/*
* Program the multicast receive filter.
*/
@@ -1207,8 +1481,8 @@ static void
hme_mcreset(sc)
struct hme_softc *sc;
{
- struct arpcom *ac = &sc->sc_stp.stp_arpcom;
- struct ifnet *ifp = &sc->sc_stp.stp_arpcom.ac_if;
+ struct arpcom *ac = &sc->sc_arpcom;
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
struct hme_cr *cr = sc->sc_cr;
struct ether_multi *enm;
struct ether_multistep step;