aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/ethernet/intel/e1000e/ich8lan.c
diff options
context:
space:
mode:
authorBruce Allan <bruce.w.allan@intel.com>2011-10-06 23:50:38 -0400
committerJeff Kirsher <jeffrey.t.kirsher@intel.com>2011-10-16 10:13:17 -0400
commita90b412cb8c7ccc1689f9ea130883d00a1f0a5bb (patch)
tree8bbc72841abb0900f950513b62e6d7f597b25c63 /drivers/net/ethernet/intel/e1000e/ich8lan.c
parent96cd8951684adaa5fd72952adef532d0b42f70e1 (diff)
e1000e: locking bug introduced by commit 67fd4fcb
Commit 67fd4fcb (e1000e: convert to stats64) added the ability to update statistics more accurately and on-demand through the net_device_ops .ndo_get_stats64 hook, but introduced a locking bug on 82577/8/9 when linked at half-duplex (seen on kernels with CONFIG_DEBUG_ATOMIC_SLEEP=y and CONFIG_PROVE_LOCKING=y). The commit introduced code paths that caused a mutex to be locked in atomic contexts, e.g. an rcu_read_lock is held when irqbalance reads the stats from /sys/class/net/ethX/statistics causing the mutex to be locked to read the Phy half-duplex statistics registers. The mutex was originally introduced to prevent concurrent accesses of resources (the NVM and Phy) shared by the driver, firmware and hardware a few years back when there was an issue with the NVM getting corrupted. It was later split into two mutexes - one for the NVM and one for the Phy when it was determined the NVM, unlike the Phy, should not be protected by the software/firmware/hardware semaphore (arbitration of which is done in part with the SWFLAG bit in the EXTCNF_CTRL register). This latter semaphore should be sufficient to prevent resource contention of the Phy in the driver (i.e. the mutex for Phy accesses is not needed), but to be sure the mutex is replaced with an atomic bit flag which will warn if any contention is possible. Also add additional debug output to help determine when the sw/fw/hw semaphore is owned by the firmware or hardware. Signed-off-by: Bruce Allan <bruce.w.allan@intel.com> Reported-by: Francois Romieu <romieu@fr.zoreil.com> Tested-by: Jeff Pieper <jeffrey.e.pieper@intel.com>
Diffstat (limited to 'drivers/net/ethernet/intel/e1000e/ich8lan.c')
-rw-r--r--drivers/net/ethernet/intel/e1000e/ich8lan.c21
1 files changed, 13 insertions, 8 deletions
diff --git a/drivers/net/ethernet/intel/e1000e/ich8lan.c b/drivers/net/ethernet/intel/e1000e/ich8lan.c
index 4f709749dbce..6a17c62cb86f 100644
--- a/drivers/net/ethernet/intel/e1000e/ich8lan.c
+++ b/drivers/net/ethernet/intel/e1000e/ich8lan.c
@@ -852,8 +852,6 @@ static void e1000_release_nvm_ich8lan(struct e1000_hw *hw)
852 mutex_unlock(&nvm_mutex); 852 mutex_unlock(&nvm_mutex);
853} 853}
854 854
855static DEFINE_MUTEX(swflag_mutex);
856
857/** 855/**
858 * e1000_acquire_swflag_ich8lan - Acquire software control flag 856 * e1000_acquire_swflag_ich8lan - Acquire software control flag
859 * @hw: pointer to the HW structure 857 * @hw: pointer to the HW structure
@@ -866,7 +864,12 @@ static s32 e1000_acquire_swflag_ich8lan(struct e1000_hw *hw)
866 u32 extcnf_ctrl, timeout = PHY_CFG_TIMEOUT; 864 u32 extcnf_ctrl, timeout = PHY_CFG_TIMEOUT;
867 s32 ret_val = 0; 865 s32 ret_val = 0;
868 866
869 mutex_lock(&swflag_mutex); 867 if (test_and_set_bit(__E1000_ACCESS_SHARED_RESOURCE,
868 &hw->adapter->state)) {
869 WARN(1, "e1000e: %s: contention for Phy access\n",
870 hw->adapter->netdev->name);
871 return -E1000_ERR_PHY;
872 }
870 873
871 while (timeout) { 874 while (timeout) {
872 extcnf_ctrl = er32(EXTCNF_CTRL); 875 extcnf_ctrl = er32(EXTCNF_CTRL);
@@ -878,7 +881,7 @@ static s32 e1000_acquire_swflag_ich8lan(struct e1000_hw *hw)
878 } 881 }
879 882
880 if (!timeout) { 883 if (!timeout) {
881 e_dbg("SW/FW/HW has locked the resource for too long.\n"); 884 e_dbg("SW has already locked the resource.\n");
882 ret_val = -E1000_ERR_CONFIG; 885 ret_val = -E1000_ERR_CONFIG;
883 goto out; 886 goto out;
884 } 887 }
@@ -898,7 +901,9 @@ static s32 e1000_acquire_swflag_ich8lan(struct e1000_hw *hw)
898 } 901 }
899 902
900 if (!timeout) { 903 if (!timeout) {
901 e_dbg("Failed to acquire the semaphore.\n"); 904 e_dbg("Failed to acquire the semaphore, FW or HW has it: "
905 "FWSM=0x%8.8x EXTCNF_CTRL=0x%8.8x)\n",
906 er32(FWSM), extcnf_ctrl);
902 extcnf_ctrl &= ~E1000_EXTCNF_CTRL_SWFLAG; 907 extcnf_ctrl &= ~E1000_EXTCNF_CTRL_SWFLAG;
903 ew32(EXTCNF_CTRL, extcnf_ctrl); 908 ew32(EXTCNF_CTRL, extcnf_ctrl);
904 ret_val = -E1000_ERR_CONFIG; 909 ret_val = -E1000_ERR_CONFIG;
@@ -907,7 +912,7 @@ static s32 e1000_acquire_swflag_ich8lan(struct e1000_hw *hw)
907 912
908out: 913out:
909 if (ret_val) 914 if (ret_val)
910 mutex_unlock(&swflag_mutex); 915 clear_bit(__E1000_ACCESS_SHARED_RESOURCE, &hw->adapter->state);
911 916
912 return ret_val; 917 return ret_val;
913} 918}
@@ -932,7 +937,7 @@ static void e1000_release_swflag_ich8lan(struct e1000_hw *hw)
932 e_dbg("Semaphore unexpectedly released by sw/fw/hw\n"); 937 e_dbg("Semaphore unexpectedly released by sw/fw/hw\n");
933 } 938 }
934 939
935 mutex_unlock(&swflag_mutex); 940 clear_bit(__E1000_ACCESS_SHARED_RESOURCE, &hw->adapter->state);
936} 941}
937 942
938/** 943/**
@@ -3139,7 +3144,7 @@ static s32 e1000_reset_hw_ich8lan(struct e1000_hw *hw)
3139 msleep(20); 3144 msleep(20);
3140 3145
3141 if (!ret_val) 3146 if (!ret_val)
3142 mutex_unlock(&swflag_mutex); 3147 clear_bit(__E1000_ACCESS_SHARED_RESOURCE, &hw->adapter->state);
3143 3148
3144 if (ctrl & E1000_CTRL_PHY_RST) { 3149 if (ctrl & E1000_CTRL_PHY_RST) {
3145 ret_val = hw->phy.ops.get_cfg_done(hw); 3150 ret_val = hw->phy.ops.get_cfg_done(hw);