Commit 5ab5a721 authored by Inaky Perez-Gonzalez's avatar Inaky Perez-Gonzalez

wimax/i2400m: fix device getting stuck in IDLE mode

The i2400m, when conected, will negotiate with the WiMAX basestation
to put the link in IDLE mode when it is not being used. Upon RX/TX
traffic, the link has to be restablished and that might require some
crypto handshakes and maybe a DHCP renew.

This process might take up to 20 (!) seconds and in some cases we were
seeing network watchdog warnings that weren't needed.

So the network watchdog timeout is updated to be slightly above that
20s threshold. As well, the driver itself will double check if the
device is stuck in IDLE mode -- if that happens, the device will be
reset (in this case the queue is also woken up to remove bogus--once
the device is reset--warnings).
Signed-off-by: default avatarInaky Perez-Gonzalez <inaky@linux.intel.com>
parent c931ceeb
...@@ -89,7 +89,10 @@ enum { ...@@ -89,7 +89,10 @@ enum {
* The MTU is 1400 or less * The MTU is 1400 or less
*/ */
I2400M_MAX_MTU = 1400, I2400M_MAX_MTU = 1400,
I2400M_TX_TIMEOUT = HZ, /* 20 secs? yep, this is the maximum timeout that the device
* might take to get out of IDLE / negotiate it with the base
* station. We add 1sec for good measure. */
I2400M_TX_TIMEOUT = 21 * HZ,
I2400M_TX_QLEN = 5, I2400M_TX_QLEN = 5,
}; };
...@@ -151,6 +154,7 @@ void i2400m_wake_tx_work(struct work_struct *ws) ...@@ -151,6 +154,7 @@ void i2400m_wake_tx_work(struct work_struct *ws)
{ {
int result; int result;
struct i2400m *i2400m = container_of(ws, struct i2400m, wake_tx_ws); struct i2400m *i2400m = container_of(ws, struct i2400m, wake_tx_ws);
struct net_device *net_dev = i2400m->wimax_dev.net_dev;
struct device *dev = i2400m_dev(i2400m); struct device *dev = i2400m_dev(i2400m);
struct sk_buff *skb = i2400m->wake_tx_skb; struct sk_buff *skb = i2400m->wake_tx_skb;
unsigned long flags; unsigned long flags;
...@@ -166,6 +170,11 @@ void i2400m_wake_tx_work(struct work_struct *ws) ...@@ -166,6 +170,11 @@ void i2400m_wake_tx_work(struct work_struct *ws)
dev_err(dev, "WAKE&TX: skb dissapeared!\n"); dev_err(dev, "WAKE&TX: skb dissapeared!\n");
goto out_put; goto out_put;
} }
/* If we have, somehow, lost the connection after this was
* queued, don't do anything; this might be the device got
* reset or just disconnected. */
if (unlikely(!netif_carrier_ok(net_dev)))
goto out_kfree;
result = i2400m_cmd_exit_idle(i2400m); result = i2400m_cmd_exit_idle(i2400m);
if (result == -EILSEQ) if (result == -EILSEQ)
result = 0; result = 0;
...@@ -176,7 +185,8 @@ void i2400m_wake_tx_work(struct work_struct *ws) ...@@ -176,7 +185,8 @@ void i2400m_wake_tx_work(struct work_struct *ws)
goto error; goto error;
} }
result = wait_event_timeout(i2400m->state_wq, result = wait_event_timeout(i2400m->state_wq,
i2400m->state != I2400M_SS_IDLE, 5 * HZ); i2400m->state != I2400M_SS_IDLE,
net_dev->watchdog_timeo - HZ/2);
if (result == 0) if (result == 0)
result = -ETIMEDOUT; result = -ETIMEDOUT;
if (result < 0) { if (result < 0) {
...@@ -187,8 +197,9 @@ void i2400m_wake_tx_work(struct work_struct *ws) ...@@ -187,8 +197,9 @@ void i2400m_wake_tx_work(struct work_struct *ws)
} }
msleep(20); /* device still needs some time or it drops it */ msleep(20); /* device still needs some time or it drops it */
result = i2400m_tx(i2400m, skb->data, skb->len, I2400M_PT_DATA); result = i2400m_tx(i2400m, skb->data, skb->len, I2400M_PT_DATA);
netif_wake_queue(i2400m->wimax_dev.net_dev);
error: error:
netif_wake_queue(net_dev);
out_kfree:
kfree_skb(skb); /* refcount transferred by _hard_start_xmit() */ kfree_skb(skb); /* refcount transferred by _hard_start_xmit() */
out_put: out_put:
i2400m_put(i2400m); i2400m_put(i2400m);
......
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