Commit 748299f2 authored by Felix Fietkau's avatar Felix Fietkau Committed by John W. Linville

ath9k: switch channel context for beaconing

Add a basic state machine for switch channel context
for beacon transmission.
Signed-off-by: default avatarFelix Fietkau <nbd@openwrt.org>
Signed-off-by: default avatarRajkumar Manoharan <rmanohar@qti.qualcomm.com>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 9a9c4fbc
...@@ -341,6 +341,28 @@ struct ath_chanctx { ...@@ -341,6 +341,28 @@ struct ath_chanctx {
bool stopped; bool stopped;
bool active; bool active;
bool assigned; bool assigned;
bool switch_after_beacon;
};
enum ath_chanctx_event {
ATH_CHANCTX_EVENT_BEACON_PREPARE,
ATH_CHANCTX_EVENT_BEACON_SENT,
ATH_CHANCTX_EVENT_TSF_TIMER,
};
enum ath_chanctx_state {
ATH_CHANCTX_STATE_IDLE,
ATH_CHANCTX_STATE_WAIT_FOR_BEACON,
ATH_CHANCTX_STATE_WAIT_FOR_TIMER,
ATH_CHANCTX_STATE_SWITCH,
};
struct ath_chanctx_sched {
bool beacon_pending;
enum ath_chanctx_state state;
u32 next_tbtt;
unsigned int channel_switch_time;
}; };
enum ath_offchannel_state { enum ath_offchannel_state {
...@@ -388,6 +410,8 @@ void ath_chanctx_offchan_switch(struct ath_softc *sc, ...@@ -388,6 +410,8 @@ void ath_chanctx_offchan_switch(struct ath_softc *sc,
struct ieee80211_channel *chan); struct ieee80211_channel *chan);
struct ath_chanctx *ath_chanctx_get_oper_chan(struct ath_softc *sc, struct ath_chanctx *ath_chanctx_get_oper_chan(struct ath_softc *sc,
bool active); bool active);
void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif,
enum ath_chanctx_event ev);
int ath_reset_internal(struct ath_softc *sc, struct ath9k_channel *hchan); int ath_reset_internal(struct ath_softc *sc, struct ath9k_channel *hchan);
int ath_startrecv(struct ath_softc *sc); int ath_startrecv(struct ath_softc *sc);
...@@ -826,6 +850,7 @@ struct ath_softc { ...@@ -826,6 +850,7 @@ struct ath_softc {
struct ath_chanctx *next_chan; struct ath_chanctx *next_chan;
spinlock_t chan_lock; spinlock_t chan_lock;
struct ath_offchannel offchannel; struct ath_offchannel offchannel;
struct ath_chanctx_sched sched;
#ifdef CONFIG_MAC80211_LEDS #ifdef CONFIG_MAC80211_LEDS
bool led_registered; bool led_registered;
......
...@@ -374,12 +374,19 @@ void ath9k_beacon_tasklet(unsigned long data) ...@@ -374,12 +374,19 @@ void ath9k_beacon_tasklet(unsigned long data)
vif = sc->beacon.bslot[slot]; vif = sc->beacon.bslot[slot];
/* EDMA devices check that in the tx completion function. */ /* EDMA devices check that in the tx completion function. */
if (!edma && ath9k_csa_is_finished(sc, vif)) if (!edma) {
if (sc->sched.beacon_pending)
ath_chanctx_event(sc, NULL,
ATH_CHANCTX_EVENT_BEACON_SENT);
if (ath9k_csa_is_finished(sc, vif))
return; return;
}
if (!vif || !vif->bss_conf.enable_beacon) if (!vif || !vif->bss_conf.enable_beacon)
return; return;
ath_chanctx_event(sc, vif, ATH_CHANCTX_EVENT_BEACON_PREPARE);
bf = ath9k_beacon_generate(sc->hw, vif); bf = ath9k_beacon_generate(sc->hw, vif);
if (sc->beacon.bmisscnt != 0) { if (sc->beacon.bmisscnt != 0) {
......
...@@ -202,10 +202,33 @@ ath_chanctx_send_ps_frame(struct ath_softc *sc, bool powersave) ...@@ -202,10 +202,33 @@ ath_chanctx_send_ps_frame(struct ath_softc *sc, bool powersave)
return sent; return sent;
} }
static bool ath_chanctx_defer_switch(struct ath_softc *sc)
{
if (sc->cur_chan == &sc->offchannel.chan)
return false;
switch (sc->sched.state) {
case ATH_CHANCTX_STATE_SWITCH:
return false;
case ATH_CHANCTX_STATE_IDLE:
if (!sc->cur_chan->switch_after_beacon)
return false;
sc->sched.state = ATH_CHANCTX_STATE_WAIT_FOR_BEACON;
break;
default:
break;
}
return true;
}
void ath_chanctx_work(struct work_struct *work) void ath_chanctx_work(struct work_struct *work)
{ {
struct ath_softc *sc = container_of(work, struct ath_softc, struct ath_softc *sc = container_of(work, struct ath_softc,
chanctx_work); chanctx_work);
struct timespec ts;
bool measure_time = false;
bool send_ps = false; bool send_ps = false;
mutex_lock(&sc->mutex); mutex_lock(&sc->mutex);
...@@ -216,10 +239,20 @@ void ath_chanctx_work(struct work_struct *work) ...@@ -216,10 +239,20 @@ void ath_chanctx_work(struct work_struct *work)
return; return;
} }
if (ath_chanctx_defer_switch(sc)) {
spin_unlock_bh(&sc->chan_lock);
mutex_unlock(&sc->mutex);
return;
}
if (sc->cur_chan != sc->next_chan) { if (sc->cur_chan != sc->next_chan) {
sc->cur_chan->stopped = true; sc->cur_chan->stopped = true;
spin_unlock_bh(&sc->chan_lock); spin_unlock_bh(&sc->chan_lock);
if (sc->next_chan == &sc->offchannel.chan) {
getrawmonotonic(&ts);
measure_time = true;
}
__ath9k_flush(sc->hw, ~0, true); __ath9k_flush(sc->hw, ~0, true);
if (ath_chanctx_send_ps_frame(sc, true)) if (ath_chanctx_send_ps_frame(sc, true))
...@@ -236,13 +269,17 @@ void ath_chanctx_work(struct work_struct *work) ...@@ -236,13 +269,17 @@ void ath_chanctx_work(struct work_struct *work)
sc->cur_chan = sc->next_chan; sc->cur_chan = sc->next_chan;
sc->cur_chan->stopped = false; sc->cur_chan->stopped = false;
sc->next_chan = NULL; sc->next_chan = NULL;
sc->sched.state = ATH_CHANCTX_STATE_IDLE;
spin_unlock_bh(&sc->chan_lock); spin_unlock_bh(&sc->chan_lock);
if (sc->sc_ah->chip_fullsleep || if (sc->sc_ah->chip_fullsleep ||
memcmp(&sc->cur_chandef, &sc->cur_chan->chandef, memcmp(&sc->cur_chandef, &sc->cur_chan->chandef,
sizeof(sc->cur_chandef))) sizeof(sc->cur_chandef))) {
ath_set_channel(sc); ath_set_channel(sc);
if (measure_time)
sc->sched.channel_switch_time =
ath9k_hw_get_tsf_offset(&ts, NULL);
}
if (send_ps) if (send_ps)
ath_chanctx_send_ps_frame(sc, false); ath_chanctx_send_ps_frame(sc, false);
...@@ -335,3 +372,46 @@ void ath_chanctx_offchan_switch(struct ath_softc *sc, ...@@ -335,3 +372,46 @@ void ath_chanctx_offchan_switch(struct ath_softc *sc,
ath_chanctx_switch(sc, &sc->offchannel.chan, &chandef); ath_chanctx_switch(sc, &sc->offchannel.chan, &chandef);
} }
void ath_chanctx_event(struct ath_softc *sc, struct ieee80211_vif *vif,
enum ath_chanctx_event ev)
{
struct ath_hw *ah = sc->sc_ah;
u32 tsf_time;
spin_lock_bh(&sc->chan_lock);
switch (ev) {
case ATH_CHANCTX_EVENT_BEACON_PREPARE:
if (sc->sched.state != ATH_CHANCTX_STATE_WAIT_FOR_BEACON)
break;
sc->sched.beacon_pending = true;
sc->sched.next_tbtt = REG_READ(ah, AR_NEXT_TBTT_TIMER);
break;
case ATH_CHANCTX_EVENT_BEACON_SENT:
if (!sc->sched.beacon_pending)
break;
sc->sched.beacon_pending = false;
if (sc->sched.state != ATH_CHANCTX_STATE_WAIT_FOR_BEACON)
break;
/* defer channel switch by a quarter beacon interval */
tsf_time = TU_TO_USEC(sc->cur_chan->beacon.beacon_interval);
tsf_time = sc->sched.next_tbtt + tsf_time / 4;
sc->sched.state = ATH_CHANCTX_STATE_WAIT_FOR_TIMER;
ath9k_hw_gen_timer_start(ah, sc->p2p_ps_timer, tsf_time,
1000000);
break;
case ATH_CHANCTX_EVENT_TSF_TIMER:
if (sc->sched.state != ATH_CHANCTX_STATE_WAIT_FOR_TIMER)
break;
sc->sched.state = ATH_CHANCTX_STATE_SWITCH;
ieee80211_queue_work(sc->hw, &sc->chanctx_work);
break;
}
spin_unlock_bh(&sc->chan_lock);
}
...@@ -1047,10 +1047,14 @@ void ath9k_calculate_summary_state(struct ath_softc *sc, ...@@ -1047,10 +1047,14 @@ void ath9k_calculate_summary_state(struct ath_softc *sc,
ath9k_hw_setopmode(ah); ath9k_hw_setopmode(ah);
ctx->switch_after_beacon = false;
if ((iter_data.nstations + iter_data.nadhocs + iter_data.nmeshes) > 0) if ((iter_data.nstations + iter_data.nadhocs + iter_data.nmeshes) > 0)
ah->imask |= ATH9K_INT_TSFOOR; ah->imask |= ATH9K_INT_TSFOOR;
else else {
ah->imask &= ~ATH9K_INT_TSFOOR; ah->imask &= ~ATH9K_INT_TSFOOR;
if (iter_data.naps == 1 && iter_data.beacons)
ctx->switch_after_beacon = true;
}
ah->imask &= ~ATH9K_INT_SWBA; ah->imask &= ~ATH9K_INT_SWBA;
if (ah->opmode == NL80211_IFTYPE_STATION) { if (ah->opmode == NL80211_IFTYPE_STATION) {
...@@ -1664,6 +1668,9 @@ void ath9k_p2p_ps_timer(void *priv) ...@@ -1664,6 +1668,9 @@ void ath9k_p2p_ps_timer(void *priv)
struct ath_node *an; struct ath_node *an;
u32 tsf; u32 tsf;
ath9k_hw_gen_timer_stop(sc->sc_ah, sc->p2p_ps_timer);
ath_chanctx_event(sc, NULL, ATH_CHANCTX_EVENT_TSF_TIMER);
if (!avp || avp->chanctx != sc->cur_chan) if (!avp || avp->chanctx != sc->cur_chan)
return; return;
......
...@@ -2617,6 +2617,8 @@ void ath_tx_edma_tasklet(struct ath_softc *sc) ...@@ -2617,6 +2617,8 @@ void ath_tx_edma_tasklet(struct ath_softc *sc)
sc->beacon.tx_processed = true; sc->beacon.tx_processed = true;
sc->beacon.tx_last = !(ts.ts_status & ATH9K_TXERR_MASK); sc->beacon.tx_last = !(ts.ts_status & ATH9K_TXERR_MASK);
ath_chanctx_event(sc, NULL,
ATH_CHANCTX_EVENT_BEACON_SENT);
ath9k_csa_update(sc); ath9k_csa_update(sc);
continue; continue;
} }
......
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