Commit 7e0f6616 authored by Ronak Doshi's avatar Ronak Doshi Committed by Khalid Elmously

vmxnet3: avoid xmit reset due to a race in vmxnet3

BugLink: http://bugs.launchpad.net/bugs/1768143

The field txNumDeferred is used by the driver to keep track of the number
of packets it has pushed to the emulation. The driver increments it on
pushing the packet to the emulation and the emulation resets it to 0 at
the end of the transmit.

There is a possibility of a race either when (a) ESX is under heavy load or
(b) workload inside VM is of low packet rate.

This race results in xmit hangs when network coalescing is disabled. This
change creates a local copy of txNumDeferred and uses it to perform ring
arithmetic.
Reported-by: default avatarNoriho Tanaka <ntanaka@vmware.com>
Signed-off-by: default avatarRonak Doshi <doshir@vmware.com>
Acked-by: default avatarShrikrishna Khare <skhare@vmware.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
(back ported from commit 7a4c003d)
Signed-off-by: default avatarJoseph Salisbury <joseph.salisbury@canonical.com>
Acked-by: default avatarStefan Bader <stefan.bader@canonical.com>
Acked-by: default avatarKleber Souza <kleber.souza@canonical.com>
Signed-off-by: default avatarKhalid Elmously <khalid.elmously@canonical.com>
parent 702c50f4
...@@ -957,6 +957,8 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq, ...@@ -957,6 +957,8 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq,
{ {
int ret; int ret;
u32 count; u32 count;
int num_pkts;
int tx_num_deferred;
unsigned long flags; unsigned long flags;
struct vmxnet3_tx_ctx ctx; struct vmxnet3_tx_ctx ctx;
union Vmxnet3_GenericDesc *gdesc; union Vmxnet3_GenericDesc *gdesc;
...@@ -1051,12 +1053,12 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq, ...@@ -1051,12 +1053,12 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq,
#else #else
gdesc = ctx.sop_txd; gdesc = ctx.sop_txd;
#endif #endif
tx_num_deferred = le32_to_cpu(tq->shared->txNumDeferred);
if (ctx.mss) { if (ctx.mss) {
gdesc->txd.hlen = ctx.eth_ip_hdr_size + ctx.l4_hdr_size; gdesc->txd.hlen = ctx.eth_ip_hdr_size + ctx.l4_hdr_size;
gdesc->txd.om = VMXNET3_OM_TSO; gdesc->txd.om = VMXNET3_OM_TSO;
gdesc->txd.msscof = ctx.mss; gdesc->txd.msscof = ctx.mss;
le32_add_cpu(&tq->shared->txNumDeferred, (skb->len - num_pkts = (skb->len - gdesc->txd.hlen + ctx.mss - 1) / ctx.mss;
gdesc->txd.hlen + ctx.mss - 1) / ctx.mss);
} else { } else {
if (skb->ip_summed == CHECKSUM_PARTIAL) { if (skb->ip_summed == CHECKSUM_PARTIAL) {
gdesc->txd.hlen = ctx.eth_ip_hdr_size; gdesc->txd.hlen = ctx.eth_ip_hdr_size;
...@@ -1067,8 +1069,10 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq, ...@@ -1067,8 +1069,10 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq,
gdesc->txd.om = 0; gdesc->txd.om = 0;
gdesc->txd.msscof = 0; gdesc->txd.msscof = 0;
} }
le32_add_cpu(&tq->shared->txNumDeferred, 1); num_pkts = 1;
} }
le32_add_cpu(&tq->shared->txNumDeferred, num_pkts);
tx_num_deferred += num_pkts;
if (skb_vlan_tag_present(skb)) { if (skb_vlan_tag_present(skb)) {
gdesc->txd.ti = 1; gdesc->txd.ti = 1;
...@@ -1094,8 +1098,7 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq, ...@@ -1094,8 +1098,7 @@ vmxnet3_tq_xmit(struct sk_buff *skb, struct vmxnet3_tx_queue *tq,
spin_unlock_irqrestore(&tq->tx_lock, flags); spin_unlock_irqrestore(&tq->tx_lock, flags);
if (le32_to_cpu(tq->shared->txNumDeferred) >= if (tx_num_deferred >= le32_to_cpu(tq->shared->txThreshold)) {
le32_to_cpu(tq->shared->txThreshold)) {
tq->shared->txNumDeferred = 0; tq->shared->txNumDeferred = 0;
VMXNET3_WRITE_BAR0_REG(adapter, VMXNET3_WRITE_BAR0_REG(adapter,
VMXNET3_REG_TXPROD + tq->qid * 8, VMXNET3_REG_TXPROD + tq->qid * 8,
......
...@@ -69,10 +69,10 @@ ...@@ -69,10 +69,10 @@
/* /*
* Version numbers * Version numbers
*/ */
#define VMXNET3_DRIVER_VERSION_STRING "1.4.7.0-k" #define VMXNET3_DRIVER_VERSION_STRING "1.4.12.0-k"
/* a 32-bit int, each byte encode a verion number in VMXNET3_DRIVER_VERSION */ /* a 32-bit int, each byte encode a verion number in VMXNET3_DRIVER_VERSION */
#define VMXNET3_DRIVER_VERSION_NUM 0x01040700 #define VMXNET3_DRIVER_VERSION_NUM 0x01040c00
#if defined(CONFIG_PCI_MSI) #if defined(CONFIG_PCI_MSI)
/* RSS only makes sense if MSI-X is supported. */ /* RSS only makes sense if MSI-X is supported. */
......
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