It seems that RealTek PCIe controllers require an explicit Tx poll

command whenever Tx completion interrupt is raised. The Tx poll
bit is cleared when all packets waiting to be transferred have been
processed. This means the second Tx poll command can be silently
ignored as the Tx poll bit could be still active while processing
of previous Tx poll command is in progress.
To address the issue re(4) used to invoke the Tx poll command in Tx
completion handler whenever it detects there are pending packets in
TxQ. However that still does not seem to completely eliminate
watchdog timeouts seen on RealTek PCIe controllers. To fix the
issue kick Tx poll command only after Tx completion interrupt is
raised as this would indicate Tx is now idle state such that it can
accept new Tx poll command again. While here apply this workaround
for PCIe based controllers as other controllers does not seem to
have this limitation.

Tested by:	Victor Balada Diaz < victor <> bsdes DOT net >
This commit is contained in:
Pyun YongHyeon 2008-12-17 08:18:11 +00:00
parent 64c44e5db8
commit 818951afd8
2 changed files with 16 additions and 10 deletions

View File

@ -1157,6 +1157,7 @@ re_attach(device_t dev)
msic = 0; msic = 0;
if (pci_find_extcap(dev, PCIY_EXPRESS, &reg) == 0) { if (pci_find_extcap(dev, PCIY_EXPRESS, &reg) == 0) {
sc->rl_flags |= RL_FLAG_PCIE;
msic = pci_msi_count(dev); msic = pci_msi_count(dev);
if (bootverbose) if (bootverbose)
device_printf(dev, "MSI count : %d\n", msic); device_printf(dev, "MSI count : %d\n", msic);
@ -2042,16 +2043,6 @@ re_txeof(struct rl_softc *sc)
/* No changes made to the TX ring, so no flush needed */ /* No changes made to the TX ring, so no flush needed */
if (sc->rl_ldata.rl_tx_free != sc->rl_ldata.rl_tx_desc_cnt) { if (sc->rl_ldata.rl_tx_free != sc->rl_ldata.rl_tx_desc_cnt) {
/*
* Some chips will ignore a second TX request issued
* while an existing transmission is in progress. If
* the transmitter goes idle but there are still
* packets waiting to be sent, we need to restart the
* channel here to flush them out. This only seems to
* be required with the PCIe devices.
*/
CSR_WRITE_1(sc, sc->rl_txstart, RL_TXSTART_START);
#ifdef RE_TX_MODERATION #ifdef RE_TX_MODERATION
/* /*
* If not all descriptors have been reaped yet, reload * If not all descriptors have been reaped yet, reload
@ -2115,6 +2106,9 @@ re_poll_locked(struct ifnet *ifp, enum poll_cmd cmd, int count)
return; return;
if (status) if (status)
CSR_WRITE_2(sc, RL_ISR, status); CSR_WRITE_2(sc, RL_ISR, status);
if ((status & (RL_ISR_TX_OK | RL_ISR_TX_DESC_UNAVAIL)) &&
(sc->rl_flags & RL_FLAG_PCIE))
CSR_WRITE_1(sc, sc->rl_txstart, RL_TXSTART_START);
/* /*
* XXX check behaviour on receiver stalls. * XXX check behaviour on receiver stalls.
@ -2176,6 +2170,17 @@ re_int_task(void *arg, int npending)
if (status & (RL_ISR_RX_OK|RL_ISR_RX_ERR|RL_ISR_FIFO_OFLOW)) if (status & (RL_ISR_RX_OK|RL_ISR_RX_ERR|RL_ISR_FIFO_OFLOW))
rval = re_rxeof(sc); rval = re_rxeof(sc);
/*
* Some chips will ignore a second TX request issued
* while an existing transmission is in progress. If
* the transmitter goes idle but there are still
* packets waiting to be sent, we need to restart the
* channel here to flush them out. This only seems to
* be required with the PCIe devices.
*/
if ((status & (RL_ISR_TX_OK | RL_ISR_TX_DESC_UNAVAIL)) &&
(sc->rl_flags & RL_FLAG_PCIE))
CSR_WRITE_1(sc, sc->rl_txstart, RL_TXSTART_START);
if (status & ( if (status & (
#ifdef RE_TX_MODERATION #ifdef RE_TX_MODERATION
RL_ISR_TIMEOUT_EXPIRED| RL_ISR_TIMEOUT_EXPIRED|

View File

@ -891,6 +891,7 @@ struct rl_softc {
#define RL_FLAG_PHY8110S 0x0800 #define RL_FLAG_PHY8110S 0x0800
#define RL_FLAG_WOLRXENB 0x1000 #define RL_FLAG_WOLRXENB 0x1000
#define RL_FLAG_MACSLEEP 0x2000 #define RL_FLAG_MACSLEEP 0x2000
#define RL_FLAG_PCIE 0x4000
#define RL_FLAG_LINK 0x8000 #define RL_FLAG_LINK 0x8000
}; };