diff options
Diffstat (limited to 'sys/dev/pci/if_ipw.c')
-rw-r--r-- | sys/dev/pci/if_ipw.c | 768 |
1 files changed, 319 insertions, 449 deletions
diff --git a/sys/dev/pci/if_ipw.c b/sys/dev/pci/if_ipw.c index e3dbebf2a7a..ad834ef122d 100644 --- a/sys/dev/pci/if_ipw.c +++ b/sys/dev/pci/if_ipw.c @@ -1,4 +1,4 @@ -/* $Id: if_ipw.c,v 1.29 2004/12/05 17:13:52 damien Exp $ */ +/* $Id: if_ipw.c,v 1.30 2004/12/05 17:46:07 damien Exp $ */ /*- * Copyright (c) 2004 @@ -80,6 +80,8 @@ static const struct ieee80211_rateset ipw_rateset_11b = int ipw_match(struct device *, void *, void *); void ipw_attach(struct device *, struct device *, void *); int ipw_detach(struct device *, int); +int ipw_dma_alloc(struct ipw_softc *); +void ipw_release(struct ipw_softc *); int ipw_media_change(struct ifnet *); void ipw_media_status(struct ifnet *, struct ifmediareq *); int ipw_newstate(struct ieee80211com *, enum ieee80211_state, int); @@ -103,12 +105,6 @@ int ipw_ioctl(struct ifnet *, u_long, caddr_t); u_int32_t ipw_read_table1(struct ipw_softc *, u_int32_t); void ipw_write_table1(struct ipw_softc *, u_int32_t, u_int32_t); int ipw_read_table2(struct ipw_softc *, u_int32_t, void *, u_int32_t *); -int ipw_dmamem_init(struct ipw_softc *); -void ipw_dmamem_stop(struct ipw_softc *); -int ipw_tx_init(struct ipw_softc *); -void ipw_tx_stop(struct ipw_softc *); -int ipw_rx_init(struct ipw_softc *); -void ipw_rx_stop(struct ipw_softc *); void ipw_stop_master(struct ipw_softc *); int ipw_reset(struct ipw_softc *); int ipw_load_ucode(struct ipw_softc *, u_char *, int); @@ -224,9 +220,8 @@ ipw_attach(struct device *parent, struct device *self, void *aux) return; } - /* set up dma memory */ - if (ipw_dmamem_init(sc) != 0) { - printf(": failed to initialize DMA memory\n"); + if (ipw_dma_alloc(sc) != 0) { + printf(": failed to allocate DMA resources\n"); return; } @@ -304,7 +299,6 @@ ipw_detach(struct device* self, int flags) struct ifnet *ifp = &sc->sc_ic.ic_if; ipw_stop(ifp, 1); - ipw_dmamem_stop(sc); #if NBPFILTER > 0 bpfdetach(ifp); @@ -312,6 +306,8 @@ ipw_detach(struct device* self, int flags) ieee80211_ifdetach(ifp); if_detach(ifp); + ipw_release(sc); + if (sc->sc_ih != NULL) { pci_intr_disestablish(sc->sc_pct, sc->sc_ih); sc->sc_ih = NULL; @@ -323,6 +319,265 @@ ipw_detach(struct device* self, int flags) } int +ipw_dma_alloc(struct ipw_softc *sc) +{ + struct ipw_soft_bd *sbd; + struct ipw_soft_hdr *shdr; + struct ipw_soft_buf *sbuf; + int i, nsegs, error; + + /* + * Allocate and map tx ring + */ + error = bus_dmamap_create(sc->sc_dmat, IPW_TBD_SZ, 1, IPW_TBD_SZ, 0, + BUS_DMA_NOWAIT, &sc->tbd_map); + if (error != 0) { + printf("%s: could not create tx ring DMA map\n"); + goto fail; + } + + error = bus_dmamem_alloc(sc->sc_dmat, IPW_TBD_SZ, PAGE_SIZE, 0, + &sc->tbd_seg, 1, &nsegs, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not allocate tx ring DMA memory\n"); + goto fail; + } + + error = bus_dmamem_map(sc->sc_dmat, &sc->tbd_seg, nsegs, IPW_TBD_SZ, + (caddr_t *)&sc->tbd_list, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not map tx ring DMA memory\n"); + goto fail; + } + + error = bus_dmamap_load(sc->sc_dmat, sc->tbd_map, sc->tbd_list, + IPW_TBD_SZ, NULL, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not load tx ring DMA map\n"); + goto fail; + } + + /* + * Allocate and map rx ring + */ + error = bus_dmamap_create(sc->sc_dmat, IPW_RBD_SZ, 1, IPW_RBD_SZ, 0, + BUS_DMA_NOWAIT, &sc->rbd_map); + if (error != 0) { + printf("%s: could not create rx ring DMA map\n"); + goto fail; + } + + error = bus_dmamem_alloc(sc->sc_dmat, IPW_RBD_SZ, PAGE_SIZE, 0, + &sc->rbd_seg, 1, &nsegs, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not allocate rx ring DMA memory\n"); + goto fail; + } + + error = bus_dmamem_map(sc->sc_dmat, &sc->rbd_seg, nsegs, IPW_RBD_SZ, + (caddr_t *)&sc->rbd_list, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not map rx ring DMA memory\n"); + goto fail; + } + + error = bus_dmamap_load(sc->sc_dmat, sc->rbd_map, sc->rbd_list, + IPW_RBD_SZ, NULL, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not load tx ring DMA map\n"); + goto fail; + } + + /* + * Allocate and map status ring + */ + error = bus_dmamap_create(sc->sc_dmat, IPW_STATUS_SZ, 1, IPW_STATUS_SZ, + 0, BUS_DMA_NOWAIT, &sc->status_map); + if (error != 0) { + printf("%s: could not create status ring DMA map\n"); + goto fail; + } + + error = bus_dmamem_alloc(sc->sc_dmat, IPW_STATUS_SZ, PAGE_SIZE, 0, + &sc->status_seg, 1, &nsegs, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not allocate status ring DMA memory\n"); + goto fail; + } + + error = bus_dmamem_map(sc->sc_dmat, &sc->status_seg, nsegs, + IPW_STATUS_SZ, (caddr_t *)&sc->status_list, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not map status ring DMA memory\n"); + goto fail; + } + + error = bus_dmamap_load(sc->sc_dmat, sc->status_map, sc->status_list, + IPW_STATUS_SZ, NULL, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not load status ring DMA map\n"); + goto fail; + } + + /* + * Allocate command DMA map + */ + error = bus_dmamap_create(sc->sc_dmat, sizeof (struct ipw_cmd), 1, + sizeof (struct ipw_cmd), 0, BUS_DMA_NOWAIT, &sc->cmd_map); + if (error != 0) { + printf("%s: could not create command DMA map\n"); + goto fail; + } + + /* + * Allocate headers DMA maps + */ + SLIST_INIT(&sc->free_shdr); + for (i = 0; i < IPW_NDATA; i++) { + shdr = &sc->shdr_list[i]; + error = bus_dmamap_create(sc->sc_dmat, sizeof (struct ipw_hdr), + 1, sizeof (struct ipw_hdr), 0, BUS_DMA_NOWAIT, &shdr->map); + if (error != 0) { + printf("%s: could not create header DMA map\n"); + goto fail; + } + SLIST_INSERT_HEAD(&sc->free_shdr, shdr, next); + } + + /* + * Allocate tx buffers DMA maps + */ + SLIST_INIT(&sc->free_sbuf); + for (i = 0; i < IPW_NDATA; i++) { + sbuf = &sc->tx_sbuf_list[i]; + error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, IPW_MAX_NSEG, + MCLBYTES, 0, BUS_DMA_NOWAIT, &sbuf->map); + if (error != 0) { + printf("%s: could not create tx DMA map\n"); + goto fail; + } + SLIST_INSERT_HEAD(&sc->free_sbuf, sbuf, next); + } + + /* + * Initialize tx ring + */ + for (i = 0; i < IPW_NTBD; i++) { + sbd = &sc->stbd_list[i]; + sbd->bd = &sc->tbd_list[i]; + sbd->type = IPW_SBD_TYPE_NOASSOC; + } + + /* + * Pre-allocate rx buffers and DMA maps + */ + for (i = 0; i < IPW_NRBD; i++) { + sbd = &sc->srbd_list[i]; + sbuf = &sc->rx_sbuf_list[i]; + sbd->bd = &sc->rbd_list[i]; + + MGETHDR(sbuf->m, M_DONTWAIT, MT_DATA); + if (sbuf->m == NULL) { + printf("%s: could not allocate rx mbuf\n"); + error = ENOMEM; + goto fail; + } + + MCLGET(sbuf->m, M_DONTWAIT); + if (!(sbuf->m->m_flags & M_EXT)) { + m_freem(sbuf->m); + printf("%s: could not allocate rx mbuf cluster\n"); + error = ENOMEM; + goto fail; + } + + error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, 1, MCLBYTES, + 0, BUS_DMA_NOWAIT, &sbuf->map); + if (error != 0) { + printf("%s: could not create rx DMA map\n"); + goto fail; + } + + error = bus_dmamap_load(sc->sc_dmat, sbuf->map, + mtod(sbuf->m, void *), MCLBYTES, NULL, BUS_DMA_NOWAIT); + if (error != 0) { + printf("%s: could not map rx DMA memory\n"); + goto fail; + } + + sbd->type = IPW_SBD_TYPE_DATA; + sbd->priv = sbuf; + sbd->bd->physaddr = htole32(sbuf->map->dm_segs[0].ds_addr); + sbd->bd->len = htole32(MCLBYTES); + } + + bus_dmamap_sync(sc->sc_dmat, sc->rbd_map, 0, IPW_RBD_SZ, + BUS_DMASYNC_PREWRITE); + + return 0; + +fail: ipw_release(sc); + return error; +} + +void +ipw_release(struct ipw_softc *sc) +{ + struct ipw_soft_buf *sbuf; + int i; + + if (sc->tbd_map != NULL) { + if (sc->tbd_list != NULL) { + bus_dmamap_unload(sc->sc_dmat, sc->tbd_map); + bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->tbd_list, + IPW_TBD_SZ); + bus_dmamem_free(sc->sc_dmat, &sc->tbd_seg, 1); + } + bus_dmamap_destroy(sc->sc_dmat, sc->tbd_map); + } + + if (sc->rbd_map != NULL) { + if (sc->rbd_list != NULL) { + bus_dmamap_unload(sc->sc_dmat, sc->rbd_map); + bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->rbd_list, + IPW_RBD_SZ); + bus_dmamem_free(sc->sc_dmat, &sc->rbd_seg, 1); + } + bus_dmamap_destroy(sc->sc_dmat, sc->rbd_map); + } + + if (sc->status_map != NULL) { + if (sc->status_list != NULL) { + bus_dmamap_unload(sc->sc_dmat, sc->status_map); + bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->status_list, + IPW_RBD_SZ); + bus_dmamem_free(sc->sc_dmat, &sc->status_seg, 1); + } + bus_dmamap_destroy(sc->sc_dmat, sc->status_map); + } + + if (sc->cmd_map != NULL) + bus_dmamap_destroy(sc->sc_dmat, sc->cmd_map); + + for (i = 0; i < IPW_NDATA; i++) + bus_dmamap_destroy(sc->sc_dmat, sc->shdr_list[i].map); + + for (i = 0; i < IPW_NDATA; i++) + bus_dmamap_destroy(sc->sc_dmat, sc->tx_sbuf_list[i].map); + + for (i = 0; i < IPW_NRBD; i++) { + sbuf = &sc->rx_sbuf_list[i]; + if (sbuf->map != NULL) { + if (sbuf->m != NULL) { + bus_dmamap_unload(sc->sc_dmat, sbuf->map); + m_freem(sbuf->m); + } + bus_dmamap_destroy(sc->sc_dmat, sbuf->map); + } + } +} + +int ipw_media_change(struct ifnet *ifp) { int error; @@ -678,7 +933,7 @@ ipw_rx_intr(struct ipw_softc *sc) void ipw_release_sbd(struct ipw_softc *sc, struct ipw_soft_bd *sbd) { - struct ieee80211com *ic; + struct ieee80211com *ic = &sc->sc_ic; struct ipw_soft_hdr *shdr; struct ipw_soft_buf *sbuf; @@ -690,19 +945,21 @@ ipw_release_sbd(struct ipw_softc *sc, struct ipw_soft_bd *sbd) case IPW_SBD_TYPE_HEADER: shdr = sbd->priv; bus_dmamap_unload(sc->sc_dmat, shdr->map); - TAILQ_INSERT_TAIL(&sc->sc_free_shdr, shdr, next); + SLIST_INSERT_HEAD(&sc->free_shdr, shdr, next); break; case IPW_SBD_TYPE_DATA: - ic = &sc->sc_ic; sbuf = sbd->priv; bus_dmamap_unload(sc->sc_dmat, sbuf->map); + SLIST_INSERT_HEAD(&sc->free_sbuf, sbuf, next); + m_freem(sbuf->m); + if (sbuf->ni != NULL && sbuf->ni != ic->ic_bss) ieee80211_free_node(ic, sbuf->ni); + /* kill watchdog timer */ sc->sc_tx_timer = 0; - TAILQ_INSERT_TAIL(&sc->sc_free_sbuf, sbuf, next); break; } sbd->type = IPW_SBD_TYPE_NOASSOC; @@ -716,15 +973,19 @@ ipw_tx_intr(struct ipw_softc *sc) r = CSR_READ_4(sc, IPW_CSR_TX_READ_INDEX); - for (i = (sc->txold + 1) % IPW_NTBD; i != r; i = (i + 1) % IPW_NTBD) + for (i = (sc->txold + 1) % IPW_NTBD; i != r; i = (i + 1) % IPW_NTBD) { ipw_release_sbd(sc, &sc->stbd_list[i]); + sc->txfree++; + } /* Remember what the firmware has processed */ sc->txold = (r == 0) ? IPW_NTBD - 1 : r - 1; /* Call start() since some buffer descriptors have been released */ - ifp->if_flags &= ~IFF_OACTIVE; - (*ifp->if_start)(ifp); + if (sc->txfree >= 1 + IPW_MAX_NSEG) { + ifp->if_flags &= ~IFF_OACTIVE; + (*ifp->if_start)(ifp); + } } int @@ -774,7 +1035,7 @@ ipw_cmd(struct ipw_softc *sc, u_int32_t type, void *data, u_int32_t len) sbd = &sc->stbd_list[sc->txcur]; - error = bus_dmamap_load(sc->sc_dmat, sc->cmd_map, sc->cmd, + error = bus_dmamap_load(sc->sc_dmat, sc->cmd_map, &sc->cmd, sizeof (struct ipw_cmd), NULL, BUS_DMA_NOWAIT); if (error != 0) { printf("%s: could not map cmd dma memory\n", @@ -782,12 +1043,12 @@ ipw_cmd(struct ipw_softc *sc, u_int32_t type, void *data, u_int32_t len) return error; } - sc->cmd->type = htole32(type); - sc->cmd->subtype = htole32(0); - sc->cmd->len = htole32(len); - sc->cmd->seq = htole32(0); + sc->cmd.type = htole32(type); + sc->cmd.subtype = htole32(0); + sc->cmd.len = htole32(len); + sc->cmd.seq = htole32(0); if (data != NULL) - bcopy(data, sc->cmd->data, len); + bcopy(data, sc->cmd.data, len); sbd->type = IPW_SBD_TYPE_COMMAND; sbd->bd->physaddr = htole32(sc->cmd_map->dm_segs[0].ds_addr); @@ -804,6 +1065,7 @@ ipw_cmd(struct ipw_softc *sc, u_int32_t type, void *data, u_int32_t len) BUS_DMASYNC_PREWRITE); sc->txcur = (sc->txcur + 1) % IPW_NTBD; + sc->txfree--; CSR_WRITE_4(sc, IPW_CSR_TX_WRITE_INDEX, sc->txcur); DPRINTFN(2, ("TX!CMD!%u!%u!%u!%u\n", type, 0, 0, len)); @@ -849,8 +1111,8 @@ ipw_tx_start(struct ifnet *ifp, struct mbuf *m, struct ieee80211_node *ni) wh = mtod(m, struct ieee80211_frame *); - shdr = TAILQ_FIRST(&sc->sc_free_shdr); - sbuf = TAILQ_FIRST(&sc->sc_free_sbuf); + shdr = SLIST_FIRST(&sc->free_shdr); + sbuf = SLIST_FIRST(&sc->free_sbuf); shdr->hdr.type = htole32(IPW_HDR_TYPE_SEND); shdr->hdr.subtype = htole32(0); @@ -890,8 +1152,8 @@ ipw_tx_start(struct ifnet *ifp, struct mbuf *m, struct ieee80211_node *ni) return error; } - TAILQ_REMOVE(&sc->sc_free_sbuf, sbuf, next); - TAILQ_REMOVE(&sc->sc_free_shdr, shdr, next); + SLIST_REMOVE_HEAD(&sc->free_sbuf, next); + SLIST_REMOVE_HEAD(&sc->free_shdr, next); sbd = &sc->stbd_list[sc->txcur]; sbd->type = IPW_SBD_TYPE_HEADER; @@ -912,6 +1174,7 @@ ipw_tx_start(struct ifnet *ifp, struct mbuf *m, struct ieee80211_node *ni) sizeof (struct ipw_bd), BUS_DMASYNC_PREWRITE); sc->txcur = (sc->txcur + 1) % IPW_NTBD; + sc->txfree--; sbuf->m = m; sbuf->ni = ni; @@ -939,6 +1202,7 @@ ipw_tx_start(struct ifnet *ifp, struct mbuf *m, struct ieee80211_node *ni) sizeof (struct ipw_bd), BUS_DMASYNC_PREWRITE); sc->txcur = (sc->txcur + 1) % IPW_NTBD; + sc->txfree--; } bus_dmamap_sync(sc->sc_dmat, shdr->map, 0, sizeof (struct ipw_hdr), @@ -950,6 +1214,9 @@ ipw_tx_start(struct ifnet *ifp, struct mbuf *m, struct ieee80211_node *ni) /* Inform firmware about this new packet */ CSR_WRITE_4(sc, IPW_CSR_TX_WRITE_INDEX, sc->txcur); + if (sc->txfree < 1 + IPW_MAX_NSEG) + ifp->if_flags |= IFF_OACTIVE; + return 0; } @@ -1153,415 +1420,6 @@ ipw_read_table2(struct ipw_softc *sc, u_int32_t off, void *buf, u_int32_t *len) return 0; } -/* - * Allocate and map DMAble memory. Do not call in interrupt context. - */ -int -ipw_dmamem_init(struct ipw_softc *sc) -{ - int error, nsegs; - char *errmsg; - - error = bus_dmamem_alloc(sc->sc_dmat, IPW_TBD_SZ, PAGE_SIZE, 0, - &sc->tbd_seg, 1, &nsegs, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not allocate tbd dma memory"; - goto fail; - } - - error = bus_dmamem_map(sc->sc_dmat, &sc->tbd_seg, nsegs, IPW_TBD_SZ, - (caddr_t *)&sc->tbd_list, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not map tbd dma memory"; - goto fail; - } - - error = bus_dmamem_alloc(sc->sc_dmat, sizeof (struct ipw_cmd), - PAGE_SIZE, 0, &sc->cmd_seg, 1, &nsegs, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not allocate cmd dma memory"; - goto fail; - } - - error = bus_dmamem_map(sc->sc_dmat, &sc->cmd_seg, nsegs, - sizeof (struct ipw_cmd), (caddr_t *)&sc->cmd, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not map cmd dma memory"; - goto fail; - } - - error = bus_dmamem_alloc(sc->sc_dmat, IPW_RBD_SZ, PAGE_SIZE, 0, - &sc->rbd_seg, 1, &nsegs, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not allocate rbd dma memory"; - goto fail; - } - - error = bus_dmamem_map(sc->sc_dmat, &sc->rbd_seg, nsegs, IPW_RBD_SZ, - (caddr_t *)&sc->rbd_list, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not map rbd dma memory"; - goto fail; - } - - error = bus_dmamem_alloc(sc->sc_dmat, IPW_STATUS_SZ, PAGE_SIZE, 0, - &sc->status_seg, 1, &nsegs, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not allocate status dma memory"; - goto fail; - } - - error = bus_dmamem_map(sc->sc_dmat, &sc->status_seg, nsegs, - IPW_STATUS_SZ, (caddr_t *)&sc->status_list, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not map status dma memory"; - goto fail; - } - - return 0; - -fail: printf("%s: %s", sc->sc_dev.dv_xname, errmsg); - ipw_dmamem_stop(sc); - - return error; -} - -/* - * Unmap and free DMAble memory. - */ -void -ipw_dmamem_stop(struct ipw_softc *sc) -{ - if (sc->tbd_list != NULL) { - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->tbd_list, - IPW_TBD_SZ); - bus_dmamem_free(sc->sc_dmat, &sc->tbd_seg, 1); - sc->tbd_list = NULL; - } - - if (sc->cmd != NULL) { - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->cmd, - sizeof (struct ipw_cmd)); - bus_dmamem_free(sc->sc_dmat, &sc->cmd_seg, 1); - sc->cmd = NULL; - } - - if (sc->rbd_list != NULL) { - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->rbd_list, - IPW_RBD_SZ); - bus_dmamem_free(sc->sc_dmat, &sc->rbd_seg, 1); - sc->rbd_list = NULL; - } - - if (sc->status_list != NULL) { - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)sc->status_list, - IPW_STATUS_SZ); - bus_dmamem_free(sc->sc_dmat, &sc->status_seg, 1); - sc->status_list = NULL; - } -} - -int -ipw_tx_init(struct ipw_softc *sc) -{ - char *errmsg; - struct ipw_bd *bd; - struct ipw_soft_bd *sbd; - struct ipw_soft_hdr *shdr; - struct ipw_soft_buf *sbuf; - int error, i; - - /* Allocate transmission buffer descriptors */ - error = bus_dmamap_create(sc->sc_dmat, IPW_TBD_SZ, 1, IPW_TBD_SZ, 0, - BUS_DMA_NOWAIT, &sc->tbd_map); - if (error != 0) { - errmsg = "could not create tbd dma map"; - goto fail; - } - - error = bus_dmamap_load(sc->sc_dmat, sc->tbd_map, sc->tbd_list, - IPW_TBD_SZ, NULL, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not load tbd dma memory"; - goto fail; - } - - sc->stbd_list = malloc(IPW_NTBD * sizeof (struct ipw_soft_bd), - M_DEVBUF, M_NOWAIT); - if (sc->stbd_list == NULL) { - errmsg = "could not allocate soft tbd"; - error = ENOMEM; - goto fail; - } - sbd = sc->stbd_list; - bd = sc->tbd_list; - for (i = 0; i < IPW_NTBD; i++, sbd++, bd++) { - sbd->type = IPW_SBD_TYPE_NOASSOC; - sbd->bd = bd; - } - - CSR_WRITE_4(sc, IPW_CSR_TX_BD_BASE, sc->tbd_map->dm_segs[0].ds_addr); - CSR_WRITE_4(sc, IPW_CSR_TX_BD_SIZE, IPW_NTBD); - CSR_WRITE_4(sc, IPW_CSR_TX_READ_INDEX, 0); - CSR_WRITE_4(sc, IPW_CSR_TX_WRITE_INDEX, 0); - sc->txold = IPW_NTBD - 1; /* latest bd index ack'ed by firmware */ - sc->txcur = 0; /* bd index to write to */ - - /* Allocate a DMA-able command */ - error = bus_dmamap_create(sc->sc_dmat, sizeof (struct ipw_cmd), 1, - sizeof (struct ipw_cmd), 0, BUS_DMA_NOWAIT, &sc->cmd_map); - if (error != 0) { - errmsg = "could not create cmd dma map"; - goto fail; - } - - /* Allocate a pool of DMA-able headers */ - sc->shdr_list = malloc(IPW_NDATA * sizeof (struct ipw_soft_hdr), - M_DEVBUF, M_NOWAIT); - if (sc->shdr_list == NULL) { - errmsg = "could not allocate soft hdr"; - error = ENOMEM; - goto fail; - } - TAILQ_INIT(&sc->sc_free_shdr); - for (i = 0, shdr = sc->shdr_list; i < IPW_NDATA; i++, shdr++) { - error = bus_dmamap_create(sc->sc_dmat, - sizeof (struct ipw_soft_hdr), 1, - sizeof (struct ipw_soft_hdr), 0, BUS_DMA_NOWAIT, - &shdr->map); - if (error != 0) { - errmsg = "could not create hdr dma map"; - goto fail; - } - TAILQ_INSERT_TAIL(&sc->sc_free_shdr, shdr, next); - } - - /* Allocate a pool of DMA-able buffers */ - sc->tx_sbuf_list = malloc(IPW_NDATA * sizeof (struct ipw_soft_buf), - M_DEVBUF, M_NOWAIT); - if (sc->tx_sbuf_list == NULL) { - errmsg = "could not allocate soft txbuf"; - error = ENOMEM; - goto fail; - } - TAILQ_INIT(&sc->sc_free_sbuf); - for (i = 0, sbuf = sc->tx_sbuf_list; i < IPW_NDATA; i++, sbuf++) { - error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, IPW_NDATA, - MCLBYTES, 0, BUS_DMA_NOWAIT, &sbuf->map); - if (error != 0) { - errmsg = "could not create txbuf dma map"; - goto fail; - } - TAILQ_INSERT_TAIL(&sc->sc_free_sbuf, sbuf, next); - } - - return 0; - -fail: printf("%s: %s\n", sc->sc_dev.dv_xname, errmsg); - ipw_tx_stop(sc); - - return error; -} - -void -ipw_tx_stop(struct ipw_softc *sc) -{ - struct ipw_soft_hdr *shdr; - struct ipw_soft_buf *sbuf; - int i; - - if (sc->tbd_map != NULL) { - if (sc->tbd_list != NULL) - bus_dmamap_unload(sc->sc_dmat, sc->tbd_map); - bus_dmamap_destroy(sc->sc_dmat, sc->tbd_map); - sc->tbd_map = NULL; - } - - if (sc->stbd_list != NULL) { - for (i = 0; i < IPW_NTBD; i++) - ipw_release_sbd(sc, &sc->stbd_list[i]); - free(sc->stbd_list, M_DEVBUF); - sc->stbd_list = NULL; - } - - if (sc->cmd_map != NULL) { - bus_dmamap_destroy(sc->sc_dmat, sc->cmd_map); - sc->cmd_map = NULL; - } - - if (sc->shdr_list != NULL) { - TAILQ_FOREACH(shdr, &sc->sc_free_shdr, next) - bus_dmamap_destroy(sc->sc_dmat, shdr->map); - free(sc->shdr_list, M_DEVBUF); - sc->shdr_list = NULL; - } - - if (sc->tx_sbuf_list != NULL) { - TAILQ_FOREACH(sbuf, &sc->sc_free_sbuf, next) - bus_dmamap_destroy(sc->sc_dmat, sbuf->map); - free(sc->tx_sbuf_list, M_DEVBUF); - sc->tx_sbuf_list = NULL; - } -} - -int -ipw_rx_init(struct ipw_softc *sc) -{ - char *errmsg; - struct ipw_bd *bd; - struct ipw_soft_bd *sbd; - struct ipw_soft_buf *sbuf; - int error, i; - - /* Allocate reception buffer descriptors */ - error = bus_dmamap_create(sc->sc_dmat, IPW_RBD_SZ, 1, IPW_RBD_SZ, 0, - BUS_DMA_NOWAIT, &sc->rbd_map); - if (error != 0) { - errmsg = "could not create rbd dma map"; - goto fail; - } - - error = bus_dmamap_load(sc->sc_dmat, sc->rbd_map, sc->rbd_list, - IPW_RBD_SZ, NULL, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not load rbd dma memory"; - goto fail; - } - - sc->srbd_list = malloc(IPW_NRBD * sizeof (struct ipw_soft_bd), - M_DEVBUF, M_NOWAIT); - if (sc->srbd_list == NULL) { - errmsg = "could not allocate soft rbd"; - error = ENOMEM; - goto fail; - } - sbd = sc->srbd_list; - bd = sc->rbd_list; - for (i = 0; i < IPW_NRBD; i++, sbd++, bd++) { - sbd->type = IPW_SBD_TYPE_NOASSOC; - sbd->bd = bd; - } - - CSR_WRITE_4(sc, IPW_CSR_RX_BD_BASE, sc->rbd_map->dm_segs[0].ds_addr); - CSR_WRITE_4(sc, IPW_CSR_RX_BD_SIZE, IPW_NRBD); - CSR_WRITE_4(sc, IPW_CSR_RX_READ_INDEX, 0); - CSR_WRITE_4(sc, IPW_CSR_RX_WRITE_INDEX, IPW_NRBD - 1); - sc->rxcur = IPW_NRBD - 1; /* latest bd index I've read */ - - /* Allocate status descriptors */ - error = bus_dmamap_create(sc->sc_dmat, IPW_STATUS_SZ, 1, IPW_STATUS_SZ, - 0, BUS_DMA_NOWAIT, &sc->status_map); - if (error != 0) { - errmsg = "could not create status dma map"; - goto fail; - } - - error = bus_dmamap_load(sc->sc_dmat, sc->status_map, sc->status_list, - IPW_STATUS_SZ, NULL, BUS_DMA_NOWAIT); - if (error != 0) { - errmsg = "could not load status dma memory"; - goto fail; - } - - CSR_WRITE_4(sc, IPW_CSR_RX_STATUS_BASE, - sc->status_map->dm_segs[0].ds_addr); - - sc->rx_sbuf_list = malloc(IPW_NRBD * sizeof (struct ipw_soft_buf), - M_DEVBUF, M_NOWAIT); - if (sc->rx_sbuf_list == NULL) { - errmsg = "could not allocate soft rxbuf"; - error = ENOMEM; - goto fail; - } - - sbuf = sc->rx_sbuf_list; - sbd = sc->srbd_list; - for (i = 0; i < IPW_NRBD; i++, sbuf++, sbd++) { - - MGETHDR(sbuf->m, M_DONTWAIT, MT_DATA); - if (sbuf->m == NULL) { - errmsg = "could not allocate rx mbuf"; - error = ENOMEM; - goto fail; - } - MCLGET(sbuf->m, M_DONTWAIT); - if (!(sbuf->m->m_flags & M_EXT)) { - m_freem(sbuf->m); - errmsg = "could not allocate rx mbuf cluster"; - error = ENOMEM; - goto fail; - } - - error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, 1, MCLBYTES, - 0, BUS_DMA_NOWAIT, &sbuf->map); - if (error != 0) { - m_freem(sbuf->m); - errmsg = "could not create rxbuf dma map"; - goto fail; - } - error = bus_dmamap_load(sc->sc_dmat, sbuf->map, - mtod(sbuf->m, void *), MCLBYTES, NULL, BUS_DMA_NOWAIT); - if (error != 0) { - bus_dmamap_destroy(sc->sc_dmat, sbuf->map); - m_freem(sbuf->m); - errmsg = "could not map rxbuf dma memory"; - goto fail; - } - sbd->type = IPW_SBD_TYPE_DATA; - sbd->priv = sbuf; - sbd->bd->physaddr = htole32(sbuf->map->dm_segs[0].ds_addr); - sbd->bd->len = htole32(MCLBYTES); - } - - return 0; - -fail: printf("%s: %s\n", sc->sc_dev.dv_xname, errmsg); - ipw_rx_stop(sc); - - return error; -} - -void -ipw_rx_stop(struct ipw_softc *sc) -{ - struct ipw_soft_bd *sbd; - struct ipw_soft_buf *sbuf; - int i; - - if (sc->rbd_map != NULL) { - if (sc->rbd_list != NULL) - bus_dmamap_unload(sc->sc_dmat, sc->rbd_map); - bus_dmamap_destroy(sc->sc_dmat, sc->rbd_map); - sc->rbd_map = NULL; - } - - if (sc->status_map != NULL) { - if (sc->status_list != NULL) - bus_dmamap_unload(sc->sc_dmat, sc->status_map); - bus_dmamap_destroy(sc->sc_dmat, sc->status_map); - sc->status_map = NULL; - } - - if (sc->srbd_list != NULL) { - for (i = 0, sbd = sc->srbd_list; i < IPW_NRBD; i++, sbd++) { - if (sbd->type == IPW_SBD_TYPE_NOASSOC) - continue; - - sbuf = sbd->priv; - bus_dmamap_unload(sc->sc_dmat, sbuf->map); - bus_dmamap_destroy(sc->sc_dmat, sbuf->map); - m_freem(sbuf->m); - } - free(sc->srbd_list, M_DEVBUF); - sc->srbd_list = NULL; - } - - if (sc->rx_sbuf_list != NULL) { - free(sc->rx_sbuf_list, M_DEVBUF); - sc->rx_sbuf_list = NULL; - } -} - void ipw_stop_master(struct ipw_softc *sc) { @@ -1996,17 +1854,25 @@ ipw_init(struct ifnet *ifp) ipw_stop_master(sc); - if ((error = ipw_rx_init(sc)) != 0) { - printf("%s: could not initialize rx queue\n", - sc->sc_dev.dv_xname); - goto fail2; - } + /* + * Setup tx, rx and status rings + */ + CSR_WRITE_4(sc, IPW_CSR_TX_BD_BASE, sc->tbd_map->dm_segs[0].ds_addr); + CSR_WRITE_4(sc, IPW_CSR_TX_BD_SIZE, IPW_NTBD); + CSR_WRITE_4(sc, IPW_CSR_TX_READ_INDEX, 0); + CSR_WRITE_4(sc, IPW_CSR_TX_WRITE_INDEX, 0); + sc->txold = IPW_NTBD - 1; /* latest bd index ack'ed by firmware */ + sc->txcur = 0; /* bd index to write to */ + sc->txfree = IPW_NTBD - 2; - if ((error = ipw_tx_init(sc)) != 0) { - printf("%s: could not initialize tx queue\n", - sc->sc_dev.dv_xname); - goto fail2; - } + CSR_WRITE_4(sc, IPW_CSR_RX_BD_BASE, sc->rbd_map->dm_segs[0].ds_addr); + CSR_WRITE_4(sc, IPW_CSR_RX_BD_SIZE, IPW_NRBD); + CSR_WRITE_4(sc, IPW_CSR_RX_READ_INDEX, 0); + CSR_WRITE_4(sc, IPW_CSR_RX_WRITE_INDEX, IPW_NRBD - 1); + sc->rxcur = IPW_NRBD - 1; /* latest bd index I've read */ + + CSR_WRITE_4(sc, IPW_CSR_RX_STATUS_BASE, + sc->status_map->dm_segs[0].ds_addr); if ((error = ipw_load_firmware(sc, fw.main, fw.main_size)) != 0) { printf("%s: could not load firmware\n", sc->sc_dev.dv_xname); @@ -2043,12 +1909,16 @@ ipw_stop(struct ifnet *ifp, int disable) { struct ipw_softc *sc = ifp->if_softc; struct ieee80211com *ic = &sc->sc_ic; + int i; ipw_stop_master(sc); CSR_WRITE_4(sc, IPW_CSR_RST, IPW_RST_SW_RESET); - ipw_tx_stop(sc); - ipw_rx_stop(sc); + /* + * Release tx buffers + */ + for (i = 0; i < IPW_NTBD; i++) + ipw_release_sbd(sc, &sc->stbd_list[i]); ifp->if_timer = 0; ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); |