Commit d8498088 authored by Florian Fainelli's avatar Florian Fainelli Committed by David S. Miller

net: systemport: fix transmit locking scheme

Our transmit locking scheme did not account for the TX ring full
interrupt. If a TX ring full interrupt fires while we are attempting to
transmit, we will cause a deadlock to occur. Fix this by making sure
that we properly disable interrupts while acquiring the spinlock.
Signed-off-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 6934e79e
...@@ -637,10 +637,11 @@ static unsigned int bcm_sysport_tx_reclaim(struct bcm_sysport_priv *priv, ...@@ -637,10 +637,11 @@ static unsigned int bcm_sysport_tx_reclaim(struct bcm_sysport_priv *priv,
struct bcm_sysport_tx_ring *ring) struct bcm_sysport_tx_ring *ring)
{ {
unsigned int released; unsigned int released;
unsigned long flags;
spin_lock(&ring->lock); spin_lock_irqsave(&ring->lock, flags);
released = __bcm_sysport_tx_reclaim(priv, ring); released = __bcm_sysport_tx_reclaim(priv, ring);
spin_unlock(&ring->lock); spin_unlock_irqrestore(&ring->lock, flags);
return released; return released;
} }
...@@ -822,6 +823,7 @@ static netdev_tx_t bcm_sysport_xmit(struct sk_buff *skb, ...@@ -822,6 +823,7 @@ static netdev_tx_t bcm_sysport_xmit(struct sk_buff *skb,
struct netdev_queue *txq; struct netdev_queue *txq;
struct dma_desc *desc; struct dma_desc *desc;
unsigned int skb_len; unsigned int skb_len;
unsigned long flags;
dma_addr_t mapping; dma_addr_t mapping;
u32 len_status; u32 len_status;
u16 queue; u16 queue;
...@@ -831,8 +833,8 @@ static netdev_tx_t bcm_sysport_xmit(struct sk_buff *skb, ...@@ -831,8 +833,8 @@ static netdev_tx_t bcm_sysport_xmit(struct sk_buff *skb,
txq = netdev_get_tx_queue(dev, queue); txq = netdev_get_tx_queue(dev, queue);
ring = &priv->tx_rings[queue]; ring = &priv->tx_rings[queue];
/* lock against tx reclaim in BH context */ /* lock against tx reclaim in BH context and TX ring full interrupt */
spin_lock(&ring->lock); spin_lock_irqsave(&ring->lock, flags);
if (unlikely(ring->desc_count == 0)) { if (unlikely(ring->desc_count == 0)) {
netif_tx_stop_queue(txq); netif_tx_stop_queue(txq);
netdev_err(dev, "queue %d awake and ring full!\n", queue); netdev_err(dev, "queue %d awake and ring full!\n", queue);
...@@ -914,7 +916,7 @@ static netdev_tx_t bcm_sysport_xmit(struct sk_buff *skb, ...@@ -914,7 +916,7 @@ static netdev_tx_t bcm_sysport_xmit(struct sk_buff *skb,
ret = NETDEV_TX_OK; ret = NETDEV_TX_OK;
out: out:
spin_unlock(&ring->lock); spin_unlock_irqrestore(&ring->lock, flags);
return ret; return ret;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment