提交 9d39e5ba 编写于 作者: J Johannes Berg 提交者: Wey-Yi Guy

iwlagn: avoid hangs when restarting device

If a device error happens while the uCode is
being loaded or initialised, we will attempt
to restart the device (which will likely fail
again, but that's not the issue here). During
this new restart, we turn off the device, but
as the uCode failed to initialise it already
is turned off. As a consequence, grabbing NIC
access will fail and cause excessive messages
and hangs.

To fix this issue, introduce a new status bit
and only attempt to reprogram the device when
it isn't already disabled.
Signed-off-by: NJohannes Berg <johannes.berg@intel.com>
Signed-off-by: NWey-Yi Guy <wey-yi.w.guy@intel.com>
上级 ff776cec
...@@ -2363,12 +2363,21 @@ void iwlagn_stop_device(struct iwl_priv *priv) ...@@ -2363,12 +2363,21 @@ void iwlagn_stop_device(struct iwl_priv *priv)
/* device going down, Stop using ICT table */ /* device going down, Stop using ICT table */
iwl_disable_ict(priv); iwl_disable_ict(priv);
iwlagn_txq_ctx_stop(priv); /*
iwlagn_rxq_stop(priv); * If a HW restart happens during firmware loading,
* then the firmware loading might call this function
/* Power-down device's busmaster DMA clocks */ * and later it might be called again due to the
iwl_write_prph(priv, APMG_CLK_DIS_REG, APMG_CLK_VAL_DMA_CLK_RQT); * restart. So don't process again if the device is
udelay(5); * already dead.
*/
if (test_bit(STATUS_DEVICE_ENABLED, &priv->status)) {
iwlagn_txq_ctx_stop(priv);
iwlagn_rxq_stop(priv);
/* Power-down device's busmaster DMA clocks */
iwl_write_prph(priv, APMG_CLK_DIS_REG, APMG_CLK_VAL_DMA_CLK_RQT);
udelay(5);
}
/* Make sure (redundant) we've released our request to stay awake */ /* Make sure (redundant) we've released our request to stay awake */
iwl_clear_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_MAC_ACCESS_REQ); iwl_clear_bit(priv, CSR_GP_CNTRL, CSR_GP_CNTRL_REG_FLAG_MAC_ACCESS_REQ);
......
...@@ -995,6 +995,8 @@ void iwl_apm_stop(struct iwl_priv *priv) ...@@ -995,6 +995,8 @@ void iwl_apm_stop(struct iwl_priv *priv)
{ {
IWL_DEBUG_INFO(priv, "Stop card, put in low power state\n"); IWL_DEBUG_INFO(priv, "Stop card, put in low power state\n");
clear_bit(STATUS_DEVICE_ENABLED, &priv->status);
/* Stop device's DMA activity */ /* Stop device's DMA activity */
iwl_apm_stop_master(priv); iwl_apm_stop_master(priv);
...@@ -1109,6 +1111,8 @@ int iwl_apm_init(struct iwl_priv *priv) ...@@ -1109,6 +1111,8 @@ int iwl_apm_init(struct iwl_priv *priv)
iwl_set_bits_prph(priv, APMG_PCIDEV_STT_REG, iwl_set_bits_prph(priv, APMG_PCIDEV_STT_REG,
APMG_PCIDEV_STT_VAL_L1_ACT_DIS); APMG_PCIDEV_STT_VAL_L1_ACT_DIS);
set_bit(STATUS_DEVICE_ENABLED, &priv->status);
out: out:
return ret; return ret;
} }
......
...@@ -592,6 +592,7 @@ void iwlcore_free_geos(struct iwl_priv *priv); ...@@ -592,6 +592,7 @@ void iwlcore_free_geos(struct iwl_priv *priv);
#define STATUS_SCAN_HW 15 #define STATUS_SCAN_HW 15
#define STATUS_POWER_PMI 16 #define STATUS_POWER_PMI 16
#define STATUS_FW_ERROR 17 #define STATUS_FW_ERROR 17
#define STATUS_DEVICE_ENABLED 18
static inline int iwl_is_ready(struct iwl_priv *priv) static inline int iwl_is_ready(struct iwl_priv *priv)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册