diff options
author | David Gwynne <dlg@cvs.openbsd.org> | 2022-01-10 04:11:14 +0000 |
---|---|---|
committer | David Gwynne <dlg@cvs.openbsd.org> | 2022-01-10 04:11:14 +0000 |
commit | ca0a2f5df465defa6b1316ba181c207f91c626a8 (patch) | |
tree | 882d1c5253b3dcf84ce296fd7bd75118ea3b105b /sys/dev/pci | |
parent | 4e82ebc09d72ad823b2707c59e10a91e3ac17e6c (diff) |
handle the status ring entries as 64bit words instead of a struct.
the status ring entries are 8 bytes/64bit, and depending on the
type of entry it has fields all over the place. this loads the
descriptor with a single 64bit read, and then shifts and masks the
bits out of it depending on the type of descriptor. this looks
cleaner for the tx completions in particular.
Diffstat (limited to 'sys/dev/pci')
-rw-r--r-- | sys/dev/pci/if_msk.c | 132 | ||||
-rw-r--r-- | sys/dev/pci/if_mskvar.h | 10 |
2 files changed, 94 insertions, 48 deletions
diff --git a/sys/dev/pci/if_msk.c b/sys/dev/pci/if_msk.c index 79ba835e5f4..3bfb1c41399 100644 --- a/sys/dev/pci/if_msk.c +++ b/sys/dev/pci/if_msk.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_msk.c,v 1.138 2022/01/09 05:42:54 jsg Exp $ */ +/* $OpenBSD: if_msk.c,v 1.139 2022/01/10 04:11:13 dlg Exp $ */ /* * Copyright (c) 1997, 1998, 1999, 2000 @@ -120,6 +120,53 @@ #include <dev/pci/if_skreg.h> #include <dev/pci/if_mskvar.h> +#define MSK_STATUS_OWN_SHIFT 63 +#define MSK_STATUS_OWN_MASK 0x1 +#define MSK_STATUS_OPCODE_SHIFT 56 +#define MSK_STATUS_OPCODE_MASK 0x7f + +#define MSK_STATUS_OWN(_d) \ + (((_d) >> MSK_STATUS_OWN_SHIFT) & MSK_STATUS_OWN_MASK) +#define MSK_STATUS_OPCODE(_d) \ + (((_d) >> MSK_STATUS_OPCODE_SHIFT) & MSK_STATUS_OPCODE_MASK) + +#define MSK_STATUS_OPCODE_RXSTAT 0x60 +#define MSK_STATUS_OPCODE_RXTIMESTAMP 0x61 +#define MSK_STATUS_OPCODE_RXVLAN 0x62 +#define MSK_STATUS_OPCODE_RXCKSUM 0x64 +#define MSK_STATUS_OPCODE_RXCKSUMVLAN \ + (MSK_STATUS_OPCODE_RXVLAN | MSK_STATUS_OPCODE_RXCKSUM) +#define MSK_STATUS_OPCODE_RXTIMEVLAN \ + (MSK_STATUS_OPCODE_RXVLAN | MSK_STATUS_OPCODE_RXTIMESTAMP) +#define MSK_STATUS_OPCODE_RSS_HASH 0x65 +#define MSK_STATUS_OPCODE_TXIDX 0x68 +#define MSK_STATUS_OPCODE_MACSEC 0x6c +#define MSK_STATUS_OPCODE_PUTIDX 0x70 + +#define MSK_STATUS_RXSTAT_PORT_SHIFT 48 +#define MSK_STATUS_RXSTAT_PORT_MASK 0x1 +#define MSK_STATUS_RXSTAT_LEN_SHIFT 32 +#define MSK_STATUS_RXSTAT_LEN_MASK 0xffff +#define MSK_STATUS_RXSTAT_STATUS_SHIFT 0 +#define MSK_STATUS_RXSTAT_STATUS_MASK 0xffffffff + +#define MSK_STATUS_RXSTAT_PORT(_d) \ + (((_d) >> MSK_STATUS_RXSTAT_PORT_SHIFT) & MSK_STATUS_RXSTAT_PORT_MASK) +#define MSK_STATUS_RXSTAT_LEN(_d) \ + (((_d) >> MSK_STATUS_RXSTAT_LEN_SHIFT) & MSK_STATUS_RXSTAT_LEN_MASK) +#define MSK_STATUS_RXSTAT_STATUS(_d) \ + (((_d) >> MSK_STATUS_RXSTAT_STATUS_SHIFT) & MSK_STATUS_RXSTAT_STATUS_MASK) + +#define MSK_STATUS_TXIDX_PORTA_SHIFT 0 +#define MSK_STATUS_TXIDX_PORTA_MASK 0xfff +#define MSK_STATUS_TXIDX_PORTB_SHIFT 24 +#define MSK_STATUS_TXIDX_PORTB_MASK 0xfff + +#define MSK_STATUS_TXIDX_PORTA(_d) \ + (((_d) >> MSK_STATUS_TXIDX_PORTA_SHIFT) & MSK_STATUS_TXIDX_PORTA_MASK) +#define MSK_STATUS_TXIDX_PORTB(_d) \ + (((_d) >> MSK_STATUS_TXIDX_PORTB_SHIFT) & MSK_STATUS_TXIDX_PORTB_MASK) + int mskc_probe(struct device *, void *, void *); void mskc_attach(struct device *, struct device *self, void *aux); int mskc_detach(struct device *, int); @@ -624,6 +671,7 @@ mskc_reset(struct sk_softc *sc) { u_int32_t imtimer_ticks, reg1; int reg; + unsigned int i; DPRINTFN(2, ("mskc_reset\n")); @@ -758,8 +806,8 @@ mskc_reset(struct sk_softc *sc) } /* Reset status ring. */ - bzero(sc->sk_status_ring, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc)); + for (i = 0; i < MSK_STATUS_RING_CNT; i++) + sc->sk_status_ring[i] = htole64(0); sc->sk_status_idx = 0; sk_win_write_4(sc, SK_STAT_BMU_CSR, SK_STAT_BMU_RESET); @@ -1138,8 +1186,8 @@ mskc_attach(struct device *parent, struct device *self, void *aux) sc->sk_pc = pc; if (bus_dmamem_alloc(sc->sc_dmatag, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc), - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc), + MSK_STATUS_RING_CNT * sizeof(uint64_t), + MSK_STATUS_RING_CNT * sizeof(uint64_t), 0, &sc->sk_status_seg, 1, &sc->sk_status_nseg, BUS_DMA_NOWAIT | BUS_DMA_ZERO)) { printf(": can't alloc status buffers\n"); @@ -1148,27 +1196,27 @@ mskc_attach(struct device *parent, struct device *self, void *aux) if (bus_dmamem_map(sc->sc_dmatag, &sc->sk_status_seg, sc->sk_status_nseg, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc), + MSK_STATUS_RING_CNT * sizeof(uint64_t), &kva, BUS_DMA_NOWAIT)) { - printf(": can't map dma buffers (%lu bytes)\n", - (ulong)(MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc))); + printf(": can't map dma buffers (%zu bytes)\n", + MSK_STATUS_RING_CNT * sizeof(uint64_t)); goto fail_3; } if (bus_dmamap_create(sc->sc_dmatag, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc), 1, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc), 0, + MSK_STATUS_RING_CNT * sizeof(uint64_t), 1, + MSK_STATUS_RING_CNT * sizeof(uint64_t), 0, BUS_DMA_NOWAIT | BUS_DMA_ALLOCNOW | BUS_DMA_64BIT, &sc->sk_status_map)) { printf(": can't create dma map\n"); goto fail_4; } if (bus_dmamap_load(sc->sc_dmatag, sc->sk_status_map, kva, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc), + MSK_STATUS_RING_CNT * sizeof(uint64_t), NULL, BUS_DMA_NOWAIT)) { printf(": can't load dma map\n"); goto fail_5; } - sc->sk_status_ring = (struct msk_status_desc *)kva; + sc->sk_status_ring = (uint64_t *)kva; /* Reset the adapter. */ mskc_reset(sc); @@ -1364,7 +1412,7 @@ mskc_attach(struct device *parent, struct device *self, void *aux) fail_4: bus_dmamem_unmap(sc->sc_dmatag, (caddr_t)sc->sk_status_ring, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc)); + MSK_STATUS_RING_CNT * sizeof(uint64_t)); fail_3: bus_dmamem_free(sc->sc_dmatag, &sc->sk_status_seg, sc->sk_status_nseg); @@ -1395,7 +1443,7 @@ mskc_detach(struct device *self, int flags) if (sc->sk_status_nseg > 0) { bus_dmamap_destroy(sc->sc_dmatag, sc->sk_status_map); bus_dmamem_unmap(sc->sc_dmatag, (caddr_t)sc->sk_status_ring, - MSK_STATUS_RING_CNT * sizeof(struct msk_status_desc)); + MSK_STATUS_RING_CNT * sizeof(uint64_t)); bus_dmamem_free(sc->sc_dmatag, &sc->sk_status_seg, sc->sk_status_nseg); } @@ -1749,7 +1797,6 @@ int msk_intr(void *xsc) { struct sk_softc *sc = xsc; - struct sk_if_softc *sc_if; struct sk_if_softc *sc_if0 = sc->sk_if[SK_PORT_A]; struct sk_if_softc *sc_if1 = sc->sk_if[SK_PORT_B]; struct mbuf_list ml[2] = { @@ -1758,8 +1805,9 @@ msk_intr(void *xsc) }; struct ifnet *ifp0 = NULL, *ifp1 = NULL; int claimed = 0; - u_int32_t status, sk_status; - struct msk_status_desc *cur_st; + u_int32_t status; + uint64_t *ring = sc->sk_status_ring; + uint64_t desc; status = CSR_READ_4(sc, SK_Y2_ISSR2); if (status == 0xffffffff) @@ -1788,43 +1836,41 @@ msk_intr(void *xsc) MSK_CDSTSYNC(sc, sc->sk_status_idx, BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE); - cur_st = &sc->sk_status_ring[sc->sk_status_idx]; - - while (cur_st->sk_opcode & SK_Y2_STOPC_OWN) { - cur_st->sk_opcode &= ~SK_Y2_STOPC_OWN; - switch (cur_st->sk_opcode) { - case SK_Y2_STOPC_RXSTAT: - sc_if = sc->sk_if[cur_st->sk_link & 0x01]; - msk_rxeof(sc_if, &ml[cur_st->sk_link & 0x01], - lemtoh16(&cur_st->sk_len), - lemtoh32(&cur_st->sk_status)); + + while (MSK_STATUS_OWN(desc = lemtoh64(&ring[sc->sk_status_idx]))) { + unsigned int opcode, port; + + ring[sc->sk_status_idx] = htole64(0); /* clear ownership */ + + opcode = MSK_STATUS_OPCODE(desc); + switch (opcode) { + case MSK_STATUS_OPCODE_RXSTAT: + port = MSK_STATUS_RXSTAT_PORT(desc); + msk_rxeof(sc->sk_if[port], &ml[port], + MSK_STATUS_RXSTAT_LEN(desc), + MSK_STATUS_RXSTAT_STATUS(desc)); break; case SK_Y2_STOPC_TXSTAT: - sk_status = lemtoh32(&cur_st->sk_status); - if (sc_if0) - msk_txeof(sc_if0, sk_status & 0xfff); + if (sc_if0) { + msk_txeof(sc_if0, + MSK_STATUS_TXIDX_PORTA(desc)); + } if (sc_if1) { - /* - * this would be easier as a 64bit - * load of the whole status descriptor, - * a shift, and a mask. - */ - unsigned int prod = (sk_status >> 24) & 0xff; - prod |= (lemtoh16(&cur_st->sk_len) & 0xf) << 8; - msk_txeof(sc_if1, prod); + msk_txeof(sc_if1, + MSK_STATUS_TXIDX_PORTB(desc)); } break; default: - printf("opcode=0x%x\n", cur_st->sk_opcode); + printf("opcode=0x%x\n", opcode); break; } - SK_INC(sc->sk_status_idx, MSK_STATUS_RING_CNT); - MSK_CDSTSYNC(sc, sc->sk_status_idx, - BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE); - cur_st = &sc->sk_status_ring[sc->sk_status_idx]; + SK_INC(sc->sk_status_idx, MSK_STATUS_RING_CNT); } + MSK_CDSTSYNC(sc, sc->sk_status_idx, + BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE); + if (status & SK_Y2_IMR_BMU) { CSR_WRITE_4(sc, SK_STAT_BMU_CSR, SK_STAT_BMU_IRQ_CLEAR); claimed = 1; diff --git a/sys/dev/pci/if_mskvar.h b/sys/dev/pci/if_mskvar.h index b00ea321ca5..8ad4f6758d6 100644 --- a/sys/dev/pci/if_mskvar.h +++ b/sys/dev/pci/if_mskvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_mskvar.h,v 1.14 2018/01/06 03:11:04 dlg Exp $ */ +/* $OpenBSD: if_mskvar.h,v 1.15 2022/01/10 04:11:13 dlg Exp $ */ /* $NetBSD: if_skvar.h,v 1.6 2005/05/30 04:35:22 christos Exp $ */ /*- @@ -26,7 +26,7 @@ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ -/* $OpenBSD: if_mskvar.h,v 1.14 2018/01/06 03:11:04 dlg Exp $ */ +/* $OpenBSD: if_mskvar.h,v 1.15 2022/01/10 04:11:13 dlg Exp $ */ /* * Copyright (c) 1997, 1998, 1999, 2000 @@ -117,7 +117,7 @@ struct msk_ring_data { #define MSK_CDOFF(x) offsetof(struct msk_ring_data, x) #define MSK_CDTXOFF(x) MSK_CDOFF(sk_tx_ring[(x)]) #define MSK_CDRXOFF(x) MSK_CDOFF(sk_rx_ring[(x)]) -#define MSK_CDSTOFF(x) ((x) * sizeof(struct msk_status_desc)) +#define MSK_CDSTOFF(x) ((x) * sizeof(uint64_t)) #define MSK_CDTXSYNC(sc, x, n, ops) \ do { \ @@ -150,7 +150,7 @@ do { \ #define MSK_CDSTSYNC(sc, x, ops) \ do { \ bus_dmamap_sync((sc)->sc_dmatag, (sc)->sk_status_map, \ - MSK_CDSTOFF((x)), sizeof(struct msk_status_desc), (ops)); \ + MSK_CDSTOFF((x)), sizeof(uint64_t), (ops)); \ } while (/*CONSTCOND*/0) #define SK_INC(x, y) (x) = (x + 1) % y @@ -175,7 +175,7 @@ struct sk_softc { u_int32_t sk_intrmask; bus_dma_tag_t sc_dmatag; struct sk_if_softc *sk_if[2]; - struct msk_status_desc *sk_status_ring; + uint64_t *sk_status_ring; bus_dmamap_t sk_status_map; bus_dma_segment_t sk_status_seg; int sk_status_nseg; |