|
@@ -2453,6 +2453,47 @@ static void xgbe_config_mmc(struct xgbe_prv_data *pdata)
|
|
|
XGMAC_IOWRITE_BITS(pdata, MMC_CR, CR, 1);
|
|
|
}
|
|
|
|
|
|
+static void xgbe_prepare_tx_stop(struct xgbe_prv_data *pdata,
|
|
|
+ struct xgbe_channel *channel)
|
|
|
+{
|
|
|
+ unsigned int tx_dsr, tx_pos, tx_qidx;
|
|
|
+ unsigned int tx_status;
|
|
|
+ unsigned long tx_timeout;
|
|
|
+
|
|
|
+ /* Calculate the status register to read and the position within */
|
|
|
+ if (channel->queue_index < DMA_DSRX_FIRST_QUEUE) {
|
|
|
+ tx_dsr = DMA_DSR0;
|
|
|
+ tx_pos = (channel->queue_index * DMA_DSR_Q_WIDTH) +
|
|
|
+ DMA_DSR0_TPS_START;
|
|
|
+ } else {
|
|
|
+ tx_qidx = channel->queue_index - DMA_DSRX_FIRST_QUEUE;
|
|
|
+
|
|
|
+ tx_dsr = DMA_DSR1 + ((tx_qidx / DMA_DSRX_QPR) * DMA_DSRX_INC);
|
|
|
+ tx_pos = ((tx_qidx % DMA_DSRX_QPR) * DMA_DSR_Q_WIDTH) +
|
|
|
+ DMA_DSRX_TPS_START;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* The Tx engine cannot be stopped if it is actively processing
|
|
|
+ * descriptors. Wait for the Tx engine to enter the stopped or
|
|
|
+ * suspended state. Don't wait forever though...
|
|
|
+ */
|
|
|
+ tx_timeout = jiffies + (XGBE_DMA_STOP_TIMEOUT * HZ);
|
|
|
+ while (time_before(jiffies, tx_timeout)) {
|
|
|
+ tx_status = XGMAC_IOREAD(pdata, tx_dsr);
|
|
|
+ tx_status = GET_BITS(tx_status, tx_pos, DMA_DSR_TPS_WIDTH);
|
|
|
+ if ((tx_status == DMA_TPS_STOPPED) ||
|
|
|
+ (tx_status == DMA_TPS_SUSPENDED))
|
|
|
+ break;
|
|
|
+
|
|
|
+ usleep_range(500, 1000);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!time_before(jiffies, tx_timeout))
|
|
|
+ netdev_info(pdata->netdev,
|
|
|
+ "timed out waiting for Tx DMA channel %u to stop\n",
|
|
|
+ channel->queue_index);
|
|
|
+}
|
|
|
+
|
|
|
static void xgbe_enable_tx(struct xgbe_prv_data *pdata)
|
|
|
{
|
|
|
struct xgbe_channel *channel;
|
|
@@ -2481,6 +2522,15 @@ static void xgbe_disable_tx(struct xgbe_prv_data *pdata)
|
|
|
struct xgbe_channel *channel;
|
|
|
unsigned int i;
|
|
|
|
|
|
+ /* Prepare for Tx DMA channel stop */
|
|
|
+ channel = pdata->channel;
|
|
|
+ for (i = 0; i < pdata->channel_count; i++, channel++) {
|
|
|
+ if (!channel->tx_ring)
|
|
|
+ break;
|
|
|
+
|
|
|
+ xgbe_prepare_tx_stop(pdata, channel);
|
|
|
+ }
|
|
|
+
|
|
|
/* Disable MAC Tx */
|
|
|
XGMAC_IOWRITE_BITS(pdata, MAC_TCR, TE, 0);
|
|
|
|
|
@@ -2572,6 +2622,15 @@ static void xgbe_powerdown_tx(struct xgbe_prv_data *pdata)
|
|
|
struct xgbe_channel *channel;
|
|
|
unsigned int i;
|
|
|
|
|
|
+ /* Prepare for Tx DMA channel stop */
|
|
|
+ channel = pdata->channel;
|
|
|
+ for (i = 0; i < pdata->channel_count; i++, channel++) {
|
|
|
+ if (!channel->tx_ring)
|
|
|
+ break;
|
|
|
+
|
|
|
+ xgbe_prepare_tx_stop(pdata, channel);
|
|
|
+ }
|
|
|
+
|
|
|
/* Disable MAC Tx */
|
|
|
XGMAC_IOWRITE_BITS(pdata, MAC_TCR, TE, 0);
|
|
|
|