Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/pci msk(4): rework rx descriptor loading to support ...
details: https://anonhg.NetBSD.org/src/rev/65b5bdd7eaf1
branches: trunk
changeset: 971615:65b5bdd7eaf1
user: jakllsch <jakllsch%NetBSD.org@localhost>
date: Thu Apr 30 01:52:08 2020 +0000
description:
msk(4): rework rx descriptor loading to support multiple segments
This paves the way to replace the driver-internal jumbo frame rx buffer
with other recieve buffers (for example MCLGET/MEXTMALLOC) in the future.
diffstat:
sys/dev/pci/if_msk.c | 174 ++++++++++++++++++++++++++++++++++++++------------
1 files changed, 130 insertions(+), 44 deletions(-)
diffs (268 lines):
diff -r 78b17f40cbb5 -r 65b5bdd7eaf1 sys/dev/pci/if_msk.c
--- a/sys/dev/pci/if_msk.c Thu Apr 30 00:48:10 2020 +0000
+++ b/sys/dev/pci/if_msk.c Thu Apr 30 01:52:08 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_msk.c,v 1.105 2020/04/29 20:03:52 jakllsch Exp $ */
+/* $NetBSD: if_msk.c,v 1.106 2020/04/30 01:52:08 jakllsch Exp $ */
/* $OpenBSD: if_msk.c,v 1.79 2009/10/15 17:54:56 deraadt Exp $ */
/*
@@ -52,7 +52,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_msk.c,v 1.105 2020/04/29 20:03:52 jakllsch Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_msk.c,v 1.106 2020/04/30 01:52:08 jakllsch Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -113,7 +113,7 @@
static void msk_init_yukon(struct sk_if_softc *);
static void msk_stop(struct ifnet *, int);
static void msk_watchdog(struct ifnet *);
-static int msk_newbuf(struct sk_if_softc *, bus_dmamap_t);
+static int msk_newbuf(struct sk_if_softc *);
static int msk_alloc_jumbo_mem(struct sk_if_softc *);
static void *msk_jalloc(struct sk_if_softc *);
static void msk_jfree(struct mbuf *, void *, size_t, void *);
@@ -472,13 +472,18 @@
}
static int
-msk_newbuf(struct sk_if_softc *sc_if, bus_dmamap_t dmamap)
+msk_newbuf(struct sk_if_softc *sc_if)
{
+ struct sk_softc *sc = sc_if->sk_softc;
struct mbuf *m_new = NULL;
struct sk_chain *c;
struct msk_rx_desc *r;
void *buf = NULL;
bus_addr_t addr;
+ bus_dmamap_t rxmap;
+ size_t i;
+ uint32_t rxidx, frag, cur, hiaddr, old_hiaddr, total;
+ uint32_t entries = 0;
MGETHDR(m_new, M_DONTWAIT, MT_DATA);
if (m_new == NULL)
@@ -499,44 +504,99 @@
m_adj(m_new, ETHER_ALIGN);
- addr = dmamap->dm_segs[0].ds_addr +
- ((vaddr_t)m_new->m_data -
- (vaddr_t)sc_if->sk_cdata.sk_jumbo_buf);
-
- if (sc_if->sk_cdata.sk_rx_hiaddr != MSK_ADDR_HI(addr)) {
- c = &sc_if->sk_cdata.sk_rx_chain[sc_if->sk_cdata.sk_rx_prod];
- r = &sc_if->sk_rdata->sk_rx_ring[sc_if->sk_cdata.sk_rx_prod];
- c->sk_mbuf = NULL;
- r->sk_addr = htole32(MSK_ADDR_HI(addr));
- r->sk_len = 0;
+ rxidx = frag = cur = sc_if->sk_cdata.sk_rx_prod;
+ rxmap = sc_if->sk_cdata.sk_rx_chain[rxidx].sk_dmamap;
+
+ if (bus_dmamap_load_mbuf(sc->sc_dmatag, rxmap, m_new, BUS_DMA_NOWAIT)) {
+ DPRINTFN(2, ("msk_newbuf: dmamap_load failed\n"));
+ m_freem(m_new);
+ return ENOBUFS;
+ }
+
+ /* Count how many rx descriptors needed. */
+ hiaddr = sc_if->sk_cdata.sk_rx_hiaddr;
+ for (total = i = 0; i < rxmap->dm_nsegs; i++) {
+ if (hiaddr != MSK_ADDR_HI(rxmap->dm_segs[i].ds_addr)) {
+ hiaddr = MSK_ADDR_HI(rxmap->dm_segs[i].ds_addr);
+ total++;
+ }
+ total++;
+ }
+
+ if (total > MSK_RX_RING_CNT - sc_if->sk_cdata.sk_rx_cnt - 1) {
+ DPRINTFN(2, ("msk_newbuf: too few descriptors free\n"));
+ bus_dmamap_unload(sc->sc_dmatag, rxmap);
+ m_freem(m_new);
+ return ENOBUFS;
+ }
+
+ DPRINTFN(2, ("msk_newbuf: dm_nsegs=%d total desc=%u\n",
+ rxmap->dm_nsegs, total));
+
+ /* Sync the DMA map. */
+ bus_dmamap_sync(sc->sc_dmatag, rxmap, 0, rxmap->dm_mapsize,
+ BUS_DMASYNC_PREREAD);
+
+ old_hiaddr = sc_if->sk_cdata.sk_rx_hiaddr;
+ for (i = 0; i < rxmap->dm_nsegs; i++) {
+ addr = rxmap->dm_segs[i].ds_addr;
+ DPRINTFN(2, ("msk_newbuf: addr %llx\n",
+ (unsigned long long)addr));
+ hiaddr = MSK_ADDR_HI(addr);
+
+ if (sc_if->sk_cdata.sk_rx_hiaddr != hiaddr) {
+ c = &sc_if->sk_cdata.sk_rx_chain[frag];
+ c->sk_mbuf = NULL;
+ r = &sc_if->sk_rdata->sk_rx_ring[frag];
+ r->sk_addr = htole32(hiaddr);
+ r->sk_len = 0;
+ r->sk_ctl = 0;
+ if (i == 0)
+ r->sk_opcode = SK_Y2_BMUOPC_ADDR64;
+ else
+ r->sk_opcode = SK_Y2_BMUOPC_ADDR64 |
+ SK_Y2_RXOPC_OWN;
+ sc_if->sk_cdata.sk_rx_hiaddr = hiaddr;
+ MSK_CDRXSYNC(sc_if, frag,
+ BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
+ SK_INC(frag, MSK_RX_RING_CNT);
+ entries++;
+ DPRINTFN(10, ("%s: rx ADDR64: %#x\n",
+ sc_if->sk_ethercom.ec_if.if_xname, hiaddr));
+ }
+
+ c = &sc_if->sk_cdata.sk_rx_chain[frag];
+ r = &sc_if->sk_rdata->sk_rx_ring[frag];
+ r->sk_addr = htole32(MSK_ADDR_LO(addr));
+ r->sk_len = htole16(rxmap->dm_segs[i].ds_len);
r->sk_ctl = 0;
- r->sk_opcode = SK_Y2_BMUOPC_ADDR64 | SK_Y2_RXOPC_OWN;
- sc_if->sk_cdata.sk_rx_hiaddr = MSK_ADDR_HI(addr);
-
- MSK_CDRXSYNC(sc_if, sc_if->sk_cdata.sk_rx_prod,
+ if (i == 0) {
+ if (hiaddr != old_hiaddr)
+ r->sk_opcode = SK_Y2_RXOPC_PACKET |
+ SK_Y2_RXOPC_OWN;
+ else
+ r->sk_opcode = SK_Y2_RXOPC_PACKET;
+ } else
+ r->sk_opcode = SK_Y2_RXOPC_BUFFER | SK_Y2_RXOPC_OWN;
+ MSK_CDRXSYNC(sc_if, frag,
BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
-
- SK_INC(sc_if->sk_cdata.sk_rx_prod, MSK_RX_RING_CNT);
- sc_if->sk_cdata.sk_rx_cnt++;
-
- DPRINTFN(10, ("%s: rx ADDR64: %#x\n",
- sc_if->sk_ethercom.ec_if.if_xname,
- (unsigned)MSK_ADDR_HI(addr)));
+ cur = frag;
+ SK_INC(frag, MSK_RX_RING_CNT);
+ entries++;
}
-
- c = &sc_if->sk_cdata.sk_rx_chain[sc_if->sk_cdata.sk_rx_prod];
- r = &sc_if->sk_rdata->sk_rx_ring[sc_if->sk_cdata.sk_rx_prod];
- c->sk_mbuf = m_new;
- r->sk_addr = htole32(MSK_ADDR_LO(addr));
- r->sk_len = htole16(SK_JLEN);
- r->sk_ctl = 0;
- r->sk_opcode = SK_Y2_RXOPC_PACKET | SK_Y2_RXOPC_OWN;
-
- MSK_CDRXSYNC(sc_if, sc_if->sk_cdata.sk_rx_prod,
+ KASSERTMSG(entries == total, "entries %u total %u", entries, total);
+
+ sc_if->sk_cdata.sk_rx_chain[rxidx].sk_dmamap =
+ sc_if->sk_cdata.sk_rx_chain[cur].sk_dmamap;
+ sc_if->sk_cdata.sk_rx_chain[cur].sk_mbuf = m_new;
+ sc_if->sk_cdata.sk_rx_chain[cur].sk_dmamap = rxmap;
+
+ sc_if->sk_rdata->sk_rx_ring[rxidx].sk_opcode |= SK_Y2_RXOPC_OWN;
+ MSK_CDRXSYNC(sc_if, rxidx,
BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
- SK_INC(sc_if->sk_cdata.sk_rx_prod, MSK_RX_RING_CNT);
- sc_if->sk_cdata.sk_rx_cnt++;
+ sc_if->sk_cdata.sk_rx_cnt += entries;
+ sc_if->sk_cdata.sk_rx_prod = frag;
return 0;
}
@@ -1189,6 +1249,20 @@
sc_if->sk_cdata.sk_tx_chain[i].sk_dmamap = dmamap;
}
+ for (i = 0; i < MSK_RX_RING_CNT; i++) {
+ sc_if->sk_cdata.sk_rx_chain[i].sk_mbuf = NULL;
+
+ if (bus_dmamap_create(sc->sc_dmatag, SK_JLEN,
+ howmany(SK_JLEN + 1, NBPG),
+ SK_JLEN, 0, BUS_DMA_NOWAIT, &dmamap)) {
+ aprint_error_dev(sc_if->sk_dev,
+ "Can't create RX dmamap\n");
+ goto fail_3;
+ }
+
+ sc_if->sk_cdata.sk_rx_chain[i].sk_dmamap = dmamap;
+ }
+
sc_if->sk_rdata = (struct msk_ring_data *)kva;
memset(sc_if->sk_rdata, 0, sizeof(struct msk_ring_data));
@@ -1303,6 +1377,11 @@
sc_if->sk_cdata.sk_tx_chain[i].sk_dmamap);
}
+ for (i = 0; i < MSK_RX_RING_CNT; i++) {
+ bus_dmamap_destroy(sc->sc_dmatag,
+ sc_if->sk_cdata.sk_rx_chain[i].sk_dmamap);
+ }
+
if (--sc->rnd_attached == 0)
rnd_detach_source(&sc->rnd_source);
@@ -2027,13 +2106,13 @@
cur = sc_if->sk_cdata.sk_rx_cons;
prod = sc_if->sk_cdata.sk_rx_prod;
- /* Sync the descriptor */
- MSK_CDRXSYNC(sc_if, cur, BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
-
DPRINTFN(2, ("msk_rxeof: cur %u prod %u rx_cnt %u\n", cur, prod,
sc_if->sk_cdata.sk_rx_cnt));
while (prod != cur) {
+ MSK_CDRXSYNC(sc_if, cur,
+ BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
+
tail = cur;
SK_INC(cur, MSK_RX_RING_CNT);
@@ -2050,10 +2129,11 @@
if (m == NULL)
return;
- dmamap = sc_if->sk_cdata.sk_rx_jumbo_map;
+ dmamap = sc_if->sk_cdata.sk_rx_chain[tail].sk_dmamap;
bus_dmamap_sync(sc_if->sk_softc->sc_dmatag, dmamap, 0,
- dmamap->dm_mapsize, BUS_DMASYNC_POSTREAD);
+ uimin(dmamap->dm_mapsize, total_len), BUS_DMASYNC_POSTREAD);
+ bus_dmamap_unload(sc->sc_dmatag, dmamap);
if (total_len < SK_MIN_FRAMELEN ||
total_len > ETHER_MAX_LEN_JUMBO ||
@@ -2132,8 +2212,7 @@
{
/* Make sure to not completely wrap around */
while (sc_if->sk_cdata.sk_rx_cnt < (MSK_RX_RING_CNT - 1)) {
- if (msk_newbuf(sc_if,
- sc_if->sk_cdata.sk_rx_jumbo_map) == ENOBUFS) {
+ if (msk_newbuf(sc_if) == ENOBUFS) {
goto schedretry;
}
}
@@ -2640,6 +2719,13 @@
/* Free RX and TX mbufs still in the queues. */
for (i = 0; i < MSK_RX_RING_CNT; i++) {
if (sc_if->sk_cdata.sk_rx_chain[i].sk_mbuf != NULL) {
+ dmamap = sc_if->sk_cdata.sk_rx_chain[i].sk_dmamap;
+
+ bus_dmamap_sync(sc->sc_dmatag, dmamap, 0,
+ dmamap->dm_mapsize, BUS_DMASYNC_POSTREAD);
+
+ bus_dmamap_unload(sc->sc_dmatag, dmamap);
+
m_freem(sc_if->sk_cdata.sk_rx_chain[i].sk_mbuf);
sc_if->sk_cdata.sk_rx_chain[i].sk_mbuf = NULL;
}
Home |
Main Index |
Thread Index |
Old Index