summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/ic/fxp.c91
-rw-r--r--sys/dev/ic/fxpvar.h13
2 files changed, 77 insertions, 27 deletions
diff --git a/sys/dev/ic/fxp.c b/sys/dev/ic/fxp.c
index d3216d8ede0..df73b19b957 100644
--- a/sys/dev/ic/fxp.c
+++ b/sys/dev/ic/fxp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: fxp.c,v 1.26 2001/08/27 22:06:52 jason Exp $ */
+/* $OpenBSD: fxp.c,v 1.27 2001/09/17 16:24:49 jason Exp $ */
/* $NetBSD: if_fxp.c,v 1.2 1997/06/05 02:01:55 thorpej Exp $ */
/*
@@ -93,12 +93,6 @@
#include <dev/ic/fxpreg.h>
#include <dev/ic/fxpvar.h>
-#ifdef __alpha__ /* XXX */
-/* XXX XXX NEED REAL DMA MAPPING SUPPORT XXX XXX */
-#undef vtophys
-#define vtophys(va) alpha_XXX_dmamap((vm_offset_t)(va))
-#endif /* __alpha__ */
-
/*
* NOTE! On the Alpha, we have an alignment constraint. The
* card DMAs the packet immediately following the RFA. However,
@@ -108,7 +102,7 @@
* aligns the packet after the Ethernet header at a 32-bit
* boundary. HOWEVER! This means that the RFA is misaligned!
*/
-#define RFA_ALIGNMENT_FUDGE 2
+#define RFA_ALIGNMENT_FUDGE (2 + sizeof(bus_dmamap_t))
/*
* Inline function to copy a 16-bit aligned 32-bit quantity.
@@ -199,12 +193,6 @@ static int tx_threshold = 64;
#define FXP_TXCB_MASK (FXP_NTXCB - 1)
/*
- * Number of receive frame area buffers. These are large so chose
- * wisely.
- */
-#define FXP_NRFABUFS 64
-
-/*
* Maximum number of seconds that the receiver can be idle before we
* assume it's dead and attempt to reset it by reprogramming the
* multicast filter. This is part of a work-around for a bug in the
@@ -291,6 +279,8 @@ fxp_attach_common(sc, enaddr, intrstr)
const char *intrstr;
{
struct ifnet *ifp;
+ struct mbuf *m;
+ bus_dmamap_t rxmap;
u_int16_t data;
int i, err;
@@ -342,11 +332,18 @@ fxp_attach_common(sc, enaddr, intrstr)
/*
* Pre-allocate our receive buffers.
*/
+ sc->sc_rxfree = 0;
for (i = 0; i < FXP_NRFABUFS; i++) {
- if (fxp_add_rfabuf(sc, NULL) != 0) {
+ if ((err = bus_dmamap_create(sc->sc_dmat, MCLBYTES, 1,
+ MCLBYTES, 0, 0, &sc->sc_rxmaps[i])) != 0) {
+ printf("%s: unable to create tx dma map %d, error %d\n",
+ sc->sc_dev.dv_xname, i, err);
goto fail;
}
}
+ for (i = 0; i < FXP_NRFABUFS; i++)
+ if (fxp_add_rfabuf(sc, NULL) != 0)
+ goto fail;
/*
* Find out how large of an SEEPROM we have.
@@ -452,10 +449,13 @@ fxp_attach_common(sc, enaddr, intrstr)
sizeof(struct fxp_cb_tx) * FXP_NTXCB);
bus_dmamem_free(sc->sc_dmat, &sc->sc_cb_seg, sc->sc_cb_nseg);
}
- /* frees entire chain */
- if (sc->rfa_headm)
- m_freem(sc->rfa_headm);
-
+ m = sc->rfa_headm;
+ while (m != NULL) {
+ rxmap = *((bus_dmamap_t *)m->m_ext.ext_buf);
+ bus_dmamap_unload(sc->sc_dmat, rxmap);
+ FXP_RXMAP_PUT(sc, rxmap);
+ m = m_free(m);
+ }
return (ENOMEM);
}
@@ -806,10 +806,15 @@ fxp_intr(arg)
*/
if (statack & (FXP_SCB_STATACK_FR | FXP_SCB_STATACK_RNR)) {
struct mbuf *m;
+ bus_dmamap_t rxmap;
u_int8_t *rfap;
rcvloop:
m = sc->rfa_headm;
rfap = m->m_ext.ext_buf + RFA_ALIGNMENT_FUDGE;
+ rxmap = *((bus_dmamap_t *)m->m_ext.ext_buf);
+ fxp_bus_dmamap_sync(sc->sc_dmat, rxmap,
+ 0, MCLBYTES, BUS_DMASYNC_POSTREAD |
+ BUS_DMASYNC_POSTWRITE);
if (*(u_int16_t *)(rfap +
offsetof(struct fxp_rfa, rfa_status)) &
@@ -854,9 +859,12 @@ rcvloop:
goto rcvloop;
}
if (rnr) {
+ rxmap = *((bus_dmamap_t *)
+ sc->rfa_headm->m_ext.ext_buf);
fxp_scb_wait(sc);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL,
- vtophys((vaddr_t)sc->rfa_headm->m_ext.ext_buf) + RFA_ALIGNMENT_FUDGE);
+ rxmap->dm_segs[0].ds_addr +
+ RFA_ALIGNMENT_FUDGE);
fxp_scb_cmd(sc, FXP_SCB_COMMAND_RU_START);
}
}
@@ -1002,11 +1010,19 @@ fxp_stop(sc, drain)
sc->sc_cbt_cnt = 0;
if (drain) {
+ bus_dmamap_t rxmap;
+ struct mbuf *m;
+
/*
* Free all the receive buffers then reallocate/reinitialize
*/
- if (sc->rfa_headm != NULL)
- m_freem(sc->rfa_headm);
+ m = sc->rfa_headm;
+ while (m != NULL) {
+ rxmap = *((bus_dmamap_t *)m->m_ext.ext_buf);
+ bus_dmamap_unload(sc->sc_dmat, rxmap);
+ FXP_RXMAP_PUT(sc, rxmap);
+ m = m_free(m);
+ }
sc->rfa_headm = NULL;
sc->rfa_tailm = NULL;
for (i = 0; i < FXP_NRFABUFS; i++) {
@@ -1065,6 +1081,7 @@ fxp_init(xsc)
struct fxp_cb_config *cbp;
struct fxp_cb_ias *cb_ias;
struct fxp_cb_tx *txp;
+ bus_dmamap_t rxmap;
int i, prm, allm, s;
s = splimp();
@@ -1218,8 +1235,9 @@ fxp_init(xsc)
* Initialize receiver buffer area - RFA.
*/
fxp_scb_wait(sc);
+ rxmap = *((bus_dmamap_t *)sc->rfa_headm->m_ext.ext_buf);
CSR_WRITE_4(sc, FXP_CSR_SCB_GENERAL,
- vtophys((vaddr_t)sc->rfa_headm->m_ext.ext_buf) + RFA_ALIGNMENT_FUDGE);
+ rxmap->dm_segs[0].ds_addr + RFA_ALIGNMENT_FUDGE);
fxp_scb_cmd(sc, FXP_SCB_COMMAND_RU_START);
/*
@@ -1281,6 +1299,7 @@ fxp_add_rfabuf(sc, oldm)
u_int32_t v;
struct mbuf *m;
u_int8_t *rfap;
+ bus_dmamap_t rxmap = NULL;
MGETHDR(m, M_DONTWAIT, MT_DATA);
if (m != NULL) {
@@ -1292,11 +1311,28 @@ fxp_add_rfabuf(sc, oldm)
m = oldm;
m->m_data = m->m_ext.ext_buf;
}
+ if (oldm == NULL) {
+ rxmap = FXP_RXMAP_GET(sc);
+ *((bus_dmamap_t *)m->m_ext.ext_buf) = rxmap;
+ bus_dmamap_load(sc->sc_dmat, rxmap,
+ m->m_ext.ext_buf, m->m_ext.ext_size, NULL,
+ BUS_DMA_NOWAIT);
+ } else if (oldm == m)
+ rxmap = *((bus_dmamap_t *)oldm->m_ext.ext_buf);
+ else {
+ rxmap = *((bus_dmamap_t *)oldm->m_ext.ext_buf);
+ bus_dmamap_unload(sc->sc_dmat, rxmap);
+ bus_dmamap_load(sc->sc_dmat, rxmap,
+ m->m_ext.ext_buf, m->m_ext.ext_size, NULL,
+ BUS_DMA_NOWAIT);
+ *mtod(m, bus_dmamap_t *) = rxmap;
+ }
} else {
if (oldm == NULL)
return 1;
m = oldm;
m->m_data = m->m_ext.ext_buf;
+ rxmap = *mtod(m, bus_dmamap_t *);
}
/*
@@ -1336,17 +1372,20 @@ fxp_add_rfabuf(sc, oldm)
*/
if (sc->rfa_headm != NULL) {
sc->rfa_tailm->m_next = m;
- v = vtophys((vaddr_t)rfap);
+ v = rxmap->dm_segs[0].ds_addr + RFA_ALIGNMENT_FUDGE;
rfap = sc->rfa_tailm->m_ext.ext_buf + RFA_ALIGNMENT_FUDGE;
fxp_lwcopy(&v,
(u_int32_t *)(rfap + offsetof(struct fxp_rfa, link_addr)));
*(u_int16_t *)(rfap + offsetof(struct fxp_rfa, rfa_control)) &=
~FXP_RFA_CONTROL_EL;
- } else {
+ } else
sc->rfa_headm = m;
- }
+
sc->rfa_tailm = m;
+ fxp_bus_dmamap_sync(sc->sc_dmat, rxmap, 0, MCLBYTES,
+ BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
+
return (m == oldm);
}
diff --git a/sys/dev/ic/fxpvar.h b/sys/dev/ic/fxpvar.h
index 06b0c4942cd..99c7f172676 100644
--- a/sys/dev/ic/fxpvar.h
+++ b/sys/dev/ic/fxpvar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: fxpvar.h,v 1.8 2001/08/25 14:55:14 jason Exp $ */
+/* $OpenBSD: fxpvar.h,v 1.9 2001/09/17 16:24:49 jason Exp $ */
/* $NetBSD: if_fxpvar.h,v 1.1 1997/06/05 02:01:58 thorpej Exp $ */
/*
@@ -46,6 +46,12 @@
#define FXP_NTXCB 128
/*
+ * Number of receive frame area buffers. These are large so chose
+ * wisely.
+ */
+#define FXP_NRFABUFS 64
+
+/*
* NOTE: Elements are ordered for optimal cacheline behavior, and NOT
* for functional grouping.
*/
@@ -98,6 +104,8 @@ struct fxp_softc {
bus_dma_segment_t sc_cb_seg;
int sc_cb_nseg;
struct fxp_ctrl *sc_ctrl;
+ bus_dmamap_t sc_rxmaps[FXP_NRFABUFS];
+ int sc_rxfree;
};
/* Macros to ease CSR access. */
@@ -118,6 +126,9 @@ extern int fxp_intr __P((void *));
extern int fxp_attach_common __P((struct fxp_softc *, u_int8_t *, const char *));
extern int fxp_detach __P((struct fxp_softc *));
+#define FXP_RXMAP_GET(sc) ((sc)->sc_rxmaps[(sc)->sc_rxfree++])
+#define FXP_RXMAP_PUT(sc,map) ((sc)->sc_rxmaps[--(sc)->sc_rxfree] = (map))
+
#ifdef __HAS_NEW_BUS_DMAMAP_SYNC
#define fxp_bus_dmamap_sync(t, m, o, l, p) \
bus_dmamap_sync((t), (m), (o), (l), (p))