summaryrefslogtreecommitdiff
path: root/sys/dev/ic
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic')
-rw-r--r--sys/dev/ic/fxp.c78
-rw-r--r--sys/dev/ic/fxpvar.h6
2 files changed, 62 insertions, 22 deletions
diff --git a/sys/dev/ic/fxp.c b/sys/dev/ic/fxp.c
index 3428da957b4..81bd01421b2 100644
--- a/sys/dev/ic/fxp.c
+++ b/sys/dev/ic/fxp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: fxp.c,v 1.17 2001/05/27 01:32:21 art Exp $ */
+/* $OpenBSD: fxp.c,v 1.18 2001/06/13 23:19:17 jason Exp $ */
/* $NetBSD: if_fxp.c,v 1.2 1997/06/05 02:01:55 thorpej Exp $ */
/*
@@ -182,6 +182,7 @@ void fxp_read_eeprom __P((struct fxp_softc *, u_int16_t *,
int, int));
void fxp_stats_update __P((void *));
void fxp_mc_setup __P((struct fxp_softc *));
+static __inline void fxp_scb_cmd __P((struct fxp_softc *, u_int8_t));
/*
* Set initial transmit threshold at 64 (512 bytes). This is
@@ -740,6 +741,8 @@ tbdinit:
}
}
+volatile int _fxp_debugit;
+
/*
* Process interface interrupts.
*/
@@ -750,7 +753,7 @@ fxp_intr(arg)
struct fxp_softc *sc = arg;
struct ifnet *ifp = &sc->arpcom.ac_if;
u_int8_t statack;
- int claimed = 0;
+ int claimed = 0, rnr;
/*
* If the interface isn't running, don't try to
@@ -767,6 +770,8 @@ fxp_intr(arg)
while ((statack = CSR_READ_1(sc, FXP_CSR_SCB_STATACK)) != 0) {
claimed = 1;
+ rnr = 0;
+
/*
* First ACK all the interrupts in this pass.
*/
@@ -782,7 +787,9 @@ fxp_intr(arg)
(txp->cb_status & FXP_CB_STATUS_C) != 0;
txp = txp->next) {
if (txp->mb_head != NULL) {
+ _fxp_debugit = 1;
m_freem(txp->mb_head);
+ _fxp_debugit = 0;
txp->mb_head = NULL;
}
sc->tx_queued--;
@@ -814,6 +821,11 @@ rcvloop:
if (*(u_int16_t *)(rfap +
offsetof(struct fxp_rfa, rfa_status)) &
FXP_RFA_STATUS_C) {
+ if (*(u_int16_t *)(rfap +
+ offsetof(struct fxp_rfa, rfa_status)) &
+ FXP_RFA_STATUS_RNR)
+ rnr = 1;
+
/*
* Remove first packet from the chain.
*/
@@ -835,7 +847,9 @@ rcvloop:
(MCLBYTES - 1);
if (total_len <
sizeof(struct ether_header)) {
+ _fxp_debugit = 2;
m_freem(m);
+ _fxp_debugit = 0;
goto rcvloop;
}
m->m_pkthdr.rcvif = ifp;
@@ -855,13 +869,12 @@ rcvloop:
}
goto rcvloop;
}
- if (statack & FXP_SCB_STATACK_RNR) {
+ if (rnr) {
fxp_scb_wait(sc);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL,
vtophys(sc->rfa_headm->m_ext.ext_buf) +
RFA_ALIGNMENT_FUDGE);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND,
- FXP_SCB_COMMAND_RU_START);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_RU_START);
}
}
}
@@ -933,8 +946,7 @@ fxp_stats_update(arg)
/*
* Start another stats dump.
*/
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND,
- FXP_SCB_COMMAND_CU_DUMPRESET);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_DUMPRESET);
} else {
/*
* A previous command is still waiting to be accepted.
@@ -1044,6 +1056,22 @@ fxp_watchdog(ifp)
fxp_init(sc);
}
+/*
+ * Submit a command to the i82557.
+ */
+static __inline void
+fxp_scb_cmd(sc, cmd)
+ struct fxp_softc *sc;
+ u_int8_t cmd;
+{
+ if (cmd == FXP_SCB_COMMAND_CU_RESUME &&
+ (sc->sc_flags & FXPF_FIX_RESUME_BUG) != 0) {
+ CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_CB_COMMAND_NOP);
+ fxp_scb_wait(sc);
+ }
+ CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, cmd);
+}
+
void
fxp_init(xsc)
void *xsc;
@@ -1068,17 +1096,17 @@ fxp_init(xsc)
* sets it up for regular linear addressing.
*/
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL, 0);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_BASE);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_BASE);
fxp_scb_wait(sc);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_RU_BASE);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_RU_BASE);
/*
* Initialize base of dump-stats buffer.
*/
fxp_scb_wait(sc);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL, vtophys(sc->fxp_stats));
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_DUMP_ADR);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_DUMP_ADR);
/*
* We temporarily use memory that contains the TxCB list to
@@ -1135,7 +1163,7 @@ fxp_init(xsc)
*/
fxp_scb_wait(sc);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL, vtophys(&cbp->cb_status));
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_START);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_START);
/* ...and wait for it to complete. */
while (!(cbp->cb_status & FXP_CB_STATUS_C));
@@ -1154,7 +1182,7 @@ fxp_init(xsc)
* Start the IAS (Individual Address Setup) command/DMA.
*/
fxp_scb_wait(sc);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_START);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_START);
/* ...and wait for it to complete. */
while (!(cb_ias->cb_status & FXP_CB_STATUS_C));
@@ -1180,7 +1208,7 @@ fxp_init(xsc)
sc->tx_queued = 1;
fxp_scb_wait(sc);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_START);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_START);
/*
* Initialize receiver buffer area - RFA.
@@ -1188,7 +1216,7 @@ fxp_init(xsc)
fxp_scb_wait(sc);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL,
vtophys(sc->rfa_headm->m_ext.ext_buf) + RFA_ALIGNMENT_FUDGE);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_RU_START);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_RU_START);
/*
* Set current media.
@@ -1345,8 +1373,18 @@ void
fxp_statchg(self)
struct device *self;
{
+ struct fxp_softc *sc = (struct fxp_softc *)self;
- /* XXX Update ifp->if_baudrate */
+ /*
+ * Determine whether or not we have to work-around the
+ * Resume Bug.
+ */
+ if (sc->sc_flags & FXPF_HAS_RESUME_BUG) {
+ if (IFM_TYPE(sc->sc_mii.mii_media_active) == IFM_10_T)
+ sc->sc_flags |= FXPF_FIX_RESUME_BUG;
+ else
+ sc->sc_flags &= ~FXPF_FIX_RESUME_BUG;
+ }
}
void
@@ -1542,11 +1580,9 @@ fxp_mc_setup(sc)
* Issue a resume in case the CU has just suspended.
*/
fxp_scb_wait(sc);
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_RESUME);
- /*
- * Set a 5 second timer just in case we don't hear from the
- * card again.
- */
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_RESUME);
+
+ /* Set a 5 watchdog timer in case the chip flakes out. */
ifp->if_timer = 5;
return;
@@ -1601,7 +1637,7 @@ fxp_mc_setup(sc)
*/
fxp_scb_wait(sc);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL, vtophys(&mcsp->cb_status));
- CSR_WRITE_1(sc, FXP_CSR_SCB_COMMAND, FXP_SCB_COMMAND_CU_START);
+ fxp_scb_cmd(sc, FXP_SCB_COMMAND_CU_START);
ifp->if_timer = 2;
return;
diff --git a/sys/dev/ic/fxpvar.h b/sys/dev/ic/fxpvar.h
index f41abdc9755..d34e0a40268 100644
--- a/sys/dev/ic/fxpvar.h
+++ b/sys/dev/ic/fxpvar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: fxpvar.h,v 1.4 2000/09/17 17:08:16 aaron Exp $ */
+/* $OpenBSD: fxpvar.h,v 1.5 2001/06/13 23:19:17 jason Exp $ */
/* $NetBSD: if_fxpvar.h,v 1.1 1997/06/05 02:01:58 thorpej Exp $ */
/*
@@ -51,6 +51,10 @@ struct fxp_softc {
struct mbuf *rfa_headm; /* first mbuf in receive frame area */
struct mbuf *rfa_tailm; /* last mbuf in receive frame area */
struct fxp_cb_tx *cbl_first; /* first active TxCB in list */
+ int sc_flags; /* misc. flags */
+#define FXPF_HAS_RESUME_BUG 0x08 /* has the resume bug */
+#define FXPF_FIX_RESUME_BUG 0x10 /* currently need to work-around
+ the resume bug */
int tx_queued; /* # of active TxCB's */
int need_mcsetup; /* multicast filter needs programming */
struct fxp_cb_tx *cbl_last; /* last active TxCB in list */