diff options
Diffstat (limited to 'sys/dev/pci/if_sk.c')
-rw-r--r-- | sys/dev/pci/if_sk.c | 213 |
1 files changed, 136 insertions, 77 deletions
diff --git a/sys/dev/pci/if_sk.c b/sys/dev/pci/if_sk.c index 72bf161d805..12166ab7a04 100644 --- a/sys/dev/pci/if_sk.c +++ b/sys/dev/pci/if_sk.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_sk.c,v 1.30 2003/05/08 04:20:05 nate Exp $ */ +/* $OpenBSD: if_sk.c,v 1.31 2003/05/14 01:54:15 nate Exp $ */ /* * Copyright (c) 1997, 1998, 1999, 2000 @@ -100,9 +100,6 @@ #include <net/bpf.h> #endif -#include <uvm/uvm_extern.h> /* for vtophys */ -#include <machine/bus.h> - #include <dev/mii/mii.h> #include <dev/mii/miivar.h> #include <dev/mii/brgphyreg.h> @@ -139,10 +136,9 @@ void sk_shutdown(void *); int sk_ifmedia_upd(struct ifnet *); void sk_ifmedia_sts(struct ifnet *, struct ifmediareq *); void sk_reset(struct sk_softc *); -int sk_newbuf(struct sk_if_softc *, struct sk_chain *, - struct mbuf *); +int sk_newbuf(struct sk_if_softc *, int, struct mbuf *, bus_dmamap_t); int sk_init_rx_ring(struct sk_if_softc *); -void sk_init_tx_ring(struct sk_if_softc *); +int sk_init_tx_ring(struct sk_if_softc *); u_int32_t sk_win_read_4(struct sk_softc *, int); u_int16_t sk_win_read_2(struct sk_softc *, int); u_int8_t sk_win_read_1(struct sk_softc *, int); @@ -486,66 +482,95 @@ sk_init_rx_ring(struct sk_if_softc *sc_if) for (i = 0; i < SK_RX_RING_CNT; i++) { cd->sk_rx_chain[i].sk_desc = &rd->sk_rx_ring[i]; - if (sk_newbuf(sc_if, &cd->sk_rx_chain[i], NULL) == ENOBUFS) { - printf("%s: failed alloc of %dth mbuf\n", - sc_if->sk_dev.dv_xname, i); - return(ENOBUFS); - } if (i == (SK_RX_RING_CNT - 1)) { - cd->sk_rx_chain[i].sk_next = - &cd->sk_rx_chain[0]; - rd->sk_rx_ring[i].sk_next = - vtophys(&rd->sk_rx_ring[0]); + cd->sk_rx_chain[i].sk_next = &cd->sk_rx_chain[0]; + rd->sk_rx_ring[i].sk_next = SK_RX_RING_ADDR(sc_if, 0); } else { - cd->sk_rx_chain[i].sk_next = - &cd->sk_rx_chain[i + 1]; - rd->sk_rx_ring[i].sk_next = - vtophys(&rd->sk_rx_ring[i + 1]); + cd->sk_rx_chain[i].sk_next = &cd->sk_rx_chain[i + 1]; + rd->sk_rx_ring[i].sk_next = SK_RX_RING_ADDR(sc_if,i+1); } } + for (i = 0; i < SK_RX_RING_CNT; i++) { + if (sk_newbuf(sc_if, i, NULL, NULL) == ENOBUFS) { + printf("%s: failed alloc of %dth mbuf\n", + sc_if->sk_dev.dv_xname, i); + return(ENOBUFS); + } + } sc_if->sk_cdata.sk_rx_prod = 0; sc_if->sk_cdata.sk_rx_cons = 0; return(0); } -void +int sk_init_tx_ring(struct sk_if_softc *sc_if) { + struct sk_softc *sc = sc_if->sk_softc; struct sk_chain_data *cd = &sc_if->sk_cdata; struct sk_ring_data *rd = sc_if->sk_rdata; + bus_dmamap_t dmamap; + struct sk_txmap_entry *entry; int i; bzero((char *)sc_if->sk_rdata->sk_tx_ring, sizeof(struct sk_tx_desc) * SK_TX_RING_CNT); + SLIST_INIT(&sc_if->sk_txmap_listhead); for (i = 0; i < SK_TX_RING_CNT; i++) { cd->sk_tx_chain[i].sk_desc = &rd->sk_tx_ring[i]; if (i == (SK_TX_RING_CNT - 1)) { - cd->sk_tx_chain[i].sk_next = - &cd->sk_tx_chain[0]; - rd->sk_tx_ring[i].sk_next = - vtophys(&rd->sk_tx_ring[0]); + cd->sk_tx_chain[i].sk_next = &cd->sk_tx_chain[0]; + rd->sk_tx_ring[i].sk_next = SK_TX_RING_ADDR(sc_if, 0); } else { - cd->sk_tx_chain[i].sk_next = - &cd->sk_tx_chain[i + 1]; - rd->sk_tx_ring[i].sk_next = - vtophys(&rd->sk_tx_ring[i + 1]); + cd->sk_tx_chain[i].sk_next = &cd->sk_tx_chain[i + 1]; + rd->sk_tx_ring[i].sk_next = SK_TX_RING_ADDR(sc_if,i+1); } + + if (bus_dmamap_create(sc->sc_dmatag, MCLBYTES, SK_NTXSEG, + MCLBYTES, 0, BUS_DMA_NOWAIT, &dmamap)) + return (ENOBUFS); + + entry = malloc(sizeof(*entry), M_DEVBUF, M_NOWAIT); + if (!entry) { + bus_dmamap_destroy(sc->sc_dmatag, dmamap); + return (ENOBUFS); + } + entry->dmamap = dmamap; + SLIST_INSERT_HEAD(&sc_if->sk_txmap_listhead, entry, link); } sc_if->sk_cdata.sk_tx_prod = 0; sc_if->sk_cdata.sk_tx_cons = 0; sc_if->sk_cdata.sk_tx_cnt = 0; + + return (0); } int -sk_newbuf(struct sk_if_softc *sc_if, struct sk_chain *c, struct mbuf *m) +sk_newbuf(struct sk_if_softc *sc_if, int i, struct mbuf *m, + bus_dmamap_t dmamap) { + struct sk_softc *sc = sc_if->sk_softc; struct mbuf *m_new = NULL; + struct sk_chain *c; struct sk_rx_desc *r; + if (dmamap == NULL) { + /* if (m) panic() */ + + if (bus_dmamap_create(sc->sc_dmatag, MCLBYTES, 1, MCLBYTES, + 0, BUS_DMA_NOWAIT, &dmamap)) { + printf("%s: can't create recv map\n", + sc_if->sk_dev.dv_xname); + return(ENOMEM); + } + } else if (m == NULL) + bus_dmamap_unload(sc->sc_dmatag, dmamap); + + sc_if->sk_cdata.sk_rx_map[i] = dmamap; + if (m == NULL) { MGETHDR(m_new, M_DONTWAIT, MT_DATA); if (m_new == NULL) { @@ -560,6 +585,14 @@ sk_newbuf(struct sk_if_softc *sc_if, struct sk_chain *c, struct mbuf *m) m_freem(m_new); return (ENOBUFS); } + + m_new->m_len = m_new->m_pkthdr.len = MCLBYTES; + + m_adj(m_new, ETHER_ALIGN); + + if (bus_dmamap_load_mbuf(sc->sc_dmatag, dmamap, m_new, + BUS_DMA_NOWAIT)) + return(ENOBUFS); } else { /* * We're re-using a previously allocated mbuf; @@ -567,21 +600,16 @@ sk_newbuf(struct sk_if_softc *sc_if, struct sk_chain *c, struct mbuf *m) * default values. */ m_new = m; + m_new->m_len = m_new->m_pkthdr.len = MCLBYTES; + m_adj(m_new, ETHER_ALIGN); m_new->m_data = m_new->m_ext.ext_buf; } - m_new->m_len = m_new->m_pkthdr.len = MCLBYTES; - - /* - * Adjust alignment so packet payload begins on a - * longword boundary. Mandatory for Alpha, useful on - * x86 too. - */ - m_adj(m_new, ETHER_ALIGN); + c = &sc_if->sk_cdata.sk_rx_chain[i]; r = c->sk_desc; c->sk_mbuf = m_new; - r->sk_data_lo = vtophys(mtod(m_new, caddr_t)); - r->sk_ctl = m_new->m_len | SK_RXSTAT; + r->sk_data_lo = dmamap->dm_segs[0].ds_addr; + r->sk_ctl = dmamap->dm_segs[0].ds_len | SK_RXSTAT; return(0); } @@ -778,7 +806,6 @@ sk_attach(struct device *parent, struct device *self, void *aux) struct ifnet *ifp; caddr_t kva; bus_dma_segment_t seg; - bus_dmamap_t dmamap; int i, rseg; sc_if->sk_port = sa->skc_port; @@ -869,17 +896,18 @@ sk_attach(struct device *parent, struct device *self, void *aux) goto fail; } if (bus_dmamap_create(sc->sc_dmatag, sizeof(struct sk_ring_data), 1, - sizeof(struct sk_ring_data), 0, BUS_DMA_NOWAIT, &dmamap)) { + sizeof(struct sk_ring_data), 0, BUS_DMA_NOWAIT, + &sc_if->sk_ring_map)) { printf("%s: can't create dma map\n", sc_if->sk_dev.dv_xname); bus_dmamem_unmap(sc->sc_dmatag, kva, sizeof(struct sk_ring_data)); bus_dmamem_free(sc->sc_dmatag, &seg, rseg); goto fail; } - if (bus_dmamap_load(sc->sc_dmatag, dmamap, kva, + if (bus_dmamap_load(sc->sc_dmatag, sc_if->sk_ring_map, kva, sizeof(struct sk_ring_data), NULL, BUS_DMA_NOWAIT)) { printf("%s: can't load dma map\n", sc_if->sk_dev.dv_xname); - bus_dmamap_destroy(sc->sc_dmatag, dmamap); + bus_dmamap_destroy(sc->sc_dmatag, sc_if->sk_ring_map); bus_dmamem_unmap(sc->sc_dmatag, kva, sizeof(struct sk_ring_data)); bus_dmamem_free(sc->sc_dmatag, &seg, rseg); @@ -1134,9 +1162,17 @@ fail: int sk_encap(struct sk_if_softc *sc_if, struct mbuf *m_head, u_int32_t *txidx) { + struct sk_softc *sc = sc_if->sk_softc; struct sk_tx_desc *f = NULL; - struct mbuf *m; u_int32_t frag, cur, cnt = 0; + int i; + struct sk_txmap_entry *entry; + bus_dmamap_t txmap; + + entry = SLIST_FIRST(&sc_if->sk_txmap_listhead); + if (entry == NULL) + return (ENOBUFS); + txmap = entry->dmamap; cur = frag = *txidx; @@ -1145,30 +1181,29 @@ sk_encap(struct sk_if_softc *sc_if, struct mbuf *m_head, u_int32_t *txidx) * the fragment pointers. Stop when we run out * of fragments or hit the end of the mbuf chain. */ - for (m = m_head; m != NULL; m = m->m_next) { - if (m->m_len != 0) { - if ((SK_TX_RING_CNT - - (sc_if->sk_cdata.sk_tx_cnt + cnt)) < 2) - return(ENOBUFS); - f = &sc_if->sk_rdata->sk_tx_ring[frag]; - f->sk_data_lo = vtophys(mtod(m, vaddr_t)); - f->sk_ctl = m->m_len | SK_OPCODE_DEFAULT; - if (cnt == 0) - f->sk_ctl |= SK_TXCTL_FIRSTFRAG; - else - f->sk_ctl |= SK_TXCTL_OWN; - cur = frag; - SK_INC(frag, SK_TX_RING_CNT); - cnt++; - } - } - - if (m != NULL) + if (bus_dmamap_load_mbuf(sc->sc_dmatag, txmap, m_head, BUS_DMA_NOWAIT)) return(ENOBUFS); + for (i = 0; i < txmap->dm_nsegs; i++) { + if ((SK_TX_RING_CNT - (sc_if->sk_cdata.sk_tx_cnt + cnt)) < 2) + return(ENOBUFS); + f = &sc_if->sk_rdata->sk_tx_ring[frag]; + f->sk_data_lo = txmap->dm_segs[i].ds_addr; + f->sk_ctl = txmap->dm_segs[i].ds_len | SK_OPCODE_DEFAULT; + if (cnt == 0) + f->sk_ctl |= SK_TXCTL_FIRSTFRAG; + else + f->sk_ctl |= SK_TXCTL_OWN; + cur = frag; + SK_INC(frag, SK_TX_RING_CNT); + cnt++; + } + + sc_if->sk_cdata.sk_tx_chain[cur].sk_mbuf = m_head; + SLIST_REMOVE_HEAD(&sc_if->sk_txmap_listhead, link); + sc_if->sk_cdata.sk_tx_map[cur] = entry; sc_if->sk_rdata->sk_tx_ring[cur].sk_ctl |= SK_TXCTL_LASTFRAG|SK_TXCTL_EOF_INTR; - sc_if->sk_cdata.sk_tx_chain[cur].sk_mbuf = m_head; sc_if->sk_rdata->sk_tx_ring[*txidx].sk_ctl |= SK_TXCTL_OWN; sc_if->sk_cdata.sk_tx_cnt += cnt; @@ -1256,25 +1291,31 @@ sk_rxeof(struct sk_if_softc *sc_if) struct ifnet *ifp = &sc_if->arpcom.ac_if; struct mbuf *m; struct sk_chain *cur_rx; - int total_len = 0; - int i; + struct sk_rx_desc *cur_desc; + int i, cur, total_len = 0; u_int32_t rxstat; + bus_dmamap_t dmamap; i = sc_if->sk_cdata.sk_rx_prod; - cur_rx = &sc_if->sk_cdata.sk_rx_chain[i]; while(!(sc_if->sk_rdata->sk_rx_ring[i].sk_ctl & SK_RXCTL_OWN)) { + cur = i; + cur_rx = &sc_if->sk_cdata.sk_rx_chain[cur]; + cur_desc = &sc_if->sk_rdata->sk_rx_ring[cur]; - cur_rx = &sc_if->sk_cdata.sk_rx_chain[i]; - rxstat = sc_if->sk_rdata->sk_rx_ring[i].sk_xmac_rxstat; + rxstat = cur_desc->sk_xmac_rxstat; m = cur_rx->sk_mbuf; cur_rx->sk_mbuf = NULL; - total_len = SK_RXBYTES(sc_if->sk_rdata->sk_rx_ring[i].sk_ctl); + total_len = SK_RXBYTES(cur_desc->sk_ctl); + + dmamap = sc_if->sk_cdata.sk_rx_map[cur]; + sc_if->sk_cdata.sk_rx_map[cur] = 0; + SK_INC(i, SK_RX_RING_CNT); if (rxstat & XM_RXSTAT_ERRFRAME) { ifp->if_ierrors++; - sk_newbuf(sc_if, cur_rx, m); + sk_newbuf(sc_if, cur, m, dmamap); continue; } @@ -1285,11 +1326,11 @@ sk_rxeof(struct sk_if_softc *sc_if) * re-used. If allocating mbufs fails, then we * have to drop the packet. */ - if (sk_newbuf(sc_if, cur_rx, NULL) == ENOBUFS) { + if (sk_newbuf(sc_if, cur, NULL, dmamap) == ENOBUFS) { struct mbuf *m0; m0 = m_devget(mtod(m, char *) - ETHER_ALIGN, total_len + ETHER_ALIGN, 0, ifp, NULL); - sk_newbuf(sc_if, cur_rx, m); + sk_newbuf(sc_if, cur, m, dmamap); if (m0 == NULL) { printf("%s: no receive buffers " "available -- packet dropped!\n", @@ -1320,9 +1361,11 @@ sk_rxeof(struct sk_if_softc *sc_if) void sk_txeof(struct sk_if_softc *sc_if) { + struct sk_softc *sc = sc_if->sk_softc; struct sk_tx_desc *cur_tx = NULL; struct ifnet *ifp = &sc_if->arpcom.ac_if; u_int32_t idx; + struct sk_txmap_entry *entry; /* * Go through our tx ring and free mbufs for those @@ -1338,6 +1381,15 @@ sk_txeof(struct sk_if_softc *sc_if) if (sc_if->sk_cdata.sk_tx_chain[idx].sk_mbuf != NULL) { m_freem(sc_if->sk_cdata.sk_tx_chain[idx].sk_mbuf); sc_if->sk_cdata.sk_tx_chain[idx].sk_mbuf = NULL; + + entry = sc_if->sk_cdata.sk_tx_map[idx]; + bus_dmamap_sync(sc->sc_dmatag, entry->dmamap, 0, + entry->dmamap->dm_mapsize, BUS_DMASYNC_POSTWRITE); + + bus_dmamap_unload(sc->sc_dmatag, entry->dmamap); + SLIST_INSERT_HEAD(&sc_if->sk_txmap_listhead, entry, + link); + sc_if->sk_cdata.sk_tx_map[idx] = NULL; } sc_if->sk_cdata.sk_tx_cnt--; SK_INC(idx, SK_TX_RING_CNT); @@ -1767,12 +1819,12 @@ sk_init(void *xsc_if) /* Configure BMUs */ SK_IF_WRITE_4(sc_if, 0, SK_RXQ1_BMU_CSR, SK_RXBMU_ONLINE); SK_IF_WRITE_4(sc_if, 0, SK_RXQ1_CURADDR_LO, - vtophys(&sc_if->sk_rdata->sk_rx_ring[0])); + SK_RX_RING_ADDR(sc_if, 0)); SK_IF_WRITE_4(sc_if, 0, SK_RXQ1_CURADDR_HI, 0); SK_IF_WRITE_4(sc_if, 1, SK_TXQS1_BMU_CSR, SK_TXBMU_ONLINE); SK_IF_WRITE_4(sc_if, 1, SK_TXQS1_CURADDR_LO, - vtophys(&sc_if->sk_rdata->sk_tx_ring[0])); + SK_TX_RING_ADDR(sc_if, 0)); SK_IF_WRITE_4(sc_if, 1, SK_TXQS1_CURADDR_HI, 0); /* Init descriptors */ @@ -1783,7 +1835,14 @@ sk_init(void *xsc_if) splx(s); return; } - sk_init_tx_ring(sc_if); + + if (sk_init_tx_ring(sc_if) == ENOBUFS) { + printf("%s: initialization failed: no " + "memory for tx buffers\n", sc_if->sk_dev.dv_xname); + sk_stop(sc_if); + splx(s); + return; + } /* Configure interrupt handling */ CSR_READ_4(sc, SK_ISSR); |