summaryrefslogtreecommitdiff
path: root/sys/arch/sparc/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/sparc/dev')
-rw-r--r--sys/arch/sparc/dev/be.c769
-rw-r--r--sys/arch/sparc/dev/bereg.h415
-rw-r--r--sys/arch/sparc/dev/bevar.h26
3 files changed, 950 insertions, 260 deletions
diff --git a/sys/arch/sparc/dev/be.c b/sys/arch/sparc/dev/be.c
index b2cafbfafb7..83aaa8453f1 100644
--- a/sys/arch/sparc/dev/be.c
+++ b/sys/arch/sparc/dev/be.c
@@ -1,7 +1,8 @@
-/* $OpenBSD: be.c,v 1.5 1998/07/11 05:47:36 deraadt Exp $ */
+/* $OpenBSD: be.c,v 1.6 1998/08/26 05:00:51 jason Exp $ */
/*
- * Copyright (c) 1998 Theo de Raadt. All rights reserved.
+ * Copyright (c) 1998 Theo de Raadt and Jason L. Wright.
+ * All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -11,13 +12,13 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
- * 3. The name of the author may not be used to endorse or promote products
+ * 3. The name of the authors may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
@@ -70,19 +71,29 @@ int bematch __P((struct device *, void *, void *));
void beattach __P((struct device *, struct device *, void *));
void beinit __P((struct besoftc *));
-void be_meminit __P((struct besoftc *));
void bestart __P((struct ifnet *));
void bestop __P((struct besoftc *));
void bewatchdog __P((struct ifnet *));
int beioctl __P((struct ifnet *, u_long, caddr_t));
+void bereset __P((struct besoftc *sc));
int beintr __P((void *));
-int betint __P((struct besoftc *));
int berint __P((struct besoftc *));
-int beqint __P((struct besoftc *));
-int beeint __P((struct besoftc *));
+int betint __P((struct besoftc *));
+int beqint __P((struct besoftc *, u_int32_t));
+int beeint __P((struct besoftc *, u_int32_t));
+int be_put __P((struct besoftc *, int, struct mbuf *));
+void be_read __P((struct besoftc *, int, int));
+struct mbuf * be_get __P((struct besoftc *, int, int));
+
+void be_tcvr_idle __P((struct besoftc *sc));
void be_tcvr_init __P((struct besoftc *sc));
void be_tcvr_setspeed __P((struct besoftc *sc));
+void be_tcvr_write __P((struct besoftc *sc, u_int8_t reg, u_int16_t val));
+void be_tcvr_write_bit __P((struct besoftc *sc, int bit));
+int be_tcvr_read_bit1 __P((struct besoftc *sc));
+int be_tcvr_read_bit2 __P((struct besoftc *sc));
+int be_tcvr_read __P((struct besoftc *sc, u_int8_t reg));
struct cfdriver be_cd = {
NULL, "be", DV_IFNET
@@ -125,9 +136,10 @@ beattach(parent, self, aux)
pri = ca->ca_ra.ra_intr[0].int_pri;
sc->sc_rev = getpropint(ca->ca_ra.ra_node, "board-version", -1);
- sc->sc_cr = mapiodev(ca->ca_ra.ra_reg, 0, sizeof(struct be_cregs));
+ sc->sc_cr = mapiodev(&ca->ca_ra.ra_reg[0], 0, sizeof(struct be_cregs));
sc->sc_br = mapiodev(&ca->ca_ra.ra_reg[1], 0, sizeof(struct be_bregs));
sc->sc_tr = mapiodev(&ca->ca_ra.ra_reg[2], 0, sizeof(struct be_tregs));
+ sc->sc_qec = qec;
sc->sc_qr = qec->sc_regs;
bestop(sc);
@@ -182,11 +194,9 @@ void
bestart(ifp)
struct ifnet *ifp;
{
-printf("start\n");
-#if 0
- struct besoftc *sc = ifp->if_softc;
+ struct besoftc *sc = (struct besoftc *)ifp->if_softc;
struct mbuf *m;
- int bix;
+ int bix, len;
if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING)
return;
@@ -194,11 +204,10 @@ printf("start\n");
bix = sc->sc_last_td;
for (;;) {
- /* XXX TODO: Some magic */
-
IF_DEQUEUE(&ifp->if_snd, m);
if (m == 0)
break;
+
#if NBPFILTER > 0
/*
* If BPF is listening on this interface, let it see the
@@ -208,24 +217,29 @@ printf("start\n");
bpf_mtap(ifp->if_bpf, m);
#endif
- /* XXX TODO
+ /*
* Copy the mbuf chain into the transmit buffer.
*/
- /* XXX TODO
+ len = be_put(sc, bix, m);
+
+ /*
* Initialize transmit registers and start transmission
*/
+ sc->sc_desc->be_txd[bix].tx_flags =
+ BE_TXD_OWN | BE_TXD_SOP | BE_TXD_EOP |
+ (len & BE_TXD_LENGTH);
+ sc->sc_cr->ctrl = BE_CR_CTRL_TWAKEUP;
- if (++bix == TX_RING_SIZE)
+ if (++bix == BE_TX_RING_SIZE)
bix = 0;
- if (++sc->sc_no_td == TX_RING_SIZE) {
+ if (++sc->sc_no_td == BE_TX_RING_SIZE) {
ifp->if_flags |= IFF_OACTIVE;
break;
}
}
sc->sc_last_td = bix;
-#endif
}
void
@@ -245,20 +259,31 @@ bestop(sc)
DELAY(20);
}
+/*
+ * Reset interface.
+ */
+void
+bereset(sc)
+ struct besoftc *sc;
+{
+ int s;
+
+ s = splnet();
+ bestop(sc);
+ beinit(sc);
+ splx(s);
+}
+
void
bewatchdog(ifp)
struct ifnet *ifp;
{
-printf("watchdog\n");
-
-#if 0
struct besoftc *sc = ifp->if_softc;
log(LOG_ERR, "%s: device timeout\n", sc->sc_dev.dv_xname);
++sc->sc_arpcom.ac_if.if_oerrors;
bereset(sc);
-#endif
}
int
@@ -266,50 +291,227 @@ beintr(v)
void *v;
{
struct besoftc *sc = (struct besoftc *)v;
- u_int32_t why;
+ u_int32_t whyq, whyb, whyc;
int r = 0;
- why = sc->sc_qr->stat;
+ whyq = sc->sc_qr->stat; /* qec status */
+ whyc = sc->sc_cr->stat; /* be channel status */
+ whyb = sc->sc_br->stat; /* be status */
+
+ if (whyq & QEC_STAT_BM)
+ r |= beeint(sc, whyb);
+
+ if (whyq & QEC_STAT_ER)
+ r |= beqint(sc, whyc);
- if (why & QEC_STAT_TX)
+ if (whyq & QEC_STAT_TX && whyc & BE_CR_STAT_TXIRQ)
r |= betint(sc);
- if (why & QEC_STAT_RX)
+
+ if (whyq & QEC_STAT_RX && whyc & BE_CR_STAT_RXIRQ)
r |= berint(sc);
- if (why & QEC_STAT_BM)
- r |= beqint(sc);
- if (why & QEC_STAT_ER)
- r |= beeint(sc);
- if (r)
- printf("%s: intr: why=%08x\n", sc->sc_dev.dv_xname, why);
+
return (r);
}
+/*
+ * QEC Interrupt.
+ */
int
-betint(sc)
+beqint(sc, why)
struct besoftc *sc;
+ u_int32_t why;
{
- return (0);
+ int r = 0, rst = 0;
+
+ if (why & BE_CR_STAT_TXIRQ)
+ r |= 1;
+ if (why & BE_CR_STAT_RXIRQ)
+ r |= 1;
+
+ if (why & BE_CR_STAT_BERROR) {
+ r |= 1;
+ rst = 1;
+ printf("%s: bigmac error\n", sc->sc_dev.dv_xname);
+ }
+
+ if (why & BE_CR_STAT_TXDERR) {
+ r |= 1;
+ rst = 1;
+ printf("%s: bogus tx descriptor\n", sc->sc_dev.dv_xname);
+ }
+
+ if (why & (BE_CR_STAT_TXLERR | BE_CR_STAT_TXPERR | BE_CR_STAT_TXSERR)) {
+ r |= 1;
+ rst = 1;
+ printf("%s: tx dma error ( ", sc->sc_dev.dv_xname);
+ if (why & BE_CR_STAT_TXLERR)
+ printf("Late ");
+ if (why & BE_CR_STAT_TXPERR)
+ printf("Parity ");
+ if (why & BE_CR_STAT_TXSERR)
+ printf("Generic ");
+ printf(")\n");
+ }
+
+ if (why & BE_CR_STAT_RXDROP) {
+ r |= 1;
+ rst = 1;
+ printf("%s: out of rx descriptors\n", sc->sc_dev.dv_xname);
+ }
+
+ if (why & BE_CR_STAT_RXSMALL) {
+ r |= 1;
+ rst = 1;
+ printf("%s: rx descriptor too small\n", sc->sc_dev.dv_xname);
+ }
+
+ if (why & (BE_CR_STAT_RXLERR | BE_CR_STAT_RXPERR | BE_CR_STAT_RXSERR)) {
+ r |= 1;
+ rst = 1;
+ printf("%s: rx dma error ( ", sc->sc_dev.dv_xname);
+ if (why & BE_CR_STAT_RXLERR)
+ printf("Late ");
+ if (why & BE_CR_STAT_RXPERR)
+ printf("Parity ");
+ if (why & BE_CR_STAT_RXSERR)
+ printf("Generic ");
+ printf(")\n");
+ }
+
+ if (!r) {
+ rst = 1;
+ printf("%s: unexpected error interrupt %08x\n",
+ sc->sc_dev.dv_xname, why);
+ }
+
+ if (rst) {
+ printf("%s: resetting\n", sc->sc_dev.dv_xname);
+ bereset(sc);
+ }
+
+ return r;
}
+/*
+ * Error interrupt.
+ */
int
-berint(sc)
+beeint(sc, why)
struct besoftc *sc;
+ u_int32_t why;
{
- return (0);
+ int r = 0, rst = 0;
+
+ if (why & BE_BR_STAT_RFIFOVF) {
+ r |= 1;
+ rst = 1;
+ printf("%s: receive fifo overrun\n", sc->sc_dev.dv_xname);
+ }
+ if (why & BE_BR_STAT_TFIFO_UND) {
+ r |= 1;
+ rst = 1;
+ printf("%s: transmit fifo underrun\n", sc->sc_dev.dv_xname);
+ }
+ if (why & BE_BR_STAT_MAXPKTERR) {
+ r |= 1;
+ rst = 1;
+ printf("%s: max packet size error\n", sc->sc_dev.dv_xname);
+ }
+ if (why & BE_BR_STAT_DTIMEXP) {
+ r |= 1;
+ printf("%s: defer timer expired\n", sc->sc_dev.dv_xname);
+ }
+
+ if (!r) {
+ rst = 1;
+ printf("%s: unexpected error interrupt %08x\n",
+ sc->sc_dev.dv_xname, why);
+ }
+
+ if (rst) {
+ printf("%s: resetting\n", sc->sc_dev.dv_xname);
+ bereset(sc);
+ }
+
+ return r;
}
+/*
+ * Transmit interrupt.
+ */
int
-beqint(sc)
+betint(sc)
struct besoftc *sc;
{
- return (0);
+ struct ifnet *ifp = &sc->sc_arpcom.ac_if;
+ int bix;
+ struct be_txd txd;
+
+ bix = sc->sc_first_td;
+
+ for (;;) {
+ if (sc->sc_no_td <= 0)
+ break;
+
+ bcopy(&sc->sc_desc->be_txd[bix], &txd, sizeof(txd));
+
+ if (txd.tx_flags & BE_TXD_OWN)
+ break;
+
+ ifp->if_flags &= ~IFF_OACTIVE;
+ ifp->if_opackets++;
+
+ if (++bix == BE_TX_RING_SIZE)
+ bix = 0;
+
+ --sc->sc_no_td;
+ }
+
+ sc->sc_first_td = bix;
+
+ bestart(ifp);
+
+ if (sc->sc_no_td == 0)
+ ifp->if_timer = 0;
+
+ return 1;
}
+/*
+ * Receive interrupt.
+ */
int
-beeint(sc)
+berint(sc)
struct besoftc *sc;
{
- return (0);
+ int bix, len;
+ struct be_rxd rxd;
+
+ bix = sc->sc_last_rd;
+
+ /*
+ * Process all buffers with valid data.
+ */
+ for (;;) {
+ bcopy(&sc->sc_desc->be_rxd[bix], &rxd, sizeof(rxd));
+
+ if (rxd.rx_flags & BE_RXD_OWN)
+ break;
+
+ len = rxd.rx_flags & BE_RXD_LENGTH;
+ be_read(sc, bix, len);
+
+ rxd.rx_flags = BE_RXD_OWN | (BE_PKT_BUF_SZ & BE_RXD_LENGTH);
+
+ bcopy(&rxd, &sc->sc_desc->be_rxd[bix], sizeof(rxd));
+
+ if (++bix == BE_RX_RING_SIZE)
+ bix = 0;
+ }
+
+ sc->sc_last_rd = bix;
+
+ return 1;
}
int
@@ -422,18 +624,6 @@ beioctl(ifp, cmd, data)
}
void
-be_tcvr_init(sc)
- struct besoftc *sc;
-{
-}
-
-void
-be_tcvr_setspeed(sc)
- struct besoftc *sc;
-{
-}
-
-void
beinit(sc)
struct besoftc *sc;
{
@@ -441,13 +631,15 @@ beinit(sc)
int s = splimp();
int i;
- bestop(sc);
+ qec_reset(sc->sc_qec);
- /* init QEC */
+ /*
+ * init QEC: 'be' specific initializations
+ */
sc->sc_qr->msize = sc->sc_memsize;
sc->sc_qr->rsize = sc->sc_memsize / 2;
sc->sc_qr->tsize = sc->sc_memsize / 2;
- sc->sc_qr->psize = 2048;
+ sc->sc_qr->psize = QEC_PSIZE_2048;
if (sc->sc_burst & SBUS_BURST_64)
i = QEC_CTRL_B64;
else if (sc->sc_burst & SBUS_BURST_32)
@@ -456,28 +648,31 @@ beinit(sc)
i = QEC_CTRL_B16;
sc->sc_qr->ctrl = QEC_CTRL_BMODE | i;
- /* Allocate memory if not done yet */
- if (sc->sc_desc_dva == NULL)
- sc->sc_desc_dva = (struct be_desc *)dvma_malloc(
- sizeof(struct be_desc), &sc->sc_desc, M_NOWAIT);
- if (sc->sc_bufs_dva == NULL)
- sc->sc_bufs_dva = (struct be_bufs *)dvma_malloc(
- sizeof(struct be_bufs), &sc->sc_bufs, M_NOWAIT);
-
- /* chain descriptors into buffers */
- sc->sc_txnew = 0;
- sc->sc_rxnew = 0;
- sc->sc_txold = 0;
- sc->sc_rxold = 0;
- for (i = 0; i < RX_RING_SIZE; i++) {
- sc->sc_desc->be_rxd[i].rx_addr =
- (u_int32_t)&sc->sc_bufs_dva->rx_buf[i][0];
- sc->sc_desc->be_rxd[i].rx_flags = 0;
- }
- for (i = 0; i < TX_RING_SIZE; i++) {
- sc->sc_desc->be_txd[i].tx_addr = 0;
+ /*
+ * Allocate descriptor ring and buffers, if not already done
+ */
+ if (sc->sc_desc == NULL)
+ sc->sc_desc_dva = (struct be_desc *) dvma_malloc(
+ sizeof(struct be_desc), &sc->sc_desc, M_NOWAIT);
+ if (sc->sc_bufs == NULL)
+ sc->sc_bufs_dva = (struct be_bufs *) dvma_malloc(
+ sizeof(struct be_bufs), &sc->sc_bufs, M_NOWAIT);
+
+ for (i = 0; i < BE_TX_RING_SIZE; i++) {
+ sc->sc_desc->be_txd[i].tx_addr =
+ (u_int32_t) &sc->sc_bufs_dva->tx_buf[i][0];
sc->sc_desc->be_txd[i].tx_flags = 0;
}
+ for (i = 0; i < BE_RX_RING_SIZE; i++) {
+ sc->sc_desc->be_rxd[i].rx_addr =
+ (u_int32_t) &sc->sc_bufs_dva->rx_buf[i][0];
+ sc->sc_desc->be_rxd[i].rx_flags =
+ BE_RXD_OWN |
+ (BE_PKT_BUF_SZ & BE_RXD_LENGTH);
+ }
+
+ sc->sc_first_td = sc->sc_last_td = sc->sc_no_td = 0;
+ sc->sc_last_rd = 0;
be_tcvr_init(sc);
be_tcvr_setspeed(sc);
@@ -496,23 +691,38 @@ beinit(sc)
sc->sc_br->htable1 = 0;
sc->sc_br->htable0 = 0;
- sc->sc_br->rx_cfg = BE_RXCFG_HENABLE | BE_RXCFG_FIFO;
+ sc->sc_br->rx_cfg = BE_BR_RXCFG_HENABLE | BE_BR_RXCFG_FIFO;
DELAY(20);
- sc->sc_br->tx_cfg = BE_TXCFG_FIFO;
+ sc->sc_br->tx_cfg = BE_BR_TXCFG_FIFO;
sc->sc_br->rand_seed = 0xbd;
- sc->sc_br->xif_cfg = BE_XCFG_ODENABLE | BE_XCFG_RESV;
+ sc->sc_br->xif_cfg = BE_BR_XCFG_ODENABLE | BE_BR_XCFG_RESV;
- sc->sc_cr->rxds = (u_int32_t)&sc->sc_desc_dva->be_rxd[0];
- sc->sc_cr->txds = (u_int32_t)&sc->sc_desc_dva->be_txd[0];
+ sc->sc_cr->rxds = (u_int32_t) &sc->sc_desc_dva->be_rxd[0];
+ sc->sc_cr->txds = (u_int32_t) &sc->sc_desc_dva->be_txd[0];
sc->sc_cr->rxwbufptr = 0;
sc->sc_cr->rxrbufptr = 0;
- sc->sc_cr->txwbufptr = sc->sc_memsize;
- sc->sc_cr->txrbufptr = sc->sc_memsize;
-
- sc->sc_br->imask = 0;
+ sc->sc_cr->txwbufptr = sc->sc_qr->tsize;
+ sc->sc_cr->txrbufptr = sc->sc_qr->tsize;
+
+ /*
+ * Turn off counter expiration interrupts as well as
+ * 'gotframe' and 'sentframe'
+ */
+ sc->sc_br->imask = BE_BR_IMASK_GOTFRAME |
+ BE_BR_IMASK_RCNTEXP |
+ BE_BR_IMASK_ACNTEXP |
+ BE_BR_IMASK_CCNTEXP |
+ BE_BR_IMASK_LCNTEXP |
+ BE_BR_IMASK_CVCNTEXP |
+ BE_BR_IMASK_NCNTEXP |
+ BE_BR_IMASK_ECNTEXP |
+ BE_BR_IMASK_LCCNTEXP |
+ BE_BR_IMASK_LCNTEXP |
+ BE_BR_IMASK_LCNTEXP |
+ BE_BR_IMASK_SENTFRAME;
sc->sc_cr->rimask = 0;
sc->sc_cr->timask = 0;
@@ -523,10 +733,391 @@ beinit(sc)
sc->sc_cr->ccnt = 0;
- sc->sc_br->tx_cfg |= BE_TXCFG_ENABLE;
- sc->sc_br->rx_cfg |= BE_RXCFG_ENABLE;
+ sc->sc_br->tx_cfg |= BE_BR_TXCFG_ENABLE;
+ sc->sc_br->rx_cfg |= BE_BR_RXCFG_ENABLE;
ifp->if_flags |= IFF_RUNNING;
ifp->if_flags &= ~IFF_OACTIVE;
splx(s);
}
+
+void
+be_tcvr_setspeed(sc)
+ struct besoftc *sc;
+{
+ int x;
+
+ be_tcvr_write(sc, PHY_BMCR, 0);
+ x = be_tcvr_read(sc, PHY_BMSR);
+}
+
+/*
+ * Set the tcvr to an idle state
+ */
+void
+be_tcvr_idle(sc)
+ struct besoftc *sc;
+{
+ struct be_tregs *tr = sc->sc_tr;
+ volatile u_int32_t x;
+ int i = 20;
+
+ while (i--) {
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO | MGMT_PAL_EXT_MDIO |
+ MGMT_PAL_OENAB;
+ x = tr->mgmt_pal;
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO | MGMT_PAL_EXT_MDIO |
+ MGMT_PAL_OENAB | MGMT_PAL_DCLOCK;
+ x = tr->mgmt_pal;
+ }
+}
+
+/*
+ * Initialize the transceiver and figure out whether we're using the
+ * external or internal one.
+ */
+void
+be_tcvr_init(sc)
+ struct besoftc *sc;
+{
+ volatile u_int32_t x;
+ struct be_tregs *tr = sc->sc_tr;
+
+ be_tcvr_idle(sc);
+
+ if (sc->sc_rev != 1) {
+ printf("%s: rev %d PAL not supported.\n",
+ sc->sc_dev.dv_xname,
+ sc->sc_rev);
+ return;
+ }
+
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO | MGMT_PAL_EXT_MDIO | MGMT_PAL_DCLOCK;
+ x = tr->mgmt_pal;
+
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO | MGMT_PAL_EXT_MDIO;
+ x = tr->mgmt_pal;
+ DELAY(200);
+
+ if (tr->mgmt_pal & MGMT_PAL_EXT_MDIO) {
+ sc->sc_tcvr_type = BE_TCVR_EXTERNAL;
+ tr->tcvr_pal = ~(TCVR_PAL_EXTLBACK | TCVR_PAL_MSENSE |
+ TCVR_PAL_LTENABLE);
+ x = tr->tcvr_pal;
+ }
+ else if (tr->mgmt_pal & MGMT_PAL_INT_MDIO) {
+ sc->sc_tcvr_type = BE_TCVR_INTERNAL;
+ tr->tcvr_pal = ~(TCVR_PAL_EXTLBACK | TCVR_PAL_MSENSE |
+ TCVR_PAL_LTENABLE | TCVR_PAL_SERIAL);
+ x = tr->tcvr_pal;
+ }
+ else {
+ printf("%s: no internal or external transceiver found.\n",
+ sc->sc_dev.dv_xname);
+ }
+}
+
+int
+be_tcvr_read(sc, reg)
+ struct besoftc *sc;
+ u_int8_t reg;
+{
+ int phy, i;
+ u_int32_t ret = 0;
+
+ if (sc->sc_tcvr_type == BE_TCVR_INTERNAL)
+ phy = BE_PHY_INTERNAL;
+ else if (sc->sc_tcvr_type == BE_TCVR_EXTERNAL)
+ phy = BE_PHY_EXTERNAL;
+ else {
+ printf("%s: invalid tcvr type\n", sc->sc_dev.dv_xname);
+ return BE_TCVR_READ_INVALID;
+ }
+
+ be_tcvr_idle(sc);
+
+ be_tcvr_write_bit(sc, 0);
+ be_tcvr_write_bit(sc, 1);
+ be_tcvr_write_bit(sc, 1);
+ be_tcvr_write_bit(sc, 0);
+
+ for (i = 4; i >= 0; i--)
+ be_tcvr_write_bit(sc, (phy >> i) & 1);
+
+ for (i = 4; i >= 0; i--)
+ be_tcvr_write_bit(sc, (reg >> i) & 1);
+
+ if (sc->sc_tcvr_type == BE_TCVR_EXTERNAL) {
+ (void) be_tcvr_read_bit2(sc);
+ (void) be_tcvr_read_bit2(sc);
+
+ for (i = 15; i >= 0; i--) {
+ int b;
+
+ b = be_tcvr_read_bit2(sc);
+ ret |= (b & 1) << i;
+ }
+
+ (void) be_tcvr_read_bit2(sc);
+ (void) be_tcvr_read_bit2(sc);
+ (void) be_tcvr_read_bit2(sc);
+ }
+ else {
+ (void) be_tcvr_read_bit1(sc);
+ (void) be_tcvr_read_bit1(sc);
+
+ for (i = 15; i >= 0; i--) {
+ int b;
+
+ b = be_tcvr_read_bit1(sc);
+ ret |= (b & 1) << i;
+ }
+
+ (void) be_tcvr_read_bit1(sc);
+ (void) be_tcvr_read_bit1(sc);
+ (void) be_tcvr_read_bit1(sc);
+ }
+ return ret;
+}
+
+int
+be_tcvr_read_bit1(sc)
+ struct besoftc *sc;
+{
+ volatile u_int32_t x;
+ struct be_tregs *tr = sc->sc_tr;
+ int ret = 0;
+
+ if (sc->sc_tcvr_type == BE_TCVR_INTERNAL) {
+ tr->mgmt_pal = MGMT_PAL_EXT_MDIO;
+ x = tr->mgmt_pal;
+ tr->mgmt_pal = MGMT_PAL_EXT_MDIO | MGMT_PAL_DCLOCK;
+ x = tr->mgmt_pal;
+ DELAY(20);
+ ret = (tr->mgmt_pal & MGMT_PAL_INT_MDIO) >> 3;
+ } else if (sc->sc_tcvr_type == BE_TCVR_EXTERNAL) {
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO;
+ x = tr->mgmt_pal;
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO | MGMT_PAL_DCLOCK;
+ x = tr->mgmt_pal;
+ DELAY(20);
+ ret = (tr->mgmt_pal & MGMT_PAL_EXT_MDIO) >> 2;
+ } else {
+ printf("%s: invalid tcvr type\n", sc->sc_dev.dv_xname);
+ }
+ return (ret & 1);
+}
+
+int
+be_tcvr_read_bit2(sc)
+ struct besoftc *sc;
+{
+ volatile u_int32_t x;
+ struct be_tregs *tr = sc->sc_tr;
+ int ret = 0;
+
+ if (sc->sc_tcvr_type == BE_TCVR_INTERNAL) {
+ tr->mgmt_pal = MGMT_PAL_EXT_MDIO;
+ x = tr->mgmt_pal;
+ DELAY(20);
+ ret = (tr->mgmt_pal & MGMT_PAL_INT_MDIO) >> 3;
+ tr->mgmt_pal = MGMT_PAL_EXT_MDIO | MGMT_PAL_DCLOCK;
+ x = tr->mgmt_pal;
+ } else if (sc->sc_tcvr_type == BE_TCVR_EXTERNAL) {
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO;
+ x = tr->mgmt_pal;
+ DELAY(20);
+ ret = (tr->mgmt_pal & MGMT_PAL_EXT_MDIO) >> 2;
+ tr->mgmt_pal = MGMT_PAL_INT_MDIO | MGMT_PAL_DCLOCK;
+ x = tr->mgmt_pal;
+ } else {
+ printf("%s: invalid tcvr type\n", sc->sc_dev.dv_xname);
+ }
+ return ret;
+}
+
+void
+be_tcvr_write(sc, reg, val)
+ struct besoftc *sc;
+ u_int8_t reg;
+ u_int16_t val;
+{
+ int phy, i;
+
+ if (sc->sc_tcvr_type == BE_TCVR_INTERNAL)
+ phy = BE_PHY_INTERNAL;
+ else if (sc->sc_tcvr_type == BE_TCVR_EXTERNAL)
+ phy = BE_PHY_EXTERNAL;
+ else {
+ printf("%s: invalid tcvr type\n", sc->sc_dev.dv_xname);
+ return;
+ }
+
+ be_tcvr_idle(sc);
+
+ be_tcvr_write_bit(sc, 0);
+ be_tcvr_write_bit(sc, 1);
+ be_tcvr_write_bit(sc, 0);
+ be_tcvr_write_bit(sc, 1);
+
+ for (i = 4; i >= 0; i--)
+ be_tcvr_write_bit(sc, (phy >> i) & 1);
+
+ for (i = 4; i >= 0; i--)
+ be_tcvr_write_bit(sc, (reg >> i) & 1);
+
+ be_tcvr_write_bit(sc, 1);
+ be_tcvr_write_bit(sc, 0);
+
+ for (i = 15; i >= 0; i--)
+ be_tcvr_write_bit(sc, (val >> i) & 1);
+}
+
+void
+be_tcvr_write_bit(sc, bit)
+ struct besoftc *sc;
+ int bit;
+{
+ volatile u_int32_t x;
+
+ if (sc->sc_tcvr_type == BE_TCVR_INTERNAL) {
+ bit = ((bit & 1) << 3) | MGMT_PAL_OENAB | MGMT_PAL_EXT_MDIO;
+ sc->sc_tr->mgmt_pal = bit;
+ x = sc->sc_tr->mgmt_pal;
+ sc->sc_tr->mgmt_pal = bit | MGMT_PAL_DCLOCK;
+ x = sc->sc_tr->mgmt_pal;
+ } else {
+ bit = ((bit & 1) << 2) | MGMT_PAL_OENAB | MGMT_PAL_INT_MDIO;
+ sc->sc_tr->mgmt_pal = bit;
+ x = sc->sc_tr->mgmt_pal;
+ sc->sc_tr->mgmt_pal = bit | MGMT_PAL_DCLOCK;
+ x = sc->sc_tr->mgmt_pal;
+ }
+}
+
+/*
+ * Routine to copy from mbuf chain to transmit buffer in
+ * network buffer memory.
+ */
+int
+be_put(sc, idx, m)
+ struct besoftc *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;
+}
+
+/*
+ * Pass a packet to the higher levels.
+ */
+void
+be_read(sc, idx, len)
+ struct besoftc *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 = be_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);
+}
+
+/*
+ * Pull data off an interface.
+ * Len is the length of data, with local net header stripped.
+ * We copy the data into mbufs. When full cluster sized units are present,
+ * we copy into clusters.
+ */
+struct mbuf *
+be_get(sc, idx, totlen)
+ struct besoftc *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], mtod(m, caddr_t), len);
+ boff += len;
+ totlen -= len;
+ *mp = m;
+ mp = &m->m_next;
+ }
+
+ return (top);
+}
diff --git a/sys/arch/sparc/dev/bereg.h b/sys/arch/sparc/dev/bereg.h
index 408a1e94407..8097d808bd6 100644
--- a/sys/arch/sparc/dev/bereg.h
+++ b/sys/arch/sparc/dev/bereg.h
@@ -1,7 +1,8 @@
-/* $OpenBSD: bereg.h,v 1.2 1998/07/05 09:25:55 deraadt Exp $ */
+/* $OpenBSD: bereg.h,v 1.3 1998/08/26 05:00:53 jason Exp $ */
/*
- * Copyright (c) 1998 Theo de Raadt. All rights reserved.
+ * Copyright (c) 1998 Theo de Raadt and Jason L. Wright.
+ * All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -11,13 +12,13 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
- * 3. The name of the author may not be used to endorse or promote products
+ * 3. The name of the authors may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
@@ -26,180 +27,278 @@
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
+/*
+ * BE Global registers
+ */
struct be_bregs {
- volatile u_int32_t xif_cfg; /* XIF config register */
- volatile u_int32_t _unused[63]; /* Reserved... */
- volatile u_int32_t status; /* Status register, clear on read */
- volatile u_int32_t imask; /* Interrupt mask register */
- volatile u_int32_t _unused2[64]; /* Reserved... */
- volatile u_int32_t tx_swreset; /* Transmitter software reset */
- volatile u_int32_t tx_cfg; /* Transmitter config register */
- volatile u_int32_t ipkt_gap1; /* Inter-packet gap 1 */
- volatile u_int32_t ipkt_gap2; /* Inter-packet gap 2 */
- volatile u_int32_t attempt_limit; /* Transmit attempt limit */
- volatile u_int32_t stime; /* Transmit slot time */
- volatile u_int32_t preamble_len; /* Size of transmit preamble */
- volatile u_int32_t preamble_pattern; /* Pattern for transmit preamble */
- volatile u_int32_t tx_sframe_delim; /* Transmit delimiter */
- volatile u_int32_t jsize; /* Toe jam... */
- volatile u_int32_t tx_pkt_max; /* Transmit max pkt size */
- volatile u_int32_t tx_pkt_min; /* Transmit min pkt size */
- volatile u_int32_t peak_attempt; /* Count of transmit peak attempts */
- volatile u_int32_t dt_ctr; /* Transmit defer timer */
- volatile u_int32_t nc_ctr; /* Transmit normal-collision counter */
- volatile u_int32_t fc_ctr; /* Transmit first-collision counter */
- volatile u_int32_t ex_ctr; /* Transmit excess-collision counter */
- volatile u_int32_t lt_ctr; /* Transmit late-collision counter */
- volatile u_int32_t rand_seed; /* Transmit random number seed */
- volatile u_int32_t tx_smachine; /* Transmit state machine */
- volatile u_int32_t _unused3[44]; /* Reserved */
- volatile u_int32_t rx_swreset; /* Receiver software reset */
- volatile u_int32_t rx_cfg; /* Receiver config register */
- volatile u_int32_t rx_pkt_max; /* Receive max pkt size */
- volatile u_int32_t rx_pkt_min; /* Receive min pkt size */
- volatile u_int32_t mac_addr2; /* Ether address register 2 */
- volatile u_int32_t mac_addr1; /* Ether address register 1 */
- volatile u_int32_t mac_addr0; /* Ether address register 0 */
- volatile u_int32_t fr_ctr; /* Receive frame receive counter */
- volatile u_int32_t gle_ctr; /* Receive giant-length error counter */
- volatile u_int32_t unale_ctr; /* Receive unaligned error counter */
- volatile u_int32_t rcrce_ctr; /* Receive CRC error counter */
- volatile u_int32_t rx_smachine; /* Receiver state machine */
- volatile u_int32_t rx_cvalid; /* Receiver code violation */
- volatile u_int32_t _unused4; /* Reserved... */
- volatile u_int32_t htable3; /* Hash table 3 */
- volatile u_int32_t htable2; /* Hash table 2 */
- volatile u_int32_t htable1; /* Hash table 1 */
- volatile u_int32_t htable0; /* Hash table 0 */
- volatile u_int32_t afilter2; /* Address filter 2 */
- volatile u_int32_t afilter1; /* Address filter 1 */
- volatile u_int32_t afilter0; /* Address filter 0 */
- volatile u_int32_t afilter_mask; /* Address filter mask */
+ volatile u_int32_t xif_cfg; /* XIF config */
+ volatile u_int32_t _unused[63]; /* reserved */
+ volatile u_int32_t stat; /* status, clear on read */
+ volatile u_int32_t imask; /* interrupt mask */
+ volatile u_int32_t _unused2[64]; /* reserved */
+ volatile u_int32_t tx_swreset; /* tx software reset */
+ volatile u_int32_t tx_cfg; /* tx config */
+ volatile u_int32_t ipkt_gap1; /* inter-packet gap 1 */
+ volatile u_int32_t ipkt_gap2; /* inter-packet gap 2 */
+ volatile u_int32_t attempt_limit; /* tx attempt limit */
+ volatile u_int32_t stime; /* tx slot time */
+ volatile u_int32_t preamble_len; /* size of tx preamble */
+ volatile u_int32_t preamble_pattern; /* pattern for tx preamble */
+ volatile u_int32_t tx_sframe_delim; /* tx delimiter */
+ volatile u_int32_t jsize; /* jam length */
+ volatile u_int32_t tx_pkt_max; /* tx max pkt size */
+ volatile u_int32_t tx_pkt_min; /* tx min pkt size */
+ volatile u_int32_t peak_attempt; /* count of tx peak attempts */
+ volatile u_int32_t dt_ctr; /* tx defer timer */
+ volatile u_int32_t nc_ctr; /* tx normal collision cntr */
+ volatile u_int32_t fc_ctr; /* tx first-collision cntr */
+ volatile u_int32_t ex_ctr; /* tx excess-collision cntr */
+ volatile u_int32_t lt_ctr; /* tx late-collision cntr */
+ volatile u_int32_t rand_seed; /* tx random number seed */
+ volatile u_int32_t tx_smachine; /* tx state machine */
+ volatile u_int32_t _unused3[44]; /* reserved */
+ volatile u_int32_t rx_swreset; /* rx software reset */
+ volatile u_int32_t rx_cfg; /* rx config register */
+ volatile u_int32_t rx_pkt_max; /* rx max pkt size */
+ volatile u_int32_t rx_pkt_min; /* rx min pkt size */
+ volatile u_int32_t mac_addr2; /* ethernet address 2 (MSB) */
+ volatile u_int32_t mac_addr1; /* ethernet address 1 */
+ volatile u_int32_t mac_addr0; /* ethernet address 0 (LSB) */
+ volatile u_int32_t fr_ctr; /* rx frame receive cntr */
+ volatile u_int32_t gle_ctr; /* rx giant-len error cntr */
+ volatile u_int32_t unale_ctr; /* rx unaligned error cntr */
+ volatile u_int32_t rcrce_ctr; /* rx CRC error cntr */
+ volatile u_int32_t rx_smachine; /* rx state machine */
+ volatile u_int32_t rx_cvalid; /* rx code violation */
+ volatile u_int32_t _unused4; /* reserved */
+ volatile u_int32_t htable3; /* hash table 3 */
+ volatile u_int32_t htable2; /* hash table 2 */
+ volatile u_int32_t htable1; /* hash table 1 */
+ volatile u_int32_t htable0; /* hash table 0 */
+ volatile u_int32_t afilter2; /* address filter 2 */
+ volatile u_int32_t afilter1; /* address filter 1 */
+ volatile u_int32_t afilter0; /* address filter 0 */
+ volatile u_int32_t afilter_mask; /* address filter mask */
};
-/* BE XIF config register. */
-#define BE_XCFG_ODENABLE 0x00000001 /* Output driver enable */
-#define BE_XCFG_RESV 0x00000002 /* Reserved, write always as 1 */
-#define BE_XCFG_MLBACK 0x00000004 /* Loopback-mode MII enable */
-#define BE_XCFG_SMODE 0x00000008 /* Enable serial mode */
-
-/* BE status register. */
-#define BE_STAT_GOTFRAME 0x00000001 /* Received a frame */
-#define BE_STAT_RCNTEXP 0x00000002 /* Receive frame counter expired */
-#define BE_STAT_ACNTEXP 0x00000004 /* Align-error counter expired */
-#define BE_STAT_CCNTEXP 0x00000008 /* CRC-error counter expired */
-#define BE_STAT_LCNTEXP 0x00000010 /* Length-error counter expired */
-#define BE_STAT_RFIFOVF 0x00000020 /* Receive FIFO overflow */
-#define BE_STAT_CVCNTEXP 0x00000040 /* Code-violation counter expired */
-#define BE_STAT_SENTFRAME 0x00000100 /* Transmitted a frame */
-#define BE_STAT_TFIFO_UND 0x00000200 /* Transmit FIFO underrun */
-#define BE_STAT_MAXPKTERR 0x00000400 /* Max-packet size error */
-#define BE_STAT_NCNTEXP 0x00000800 /* Normal-collision counter expired */
-#define BE_STAT_ECNTEXP 0x00001000 /* Excess-collision counter expired */
-#define BE_STAT_LCCNTEXP 0x00002000 /* Late-collision counter expired */
-#define BE_STAT_FCNTEXP 0x00004000 /* First-collision counter expired */
-#define BE_STAT_DTIMEXP 0x00008000 /* Defer-timer expired */
-
-/* BE interrupt mask register. */
-#define BE_IMASK_GOTFRAME 0x00000001 /* Received a frame */
-#define BE_IMASK_RCNTEXP 0x00000002 /* Receive frame counter expired */
-#define BE_IMASK_ACNTEXP 0x00000004 /* Align-error counter expired */
-#define BE_IMASK_CCNTEXP 0x00000008 /* CRC-error counter expired */
-#define BE_IMASK_LCNTEXP 0x00000010 /* Length-error counter expired */
-#define BE_IMASK_RFIFOVF 0x00000020 /* Receive FIFO overflow */
-#define BE_IMASK_CVCNTEXP 0x00000040 /* Code-violation counter expired */
-#define BE_IMASK_SENTFRAME 0x00000100 /* Transmitted a frame */
-#define BE_IMASK_TFIFO_UND 0x00000200 /* Transmit FIFO underrun */
-#define BE_IMASK_MAXPKTERR 0x00000400 /* Max-packet size error */
-#define BE_IMASK_NCNTEXP 0x00000800 /* Normal-collision counter expired */
-#define BE_IMASK_ECNTEXP 0x00001000 /* Excess-collision counter expired */
-#define BE_IMASK_LCCNTEXP 0x00002000 /* Late-collision counter expired */
-#define BE_IMASK_FCNTEXP 0x00004000 /* First-collision counter expired */
-#define BE_IMASK_DTIMEXP 0x00008000 /* Defer-timer expired */
-
-/* BE transmit config register. */
-#define BE_TXCFG_ENABLE 0x00000001 /* Enable the transmitter */
-#define BE_TXCFG_FIFO 0x00000010 /* Default tx fthresh... */
-#define BE_TXCFG_SMODE 0x00000020 /* Enable slow transmit mode */
-#define BE_TXCFG_CIGN 0x00000040 /* Ignore transmit collisions */
-#define BE_TXCFG_FCSOFF 0x00000080 /* Do not emit FCS */
-#define BE_TXCFG_DBACKOFF 0x00000100 /* Disable backoff */
-#define BE_TXCFG_FULLDPLX 0x00000200 /* Enable full-duplex */
-
-/* BE receive config register. */
-#define BE_RXCFG_ENABLE 0x00000001 /* Enable the receiver */
-#define BE_RXCFG_FIFO 0x0000000e /* Default rx fthresh... */
-#define BE_RXCFG_PSTRIP 0x00000020 /* Pad byte strip enable */
-#define BE_RXCFG_PMISC 0x00000040 /* Enable promiscous mode */
-#define BE_RXCFG_DERR 0x00000080 /* Disable error checking */
-#define BE_RXCFG_DCRCS 0x00000100 /* Disable CRC stripping */
-#define BE_RXCFG_ME 0x00000200 /* Receive packets addressed to me */
-#define BE_RXCFG_PGRP 0x00000400 /* Enable promisc group mode */
-#define BE_RXCFG_HENABLE 0x00000800 /* Enable the hash filter */
-#define BE_RXCFG_AENABLE 0x00001000 /* Enable the address filter */
+/* be_bregs.xif_cfg: XIF config. */
+#define BE_BR_XCFG_ODENABLE 0x00000001 /* output driver enable */
+#define BE_BR_XCFG_RESV 0x00000002 /* reserved, write as 1 */
+#define BE_BR_XCFG_MLBACK 0x00000004 /* loopback-mode mii enable */
+#define BE_BR_XCFG_SMODE 0x00000008 /* enable serial mode */
+
+/* be_bregs.stat: status, clear on read. */
+#define BE_BR_STAT_GOTFRAME 0x00000001 /* received a frame */
+#define BE_BR_STAT_RCNTEXP 0x00000002 /* rx frame cntr expired */
+#define BE_BR_STAT_ACNTEXP 0x00000004 /* align-error cntr expired */
+#define BE_BR_STAT_CCNTEXP 0x00000008 /* crc-error cntr expired */
+#define BE_BR_STAT_LCNTEXP 0x00000010 /* length-error cntr expired */
+#define BE_BR_STAT_RFIFOVF 0x00000020 /* rx fifo overflow */
+#define BE_BR_STAT_CVCNTEXP 0x00000040 /* code-violation cntr exprd */
+#define BE_BR_STAT_SENTFRAME 0x00000100 /* transmitted a frame */
+#define BE_BR_STAT_TFIFO_UND 0x00000200 /* tx fifo underrun */
+#define BE_BR_STAT_MAXPKTERR 0x00000400 /* max-packet size error */
+#define BE_BR_STAT_NCNTEXP 0x00000800 /* normal-collision cntr exp */
+#define BE_BR_STAT_ECNTEXP 0x00001000 /* excess-collision cntr exp */
+#define BE_BR_STAT_LCCNTEXP 0x00002000 /* late-collision cntr exp */
+#define BE_BR_STAT_FCNTEXP 0x00004000 /* first-collision cntr exp */
+#define BE_BR_STAT_DTIMEXP 0x00008000 /* defer-timer expired */
+
+/* be_bregs.imask: interrupt mask. */
+#define BE_BR_IMASK_GOTFRAME 0x00000001 /* received a frame */
+#define BE_BR_IMASK_RCNTEXP 0x00000002 /* rx frame cntr expired */
+#define BE_BR_IMASK_ACNTEXP 0x00000004 /* align-error cntr expired */
+#define BE_BR_IMASK_CCNTEXP 0x00000008 /* crc-error cntr expired */
+#define BE_BR_IMASK_LCNTEXP 0x00000010 /* length-error cntr expired */
+#define BE_BR_IMASK_RFIFOVF 0x00000020 /* rx fifo overflow */
+#define BE_BR_IMASK_CVCNTEXP 0x00000040 /* code-violation cntr exprd */
+#define BE_BR_IMASK_SENTFRAME 0x00000100 /* transmitted a frame */
+#define BE_BR_IMASK_TFIFO_UND 0x00000200 /* tx fifo underrun */
+#define BE_BR_IMASK_MAXPKTERR 0x00000400 /* max-packet size error */
+#define BE_BR_IMASK_NCNTEXP 0x00000800 /* normal-collision cntr exp */
+#define BE_BR_IMASK_ECNTEXP 0x00001000 /* excess-collision cntr exp */
+#define BE_BR_IMASK_LCCNTEXP 0x00002000 /* late-collision cntr exp */
+#define BE_BR_IMASK_FCNTEXP 0x00004000 /* first-collision cntr exp */
+#define BE_BR_IMASK_DTIMEXP 0x00008000 /* defer-timer expired */
+
+/* be_bregs.tx_cfg: tx config. */
+#define BE_BR_TXCFG_ENABLE 0x00000001 /* enable the transmitter */
+#define BE_BR_TXCFG_FIFO 0x00000010 /* default tx fthresh */
+#define BE_BR_TXCFG_SMODE 0x00000020 /* enable slow transmit mode */
+#define BE_BR_TXCFG_CIGN 0x00000040 /* ignore tx collisions */
+#define BE_BR_TXCFG_FCSOFF 0x00000080 /* do not emit fcs */
+#define BE_BR_TXCFG_DBACKOFF 0x00000100 /* disable backoff */
+#define BE_BR_TXCFG_FULLDPLX 0x00000200 /* enable full-duplex */
+/* be_bregs.rx_cfg: rx config. */
+#define BE_BR_RXCFG_ENABLE 0x00000001 /* enable the receiver */
+#define BE_BR_RXCFG_FIFO 0x0000000e /* default rx fthresh */
+#define BE_BR_RXCFG_PSTRIP 0x00000020 /* pad byte strip enable */
+#define BE_BR_RXCFG_PMISC 0x00000040 /* enable promiscous mode */
+#define BE_BR_RXCFG_DERR 0x00000080 /* disable error checking */
+#define BE_BR_RXCFG_DCRCS 0x00000100 /* disable crc stripping */
+#define BE_BR_RXCFG_ME 0x00000200 /* receive packets for me */
+#define BE_BR_RXCFG_PGRP 0x00000400 /* enable promisc group mode */
+#define BE_BR_RXCFG_HENABLE 0x00000800 /* enable hash filter */
+#define BE_BR_RXCFG_AENABLE 0x00001000 /* enable address filter */
+
+/*
+ * BE Channel registers
+ */
struct be_cregs {
- volatile u_int32_t ctrl; /* Control */
- volatile u_int32_t stat; /* Status */
- volatile u_int32_t rxds; /* RX descriptor ring ptr */
- volatile u_int32_t txds; /* TX descriptor ring ptr */
- volatile u_int32_t rimask; /* RX Interrupt Mask */
- volatile u_int32_t timask; /* TX Interrupt Mask */
- volatile u_int32_t qmask; /* QEC Error Interrupt Mask */
- volatile u_int32_t bmask; /* BE Error Interrupt Mask */
- volatile u_int32_t rxwbufptr; /* Local memory rx write ptr */
- volatile u_int32_t rxrbufptr; /* Local memory rx read ptr */
- volatile u_int32_t txwbufptr; /* Local memory tx write ptr */
- volatile u_int32_t txrbufptr; /* Local memory tx read ptr */
- volatile u_int32_t ccnt; /* Collision Counter */
+ volatile u_int32_t ctrl; /* control */
+ volatile u_int32_t stat; /* status */
+ volatile u_int32_t rxds; /* rx descriptor ring ptr */
+ volatile u_int32_t txds; /* tx descriptor ring ptr */
+ volatile u_int32_t rimask; /* rx interrupt mask */
+ volatile u_int32_t timask; /* tx interrupt mask */
+ volatile u_int32_t qmask; /* qec error interrupt mask */
+ volatile u_int32_t bmask; /* be error interrupt mask */
+ volatile u_int32_t rxwbufptr; /* local memory rx write ptr */
+ volatile u_int32_t rxrbufptr; /* local memory rx read ptr */
+ volatile u_int32_t txwbufptr; /* local memory tx write ptr */
+ volatile u_int32_t txrbufptr; /* local memory tx read ptr */
+ volatile u_int32_t ccnt; /* collision counter */
};
+/* be_cregs.ctrl: control. */
+#define BE_CR_CTRL_TWAKEUP 0x00000001 /* tx dma wakeup */
+
+/* be_cregs.stat: status. */
+#define BE_CR_STAT_BERROR 0x80000000 /* be error */
+#define BE_CR_STAT_TXIRQ 0x00200000 /* tx interrupt */
+#define BE_CR_STAT_TXDERR 0x00080000 /* tx descriptor is bad */
+#define BE_CR_STAT_TXLERR 0x00040000 /* tx late error */
+#define BE_CR_STAT_TXPERR 0x00020000 /* tx parity error */
+#define BE_CR_STAT_TXSERR 0x00010000 /* tx sbus error ack */
+#define BE_CR_STAT_RXIRQ 0x00000020 /* rx interrupt */
+#define BE_CR_STAT_RXDROP 0x00000010 /* rx packet dropped */
+#define BE_CR_STAT_RXSMALL 0x00000008 /* rx buffer too small */
+#define BE_CR_STAT_RXLERR 0x00000004 /* rx late error */
+#define BE_CR_STAT_RXPERR 0x00000002 /* rx parity error */
+#define BE_CR_STAT_RXSERR 0x00000001 /* rx sbus error ack */
+
+/* be_cregs.qmask: qec error interrupt mask. */
+#define BE_CR_QMASK_TXDERR 0x00080000 /* tx descriptor is bad */
+#define BE_CR_QMASK_TXLERR 0x00040000 /* tx late error */
+#define BE_CR_QMASK_TXPERR 0x00020000 /* tx parity error */
+#define BE_CR_QMASK_TXSERR 0x00010000 /* tx sbus error ack */
+#define BE_CR_QMASK_RXDROP 0x00000010 /* rx packet dropped */
+#define BE_CR_QMASK_RXSMALL 0x00000008 /* rx buffer too small */
+#define BE_CR_QMASK_RXLERR 0x00000004 /* rx late error */
+#define BE_CR_QMASK_RXPERR 0x00000002 /* rx parity error */
+#define BE_CR_QMASK_RXSERR 0x00000001 /* rx sbus error ack */
+
+/*
+ * BE Tranceiver registers
+ */
struct be_tregs {
- u_int32_t tcvr_pal;
- u_int32_t mgmt_pal;
+ volatile u_int32_t tcvr_pal; /* tranceiver pal */
+ volatile u_int32_t mgmt_pal; /* management pal */
};
+/* be_tregs.tcvr_pal: tranceiver pal */
+#define TCVR_PAL_SERIAL 0x00000001 /* serial mode enable */
+#define TCVR_PAL_EXTLBACK 0x00000002 /* external loopback */
+#define TCVR_PAL_MSENSE 0x00000004 /* media sense */
+#define TCVR_PAL_LTENABLE 0x00000008 /* link test enable */
+#define TCVR_PAL_LTSTATUS 0x00000010 /* link test status: p1 only */
+
+/* be_tregs.mgmt_pal: management pal */
+#define MGMT_PAL_DCLOCK 0x00000001 /* data clock strobe */
+#define MGMT_PAL_OENAB 0x00000002 /* output enable */
+#define MGMT_PAL_MDIO 0x00000004 /* MDIO data/attached */
+#define MGMT_PAL_EXT_MDIO MGMT_PAL_MDIO /* external mdio */
+#define MGMT_PAL_TIMEO 0x00000008 /* tx enable timeout error */
+#define MGMT_PAL_INT_MDIO MGMT_PAL_TIMEO /* internal mdio */
+
+/*
+ * BE receive descriptor
+ */
struct be_rxd {
- u_int32_t rx_flags;
- u_int32_t rx_addr;
+ volatile u_int32_t rx_flags; /* rx descriptor flags */
+ volatile u_int32_t rx_addr; /* rx buffer address */
};
-#define RXD_OWN 0x80000000 /* Ownership. */
-#define RXD_UPDATE 0x10000000 /* Being Updated? */
-#define RXD_LENGTH 0x000007ff /* Packet Length. */
+#define BE_RXD_OWN 0x80000000 /* ownership: 1=hw, 0=sw */
+#define BE_RXD_UPDATE 0x10000000 /* being updated? */
+#define BE_RXD_LENGTH 0x00001fff /* packet length */
+/*
+ * BE transmit descriptor
+ */
struct be_txd {
- u_int32_t tx_flags;
- u_int32_t tx_addr;
+ volatile u_int32_t tx_flags; /* tx descriptor flags */
+ volatile u_int32_t tx_addr; /* tx buffer address */
};
-#define TXD_OWN 0x80000000 /* Ownership. */
-#define TXD_SOP 0x40000000 /* Start Of Packet */
-#define TXD_EOP 0x20000000 /* End Of Packet */
-#define TXD_UPDATE 0x10000000 /* Being Updated? */
-#define TXD_LENGTH 0x000007ff /* Packet Length. */
+#define BE_TXD_OWN 0x80000000 /* ownership: 1=hw, 0=sw */
+#define BE_TXD_SOP 0x40000000 /* start of packet marker */
+#define BE_TXD_EOP 0x20000000 /* end of packet marker */
+#define BE_TXD_UPDATE 0x10000000 /* being updated? */
+#define BE_TXD_LENGTH 0x00001fff /* packet length */
+/* Buffer and Ring sizes: fixed ring size */
+#define BE_TX_RING_MAXSIZE 256 /* maximum tx ring size */
+#define BE_RX_RING_MAXSIZE 256 /* maximum rx ring size */
+#define BE_TX_RING_SIZE BE_TX_RING_MAXSIZE
+#define BE_RX_RING_SIZE BE_RX_RING_MAXSIZE
+#define BE_PKT_BUF_SZ 2048
-
-
-#define TX_RING_MAXSIZE 256
-#define RX_RING_MAXSIZE 256
-#define TX_RING_SIZE 256
-#define RX_RING_SIZE 256
-
-#define SUN4C_PKT_BUF_SZ 1546
-#define SUN4C_RX_BUFF_SIZE SUN4C_PKT_BUF_SZ
-#define SUN4C_TX_BUFF_SIZE SUN4C_PKT_BUF_SZ
-#define SUN4C_RX_RING_SIZE 16
-#define SUN4C_TX_RING_SIZE 16
-
+/*
+ * BE descriptor rings
+ */
struct be_desc {
- struct be_rxd be_rxd[RX_RING_MAXSIZE];
- struct be_txd be_txd[TX_RING_MAXSIZE];
+ struct be_rxd be_rxd[BE_RX_RING_MAXSIZE];
+ struct be_txd be_txd[BE_TX_RING_MAXSIZE];
};
+/*
+ * BE packet buffers
+ */
struct be_bufs {
- char tx_buf[SUN4C_TX_RING_SIZE][SUN4C_TX_BUFF_SIZE];
- char pad[2];
- char rx_buf[SUN4C_RX_RING_SIZE][SUN4C_RX_BUFF_SIZE];
+ char rx_buf[BE_RX_RING_SIZE][BE_PKT_BUF_SZ];
+ char tx_buf[BE_TX_RING_SIZE][BE_PKT_BUF_SZ];
};
+
+/* PHY addresses */
+#define BE_PHY_EXTERNAL 0
+#define BE_PHY_INTERNAL 1
+
+/* Tranceiver types */
+#define BE_TCVR_NONE 0
+#define BE_TCVR_INTERNAL 1
+#define BE_TCVR_EXTERNAL 2
+
+#define BE_TCVR_READ_INVALID 0xff000000
+
+#define PHY_BMCR 0x00 /* Basic Mode Control Register */
+#define PHY_BMSR 0x01 /* Basic Mode Status Register */
+
+/*
+ * Basic Mode Control Register (BMCR)
+ */
+#define PHY_BMCR_RESET 0x8000 /* Software reset */
+#define PHY_BMCR_LOOPBACK 0x4000 /* Lookback enable */
+#define PHY_BMCR_SPEED 0x2000 /* 1=100Mb, 0=10Mb */
+#define PHY_BMCR_ANE 0x1000 /* Auto-Negiation enable */
+#define PHY_BMCR_PDOWN 0x0800 /* power down the chip */
+#define PHY_BMCR_ISOLATE 0x0400 /* Isolate the chip */
+#define PHY_BMCR_RAN 0x0200 /* Restart autonegotiation */
+#define PHY_BMCR_DUPLEX 0x0100 /* 1=full, 0=half */
+#define PHY_BMCR_COLLISONTEST 0x0080 /* Create collisions on TX */
+
+/*
+ * Basic Mode Status Register (BMSR)
+ */
+#define PHY_BMSR_100BASET4 0x8000 /* 100BaseT4 capable? */
+#define PHY_BMSR_100BASETX_FULL 0x4000 /* 100BaseTX full duplex? */
+#define PHY_BMSR_100BASETX_HALF 0x2000 /* 100BaseTX half duplex? */
+#define PHY_BMSR_10BASET_FULL 0x1000 /* 10BaseT full duplex? */
+#define PHY_BMSR_10BASET_HALF 0x0800 /* 10BaseT half duplex? */
+#define PHY_BMSR_ANCOMPLETE 0x0020 /* auto-negotiation complete? */
+#define PHY_BMSR_REMOTEFAULT 0x0010 /* Fault condition seen? */
+#define PHY_BMSR_ANC 0x0008 /* Can auto-negotiate? */
+#define PHY_BMSR_LINKSTATUS 0x0004 /* Link established? */
+#define PHY_BMSR_JABBER 0x0002 /* Jabber detected? */
+#define PHY_BMSR_EXTENDED 0x0001 /* Extended registers? */
diff --git a/sys/arch/sparc/dev/bevar.h b/sys/arch/sparc/dev/bevar.h
index 9f526f82578..572b642ffba 100644
--- a/sys/arch/sparc/dev/bevar.h
+++ b/sys/arch/sparc/dev/bevar.h
@@ -1,7 +1,8 @@
-/* $OpenBSD: bevar.h,v 1.4 1998/07/05 09:25:56 deraadt Exp $ */
+/* $OpenBSD: bevar.h,v 1.5 1998/08/26 05:00:54 jason Exp $ */
/*
- * Copyright (c) 1998 Theo de Raadt. All rights reserved.
+ * Copyright (c) 1998 Theo de Raadt and Jason L. Wright.
+ * All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
@@ -11,13 +12,13 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
- * 3. The name of the author may not be used to endorse or promote products
+ * 3. The name of the authors may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
@@ -29,11 +30,10 @@
struct besoftc {
struct device sc_dev;
struct sbusdev sc_sd; /* sbus device */
- struct arpcom sc_arpcom;
struct intrhand sc_ih; /* interrupt vectoring */
- struct dma_softc *sc_dma; /* pointer to my dma */
+ struct arpcom sc_arpcom; /* ethernet common */
- u_long sc_laddr; /* DMA address */
+ struct qec_softc *sc_qec; /* QEC parent */
struct qecregs *sc_qr; /* QEC registers */
struct be_bregs *sc_br; /* registers */
struct be_cregs *sc_cr; /* registers */
@@ -41,16 +41,16 @@ struct besoftc {
void *sc_mem;
int sc_memsize;
- long sc_addr;
int sc_conf3;
u_int sc_rev;
int sc_promisc;
int sc_burst;
+ int sc_tcvr_type;
- int sc_txnew, sc_txold;
- int sc_rxnew, sc_rxold;
-
+ struct be_bufs *sc_bufs, *sc_bufs_dva;
struct be_desc *sc_desc, *sc_desc_dva;
- struct be_bufs *sc_bufs, *sc_bufs_dva;
+
+ int sc_no_td, sc_first_td, sc_last_td;
+ int sc_last_rd;
};