- restructure tx descriptor handling code to decouple
number of tx descriptors, number of rx descriptors, and number of mbufs. - bump number of tx descriptors for rtl8169. 64 doesn't seem to be sufficient when doing TSO.
This commit is contained in:
parent
93bf0b6883
commit
e012617b1e
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: rtl8169.c,v 1.14 2005/03/14 10:08:17 yamt Exp $ */
|
||||
/* $NetBSD: rtl8169.c,v 1.15 2005/03/23 20:23:08 yamt Exp $ */
|
||||
|
||||
/*
|
||||
* Copyright (c) 1997, 1998-2003
|
||||
@ -599,6 +599,8 @@ re_attach(struct rtk_softc *sc)
|
||||
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++)
|
||||
eaddr[i] = CSR_READ_1(sc, RTK_IDR0 + i);
|
||||
|
||||
sc->rtk_ldata.rtk_tx_desc_cnt = RTK_TX_DESC_CNT_8169;
|
||||
} else {
|
||||
|
||||
/* Set RX length mask */
|
||||
@ -618,14 +620,24 @@ re_attach(struct rtk_softc *sc)
|
||||
eaddr[(i * 2) + 0] = val & 0xff;
|
||||
eaddr[(i * 2) + 1] = val >> 8;
|
||||
}
|
||||
|
||||
sc->rtk_ldata.rtk_tx_desc_cnt = RTK_TX_DESC_CNT_8139;
|
||||
}
|
||||
|
||||
aprint_normal("%s: Ethernet address %s\n",
|
||||
sc->sc_dev.dv_xname, ether_sprintf(eaddr));
|
||||
|
||||
if (sc->rtk_ldata.rtk_tx_desc_cnt >
|
||||
PAGE_SIZE / sizeof(struct rtk_desc)) {
|
||||
sc->rtk_ldata.rtk_tx_desc_cnt =
|
||||
PAGE_SIZE / sizeof(struct rtk_desc);
|
||||
}
|
||||
|
||||
aprint_verbose("%s: using %d tx descriptors\n",
|
||||
sc->sc_dev.dv_xname, sc->rtk_ldata.rtk_tx_desc_cnt);
|
||||
|
||||
/* Allocate DMA'able memory for the TX ring */
|
||||
if ((error = bus_dmamem_alloc(sc->sc_dmat, RTK_TX_LIST_SZ,
|
||||
if ((error = bus_dmamem_alloc(sc->sc_dmat, RTK_TX_LIST_SZ(sc),
|
||||
RTK_ETHER_ALIGN, 0, &sc->rtk_ldata.rtk_tx_listseg,
|
||||
1, &sc->rtk_ldata.rtk_tx_listnseg, BUS_DMA_NOWAIT)) != 0) {
|
||||
aprint_error("%s: can't allocate tx listseg, error = %d\n",
|
||||
@ -635,17 +647,17 @@ re_attach(struct rtk_softc *sc)
|
||||
|
||||
/* Load the map for the TX ring. */
|
||||
if ((error = bus_dmamem_map(sc->sc_dmat, &sc->rtk_ldata.rtk_tx_listseg,
|
||||
sc->rtk_ldata.rtk_tx_listnseg, RTK_TX_LIST_SZ,
|
||||
sc->rtk_ldata.rtk_tx_listnseg, RTK_TX_LIST_SZ(sc),
|
||||
(caddr_t *)&sc->rtk_ldata.rtk_tx_list,
|
||||
BUS_DMA_NOWAIT)) != 0) {
|
||||
aprint_error("%s: can't map tx list, error = %d\n",
|
||||
sc->sc_dev.dv_xname, error);
|
||||
goto fail_1;
|
||||
}
|
||||
memset(sc->rtk_ldata.rtk_tx_list, 0, RTK_TX_LIST_SZ);
|
||||
memset(sc->rtk_ldata.rtk_tx_list, 0, RTK_TX_LIST_SZ(sc));
|
||||
|
||||
if ((error = bus_dmamap_create(sc->sc_dmat, RTK_TX_LIST_SZ, 1,
|
||||
RTK_TX_LIST_SZ, 0, BUS_DMA_ALLOCNOW,
|
||||
if ((error = bus_dmamap_create(sc->sc_dmat, RTK_TX_LIST_SZ(sc), 1,
|
||||
RTK_TX_LIST_SZ(sc), 0, BUS_DMA_ALLOCNOW,
|
||||
&sc->rtk_ldata.rtk_tx_list_map)) != 0) {
|
||||
aprint_error("%s: can't create tx list map, error = %d\n",
|
||||
sc->sc_dev.dv_xname, error);
|
||||
@ -655,19 +667,19 @@ re_attach(struct rtk_softc *sc)
|
||||
|
||||
if ((error = bus_dmamap_load(sc->sc_dmat,
|
||||
sc->rtk_ldata.rtk_tx_list_map, sc->rtk_ldata.rtk_tx_list,
|
||||
RTK_TX_LIST_SZ, NULL, BUS_DMA_NOWAIT)) != 0) {
|
||||
RTK_TX_LIST_SZ(sc), NULL, BUS_DMA_NOWAIT)) != 0) {
|
||||
aprint_error("%s: can't load tx list, error = %d\n",
|
||||
sc->sc_dev.dv_xname, error);
|
||||
goto fail_3;
|
||||
}
|
||||
|
||||
/* Create DMA maps for TX buffers */
|
||||
for (i = 0; i < RTK_TX_DESC_CNT; i++) {
|
||||
for (i = 0; i < RTK_TX_QLEN; i++) {
|
||||
error = bus_dmamap_create(sc->sc_dmat,
|
||||
round_page(IP_MAXPACKET),
|
||||
RTK_TX_DESC_CNT - 4, RTK_TDESC_CMD_FRAGLEN,
|
||||
RTK_TX_DESC_CNT(sc) - 4, RTK_TDESC_CMD_FRAGLEN,
|
||||
0, BUS_DMA_ALLOCNOW,
|
||||
&sc->rtk_ldata.rtk_tx_dmamap[i]);
|
||||
&sc->rtk_ldata.rtk_txq[i].txq_dmamap);
|
||||
if (error) {
|
||||
aprint_error("%s: can't create DMA map for TX\n",
|
||||
sc->sc_dev.dv_xname);
|
||||
@ -693,7 +705,7 @@ re_attach(struct rtk_softc *sc)
|
||||
sc->sc_dev.dv_xname, error);
|
||||
goto fail_5;
|
||||
}
|
||||
memset(sc->rtk_ldata.rtk_rx_list, 0, RTK_TX_LIST_SZ);
|
||||
memset(sc->rtk_ldata.rtk_rx_list, 0, RTK_RX_LIST_SZ);
|
||||
|
||||
if ((error = bus_dmamap_create(sc->sc_dmat, RTK_RX_LIST_SZ, 1,
|
||||
RTK_RX_LIST_SZ, 0, BUS_DMA_ALLOCNOW,
|
||||
@ -809,10 +821,10 @@ fail_5:
|
||||
|
||||
fail_4:
|
||||
/* Destroy DMA maps for TX buffers. */
|
||||
for (i = 0; i < RTK_TX_DESC_CNT; i++)
|
||||
if (sc->rtk_ldata.rtk_tx_dmamap[i] != NULL)
|
||||
for (i = 0; i < RTK_TX_QLEN; i++)
|
||||
if (sc->rtk_ldata.rtk_txq[i].txq_dmamap != NULL)
|
||||
bus_dmamap_destroy(sc->sc_dmat,
|
||||
sc->rtk_ldata.rtk_tx_dmamap[i]);
|
||||
sc->rtk_ldata.rtk_txq[i].txq_dmamap);
|
||||
|
||||
/* Free DMA'able memory for the TX ring. */
|
||||
bus_dmamap_unload(sc->sc_dmat, sc->rtk_ldata.rtk_tx_list_map);
|
||||
@ -820,7 +832,7 @@ fail_3:
|
||||
bus_dmamap_destroy(sc->sc_dmat, sc->rtk_ldata.rtk_tx_list_map);
|
||||
fail_2:
|
||||
bus_dmamem_unmap(sc->sc_dmat,
|
||||
(caddr_t)sc->rtk_ldata.rtk_tx_list, RTK_TX_LIST_SZ);
|
||||
(caddr_t)sc->rtk_ldata.rtk_tx_list, RTK_TX_LIST_SZ(sc));
|
||||
fail_1:
|
||||
bus_dmamem_free(sc->sc_dmat,
|
||||
&sc->rtk_ldata.rtk_tx_listseg, sc->rtk_ldata.rtk_tx_listnseg);
|
||||
@ -899,16 +911,16 @@ re_detach(struct rtk_softc *sc)
|
||||
&sc->rtk_ldata.rtk_rx_listseg, sc->rtk_ldata.rtk_rx_listnseg);
|
||||
|
||||
/* Destroy DMA maps for TX buffers. */
|
||||
for (i = 0; i < RTK_TX_DESC_CNT; i++)
|
||||
if (sc->rtk_ldata.rtk_tx_dmamap[i] != NULL)
|
||||
for (i = 0; i < RTK_TX_QLEN; i++)
|
||||
if (sc->rtk_ldata.rtk_txq[i].txq_dmamap != NULL)
|
||||
bus_dmamap_destroy(sc->sc_dmat,
|
||||
sc->rtk_ldata.rtk_tx_dmamap[i]);
|
||||
sc->rtk_ldata.rtk_txq[i].txq_dmamap);
|
||||
|
||||
/* Free DMA'able memory for the TX ring. */
|
||||
bus_dmamap_unload(sc->sc_dmat, sc->rtk_ldata.rtk_tx_list_map);
|
||||
bus_dmamap_destroy(sc->sc_dmat, sc->rtk_ldata.rtk_tx_list_map);
|
||||
bus_dmamem_unmap(sc->sc_dmat,
|
||||
(caddr_t)sc->rtk_ldata.rtk_tx_list, RTK_TX_LIST_SZ);
|
||||
(caddr_t)sc->rtk_ldata.rtk_tx_list, RTK_TX_LIST_SZ(sc));
|
||||
bus_dmamem_free(sc->sc_dmat,
|
||||
&sc->rtk_ldata.rtk_tx_listseg, sc->rtk_ldata.rtk_tx_listnseg);
|
||||
|
||||
@ -1030,14 +1042,10 @@ re_newbuf(struct rtk_softc *sc, int idx, struct mbuf *m)
|
||||
cmdstat = map->dm_segs[0].ds_len;
|
||||
d->rtk_bufaddr_lo = htole32(RTK_ADDR_LO(map->dm_segs[0].ds_addr));
|
||||
d->rtk_bufaddr_hi = htole32(RTK_ADDR_HI(map->dm_segs[0].ds_addr));
|
||||
cmdstat |= RTK_TDESC_CMD_SOF;
|
||||
if (idx == (RTK_RX_DESC_CNT - 1))
|
||||
cmdstat |= RTK_TDESC_CMD_EOR;
|
||||
cmdstat |= RTK_RDESC_CMD_EOR;
|
||||
d->rtk_cmdstat = htole32(cmdstat);
|
||||
|
||||
d->rtk_cmdstat |= htole32(RTK_TDESC_CMD_EOF);
|
||||
|
||||
|
||||
sc->rtk_ldata.rtk_rx_list[idx].rtk_cmdstat |=
|
||||
htole32(RTK_RDESC_CMD_OWN);
|
||||
sc->rtk_ldata.rtk_rx_mbuf[idx] = m;
|
||||
@ -1056,16 +1064,20 @@ out:
|
||||
static int
|
||||
re_tx_list_init(struct rtk_softc *sc)
|
||||
{
|
||||
memset((char *)sc->rtk_ldata.rtk_tx_list, 0, RTK_TX_LIST_SZ);
|
||||
memset((char *)&sc->rtk_ldata.rtk_tx_mbuf, 0,
|
||||
(RTK_TX_DESC_CNT * sizeof(struct mbuf *)));
|
||||
int i;
|
||||
|
||||
memset(sc->rtk_ldata.rtk_tx_list, 0, RTK_TX_LIST_SZ(sc));
|
||||
for (i = 0; i < RTK_TX_QLEN; i++) {
|
||||
sc->rtk_ldata.rtk_txq[i].txq_mbuf = NULL;
|
||||
}
|
||||
|
||||
bus_dmamap_sync(sc->sc_dmat,
|
||||
sc->rtk_ldata.rtk_tx_list_map, 0,
|
||||
sc->rtk_ldata.rtk_tx_list_map->dm_mapsize, BUS_DMASYNC_PREWRITE);
|
||||
sc->rtk_ldata.rtk_tx_prodidx = 0;
|
||||
sc->rtk_ldata.rtk_tx_considx = 0;
|
||||
sc->rtk_ldata.rtk_tx_free = RTK_TX_DESC_CNT;
|
||||
sc->rtk_ldata.rtk_txq_prodidx = 0;
|
||||
sc->rtk_ldata.rtk_txq_considx = 0;
|
||||
sc->rtk_ldata.rtk_tx_free = RTK_TX_DESC_CNT(sc);
|
||||
sc->rtk_ldata.rtk_tx_nextfree = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -1148,7 +1160,7 @@ re_rxeof(struct rtk_softc *sc)
|
||||
sc->rtk_tail = m;
|
||||
}
|
||||
re_newbuf(sc, i, NULL);
|
||||
RTK_DESC_INC(i);
|
||||
RTK_RX_DESC_INC(sc, i);
|
||||
continue;
|
||||
}
|
||||
|
||||
@ -1182,7 +1194,7 @@ re_rxeof(struct rtk_softc *sc)
|
||||
sc->rtk_head = sc->rtk_tail = NULL;
|
||||
}
|
||||
re_newbuf(sc, i, m);
|
||||
RTK_DESC_INC(i);
|
||||
RTK_RX_DESC_INC(sc, i);
|
||||
continue;
|
||||
}
|
||||
|
||||
@ -1198,11 +1210,11 @@ re_rxeof(struct rtk_softc *sc)
|
||||
sc->rtk_head = sc->rtk_tail = NULL;
|
||||
}
|
||||
re_newbuf(sc, i, m);
|
||||
RTK_DESC_INC(i);
|
||||
RTK_RX_DESC_INC(sc, i);
|
||||
continue;
|
||||
}
|
||||
|
||||
RTK_DESC_INC(i);
|
||||
RTK_RX_DESC_INC(sc, i);
|
||||
|
||||
if (sc->rtk_head != NULL) {
|
||||
m->m_len = total_len % (MCLBYTES - RTK_ETHER_ALIGN);
|
||||
@ -1284,11 +1296,10 @@ static void
|
||||
re_txeof(struct rtk_softc *sc)
|
||||
{
|
||||
struct ifnet *ifp;
|
||||
u_int32_t txstat;
|
||||
int idx;
|
||||
|
||||
ifp = &sc->ethercom.ec_if;
|
||||
idx = sc->rtk_ldata.rtk_tx_considx;
|
||||
idx = sc->rtk_ldata.rtk_txq_considx;
|
||||
|
||||
/* Invalidate the TX descriptor list */
|
||||
|
||||
@ -1297,40 +1308,39 @@ re_txeof(struct rtk_softc *sc)
|
||||
0, sc->rtk_ldata.rtk_tx_list_map->dm_mapsize,
|
||||
BUS_DMASYNC_POSTREAD);
|
||||
|
||||
while (idx != sc->rtk_ldata.rtk_tx_prodidx) {
|
||||
while (idx != sc->rtk_ldata.rtk_txq_prodidx) {
|
||||
struct rtk_txq *txq = &sc->rtk_ldata.rtk_txq[idx];
|
||||
int descidx = txq->txq_descidx;
|
||||
u_int32_t txstat;
|
||||
|
||||
txstat = le32toh(sc->rtk_ldata.rtk_tx_list[idx].rtk_cmdstat);
|
||||
KASSERT(txq->txq_mbuf != NULL);
|
||||
|
||||
txstat =
|
||||
le32toh(sc->rtk_ldata.rtk_tx_list[descidx].rtk_cmdstat);
|
||||
KASSERT((txstat & RTK_TDESC_CMD_EOF) != 0);
|
||||
if (txstat & RTK_TDESC_CMD_OWN)
|
||||
break;
|
||||
|
||||
/*
|
||||
* We only stash mbufs in the last descriptor
|
||||
* in a fragment chain, which also happens to
|
||||
* be the only place where the TX status bits
|
||||
* are valid.
|
||||
*/
|
||||
sc->rtk_ldata.rtk_tx_free += txq->txq_dmamap->dm_nsegs;
|
||||
KASSERT(sc->rtk_ldata.rtk_tx_free <= RTK_TX_DESC_CNT(sc));
|
||||
bus_dmamap_unload(sc->sc_dmat, txq->txq_dmamap);
|
||||
m_freem(txq->txq_mbuf);
|
||||
txq->txq_mbuf = NULL;
|
||||
|
||||
if (txstat & RTK_TDESC_CMD_EOF) {
|
||||
m_freem(sc->rtk_ldata.rtk_tx_mbuf[idx]);
|
||||
sc->rtk_ldata.rtk_tx_mbuf[idx] = NULL;
|
||||
bus_dmamap_unload(sc->sc_dmat,
|
||||
sc->rtk_ldata.rtk_tx_dmamap[idx]);
|
||||
if (txstat & (RTK_TDESC_STAT_EXCESSCOL |
|
||||
RTK_TDESC_STAT_COLCNT))
|
||||
if (txstat & (RTK_TDESC_STAT_EXCESSCOL | RTK_TDESC_STAT_COLCNT))
|
||||
ifp->if_collisions++;
|
||||
if (txstat & RTK_TDESC_STAT_TXERRSUM)
|
||||
ifp->if_oerrors++;
|
||||
else
|
||||
ifp->if_opackets++;
|
||||
}
|
||||
sc->rtk_ldata.rtk_tx_free++;
|
||||
RTK_DESC_INC(idx);
|
||||
|
||||
idx = (idx + 1) % RTK_TX_QLEN;
|
||||
}
|
||||
|
||||
/* No changes made to the TX ring, so no flush needed */
|
||||
|
||||
if (idx != sc->rtk_ldata.rtk_tx_considx) {
|
||||
sc->rtk_ldata.rtk_tx_considx = idx;
|
||||
if (idx != sc->rtk_ldata.rtk_txq_considx) {
|
||||
sc->rtk_ldata.rtk_txq_considx = idx;
|
||||
ifp->if_flags &= ~IFF_OACTIVE;
|
||||
ifp->if_timer = 0;
|
||||
}
|
||||
@ -1341,7 +1351,7 @@ re_txeof(struct rtk_softc *sc)
|
||||
* interrupt that will cause us to re-enter this routine.
|
||||
* This is done in case the transmitter has gone idle.
|
||||
*/
|
||||
if (sc->rtk_ldata.rtk_tx_free != RTK_TX_DESC_CNT)
|
||||
if (sc->rtk_ldata.rtk_tx_free != RTK_TX_DESC_CNT(sc))
|
||||
CSR_WRITE_4(sc, RTK_TIMERCNT, 1);
|
||||
|
||||
return;
|
||||
@ -1495,10 +1505,11 @@ static int
|
||||
re_encap(struct rtk_softc *sc, struct mbuf *m, int *idx)
|
||||
{
|
||||
bus_dmamap_t map;
|
||||
int error, i, curidx;
|
||||
int error, i, startidx, curidx;
|
||||
struct m_tag *mtag;
|
||||
struct rtk_desc *d;
|
||||
u_int32_t cmdstat, rtk_flags;
|
||||
struct rtk_txq *txq;
|
||||
|
||||
if (sc->rtk_ldata.rtk_tx_free <= 4)
|
||||
return EFBIG;
|
||||
@ -1525,7 +1536,8 @@ re_encap(struct rtk_softc *sc, struct mbuf *m, int *idx)
|
||||
rtk_flags |= RTK_TDESC_CMD_UDPCSUM;
|
||||
}
|
||||
|
||||
map = sc->rtk_ldata.rtk_tx_dmamap[*idx];
|
||||
txq = &sc->rtk_ldata.rtk_txq[*idx];
|
||||
map = txq->txq_dmamap;
|
||||
error = bus_dmamap_load_mbuf(sc->sc_dmat, map, m, BUS_DMA_NOWAIT);
|
||||
|
||||
if (error) {
|
||||
@ -1553,14 +1565,14 @@ re_encap(struct rtk_softc *sc, struct mbuf *m, int *idx)
|
||||
* reception.)
|
||||
*/
|
||||
i = 0;
|
||||
curidx = *idx;
|
||||
curidx = startidx = sc->rtk_ldata.rtk_tx_nextfree;
|
||||
while (1) {
|
||||
d = &sc->rtk_ldata.rtk_tx_list[curidx];
|
||||
if (le32toh(d->rtk_cmdstat) & RTK_RDESC_STAT_OWN) {
|
||||
if (le32toh(d->rtk_cmdstat) & RTK_TDESC_STAT_OWN) {
|
||||
while (i > 0) {
|
||||
sc->rtk_ldata.rtk_tx_list[
|
||||
(curidx + RTK_TX_DESC_CNT - i) %
|
||||
RTK_TX_DESC_CNT].rtk_cmdstat = 0;
|
||||
(curidx + RTK_TX_DESC_CNT(sc) - i) %
|
||||
RTK_TX_DESC_CNT(sc)].rtk_cmdstat = 0;
|
||||
i--;
|
||||
}
|
||||
error = ENOBUFS;
|
||||
@ -1576,26 +1588,18 @@ re_encap(struct rtk_softc *sc, struct mbuf *m, int *idx)
|
||||
cmdstat |= RTK_TDESC_CMD_SOF;
|
||||
else
|
||||
cmdstat |= RTK_TDESC_CMD_OWN;
|
||||
if (curidx == (RTK_RX_DESC_CNT - 1))
|
||||
if (curidx == (RTK_TX_DESC_CNT(sc) - 1))
|
||||
cmdstat |= RTK_TDESC_CMD_EOR;
|
||||
d->rtk_cmdstat = htole32(cmdstat | rtk_flags);
|
||||
i++;
|
||||
if (i == map->dm_nsegs)
|
||||
break;
|
||||
RTK_DESC_INC(curidx);
|
||||
RTK_TX_DESC_INC(sc, curidx);
|
||||
}
|
||||
|
||||
d->rtk_cmdstat |= htole32(RTK_TDESC_CMD_EOF);
|
||||
|
||||
/*
|
||||
* Insure that the map for this transmission
|
||||
* is placed at the array index of the last descriptor
|
||||
* in this chain.
|
||||
*/
|
||||
sc->rtk_ldata.rtk_tx_dmamap[*idx] =
|
||||
sc->rtk_ldata.rtk_tx_dmamap[curidx];
|
||||
sc->rtk_ldata.rtk_tx_dmamap[curidx] = map;
|
||||
sc->rtk_ldata.rtk_tx_mbuf[curidx] = m;
|
||||
txq->txq_mbuf = m;
|
||||
sc->rtk_ldata.rtk_tx_free -= map->dm_nsegs;
|
||||
|
||||
/*
|
||||
@ -1605,7 +1609,7 @@ re_encap(struct rtk_softc *sc, struct mbuf *m, int *idx)
|
||||
*/
|
||||
|
||||
if ((mtag = VLAN_OUTPUT_TAG(&sc->ethercom, m)) != NULL) {
|
||||
sc->rtk_ldata.rtk_tx_list[*idx].rtk_vlanctl =
|
||||
sc->rtk_ldata.rtk_tx_list[startidx].rtk_vlanctl =
|
||||
htole32(htons(VLAN_TAG_VALUE(mtag)) |
|
||||
RTK_TDESC_VLANCTL_TAG);
|
||||
}
|
||||
@ -1614,12 +1618,14 @@ re_encap(struct rtk_softc *sc, struct mbuf *m, int *idx)
|
||||
|
||||
sc->rtk_ldata.rtk_tx_list[curidx].rtk_cmdstat |=
|
||||
htole32(RTK_TDESC_CMD_OWN);
|
||||
if (*idx != curidx)
|
||||
sc->rtk_ldata.rtk_tx_list[*idx].rtk_cmdstat |=
|
||||
if (startidx != curidx)
|
||||
sc->rtk_ldata.rtk_tx_list[startidx].rtk_cmdstat |=
|
||||
htole32(RTK_TDESC_CMD_OWN);
|
||||
|
||||
RTK_DESC_INC(curidx);
|
||||
*idx = curidx;
|
||||
txq->txq_descidx = curidx;
|
||||
RTK_TX_DESC_INC(sc, curidx);
|
||||
sc->rtk_ldata.rtk_tx_nextfree = curidx;
|
||||
*idx = (*idx + 1) % RTK_TX_QLEN;
|
||||
|
||||
return 0;
|
||||
|
||||
@ -1642,8 +1648,8 @@ re_start(struct ifnet *ifp)
|
||||
|
||||
sc = ifp->if_softc;
|
||||
|
||||
idx = sc->rtk_ldata.rtk_tx_prodidx;
|
||||
while (sc->rtk_ldata.rtk_tx_mbuf[idx] == NULL) {
|
||||
idx = sc->rtk_ldata.rtk_txq_prodidx;
|
||||
while (sc->rtk_ldata.rtk_txq[idx].txq_mbuf == NULL) {
|
||||
int error;
|
||||
|
||||
IF_DEQUEUE(&ifp->if_snd, m_head);
|
||||
@ -1652,7 +1658,7 @@ re_start(struct ifnet *ifp)
|
||||
|
||||
error = re_encap(sc, m_head, &idx);
|
||||
if (error == EFBIG &&
|
||||
sc->rtk_ldata.rtk_tx_free == RTK_TX_DESC_CNT) {
|
||||
sc->rtk_ldata.rtk_tx_free == RTK_TX_DESC_CNT(sc)) {
|
||||
ifp->if_oerrors++;
|
||||
m_freem(m_head);
|
||||
continue;
|
||||
@ -1679,7 +1685,7 @@ re_start(struct ifnet *ifp)
|
||||
0, sc->rtk_ldata.rtk_tx_list_map->dm_mapsize,
|
||||
BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD);
|
||||
|
||||
sc->rtk_ldata.rtk_tx_prodidx = idx;
|
||||
sc->rtk_ldata.rtk_txq_prodidx = idx;
|
||||
|
||||
/*
|
||||
* RealTek put the TX poll request register in a different
|
||||
@ -2016,12 +2022,12 @@ re_stop(struct ifnet *ifp, int disable)
|
||||
}
|
||||
|
||||
/* Free the TX list buffers. */
|
||||
for (i = 0; i < RTK_TX_DESC_CNT; i++) {
|
||||
if (sc->rtk_ldata.rtk_tx_mbuf[i] != NULL) {
|
||||
for (i = 0; i < RTK_TX_QLEN; i++) {
|
||||
if (sc->rtk_ldata.rtk_txq[i].txq_mbuf != NULL) {
|
||||
bus_dmamap_unload(sc->sc_dmat,
|
||||
sc->rtk_ldata.rtk_tx_dmamap[i]);
|
||||
m_freem(sc->rtk_ldata.rtk_tx_mbuf[i]);
|
||||
sc->rtk_ldata.rtk_tx_mbuf[i] = NULL;
|
||||
sc->rtk_ldata.rtk_txq[i].txq_dmamap);
|
||||
m_freem(sc->rtk_ldata.rtk_txq[i].txq_mbuf);
|
||||
sc->rtk_ldata.rtk_txq[i].txq_mbuf = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: rtl81x9reg.h,v 1.11 2005/03/12 08:01:51 yamt Exp $ */
|
||||
/* $NetBSD: rtl81x9reg.h,v 1.12 2005/03/23 20:23:08 yamt Exp $ */
|
||||
|
||||
/*
|
||||
* Copyright (c) 1997, 1998
|
||||
@ -553,12 +553,11 @@ struct rtk_stats {
|
||||
};
|
||||
|
||||
#define RTK_RX_DESC_CNT 64
|
||||
#define RTK_TX_DESC_CNT 64
|
||||
#define RTK_TX_DESC_CNT_8139 64
|
||||
#define RTK_TX_DESC_CNT_8169 1024
|
||||
#define RTK_RX_LIST_SZ (RTK_RX_DESC_CNT * sizeof(struct rtk_desc))
|
||||
#define RTK_TX_LIST_SZ (RTK_TX_DESC_CNT * sizeof(struct rtk_desc))
|
||||
#define RTK_RING_ALIGN 256
|
||||
#define RTK_IFQ_MAXLEN 512
|
||||
#define RTK_DESC_INC(x) (x = (x + 1) % RTK_TX_DESC_CNT)
|
||||
#define RTK_OWN(x) (le32toh((x)->rtk_cmdstat) & RTK_RDESC_STAT_OWN)
|
||||
#define RTK_RXBYTES(x) (le32toh((x)->rtk_cmdstat) & sc->rtk_rxlenmask)
|
||||
#define RTK_PKTSZ(x) ((x)/* >> 3*/)
|
||||
|
@ -1,4 +1,4 @@
|
||||
/* $NetBSD: rtl81x9var.h,v 1.17 2005/02/04 02:10:37 perry Exp $ */
|
||||
/* $NetBSD: rtl81x9var.h,v 1.18 2005/03/23 20:23:08 yamt Exp $ */
|
||||
|
||||
/*
|
||||
* Copyright (c) 1997, 1998
|
||||
@ -81,6 +81,8 @@ struct rtk_mii_frame {
|
||||
#define RTK_ISCPLUS(x) ((x)->rtk_type == RTK_8139CPLUS || \
|
||||
(x)->rtk_type == RTK_8169)
|
||||
|
||||
#define RTK_TX_QLEN 64
|
||||
|
||||
/*
|
||||
* The 8139C+ and 8160 gigE chips support descriptor-based TX
|
||||
* and RX. In fact, they even support TCP large send. Descriptors
|
||||
@ -89,22 +91,28 @@ struct rtk_mii_frame {
|
||||
*/
|
||||
|
||||
struct rtk_list_data {
|
||||
struct mbuf *rtk_tx_mbuf[RTK_TX_DESC_CNT];
|
||||
struct rtk_txq {
|
||||
struct mbuf *txq_mbuf;
|
||||
bus_dmamap_t txq_dmamap;
|
||||
int txq_descidx;
|
||||
} rtk_txq[RTK_TX_QLEN];
|
||||
int rtk_txq_considx;
|
||||
int rtk_txq_prodidx;
|
||||
bus_dmamap_t rtk_tx_list_map;
|
||||
struct rtk_desc *rtk_tx_list;
|
||||
bus_dma_segment_t rtk_tx_listseg;
|
||||
int rtk_tx_free; /* # of free descriptors */
|
||||
int rtk_tx_nextfree; /* next descriptor to use */
|
||||
int rtk_tx_desc_cnt; /* # of descriptors */
|
||||
int rtk_tx_listnseg;
|
||||
|
||||
struct mbuf *rtk_rx_mbuf[RTK_RX_DESC_CNT];
|
||||
int rtk_tx_prodidx;
|
||||
int rtk_rx_prodidx;
|
||||
int rtk_tx_considx;
|
||||
int rtk_tx_free;
|
||||
bus_dmamap_t rtk_tx_dmamap[RTK_TX_DESC_CNT];
|
||||
bus_dmamap_t rtk_rx_dmamap[RTK_RX_DESC_CNT];
|
||||
bus_dmamap_t rtk_rx_list_map;
|
||||
struct rtk_desc *rtk_rx_list;
|
||||
bus_dma_segment_t rtk_rx_listseg;
|
||||
int rtk_rx_prodidx;
|
||||
int rtk_rx_listnseg;
|
||||
bus_dmamap_t rtk_tx_list_map;
|
||||
struct rtk_desc *rtk_tx_list;
|
||||
bus_dma_segment_t rtk_tx_listseg;
|
||||
int rtk_tx_listnseg;
|
||||
};
|
||||
struct rtk_tx_desc {
|
||||
SIMPLEQ_ENTRY(rtk_tx_desc) txd_q;
|
||||
@ -154,6 +162,15 @@ struct rtk_softc {
|
||||
#endif
|
||||
};
|
||||
|
||||
#define RTK_TX_DESC_CNT(sc) \
|
||||
((sc)->rtk_ldata.rtk_tx_desc_cnt)
|
||||
#define RTK_TX_LIST_SZ(sc) \
|
||||
(RTK_TX_DESC_CNT(sc) * sizeof(struct rtk_desc))
|
||||
#define RTK_TX_DESC_INC(sc, x) \
|
||||
((x) = ((x) + 1) % RTK_TX_DESC_CNT(sc))
|
||||
#define RTK_RX_DESC_INC(sc, x) \
|
||||
((x) = ((x) + 1) % RTK_RX_DESC_CNT)
|
||||
|
||||
#define RTK_ATTACHED 0x00000001 /* attach has succeeded */
|
||||
#define RTK_ENABLED 0x00000002 /* chip is enabled */
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user