Commit 167bf96d authored by Sujith Manoharan's avatar Sujith Manoharan Committed by John W. Linville

ath9k: Fix beacon miss handling

The NoA duration for a GO is half the beacon interval
and a concurrent context like a STA can be active only
for that duration, before switching back to the GO's
operating channel.

Currently, when multiple beacons are missed, the dwell
time for the STA context is extended to improve the
chances of receiving a beacon. But the NoA is not updated
and this will cause problems since the GO is offline
for a period that is longer than the advertised duration.

Fix this by ensuring that the NoA is updated first before
extending the time slot for the STA context. Also make
sure that non-periodic NoA is used for a one-time, longer
absence period.
Signed-off-by: default avatarSujith Manoharan <c_manoha@qca.qualcomm.com>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 4899827d
...@@ -380,6 +380,7 @@ struct ath_chanctx_sched { ...@@ -380,6 +380,7 @@ struct ath_chanctx_sched {
bool offchannel_pending; bool offchannel_pending;
bool wait_switch; bool wait_switch;
bool force_noa_update; bool force_noa_update;
bool extend_absence;
enum ath_chanctx_state state; enum ath_chanctx_state state;
u8 beacon_miss; u8 beacon_miss;
......
...@@ -371,13 +371,6 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif, ...@@ -371,13 +371,6 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif,
sc->sched.switch_start_time = tsf_time; sc->sched.switch_start_time = tsf_time;
sc->cur_chan->last_beacon = sc->sched.next_tbtt; sc->cur_chan->last_beacon = sc->sched.next_tbtt;
/* If at least two consecutive beacons were missed on the STA
* chanctx, stay on the STA channel for one extra beacon period,
* to resync the timer properly.
*/
if (ctx->active && sc->sched.beacon_miss >= 2)
sc->sched.offchannel_duration = 3 * beacon_int / 2;
/* /*
* If an offchannel switch is scheduled to happen after * If an offchannel switch is scheduled to happen after
* a beacon transmission, update the NoA with one-shot * a beacon transmission, update the NoA with one-shot
...@@ -405,6 +398,26 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif, ...@@ -405,6 +398,26 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif,
break; break;
} }
/*
* Clear the extend_absence flag if it had been
* set during the previous beacon transmission,
* since we need to revert to the normal NoA
* schedule.
*/
if (ctx->active && sc->sched.extend_absence) {
avp->noa_duration = 0;
sc->sched.extend_absence = false;
}
/* If at least two consecutive beacons were missed on the STA
* chanctx, stay on the STA channel for one extra beacon period,
* to resync the timer properly.
*/
if (ctx->active && sc->sched.beacon_miss >= 2) {
avp->noa_duration = 0;
sc->sched.extend_absence = true;
}
/* Prevent wrap-around issues */ /* Prevent wrap-around issues */
if (avp->noa_duration && tsf_time - avp->noa_start > BIT(30)) if (avp->noa_duration && tsf_time - avp->noa_start > BIT(30))
avp->noa_duration = 0; avp->noa_duration = 0;
...@@ -418,11 +431,17 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif, ...@@ -418,11 +431,17 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif,
(!avp->noa_duration || sc->sched.force_noa_update)) { (!avp->noa_duration || sc->sched.force_noa_update)) {
avp->noa_index++; avp->noa_index++;
avp->noa_start = tsf_time; avp->noa_start = tsf_time;
if (sc->sched.extend_absence)
avp->noa_duration = (3 * beacon_int / 2) +
sc->sched.channel_switch_time;
else
avp->noa_duration = avp->noa_duration =
TU_TO_USEC(cur_conf->beacon_interval) / 2 + TU_TO_USEC(cur_conf->beacon_interval) / 2 +
sc->sched.channel_switch_time; sc->sched.channel_switch_time;
if (test_bit(ATH_OP_SCANNING, &common->op_flags)) if (test_bit(ATH_OP_SCANNING, &common->op_flags) ||
sc->sched.extend_absence)
avp->periodic_noa = false; avp->periodic_noa = false;
else else
avp->periodic_noa = true; avp->periodic_noa = true;
...@@ -520,7 +539,8 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif, ...@@ -520,7 +539,8 @@ void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif,
sc->sched.wait_switch = false; sc->sched.wait_switch = false;
tsf_time = TU_TO_USEC(cur_conf->beacon_interval) / 2; tsf_time = TU_TO_USEC(cur_conf->beacon_interval) / 2;
if (sc->sched.beacon_miss >= 2) {
if (sc->sched.extend_absence) {
sc->sched.beacon_miss = 0; sc->sched.beacon_miss = 0;
tsf_time *= 3; tsf_time *= 3;
} }
......
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