aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net
diff options
context:
space:
mode:
authorBruce Allan <bruce.w.allan@intel.com>2012-03-20 20:39:12 -0400
committerJeff Kirsher <jeffrey.t.kirsher@intel.com>2012-04-03 18:25:08 -0400
commitbb9e44d0d0f45da356c39e485edacff6e14ba961 (patch)
treead4051375ad12407b3fdec34431a7a4da4009b0f /drivers/net
parentaacc1bea190d731755a65cb8ec31dd756f4e263e (diff)
e1000e: prevent oops when adapter is being closed and reset simultaneously
When the adapter is closed while it is simultaneously going through a reset, it can cause a null-pointer dereference when the two different code paths simultaneously cleanup up the Tx/Rx resources. Signed-off-by: Bruce Allan <bruce.w.allan@intel.com> Tested-by: Aaron Brown <aaron.f.brown@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
Diffstat (limited to 'drivers/net')
-rw-r--r--drivers/net/ethernet/intel/e1000e/e1000.h6
-rw-r--r--drivers/net/ethernet/intel/e1000e/netdev.c9
2 files changed, 15 insertions, 0 deletions
diff --git a/drivers/net/ethernet/intel/e1000e/e1000.h b/drivers/net/ethernet/intel/e1000e/e1000.h
index 86cdd4793992..b83897f76ee3 100644
--- a/drivers/net/ethernet/intel/e1000e/e1000.h
+++ b/drivers/net/ethernet/intel/e1000e/e1000.h
@@ -161,6 +161,12 @@ struct e1000_info;
161/* Time to wait before putting the device into D3 if there's no link (in ms). */ 161/* Time to wait before putting the device into D3 if there's no link (in ms). */
162#define LINK_TIMEOUT 100 162#define LINK_TIMEOUT 100
163 163
164/*
165 * Count for polling __E1000_RESET condition every 10-20msec.
166 * Experimentation has shown the reset can take approximately 210msec.
167 */
168#define E1000_CHECK_RESET_COUNT 25
169
164#define DEFAULT_RDTR 0 170#define DEFAULT_RDTR 0
165#define DEFAULT_RADV 8 171#define DEFAULT_RADV 8
166#define BURST_RDTR 0x20 172#define BURST_RDTR 0x20
diff --git a/drivers/net/ethernet/intel/e1000e/netdev.c b/drivers/net/ethernet/intel/e1000e/netdev.c
index 2c38a65ade87..6878601f20c5 100644
--- a/drivers/net/ethernet/intel/e1000e/netdev.c
+++ b/drivers/net/ethernet/intel/e1000e/netdev.c
@@ -3968,6 +3968,10 @@ static int e1000_close(struct net_device *netdev)
3968{ 3968{
3969 struct e1000_adapter *adapter = netdev_priv(netdev); 3969 struct e1000_adapter *adapter = netdev_priv(netdev);
3970 struct pci_dev *pdev = adapter->pdev; 3970 struct pci_dev *pdev = adapter->pdev;
3971 int count = E1000_CHECK_RESET_COUNT;
3972
3973 while (test_bit(__E1000_RESETTING, &adapter->state) && count--)
3974 usleep_range(10000, 20000);
3971 3975
3972 WARN_ON(test_bit(__E1000_RESETTING, &adapter->state)); 3976 WARN_ON(test_bit(__E1000_RESETTING, &adapter->state));
3973 3977
@@ -5472,6 +5476,11 @@ static int __e1000_shutdown(struct pci_dev *pdev, bool *enable_wake,
5472 netif_device_detach(netdev); 5476 netif_device_detach(netdev);
5473 5477
5474 if (netif_running(netdev)) { 5478 if (netif_running(netdev)) {
5479 int count = E1000_CHECK_RESET_COUNT;
5480
5481 while (test_bit(__E1000_RESETTING, &adapter->state) && count--)
5482 usleep_range(10000, 20000);
5483
5475 WARN_ON(test_bit(__E1000_RESETTING, &adapter->state)); 5484 WARN_ON(test_bit(__E1000_RESETTING, &adapter->state));
5476 e1000e_down(adapter); 5485 e1000e_down(adapter);
5477 e1000_free_irq(adapter); 5486 e1000_free_irq(adapter);