Commit c04f9f22 authored by Wey-Yi Guy's avatar Wey-Yi Guy

iwlwifi: fix fw_restart module parameter

fw_restart module parameter was broken by the recent check for stuck
queue patch, driver check the fx_restart module parameter
before reload the firmware; but the stuck queue timer kick in after
firmware error and reload the firmware even fw_restart=0. In this case,
driver should not reload the firmware, it is important to help debugging
uCode error.

The only case we can ignore the module parameter is when user request
firmware reload from debugfs, which can bypass the checking and perform
firmware reload all the time.
Signed-off-by: default avatarWey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: default avatarReinette Chatre <reinette.chatre@intel.com>
parent 88c1f4f6
...@@ -2627,7 +2627,7 @@ static void iwl_force_rf_reset(struct iwl_priv *priv) ...@@ -2627,7 +2627,7 @@ static void iwl_force_rf_reset(struct iwl_priv *priv)
} }
int iwl_force_reset(struct iwl_priv *priv, int mode) int iwl_force_reset(struct iwl_priv *priv, int mode, bool external)
{ {
struct iwl_force_reset *force_reset; struct iwl_force_reset *force_reset;
...@@ -2640,12 +2640,14 @@ int iwl_force_reset(struct iwl_priv *priv, int mode) ...@@ -2640,12 +2640,14 @@ int iwl_force_reset(struct iwl_priv *priv, int mode)
} }
force_reset = &priv->force_reset[mode]; force_reset = &priv->force_reset[mode];
force_reset->reset_request_count++; force_reset->reset_request_count++;
if (force_reset->last_force_reset_jiffies && if (!external) {
time_after(force_reset->last_force_reset_jiffies + if (force_reset->last_force_reset_jiffies &&
force_reset->reset_duration, jiffies)) { time_after(force_reset->last_force_reset_jiffies +
IWL_DEBUG_INFO(priv, "force reset rejected\n"); force_reset->reset_duration, jiffies)) {
force_reset->reset_reject_count++; IWL_DEBUG_INFO(priv, "force reset rejected\n");
return -EAGAIN; force_reset->reset_reject_count++;
return -EAGAIN;
}
} }
force_reset->reset_success_count++; force_reset->reset_success_count++;
force_reset->last_force_reset_jiffies = jiffies; force_reset->last_force_reset_jiffies = jiffies;
...@@ -2655,6 +2657,19 @@ int iwl_force_reset(struct iwl_priv *priv, int mode) ...@@ -2655,6 +2657,19 @@ int iwl_force_reset(struct iwl_priv *priv, int mode)
iwl_force_rf_reset(priv); iwl_force_rf_reset(priv);
break; break;
case IWL_FW_RESET: case IWL_FW_RESET:
/*
* if the request is from external(ex: debugfs),
* then always perform the request in regardless the module
* parameter setting
* if the request is from internal (uCode error or driver
* detect failure), then fw_restart module parameter
* need to be check before performing firmware reload
*/
if (!external && !priv->cfg->mod_params->restart_fw) {
IWL_DEBUG_INFO(priv, "Cancel firmware reload based on "
"module parameter setting\n");
break;
}
IWL_ERR(priv, "On demand firmware reload\n"); IWL_ERR(priv, "On demand firmware reload\n");
/* Set the FW error flag -- cleared on iwl_down */ /* Set the FW error flag -- cleared on iwl_down */
set_bit(STATUS_FW_ERROR, &priv->status); set_bit(STATUS_FW_ERROR, &priv->status);
...@@ -2713,7 +2728,7 @@ static int iwl_check_stuck_queue(struct iwl_priv *priv, int cnt) ...@@ -2713,7 +2728,7 @@ static int iwl_check_stuck_queue(struct iwl_priv *priv, int cnt)
"queue %d stuck %d time. Fw reload.\n", "queue %d stuck %d time. Fw reload.\n",
q->id, q->repeat_same_read_ptr); q->id, q->repeat_same_read_ptr);
q->repeat_same_read_ptr = 0; q->repeat_same_read_ptr = 0;
iwl_force_reset(priv, IWL_FW_RESET); iwl_force_reset(priv, IWL_FW_RESET, false);
} else { } else {
q->repeat_same_read_ptr++; q->repeat_same_read_ptr++;
IWL_DEBUG_RADIO(priv, IWL_DEBUG_RADIO(priv,
......
...@@ -525,7 +525,7 @@ int iwl_mac_hw_scan(struct ieee80211_hw *hw, ...@@ -525,7 +525,7 @@ int iwl_mac_hw_scan(struct ieee80211_hw *hw,
struct cfg80211_scan_request *req); struct cfg80211_scan_request *req);
void iwl_bg_start_internal_scan(struct work_struct *work); void iwl_bg_start_internal_scan(struct work_struct *work);
void iwl_internal_short_hw_scan(struct iwl_priv *priv); void iwl_internal_short_hw_scan(struct iwl_priv *priv);
int iwl_force_reset(struct iwl_priv *priv, int mode); int iwl_force_reset(struct iwl_priv *priv, int mode, bool external);
u16 iwl_fill_probe_req(struct iwl_priv *priv, struct ieee80211_mgmt *frame, u16 iwl_fill_probe_req(struct iwl_priv *priv, struct ieee80211_mgmt *frame,
const u8 *ta, const u8 *ie, int ie_len, int left); const u8 *ta, const u8 *ie, int ie_len, int left);
void iwl_setup_rx_scan_handlers(struct iwl_priv *priv); void iwl_setup_rx_scan_handlers(struct iwl_priv *priv);
......
...@@ -1487,7 +1487,7 @@ static ssize_t iwl_dbgfs_force_reset_write(struct file *file, ...@@ -1487,7 +1487,7 @@ static ssize_t iwl_dbgfs_force_reset_write(struct file *file,
switch (reset) { switch (reset) {
case IWL_RF_RESET: case IWL_RF_RESET:
case IWL_FW_RESET: case IWL_FW_RESET:
ret = iwl_force_reset(priv, reset); ret = iwl_force_reset(priv, reset, true);
break; break;
default: default:
return -EINVAL; return -EINVAL;
......
...@@ -238,7 +238,7 @@ void iwl_recover_from_statistics(struct iwl_priv *priv, ...@@ -238,7 +238,7 @@ void iwl_recover_from_statistics(struct iwl_priv *priv,
*/ */
IWL_ERR(priv, "low ack count detected, " IWL_ERR(priv, "low ack count detected, "
"restart firmware\n"); "restart firmware\n");
if (!iwl_force_reset(priv, IWL_FW_RESET)) if (!iwl_force_reset(priv, IWL_FW_RESET, false))
return; return;
} }
} }
...@@ -249,7 +249,7 @@ void iwl_recover_from_statistics(struct iwl_priv *priv, ...@@ -249,7 +249,7 @@ void iwl_recover_from_statistics(struct iwl_priv *priv,
* high plcp error detected * high plcp error detected
* reset Radio * reset Radio
*/ */
iwl_force_reset(priv, IWL_RF_RESET); iwl_force_reset(priv, IWL_RF_RESET, false);
} }
} }
} }
......
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