iwlwifi: dvm: fix the passive-no-RX workaround
Alex Romosan reported that since the mac80211 changes in "mac80211: start auth/assoc timeout on frame status" and the subsequent fixes in "mac80211: fix auth/assoc timeout handling" (commits1672c0e319
and89afe614c0
) there's sometimes an issue connecting to a 5 GHz network with the iwlwifi DVM driver. The reason appears to be that since these commits any bad TX status makes mac80211 immediately try again, causing all of the authentication attempts to be quickly rejected by the firmware as it hasn't heard a beacon yet. Before, it would wait for the timeout regardless of status. To fix this, invoke the passive-no-RX workaround when not associated yet as well. This will cause the first frame to get lost, but then the driver will stop the queues and the second attempt will only be transmitted after hearing a beacon, thus delaying it appropriately to not make the firmware reject it again. Reported-by: Alex Romosan <romosan@sycorax.lbl.gov> Tested-by: Alex Romosan <romosan@sycorax.lbl.gov> Signed-off-by: Johannes Berg <johannes.berg@intel.com>
This commit is contained in:
parent
2d5d50ee59
commit
b8730b403a
|
@ -1419,6 +1419,14 @@ void iwlagn_bss_info_changed(struct ieee80211_hw *hw,
|
||||||
|
|
||||||
mutex_lock(&priv->mutex);
|
mutex_lock(&priv->mutex);
|
||||||
|
|
||||||
|
if (changes & BSS_CHANGED_IDLE && bss_conf->idle) {
|
||||||
|
/*
|
||||||
|
* If we go idle, then clearly no "passive-no-rx"
|
||||||
|
* workaround is needed any more, this is a reset.
|
||||||
|
*/
|
||||||
|
iwlagn_lift_passive_no_rx(priv);
|
||||||
|
}
|
||||||
|
|
||||||
if (unlikely(!iwl_is_ready(priv))) {
|
if (unlikely(!iwl_is_ready(priv))) {
|
||||||
IWL_DEBUG_MAC80211(priv, "leave - not ready\n");
|
IWL_DEBUG_MAC80211(priv, "leave - not ready\n");
|
||||||
mutex_unlock(&priv->mutex);
|
mutex_unlock(&priv->mutex);
|
||||||
|
@ -1450,16 +1458,6 @@ void iwlagn_bss_info_changed(struct ieee80211_hw *hw,
|
||||||
priv->timestamp = bss_conf->sync_tsf;
|
priv->timestamp = bss_conf->sync_tsf;
|
||||||
ctx->staging.filter_flags |= RXON_FILTER_ASSOC_MSK;
|
ctx->staging.filter_flags |= RXON_FILTER_ASSOC_MSK;
|
||||||
} else {
|
} else {
|
||||||
/*
|
|
||||||
* If we disassociate while there are pending
|
|
||||||
* frames, just wake up the queues and let the
|
|
||||||
* frames "escape" ... This shouldn't really
|
|
||||||
* be happening to start with, but we should
|
|
||||||
* not get stuck in this case either since it
|
|
||||||
* can happen if userspace gets confused.
|
|
||||||
*/
|
|
||||||
iwlagn_lift_passive_no_rx(priv);
|
|
||||||
|
|
||||||
ctx->staging.filter_flags &= ~RXON_FILTER_ASSOC_MSK;
|
ctx->staging.filter_flags &= ~RXON_FILTER_ASSOC_MSK;
|
||||||
|
|
||||||
if (ctx->ctxid == IWL_RXON_CTX_BSS)
|
if (ctx->ctxid == IWL_RXON_CTX_BSS)
|
||||||
|
|
|
@ -1192,7 +1192,7 @@ int iwlagn_rx_reply_tx(struct iwl_priv *priv, struct iwl_rx_cmd_buffer *rxb,
|
||||||
memset(&info->status, 0, sizeof(info->status));
|
memset(&info->status, 0, sizeof(info->status));
|
||||||
|
|
||||||
if (status == TX_STATUS_FAIL_PASSIVE_NO_RX &&
|
if (status == TX_STATUS_FAIL_PASSIVE_NO_RX &&
|
||||||
iwl_is_associated_ctx(ctx) && ctx->vif &&
|
ctx->vif &&
|
||||||
ctx->vif->type == NL80211_IFTYPE_STATION) {
|
ctx->vif->type == NL80211_IFTYPE_STATION) {
|
||||||
/* block and stop all queues */
|
/* block and stop all queues */
|
||||||
priv->passive_no_rx = true;
|
priv->passive_no_rx = true;
|
||||||
|
|
Loading…
Reference in New Issue