net: macb: Apply RXUBR workaround only to versions with errata
The interrupt handler contains a workaround for RX hang applicable to Zynq and AT91RM9200 only. Subsequent versions do not need this workaround. This workaround unnecessarily resets RX whenever RX used bit read is observed, which can be often under heavy traffic. There is no other action performed on RX UBR interrupt. Hence introduce a CAPS mask; enable this interrupt and workaround only on affected versions. Signed-off-by: Harini Katakam <harini.katakam@xilinx.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
b3379a424d
commit
e501070e4d
@ -643,6 +643,7 @@
|
|||||||
#define MACB_CAPS_JUMBO 0x00000020
|
#define MACB_CAPS_JUMBO 0x00000020
|
||||||
#define MACB_CAPS_GEM_HAS_PTP 0x00000040
|
#define MACB_CAPS_GEM_HAS_PTP 0x00000040
|
||||||
#define MACB_CAPS_BD_RD_PREFETCH 0x00000080
|
#define MACB_CAPS_BD_RD_PREFETCH 0x00000080
|
||||||
|
#define MACB_CAPS_NEEDS_RSTONUBR 0x00000100
|
||||||
#define MACB_CAPS_FIFO_MODE 0x10000000
|
#define MACB_CAPS_FIFO_MODE 0x10000000
|
||||||
#define MACB_CAPS_GIGABIT_MODE_AVAILABLE 0x20000000
|
#define MACB_CAPS_GIGABIT_MODE_AVAILABLE 0x20000000
|
||||||
#define MACB_CAPS_SG_DISABLED 0x40000000
|
#define MACB_CAPS_SG_DISABLED 0x40000000
|
||||||
@ -1214,6 +1215,8 @@ struct macb {
|
|||||||
|
|
||||||
int rx_bd_rd_prefetch;
|
int rx_bd_rd_prefetch;
|
||||||
int tx_bd_rd_prefetch;
|
int tx_bd_rd_prefetch;
|
||||||
|
|
||||||
|
u32 rx_intr_mask;
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef CONFIG_MACB_USE_HWSTAMP
|
#ifdef CONFIG_MACB_USE_HWSTAMP
|
||||||
|
@ -56,8 +56,7 @@
|
|||||||
/* level of occupied TX descriptors under which we wake up TX process */
|
/* level of occupied TX descriptors under which we wake up TX process */
|
||||||
#define MACB_TX_WAKEUP_THRESH(bp) (3 * (bp)->tx_ring_size / 4)
|
#define MACB_TX_WAKEUP_THRESH(bp) (3 * (bp)->tx_ring_size / 4)
|
||||||
|
|
||||||
#define MACB_RX_INT_FLAGS (MACB_BIT(RCOMP) | MACB_BIT(RXUBR) \
|
#define MACB_RX_INT_FLAGS (MACB_BIT(RCOMP) | MACB_BIT(ISR_ROVR))
|
||||||
| MACB_BIT(ISR_ROVR))
|
|
||||||
#define MACB_TX_ERR_FLAGS (MACB_BIT(ISR_TUND) \
|
#define MACB_TX_ERR_FLAGS (MACB_BIT(ISR_TUND) \
|
||||||
| MACB_BIT(ISR_RLE) \
|
| MACB_BIT(ISR_RLE) \
|
||||||
| MACB_BIT(TXERR))
|
| MACB_BIT(TXERR))
|
||||||
@ -1270,7 +1269,7 @@ static int macb_poll(struct napi_struct *napi, int budget)
|
|||||||
queue_writel(queue, ISR, MACB_BIT(RCOMP));
|
queue_writel(queue, ISR, MACB_BIT(RCOMP));
|
||||||
napi_reschedule(napi);
|
napi_reschedule(napi);
|
||||||
} else {
|
} else {
|
||||||
queue_writel(queue, IER, MACB_RX_INT_FLAGS);
|
queue_writel(queue, IER, bp->rx_intr_mask);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1288,7 +1287,7 @@ static void macb_hresp_error_task(unsigned long data)
|
|||||||
u32 ctrl;
|
u32 ctrl;
|
||||||
|
|
||||||
for (q = 0, queue = bp->queues; q < bp->num_queues; ++q, ++queue) {
|
for (q = 0, queue = bp->queues; q < bp->num_queues; ++q, ++queue) {
|
||||||
queue_writel(queue, IDR, MACB_RX_INT_FLAGS |
|
queue_writel(queue, IDR, bp->rx_intr_mask |
|
||||||
MACB_TX_INT_FLAGS |
|
MACB_TX_INT_FLAGS |
|
||||||
MACB_BIT(HRESP));
|
MACB_BIT(HRESP));
|
||||||
}
|
}
|
||||||
@ -1318,7 +1317,7 @@ static void macb_hresp_error_task(unsigned long data)
|
|||||||
|
|
||||||
/* Enable interrupts */
|
/* Enable interrupts */
|
||||||
queue_writel(queue, IER,
|
queue_writel(queue, IER,
|
||||||
MACB_RX_INT_FLAGS |
|
bp->rx_intr_mask |
|
||||||
MACB_TX_INT_FLAGS |
|
MACB_TX_INT_FLAGS |
|
||||||
MACB_BIT(HRESP));
|
MACB_BIT(HRESP));
|
||||||
}
|
}
|
||||||
@ -1372,14 +1371,14 @@ static irqreturn_t macb_interrupt(int irq, void *dev_id)
|
|||||||
(unsigned int)(queue - bp->queues),
|
(unsigned int)(queue - bp->queues),
|
||||||
(unsigned long)status);
|
(unsigned long)status);
|
||||||
|
|
||||||
if (status & MACB_RX_INT_FLAGS) {
|
if (status & bp->rx_intr_mask) {
|
||||||
/* There's no point taking any more interrupts
|
/* There's no point taking any more interrupts
|
||||||
* until we have processed the buffers. The
|
* until we have processed the buffers. The
|
||||||
* scheduling call may fail if the poll routine
|
* scheduling call may fail if the poll routine
|
||||||
* is already scheduled, so disable interrupts
|
* is already scheduled, so disable interrupts
|
||||||
* now.
|
* now.
|
||||||
*/
|
*/
|
||||||
queue_writel(queue, IDR, MACB_RX_INT_FLAGS);
|
queue_writel(queue, IDR, bp->rx_intr_mask);
|
||||||
if (bp->caps & MACB_CAPS_ISR_CLEAR_ON_WRITE)
|
if (bp->caps & MACB_CAPS_ISR_CLEAR_ON_WRITE)
|
||||||
queue_writel(queue, ISR, MACB_BIT(RCOMP));
|
queue_writel(queue, ISR, MACB_BIT(RCOMP));
|
||||||
|
|
||||||
@ -1412,8 +1411,9 @@ static irqreturn_t macb_interrupt(int irq, void *dev_id)
|
|||||||
/* There is a hardware issue under heavy load where DMA can
|
/* There is a hardware issue under heavy load where DMA can
|
||||||
* stop, this causes endless "used buffer descriptor read"
|
* stop, this causes endless "used buffer descriptor read"
|
||||||
* interrupts but it can be cleared by re-enabling RX. See
|
* interrupts but it can be cleared by re-enabling RX. See
|
||||||
* the at91 manual, section 41.3.1 or the Zynq manual
|
* the at91rm9200 manual, section 41.3.1 or the Zynq manual
|
||||||
* section 16.7.4 for details.
|
* section 16.7.4 for details. RXUBR is only enabled for
|
||||||
|
* these two versions.
|
||||||
*/
|
*/
|
||||||
if (status & MACB_BIT(RXUBR)) {
|
if (status & MACB_BIT(RXUBR)) {
|
||||||
ctrl = macb_readl(bp, NCR);
|
ctrl = macb_readl(bp, NCR);
|
||||||
@ -2259,7 +2259,7 @@ static void macb_init_hw(struct macb *bp)
|
|||||||
|
|
||||||
/* Enable interrupts */
|
/* Enable interrupts */
|
||||||
queue_writel(queue, IER,
|
queue_writel(queue, IER,
|
||||||
MACB_RX_INT_FLAGS |
|
bp->rx_intr_mask |
|
||||||
MACB_TX_INT_FLAGS |
|
MACB_TX_INT_FLAGS |
|
||||||
MACB_BIT(HRESP));
|
MACB_BIT(HRESP));
|
||||||
}
|
}
|
||||||
@ -3907,6 +3907,7 @@ static const struct macb_config sama5d4_config = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static const struct macb_config emac_config = {
|
static const struct macb_config emac_config = {
|
||||||
|
.caps = MACB_CAPS_NEEDS_RSTONUBR,
|
||||||
.clk_init = at91ether_clk_init,
|
.clk_init = at91ether_clk_init,
|
||||||
.init = at91ether_init,
|
.init = at91ether_init,
|
||||||
};
|
};
|
||||||
@ -3928,7 +3929,8 @@ static const struct macb_config zynqmp_config = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static const struct macb_config zynq_config = {
|
static const struct macb_config zynq_config = {
|
||||||
.caps = MACB_CAPS_GIGABIT_MODE_AVAILABLE | MACB_CAPS_NO_GIGABIT_HALF,
|
.caps = MACB_CAPS_GIGABIT_MODE_AVAILABLE | MACB_CAPS_NO_GIGABIT_HALF |
|
||||||
|
MACB_CAPS_NEEDS_RSTONUBR,
|
||||||
.dma_burst_length = 16,
|
.dma_burst_length = 16,
|
||||||
.clk_init = macb_clk_init,
|
.clk_init = macb_clk_init,
|
||||||
.init = macb_init,
|
.init = macb_init,
|
||||||
@ -4083,6 +4085,10 @@ static int macb_probe(struct platform_device *pdev)
|
|||||||
macb_dma_desc_get_size(bp);
|
macb_dma_desc_get_size(bp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bp->rx_intr_mask = MACB_RX_INT_FLAGS;
|
||||||
|
if (bp->caps & MACB_CAPS_NEEDS_RSTONUBR)
|
||||||
|
bp->rx_intr_mask |= MACB_BIT(RXUBR);
|
||||||
|
|
||||||
mac = of_get_mac_address(np);
|
mac = of_get_mac_address(np);
|
||||||
if (mac) {
|
if (mac) {
|
||||||
ether_addr_copy(bp->dev->dev_addr, mac);
|
ether_addr_copy(bp->dev->dev_addr, mac);
|
||||||
|
Loading…
Reference in New Issue
Block a user