aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/ethernet
diff options
context:
space:
mode:
authorSowmini Varadhan <sowmini.varadhan@oracle.com>2014-11-08 20:41:53 -0500
committerDavid S. Miller <davem@davemloft.net>2014-11-10 21:05:36 -0500
commitb0cffed54338e19e3cc46c9963478223eee0d560 (patch)
tree1da52bafeb4401687b81c58d682b103d2289386a /drivers/net/ethernet
parent6f6e741f6de5ed5eedf7d19c9c0f8c73eaa0da7d (diff)
sunvnet: Fix race between vnet_start_xmit() and vnet_ack()
When vnet_start_xmit() is concurrent with vnet_ack(), we may have a race that looks like: thread 1 thread 2 vnet_start_xmit vnet_event_napi -> vnet_rx __vnet_tx_trigger for some desc X at this point dr->prod == X peer sends back a stopped ack for X we process X, but X == dr->prod so we bail out in vnet_ack with !idx_is_pending update dr->prod As a result of the fact that we never processed the stopped ack for X, the Tx path is led to incorrectly believe that the peer is still "started" and reading, but the peer has stopped reading, which will ultimately end in flow-control assertions. The fix is to synchronize the above 2 paths on the netif_tx_lock. Signed-off-by: Sowmini Varadhan <sowmini.varadhan@oracle.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/ethernet')
-rw-r--r--drivers/net/ethernet/sun/sunvnet.c10
1 files changed, 6 insertions, 4 deletions
diff --git a/drivers/net/ethernet/sun/sunvnet.c b/drivers/net/ethernet/sun/sunvnet.c
index 5c5fb59adf76..deb395a82e50 100644
--- a/drivers/net/ethernet/sun/sunvnet.c
+++ b/drivers/net/ethernet/sun/sunvnet.c
@@ -559,15 +559,17 @@ static int vnet_ack(struct vnet_port *port, void *msgbuf)
559 return 0; 559 return 0;
560 560
561 end = pkt->end_idx; 561 end = pkt->end_idx;
562 if (unlikely(!idx_is_pending(dr, end)))
563 return 0;
564
565 vp = port->vp; 562 vp = port->vp;
566 dev = vp->dev; 563 dev = vp->dev;
564 netif_tx_lock(dev);
565 if (unlikely(!idx_is_pending(dr, end))) {
566 netif_tx_unlock(dev);
567 return 0;
568 }
569
567 /* sync for race conditions with vnet_start_xmit() and tell xmit it 570 /* sync for race conditions with vnet_start_xmit() and tell xmit it
568 * is time to send a trigger. 571 * is time to send a trigger.
569 */ 572 */
570 netif_tx_lock(dev);
571 dr->cons = next_idx(end, dr); 573 dr->cons = next_idx(end, dr);
572 desc = vio_dring_entry(dr, dr->cons); 574 desc = vio_dring_entry(dr, dr->cons);
573 if (desc->hdr.state == VIO_DESC_READY && port->start_cons) { 575 if (desc->hdr.state == VIO_DESC_READY && port->start_cons) {