Commit 4b7679a5 authored by Johannes Berg's avatar Johannes Berg Committed by John W. Linville

mac80211: clean up rate control API

Long awaited, hard work. This patch totally cleans up the rate control
API to remove the requirement to include internal headers outside of
net/mac80211/.

There's one internal use in the PID algorithm left for mesh networking,
we'll have to figure out a way to clean that one up and decide how to
do the peer link evaluation, possibly independent of the rate control
algorithm or via new API.

Additionally, ath9k is left using the cross-inclusion hack for now, we
will add new API where necessary to make this work properly, but right
now I'm not expert enough to do it. It's still off better than before.
Signed-off-by: default avatarJohannes Berg <johannes@sipsolutions.net>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 2ff6a6d4
...@@ -20,6 +20,7 @@ ...@@ -20,6 +20,7 @@
*/ */
#include "core.h" #include "core.h"
/* FIXME: remove this include! */
#include "../net/mac80211/rate.h" #include "../net/mac80211/rate.h"
static u32 tx_triglevel_max; static u32 tx_triglevel_max;
...@@ -1812,20 +1813,18 @@ static void ath_rc_sib_init(struct ath_rate_node *ath_rc_priv) ...@@ -1812,20 +1813,18 @@ static void ath_rc_sib_init(struct ath_rate_node *ath_rc_priv)
} }
static void ath_setup_rates(struct ieee80211_local *local, struct sta_info *sta) static void ath_setup_rates(struct ath_softc *sc,
struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta,
struct ath_rate_node *rc_priv)
{ {
struct ieee80211_supported_band *sband;
struct ieee80211_hw *hw = local_to_hw(local);
struct ath_softc *sc = hw->priv;
struct ath_rate_node *rc_priv = sta->rate_ctrl_priv;
int i, j = 0; int i, j = 0;
DPRINTF(sc, ATH_DBG_RATE, "%s\n", __func__); DPRINTF(sc, ATH_DBG_RATE, "%s\n", __func__);
sband = local->hw.wiphy->bands[local->hw.conf.channel->band];
for (i = 0; i < sband->n_bitrates; i++) { for (i = 0; i < sband->n_bitrates; i++) {
if (sta->sta.supp_rates[local->hw.conf.channel->band] & BIT(i)) { if (sta->supp_rates[sband->band] & BIT(i)) {
rc_priv->neg_rates.rs_rates[j] rc_priv->neg_rates.rs_rates[j]
= (sband->bitrates[i].bitrate * 2) / 10; = (sband->bitrates[i].bitrate * 2) / 10;
j++; j++;
...@@ -1852,19 +1851,17 @@ void ath_rc_node_update(struct ieee80211_hw *hw, struct ath_rate_node *rc_priv) ...@@ -1852,19 +1851,17 @@ void ath_rc_node_update(struct ieee80211_hw *hw, struct ath_rate_node *rc_priv)
} }
/* Rate Control callbacks */ /* Rate Control callbacks */
static void ath_tx_status(void *priv, struct net_device *dev, static void ath_tx_status(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta, void *priv_sta,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct ath_softc *sc = priv; struct ath_softc *sc = priv;
struct ath_tx_info_priv *tx_info_priv; struct ath_tx_info_priv *tx_info_priv;
struct ath_node *an; struct ath_node *an;
struct sta_info *sta;
struct ieee80211_local *local;
struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb); struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb);
struct ieee80211_hdr *hdr; struct ieee80211_hdr *hdr;
__le16 fc; __le16 fc;
local = hw_to_local(sc->hw);
hdr = (struct ieee80211_hdr *)skb->data; hdr = (struct ieee80211_hdr *)skb->data;
fc = hdr->frame_control; fc = hdr->frame_control;
tx_info_priv = (struct ath_tx_info_priv *)tx_info->driver_data[0]; tx_info_priv = (struct ath_tx_info_priv *)tx_info->driver_data[0];
...@@ -1873,8 +1870,7 @@ static void ath_tx_status(void *priv, struct net_device *dev, ...@@ -1873,8 +1870,7 @@ static void ath_tx_status(void *priv, struct net_device *dev,
an = ath_node_find(sc, hdr->addr1); an = ath_node_find(sc, hdr->addr1);
spin_unlock_bh(&sc->node_lock); spin_unlock_bh(&sc->node_lock);
sta = sta_info_get(local, hdr->addr1); if (!an || !priv_sta || !ieee80211_is_data(fc)) {
if (!an || !sta || !ieee80211_is_data(fc)) {
if (tx_info->driver_data[0] != NULL) { if (tx_info->driver_data[0] != NULL) {
kfree(tx_info->driver_data[0]); kfree(tx_info->driver_data[0]);
tx_info->driver_data[0] = NULL; tx_info->driver_data[0] = NULL;
...@@ -1882,24 +1878,22 @@ static void ath_tx_status(void *priv, struct net_device *dev, ...@@ -1882,24 +1878,22 @@ static void ath_tx_status(void *priv, struct net_device *dev,
return; return;
} }
if (tx_info->driver_data[0] != NULL) { if (tx_info->driver_data[0] != NULL) {
ath_rate_tx_complete(sc, an, sta->rate_ctrl_priv, tx_info_priv); ath_rate_tx_complete(sc, an, priv_sta, tx_info_priv);
kfree(tx_info->driver_data[0]); kfree(tx_info->driver_data[0]);
tx_info->driver_data[0] = NULL; tx_info->driver_data[0] = NULL;
} }
} }
static void ath_tx_aggr_resp(struct ath_softc *sc, static void ath_tx_aggr_resp(struct ath_softc *sc,
struct sta_info *sta, struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta,
struct ath_node *an, struct ath_node *an,
u8 tidno) u8 tidno)
{ {
struct ieee80211_hw *hw = sc->hw;
struct ieee80211_local *local;
struct ath_atx_tid *txtid; struct ath_atx_tid *txtid;
struct ieee80211_supported_band *sband;
u16 buffersize = 0; u16 buffersize = 0;
int state; int state;
DECLARE_MAC_BUF(mac); struct sta_info *si;
if (!(sc->sc_flags & SC_OP_TXAGGR)) if (!(sc->sc_flags & SC_OP_TXAGGR))
return; return;
...@@ -1908,11 +1902,16 @@ static void ath_tx_aggr_resp(struct ath_softc *sc, ...@@ -1908,11 +1902,16 @@ static void ath_tx_aggr_resp(struct ath_softc *sc,
if (!txtid->paused) if (!txtid->paused)
return; return;
local = hw_to_local(sc->hw); /*
sband = hw->wiphy->bands[hw->conf.channel->band]; * XXX: This is entirely busted, we aren't supposed to
* access the sta from here because it's internal
* to mac80211, and looking at the state without
* locking is wrong too.
*/
si = container_of(sta, struct sta_info, sta);
buffersize = IEEE80211_MIN_AMPDU_BUF << buffersize = IEEE80211_MIN_AMPDU_BUF <<
sband->ht_info.ampdu_factor; /* FIXME */ sband->ht_info.ampdu_factor; /* FIXME */
state = sta->ampdu_mlme.tid_state_tx[tidno]; state = si->ampdu_mlme.tid_state_tx[tidno];
if (state & HT_ADDBA_RECEIVED_MSK) { if (state & HT_ADDBA_RECEIVED_MSK) {
txtid->addba_exchangecomplete = 1; txtid->addba_exchangecomplete = 1;
...@@ -1928,18 +1927,15 @@ static void ath_tx_aggr_resp(struct ath_softc *sc, ...@@ -1928,18 +1927,15 @@ static void ath_tx_aggr_resp(struct ath_softc *sc,
} }
} }
static void ath_get_rate(void *priv, struct net_device *dev, static void ath_get_rate(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_supported_band *sband, struct ieee80211_sta *sta, void *priv_sta,
struct sk_buff *skb, struct sk_buff *skb, struct rate_selection *sel)
struct rate_selection *sel)
{ {
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data; struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); struct ath_softc *sc = priv;
struct sta_info *sta;
struct ath_softc *sc = (struct ath_softc *)priv;
struct ieee80211_hw *hw = sc->hw; struct ieee80211_hw *hw = sc->hw;
struct ath_tx_info_priv *tx_info_priv; struct ath_tx_info_priv *tx_info_priv;
struct ath_rate_node *ath_rc_priv; struct ath_rate_node *ath_rc_priv = priv_sta;
struct ath_node *an; struct ath_node *an;
struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb); struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb);
int is_probe = FALSE, chk, ret; int is_probe = FALSE, chk, ret;
...@@ -1955,8 +1951,7 @@ static void ath_get_rate(void *priv, struct net_device *dev, ...@@ -1955,8 +1951,7 @@ static void ath_get_rate(void *priv, struct net_device *dev,
ASSERT(tx_info->driver_data[0] != NULL); ASSERT(tx_info->driver_data[0] != NULL);
tx_info_priv = (struct ath_tx_info_priv *)tx_info->driver_data[0]; tx_info_priv = (struct ath_tx_info_priv *)tx_info->driver_data[0];
sta = sta_info_get(local, hdr->addr1); lowest_idx = rate_lowest_index(sband, sta);
lowest_idx = rate_lowest_index(local, sband, sta);
tx_info_priv->min_rate = (sband->bitrates[lowest_idx].bitrate * 2) / 10; tx_info_priv->min_rate = (sband->bitrates[lowest_idx].bitrate * 2) / 10;
/* lowest rate for management and multicast/broadcast frames */ /* lowest rate for management and multicast/broadcast frames */
if (!ieee80211_is_data(fc) || if (!ieee80211_is_data(fc) ||
...@@ -1965,8 +1960,6 @@ static void ath_get_rate(void *priv, struct net_device *dev, ...@@ -1965,8 +1960,6 @@ static void ath_get_rate(void *priv, struct net_device *dev,
return; return;
} }
ath_rc_priv = sta->rate_ctrl_priv;
/* Find tx rate for unicast frames */ /* Find tx rate for unicast frames */
ath_rate_findrate(sc, ath_rc_priv, ath_rate_findrate(sc, ath_rc_priv,
ATH_11N_TXMAXTRY, 4, ATH_11N_TXMAXTRY, 4,
...@@ -1975,8 +1968,7 @@ static void ath_get_rate(void *priv, struct net_device *dev, ...@@ -1975,8 +1968,7 @@ static void ath_get_rate(void *priv, struct net_device *dev,
&is_probe, &is_probe,
false); false);
if (is_probe) if (is_probe)
sel->probe_idx = ((struct ath_tx_ratectrl *) sel->probe_idx = ath_rc_priv->tx_ratectrl.probe_rate;
sta->rate_ctrl_priv)->probe_rate;
/* Ratecontrol sometimes returns invalid rate index */ /* Ratecontrol sometimes returns invalid rate index */
if (tx_info_priv->rcs[0].rix != 0xff) if (tx_info_priv->rcs[0].rix != 0xff)
...@@ -2020,37 +2012,31 @@ static void ath_get_rate(void *priv, struct net_device *dev, ...@@ -2020,37 +2012,31 @@ static void ath_get_rate(void *priv, struct net_device *dev,
__func__, __func__,
print_mac(mac, hdr->addr1)); print_mac(mac, hdr->addr1));
} else if (chk == AGGR_EXCHANGE_PROGRESS) } else if (chk == AGGR_EXCHANGE_PROGRESS)
ath_tx_aggr_resp(sc, sta, an, tid); ath_tx_aggr_resp(sc, sband, sta, an, tid);
} }
} }
} }
static void ath_rate_init(void *priv, void *priv_sta, static void ath_rate_init(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_local *local, struct ieee80211_sta *sta, void *priv_sta)
struct sta_info *sta)
{ {
struct ieee80211_supported_band *sband; struct ath_softc *sc = priv;
struct ieee80211_hw *hw = local_to_hw(local);
struct ieee80211_conf *conf = &local->hw.conf;
struct ath_softc *sc = hw->priv;
struct ath_rate_node *ath_rc_priv = priv_sta; struct ath_rate_node *ath_rc_priv = priv_sta;
int i, j = 0; int i, j = 0;
DPRINTF(sc, ATH_DBG_RATE, "%s\n", __func__); DPRINTF(sc, ATH_DBG_RATE, "%s\n", __func__);
sband = local->hw.wiphy->bands[local->hw.conf.channel->band]; ath_setup_rates(sc, sband, sta, ath_rc_priv);
if (sc->hw->conf.flags & IEEE80211_CONF_SUPPORT_HT_MODE) {
ath_setup_rates(local, sta);
if (conf->flags & IEEE80211_CONF_SUPPORT_HT_MODE) {
for (i = 0; i < MCS_SET_SIZE; i++) { for (i = 0; i < MCS_SET_SIZE; i++) {
if (conf->ht_conf.supp_mcs_set[i/8] & (1<<(i%8))) if (sc->hw->conf.ht_conf.supp_mcs_set[i/8] & (1<<(i%8)))
ath_rc_priv->neg_ht_rates.rs_rates[j++] = i; ath_rc_priv->neg_ht_rates.rs_rates[j++] = i;
if (j == ATH_RATE_MAX) if (j == ATH_RATE_MAX)
break; break;
} }
ath_rc_priv->neg_ht_rates.rs_nrates = j; ath_rc_priv->neg_ht_rates.rs_nrates = j;
} }
ath_rc_node_update(hw, priv_sta); ath_rc_node_update(sc->hw, priv_sta);
} }
static void ath_rate_clear(void *priv) static void ath_rate_clear(void *priv)
...@@ -2058,13 +2044,12 @@ static void ath_rate_clear(void *priv) ...@@ -2058,13 +2044,12 @@ static void ath_rate_clear(void *priv)
return; return;
} }
static void *ath_rate_alloc(struct ieee80211_local *local) static void *ath_rate_alloc(struct ieee80211_hw *hw, struct dentry *debugfsdir)
{ {
struct ieee80211_hw *hw = local_to_hw(local);
struct ath_softc *sc = hw->priv; struct ath_softc *sc = hw->priv;
DPRINTF(sc, ATH_DBG_RATE, "%s\n", __func__); DPRINTF(sc, ATH_DBG_RATE, "%s\n", __func__);
return local->hw.priv; return hw->priv;
} }
static void ath_rate_free(void *priv) static void ath_rate_free(void *priv)
...@@ -2072,7 +2057,7 @@ static void ath_rate_free(void *priv) ...@@ -2072,7 +2057,7 @@ static void ath_rate_free(void *priv)
return; return;
} }
static void *ath_rate_alloc_sta(void *priv, gfp_t gfp) static void *ath_rate_alloc_sta(void *priv, struct ieee80211_sta *sta, gfp_t gfp)
{ {
struct ath_softc *sc = priv; struct ath_softc *sc = priv;
struct ath_vap *avp = sc->sc_vaps[0]; struct ath_vap *avp = sc->sc_vaps[0];
...@@ -2092,7 +2077,8 @@ static void *ath_rate_alloc_sta(void *priv, gfp_t gfp) ...@@ -2092,7 +2077,8 @@ static void *ath_rate_alloc_sta(void *priv, gfp_t gfp)
return rate_priv; return rate_priv;
} }
static void ath_rate_free_sta(void *priv, void *priv_sta) static void ath_rate_free_sta(void *priv, struct ieee80211_sta *sta,
void *priv_sta)
{ {
struct ath_rate_node *rate_priv = priv_sta; struct ath_rate_node *rate_priv = priv_sta;
struct ath_softc *sc = priv; struct ath_softc *sc = priv;
...@@ -2111,7 +2097,7 @@ static struct rate_control_ops ath_rate_ops = { ...@@ -2111,7 +2097,7 @@ static struct rate_control_ops ath_rate_ops = {
.alloc = ath_rate_alloc, .alloc = ath_rate_alloc,
.free = ath_rate_free, .free = ath_rate_free,
.alloc_sta = ath_rate_alloc_sta, .alloc_sta = ath_rate_alloc_sta,
.free_sta = ath_rate_free_sta .free_sta = ath_rate_free_sta,
}; };
int ath_rate_control_register(void) int ath_rate_control_register(void)
......
...@@ -36,8 +36,6 @@ ...@@ -36,8 +36,6 @@
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include "../net/mac80211/rate.h"
#include "iwl-3945.h" #include "iwl-3945.h"
#define RS_NAME "iwl-3945-rs" #define RS_NAME "iwl-3945-rs"
...@@ -319,10 +317,10 @@ static void iwl3945_collect_tx_data(struct iwl3945_rs_sta *rs_sta, ...@@ -319,10 +317,10 @@ static void iwl3945_collect_tx_data(struct iwl3945_rs_sta *rs_sta,
} }
} }
static void rs_rate_init(void *priv_rate, void *priv_sta, static void rs_rate_init(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_local *local, struct sta_info *sta) struct ieee80211_sta *sta, void *priv_sta)
{ {
struct iwl3945_rs_sta *rs_sta = (void *)sta->rate_ctrl_priv; struct iwl3945_rs_sta *rs_sta = priv_sta;
int i; int i;
IWL_DEBUG_RATE("enter\n"); IWL_DEBUG_RATE("enter\n");
...@@ -333,22 +331,22 @@ static void rs_rate_init(void *priv_rate, void *priv_sta, ...@@ -333,22 +331,22 @@ static void rs_rate_init(void *priv_rate, void *priv_sta,
* after assoc.. */ * after assoc.. */
for (i = IWL_RATE_COUNT - 1; i >= 0; i--) { for (i = IWL_RATE_COUNT - 1; i >= 0; i--) {
if (sta->sta.supp_rates[local->hw.conf.channel->band] & (1 << i)) { if (sta->supp_rates[sband->band] & (1 << i)) {
rs_sta->last_txrate_idx = i; rs_sta->last_txrate_idx = i;
break; break;
} }
} }
/* For 5 GHz band it start at IWL_FIRST_OFDM_RATE */ /* For 5 GHz band it start at IWL_FIRST_OFDM_RATE */
if (local->hw.conf.channel->band == IEEE80211_BAND_5GHZ) if (sband->band == IEEE80211_BAND_5GHZ)
rs_sta->last_txrate_idx += IWL_FIRST_OFDM_RATE; rs_sta->last_txrate_idx += IWL_FIRST_OFDM_RATE;
IWL_DEBUG_RATE("leave\n"); IWL_DEBUG_RATE("leave\n");
} }
static void *rs_alloc(struct ieee80211_local *local) static void *rs_alloc(struct ieee80211_hw *hw, struct dentry *debugfsdir)
{ {
return local->hw.priv; return hw->priv;
} }
/* rate scale requires free function to be implemented */ /* rate scale requires free function to be implemented */
...@@ -356,17 +354,24 @@ static void rs_free(void *priv) ...@@ -356,17 +354,24 @@ static void rs_free(void *priv)
{ {
return; return;
} }
static void rs_clear(void *priv) static void rs_clear(void *priv)
{ {
return; return;
} }
static void *rs_alloc_sta(void *priv, gfp_t gfp) static void *rs_alloc_sta(void *priv, struct ieee80211_sta *sta, gfp_t gfp)
{ {
struct iwl3945_rs_sta *rs_sta; struct iwl3945_rs_sta *rs_sta;
struct iwl3945_sta_priv *psta = (void *) sta->drv_priv;
int i; int i;
/*
* XXX: If it's using sta->drv_priv anyway, it might
* as well just put all the information there.
*/
IWL_DEBUG_RATE("enter\n"); IWL_DEBUG_RATE("enter\n");
rs_sta = kzalloc(sizeof(struct iwl3945_rs_sta), gfp); rs_sta = kzalloc(sizeof(struct iwl3945_rs_sta), gfp);
...@@ -375,6 +380,8 @@ static void *rs_alloc_sta(void *priv, gfp_t gfp) ...@@ -375,6 +380,8 @@ static void *rs_alloc_sta(void *priv, gfp_t gfp)
return NULL; return NULL;
} }
psta->rs_sta = rs_sta;
spin_lock_init(&rs_sta->lock); spin_lock_init(&rs_sta->lock);
rs_sta->start_rate = IWL_RATE_INVALID; rs_sta->start_rate = IWL_RATE_INVALID;
...@@ -400,10 +407,14 @@ static void *rs_alloc_sta(void *priv, gfp_t gfp) ...@@ -400,10 +407,14 @@ static void *rs_alloc_sta(void *priv, gfp_t gfp)
return rs_sta; return rs_sta;
} }
static void rs_free_sta(void *priv, void *priv_sta) static void rs_free_sta(void *priv, struct ieee80211_sta *sta,
void *priv_sta)
{ {
struct iwl3945_sta_priv *psta = (void *) sta->drv_priv;
struct iwl3945_rs_sta *rs_sta = priv_sta; struct iwl3945_rs_sta *rs_sta = priv_sta;
psta->rs_sta = NULL;
IWL_DEBUG_RATE("enter\n"); IWL_DEBUG_RATE("enter\n");
del_timer_sync(&rs_sta->rate_scale_flush); del_timer_sync(&rs_sta->rate_scale_flush);
kfree(rs_sta); kfree(rs_sta);
...@@ -445,26 +456,19 @@ static int rs_adjust_next_rate(struct iwl3945_priv *priv, int rate) ...@@ -445,26 +456,19 @@ static int rs_adjust_next_rate(struct iwl3945_priv *priv, int rate)
* NOTE: Uses iwl3945_priv->retry_rate for the # of retries attempted by * NOTE: Uses iwl3945_priv->retry_rate for the # of retries attempted by
* the hardware for each rate. * the hardware for each rate.
*/ */
static void rs_tx_status(void *priv_rate, static void rs_tx_status(void *priv_rate, struct ieee80211_supported_band *sband,
struct net_device *dev, struct ieee80211_sta *sta, void *priv_sta,
struct sk_buff *skb) struct sk_buff *skb)
{ {
u8 retries, current_count; u8 retries, current_count;
int scale_rate_index, first_index, last_index; int scale_rate_index, first_index, last_index;
unsigned long flags; unsigned long flags;
struct sta_info *sta;
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
struct iwl3945_priv *priv = (struct iwl3945_priv *)priv_rate; struct iwl3945_priv *priv = (struct iwl3945_priv *)priv_rate;
struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); struct iwl3945_rs_sta *rs_sta = priv_sta;
struct iwl3945_rs_sta *rs_sta;
struct ieee80211_supported_band *sband;
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
IWL_DEBUG_RATE("enter\n"); IWL_DEBUG_RATE("enter\n");
sband = local->hw.wiphy->bands[local->hw.conf.channel->band];
retries = info->status.retry_count; retries = info->status.retry_count;
first_index = sband->bitrates[info->tx_rate_idx].hw_value; first_index = sband->bitrates[info->tx_rate_idx].hw_value;
if ((first_index < 0) || (first_index >= IWL_RATE_COUNT)) { if ((first_index < 0) || (first_index >= IWL_RATE_COUNT)) {
...@@ -472,17 +476,11 @@ static void rs_tx_status(void *priv_rate, ...@@ -472,17 +476,11 @@ static void rs_tx_status(void *priv_rate,
return; return;
} }
rcu_read_lock(); if (!priv_sta) {
sta = sta_info_get(local, hdr->addr1);
if (!sta || !sta->rate_ctrl_priv) {
rcu_read_unlock();
IWL_DEBUG_RATE("leave: No STA priv data to update!\n"); IWL_DEBUG_RATE("leave: No STA priv data to update!\n");
return; return;
} }
rs_sta = (void *)sta->rate_ctrl_priv;
rs_sta->tx_packets++; rs_sta->tx_packets++;
scale_rate_index = first_index; scale_rate_index = first_index;
...@@ -549,8 +547,6 @@ static void rs_tx_status(void *priv_rate, ...@@ -549,8 +547,6 @@ static void rs_tx_status(void *priv_rate,
spin_unlock_irqrestore(&rs_sta->lock, flags); spin_unlock_irqrestore(&rs_sta->lock, flags);
rcu_read_unlock();
IWL_DEBUG_RATE("leave\n"); IWL_DEBUG_RATE("leave\n");
return; return;
...@@ -634,16 +630,15 @@ static u16 iwl3945_get_adjacent_rate(struct iwl3945_rs_sta *rs_sta, ...@@ -634,16 +630,15 @@ static u16 iwl3945_get_adjacent_rate(struct iwl3945_rs_sta *rs_sta,
* rate table and must reference the driver allocated rate table * rate table and must reference the driver allocated rate table
* *
*/ */
static void rs_get_rate(void *priv_rate, struct net_device *dev, static void rs_get_rate(void *priv_r, struct ieee80211_supported_band *sband,
struct ieee80211_supported_band *sband, struct ieee80211_sta *sta, void *priv_sta,
struct sk_buff *skb, struct sk_buff *skb, struct rate_selection *sel)
struct rate_selection *sel)
{ {
u8 low = IWL_RATE_INVALID; u8 low = IWL_RATE_INVALID;
u8 high = IWL_RATE_INVALID; u8 high = IWL_RATE_INVALID;
u16 high_low; u16 high_low;
int index; int index;
struct iwl3945_rs_sta *rs_sta; struct iwl3945_rs_sta *rs_sta = priv_sta;
struct iwl3945_rate_scale_data *window = NULL; struct iwl3945_rate_scale_data *window = NULL;
int current_tpt = IWL_INV_TPT; int current_tpt = IWL_INV_TPT;
int low_tpt = IWL_INV_TPT; int low_tpt = IWL_INV_TPT;
...@@ -651,34 +646,25 @@ static void rs_get_rate(void *priv_rate, struct net_device *dev, ...@@ -651,34 +646,25 @@ static void rs_get_rate(void *priv_rate, struct net_device *dev,
u32 fail_count; u32 fail_count;
s8 scale_action = 0; s8 scale_action = 0;
unsigned long flags; unsigned long flags;
struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr);
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data; struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
struct sta_info *sta;
u16 fc, rate_mask; u16 fc, rate_mask;
struct iwl3945_priv *priv = (struct iwl3945_priv *)priv_rate; struct iwl3945_priv *priv = (struct iwl3945_priv *)priv_r;
DECLARE_MAC_BUF(mac); DECLARE_MAC_BUF(mac);
IWL_DEBUG_RATE("enter\n"); IWL_DEBUG_RATE("enter\n");
rcu_read_lock();
sta = sta_info_get(local, hdr->addr1);
/* Send management frames and broadcast/multicast data using lowest /* Send management frames and broadcast/multicast data using lowest
* rate. */ * rate. */
fc = le16_to_cpu(hdr->frame_control); fc = le16_to_cpu(hdr->frame_control);
if ((fc & IEEE80211_FCTL_FTYPE) != IEEE80211_FTYPE_DATA || if ((fc & IEEE80211_FCTL_FTYPE) != IEEE80211_FTYPE_DATA ||
is_multicast_ether_addr(hdr->addr1) || is_multicast_ether_addr(hdr->addr1) ||
!sta || !sta->rate_ctrl_priv) { !sta || !priv_sta) {
IWL_DEBUG_RATE("leave: No STA priv data to update!\n"); IWL_DEBUG_RATE("leave: No STA priv data to update!\n");
sel->rate_idx = rate_lowest_index(local, sband, sta); sel->rate_idx = rate_lowest_index(sband, sta);
rcu_read_unlock();
return; return;
} }
rs_sta = (void *)sta->rate_ctrl_priv; rate_mask = sta->supp_rates[sband->band];
rate_mask = sta->sta.supp_rates[sband->band];
index = min(rs_sta->last_txrate_idx & 0xffff, IWL_RATE_COUNT - 1); index = min(rs_sta->last_txrate_idx & 0xffff, IWL_RATE_COUNT - 1);
if (sband->band == IEEE80211_BAND_5GHZ) if (sband->band == IEEE80211_BAND_5GHZ)
...@@ -811,8 +797,6 @@ static void rs_get_rate(void *priv_rate, struct net_device *dev, ...@@ -811,8 +797,6 @@ static void rs_get_rate(void *priv_rate, struct net_device *dev,
else else
sel->rate_idx = rs_sta->last_txrate_idx; sel->rate_idx = rs_sta->last_txrate_idx;
rcu_read_unlock();
IWL_DEBUG_RATE("leave: %d\n", index); IWL_DEBUG_RATE("leave: %d\n", index);
} }
...@@ -829,114 +813,28 @@ static struct rate_control_ops rs_ops = { ...@@ -829,114 +813,28 @@ static struct rate_control_ops rs_ops = {
.free_sta = rs_free_sta, .free_sta = rs_free_sta,
}; };
int iwl3945_fill_rs_info(struct ieee80211_hw *hw, char *buf, u8 sta_id)
{
struct ieee80211_local *local = hw_to_local(hw);
struct iwl3945_priv *priv = hw->priv;
struct iwl3945_rs_sta *rs_sta;
struct sta_info *sta;
unsigned long flags;
int count = 0, i;
u32 samples = 0, success = 0, good = 0;
unsigned long now = jiffies;
u32 max_time = 0;
rcu_read_lock();
sta = sta_info_get(local, priv->stations[sta_id].sta.sta.addr);
if (!sta || !sta->rate_ctrl_priv) {
if (sta)
IWL_DEBUG_RATE("leave - no private rate data!\n");
else
IWL_DEBUG_RATE("leave - no station!\n");
rcu_read_unlock();
return sprintf(buf, "station %d not found\n", sta_id);
}
rs_sta = (void *)sta->rate_ctrl_priv;
spin_lock_irqsave(&rs_sta->lock, flags);
i = IWL_RATE_54M_INDEX;
while (1) {
u64 mask;
int j;
count +=
sprintf(&buf[count], " %2dMbs: ", iwl3945_rates[i].ieee / 2);
mask = (1ULL << (IWL_RATE_MAX_WINDOW - 1));
for (j = 0; j < IWL_RATE_MAX_WINDOW; j++, mask >>= 1)
buf[count++] =
(rs_sta->win[i].data & mask) ? '1' : '0';
samples += rs_sta->win[i].counter;
good += rs_sta->win[i].success_counter;
success += rs_sta->win[i].success_counter *
iwl3945_rates[i].ieee;
if (rs_sta->win[i].stamp) {
int delta =
jiffies_to_msecs(now - rs_sta->win[i].stamp);
if (delta > max_time)
max_time = delta;
count += sprintf(&buf[count], "%5dms\n", delta);
} else
buf[count++] = '\n';
j = iwl3945_get_prev_ieee_rate(i);
if (j == i)
break;
i = j;
}
spin_unlock_irqrestore(&rs_sta->lock, flags);
rcu_read_unlock();
/* Display the average rate of all samples taken.
*
* NOTE: We multiple # of samples by 2 since the IEEE measurement
* added from iwl3945_rates is actually 2X the rate */
if (samples)
count += sprintf(
&buf[count],
"\nAverage rate is %3d.%02dMbs over last %4dms\n"
"%3d%% success (%d good packets over %d tries)\n",
success / (2 * samples), (success * 5 / samples) % 10,
max_time, good * 100 / samples, good, samples);
else
count += sprintf(&buf[count], "\nAverage rate: 0Mbs\n");
return count;
}
void iwl3945_rate_scale_init(struct ieee80211_hw *hw, s32 sta_id) void iwl3945_rate_scale_init(struct ieee80211_hw *hw, s32 sta_id)
{ {
struct iwl3945_priv *priv = hw->priv; struct iwl3945_priv *priv = hw->priv;
s32 rssi = 0; s32 rssi = 0;
unsigned long flags; unsigned long flags;
struct ieee80211_local *local = hw_to_local(hw);
struct iwl3945_rs_sta *rs_sta; struct iwl3945_rs_sta *rs_sta;
struct sta_info *sta; struct ieee80211_sta *sta;
struct iwl3945_sta_priv *psta;
IWL_DEBUG_RATE("enter\n"); IWL_DEBUG_RATE("enter\n");
if (!local->rate_ctrl->ops->name ||
strcmp(local->rate_ctrl->ops->name, RS_NAME)) {
IWL_WARNING("iwl-3945-rs not selected as rate control algo!\n");
IWL_DEBUG_RATE("leave - mac80211 picked the wrong RC algo.\n");
return;
}
rcu_read_lock(); rcu_read_lock();
sta = sta_info_get(local, priv->stations[sta_id].sta.sta.addr); sta = ieee80211_find_sta(hw, priv->stations[sta_id].sta.sta.addr);
if (!sta || !sta->rate_ctrl_priv) { psta = (void *) sta->drv_priv;
if (!sta || !psta) {
IWL_DEBUG_RATE("leave - no private rate data!\n"); IWL_DEBUG_RATE("leave - no private rate data!\n");
rcu_read_unlock(); rcu_read_unlock();
return; return;
} }
rs_sta = (void *)sta->rate_ctrl_priv; rs_sta = psta->rs_sta;
spin_lock_irqsave(&rs_sta->lock, flags); spin_lock_irqsave(&rs_sta->lock, flags);
......
...@@ -175,15 +175,6 @@ static inline u8 iwl3945_get_prev_ieee_rate(u8 rate_index) ...@@ -175,15 +175,6 @@ static inline u8 iwl3945_get_prev_ieee_rate(u8 rate_index)
return rate; return rate;
} }
/**
* iwl3945_fill_rs_info - Fill an output text buffer with the rate representation
*
* NOTE: This is provided as a quick mechanism for a user to visualize
* the performance of the rate control algorithm and is not meant to be
* parsed software.
*/
extern int iwl3945_fill_rs_info(struct ieee80211_hw *, char *buf, u8 sta_id);
/** /**
* iwl3945_rate_scale_init - Initialize the rate scale table based on assoc info * iwl3945_rate_scale_init - Initialize the rate scale table based on assoc info
* *
......
...@@ -73,6 +73,10 @@ extern struct pci_device_id iwl3945_hw_card_ids[]; ...@@ -73,6 +73,10 @@ extern struct pci_device_id iwl3945_hw_card_ids[];
extern int iwl3945_param_hwcrypto; extern int iwl3945_param_hwcrypto;
extern int iwl3945_param_queues_num; extern int iwl3945_param_queues_num;
struct iwl3945_sta_priv {
struct iwl3945_rs_sta *rs_sta;
};
enum iwl3945_antenna { enum iwl3945_antenna {
IWL_ANTENNA_DIVERSITY, IWL_ANTENNA_DIVERSITY,
IWL_ANTENNA_MAIN, IWL_ANTENNA_MAIN,
......
This diff is collapsed.
...@@ -7370,15 +7370,6 @@ static ssize_t show_temperature(struct device *d, ...@@ -7370,15 +7370,6 @@ static ssize_t show_temperature(struct device *d,
static DEVICE_ATTR(temperature, S_IRUGO, show_temperature, NULL); static DEVICE_ATTR(temperature, S_IRUGO, show_temperature, NULL);
static ssize_t show_rs_window(struct device *d,
struct device_attribute *attr,
char *buf)
{
struct iwl3945_priv *priv = d->driver_data;
return iwl3945_fill_rs_info(priv->hw, buf, IWL_AP_ID);
}
static DEVICE_ATTR(rs_window, S_IRUGO, show_rs_window, NULL);
static ssize_t show_tx_power(struct device *d, static ssize_t show_tx_power(struct device *d,
struct device_attribute *attr, char *buf) struct device_attribute *attr, char *buf)
{ {
...@@ -7840,7 +7831,6 @@ static struct attribute *iwl3945_sysfs_entries[] = { ...@@ -7840,7 +7831,6 @@ static struct attribute *iwl3945_sysfs_entries[] = {
#endif #endif
&dev_attr_power_level.attr, &dev_attr_power_level.attr,
&dev_attr_retry_rate.attr, &dev_attr_retry_rate.attr,
&dev_attr_rs_window.attr,
&dev_attr_statistics.attr, &dev_attr_statistics.attr,
&dev_attr_status.attr, &dev_attr_status.attr,
&dev_attr_temperature.attr, &dev_attr_temperature.attr,
...@@ -7908,6 +7898,7 @@ static int iwl3945_pci_probe(struct pci_dev *pdev, const struct pci_device_id *e ...@@ -7908,6 +7898,7 @@ static int iwl3945_pci_probe(struct pci_dev *pdev, const struct pci_device_id *e
SET_IEEE80211_DEV(hw, &pdev->dev); SET_IEEE80211_DEV(hw, &pdev->dev);
hw->rate_control_algorithm = "iwl-3945-rs"; hw->rate_control_algorithm = "iwl-3945-rs";
hw->sta_data_size = sizeof(struct iwl3945_sta_priv);
IWL_DEBUG_INFO("*** LOAD DRIVER ***\n"); IWL_DEBUG_INFO("*** LOAD DRIVER ***\n");
priv = hw->priv; priv = hw->priv;
......
...@@ -1800,4 +1800,72 @@ void ieee80211_notify_mac(struct ieee80211_hw *hw, ...@@ -1800,4 +1800,72 @@ void ieee80211_notify_mac(struct ieee80211_hw *hw,
struct ieee80211_sta *ieee80211_find_sta(struct ieee80211_hw *hw, struct ieee80211_sta *ieee80211_find_sta(struct ieee80211_hw *hw,
const u8 *addr); const u8 *addr);
/* Rate control API */
/**
* struct rate_selection - rate information for/from rate control algorithms
*
* @rate_idx: selected transmission rate index
* @nonerp_idx: Non-ERP rate to use instead if ERP cannot be used
* @probe_idx: rate for probing (or -1)
* @max_rate_idx: maximum rate index that can be used, this is
* input to the algorithm and will be enforced
*/
struct rate_selection {
s8 rate_idx, nonerp_idx, probe_idx, max_rate_idx;
};
struct rate_control_ops {
struct module *module;
const char *name;
void *(*alloc)(struct ieee80211_hw *hw, struct dentry *debugfsdir);
void (*clear)(void *priv);
void (*free)(void *priv);
void *(*alloc_sta)(void *priv, struct ieee80211_sta *sta, gfp_t gfp);
void (*rate_init)(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta, void *priv_sta);
void (*free_sta)(void *priv, struct ieee80211_sta *sta,
void *priv_sta);
void (*tx_status)(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta, void *priv_sta,
struct sk_buff *skb);
void (*get_rate)(void *priv, struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta, void *priv_sta,
struct sk_buff *skb,
struct rate_selection *sel);
void (*add_sta_debugfs)(void *priv, void *priv_sta,
struct dentry *dir);
void (*remove_sta_debugfs)(void *priv, void *priv_sta);
};
static inline int rate_supported(struct ieee80211_sta *sta,
enum ieee80211_band band,
int index)
{
return (sta == NULL || sta->supp_rates[band] & BIT(index));
}
static inline s8
rate_lowest_index(struct ieee80211_supported_band *sband,
struct ieee80211_sta *sta)
{
int i;
for (i = 0; i < sband->n_bitrates; i++)
if (rate_supported(sta, sband->band, i))
return i;
/* warn when we cannot find a rate. */
WARN_ON(1);
return 0;
}
int ieee80211_rate_control_register(struct rate_control_ops *ops);
void ieee80211_rate_control_unregister(struct rate_control_ops *ops);
#endif /* MAC80211_H */ #endif /* MAC80211_H */
...@@ -693,7 +693,7 @@ static int ieee80211_add_station(struct wiphy *wiphy, struct net_device *dev, ...@@ -693,7 +693,7 @@ static int ieee80211_add_station(struct wiphy *wiphy, struct net_device *dev,
sta_apply_parameters(local, sta, params); sta_apply_parameters(local, sta, params);
rate_control_rate_init(sta, local); rate_control_rate_init(sta);
rcu_read_lock(); rcu_read_lock();
......
...@@ -724,6 +724,8 @@ struct ieee80211_local { ...@@ -724,6 +724,8 @@ struct ieee80211_local {
#ifdef CONFIG_MAC80211_DEBUGFS #ifdef CONFIG_MAC80211_DEBUGFS
struct local_debugfsdentries { struct local_debugfsdentries {
struct dentry *rcdir;
struct dentry *rcname;
struct dentry *frequency; struct dentry *frequency;
struct dentry *antenna_sel_tx; struct dentry *antenna_sel_tx;
struct dentry *antenna_sel_rx; struct dentry *antenna_sel_rx;
......
...@@ -542,6 +542,7 @@ void ieee80211_tx_status(struct ieee80211_hw *hw, struct sk_buff *skb) ...@@ -542,6 +542,7 @@ void ieee80211_tx_status(struct ieee80211_hw *hw, struct sk_buff *skb)
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
u16 frag, type; u16 frag, type;
__le16 fc; __le16 fc;
struct ieee80211_supported_band *sband;
struct ieee80211_tx_status_rtap_hdr *rthdr; struct ieee80211_tx_status_rtap_hdr *rthdr;
struct ieee80211_sub_if_data *sdata; struct ieee80211_sub_if_data *sdata;
struct net_device *prev_dev = NULL; struct net_device *prev_dev = NULL;
...@@ -588,7 +589,8 @@ void ieee80211_tx_status(struct ieee80211_hw *hw, struct sk_buff *skb) ...@@ -588,7 +589,8 @@ void ieee80211_tx_status(struct ieee80211_hw *hw, struct sk_buff *skb)
sta->tx_retry_count += info->status.retry_count; sta->tx_retry_count += info->status.retry_count;
} }
rate_control_tx_status(local->mdev, skb); sband = local->hw.wiphy->bands[info->band];
rate_control_tx_status(local, sband, sta, skb);
} }
rcu_read_unlock(); rcu_read_unlock();
......
...@@ -1323,7 +1323,7 @@ static void ieee80211_rx_mgmt_assoc_resp(struct ieee80211_sub_if_data *sdata, ...@@ -1323,7 +1323,7 @@ static void ieee80211_rx_mgmt_assoc_resp(struct ieee80211_sub_if_data *sdata,
ieee80211_handle_ht(local, 1, &sta->sta.ht_info, &bss_info); ieee80211_handle_ht(local, 1, &sta->sta.ht_info, &bss_info);
} }
rate_control_rate_init(sta, local); rate_control_rate_init(sta);
if (elems.wmm_param) { if (elems.wmm_param) {
set_sta_flags(sta, WLAN_STA_WME); set_sta_flags(sta, WLAN_STA_WME);
...@@ -2342,7 +2342,7 @@ struct sta_info *ieee80211_ibss_add_sta(struct ieee80211_sub_if_data *sdata, ...@@ -2342,7 +2342,7 @@ struct sta_info *ieee80211_ibss_add_sta(struct ieee80211_sub_if_data *sdata,
sta->sta.supp_rates[band] = supp_rates | sta->sta.supp_rates[band] = supp_rates |
ieee80211_mandatory_rates(local, band); ieee80211_mandatory_rates(local, band);
rate_control_rate_init(sta, local); rate_control_rate_init(sta);
if (sta_info_insert(sta)) if (sta_info_insert(sta))
return NULL; return NULL;
......
...@@ -12,6 +12,7 @@ ...@@ -12,6 +12,7 @@
#include <linux/rtnetlink.h> #include <linux/rtnetlink.h>
#include "rate.h" #include "rate.h"
#include "ieee80211_i.h" #include "ieee80211_i.h"
#include "debugfs.h"
struct rate_control_alg { struct rate_control_alg {
struct list_head list; struct list_head list;
...@@ -127,19 +128,46 @@ static void ieee80211_rate_control_ops_put(struct rate_control_ops *ops) ...@@ -127,19 +128,46 @@ static void ieee80211_rate_control_ops_put(struct rate_control_ops *ops)
module_put(ops->module); module_put(ops->module);
} }
#ifdef CONFIG_MAC80211_DEBUGFS
static ssize_t rcname_read(struct file *file, char __user *userbuf,
size_t count, loff_t *ppos)
{
struct rate_control_ref *ref = file->private_data;
int len = strlen(ref->ops->name);
return simple_read_from_buffer(userbuf, count, ppos,
ref->ops->name, len);
}
static const struct file_operations rcname_ops = {
.read = rcname_read,
.open = mac80211_open_file_generic,
};
#endif
struct rate_control_ref *rate_control_alloc(const char *name, struct rate_control_ref *rate_control_alloc(const char *name,
struct ieee80211_local *local) struct ieee80211_local *local)
{ {
struct dentry *debugfsdir = NULL;
struct rate_control_ref *ref; struct rate_control_ref *ref;
ref = kmalloc(sizeof(struct rate_control_ref), GFP_KERNEL); ref = kmalloc(sizeof(struct rate_control_ref), GFP_KERNEL);
if (!ref) if (!ref)
goto fail_ref; goto fail_ref;
kref_init(&ref->kref); kref_init(&ref->kref);
ref->local = local;
ref->ops = ieee80211_rate_control_ops_get(name); ref->ops = ieee80211_rate_control_ops_get(name);
if (!ref->ops) if (!ref->ops)
goto fail_ops; goto fail_ops;
ref->priv = ref->ops->alloc(local);
#ifdef CONFIG_MAC80211_DEBUGFS
debugfsdir = debugfs_create_dir("rc", local->hw.wiphy->debugfsdir);
local->debugfs.rcdir = debugfsdir;
local->debugfs.rcname = debugfs_create_file("name", 0400, debugfsdir,
ref, &rcname_ops);
#endif
ref->priv = ref->ops->alloc(&local->hw, debugfsdir);
if (!ref->priv) if (!ref->priv)
goto fail_priv; goto fail_priv;
return ref; return ref;
...@@ -158,29 +186,46 @@ static void rate_control_release(struct kref *kref) ...@@ -158,29 +186,46 @@ static void rate_control_release(struct kref *kref)
ctrl_ref = container_of(kref, struct rate_control_ref, kref); ctrl_ref = container_of(kref, struct rate_control_ref, kref);
ctrl_ref->ops->free(ctrl_ref->priv); ctrl_ref->ops->free(ctrl_ref->priv);
#ifdef CONFIG_MAC80211_DEBUGFS
debugfs_remove(ctrl_ref->local->debugfs.rcname);
ctrl_ref->local->debugfs.rcname = NULL;
debugfs_remove(ctrl_ref->local->debugfs.rcdir);
ctrl_ref->local->debugfs.rcdir = NULL;
#endif
ieee80211_rate_control_ops_put(ctrl_ref->ops); ieee80211_rate_control_ops_put(ctrl_ref->ops);
kfree(ctrl_ref); kfree(ctrl_ref);
} }
void rate_control_get_rate(struct net_device *dev, void rate_control_get_rate(struct ieee80211_sub_if_data *sdata,
struct ieee80211_supported_band *sband, struct ieee80211_supported_band *sband,
struct sk_buff *skb, struct sta_info *sta, struct sk_buff *skb,
struct rate_selection *sel) struct rate_selection *sel)
{ {
struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); struct rate_control_ref *ref = sdata->local->rate_ctrl;
struct rate_control_ref *ref = local->rate_ctrl; void *priv_sta = NULL;
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) skb->data; struct ieee80211_sta *ista = NULL;
struct sta_info *sta;
int i; int i;
rcu_read_lock();
sta = sta_info_get(local, hdr->addr1);
sel->rate_idx = -1; sel->rate_idx = -1;
sel->nonerp_idx = -1; sel->nonerp_idx = -1;
sel->probe_idx = -1; sel->probe_idx = -1;
sel->max_rate_idx = sdata->max_ratectrl_rateidx;
if (sta) {
ista = &sta->sta;
priv_sta = sta->rate_ctrl_priv;
}
if (sta && sdata->force_unicast_rateidx > -1)
sel->rate_idx = sdata->force_unicast_rateidx;
else
ref->ops->get_rate(ref->priv, sband, ista, priv_sta, skb, sel);
ref->ops->get_rate(ref->priv, dev, sband, skb, sel); if (sdata->max_ratectrl_rateidx > -1 &&
sel->rate_idx > sdata->max_ratectrl_rateidx)
sel->rate_idx = sdata->max_ratectrl_rateidx;
BUG_ON(sel->rate_idx < 0); BUG_ON(sel->rate_idx < 0);
...@@ -191,13 +236,11 @@ void rate_control_get_rate(struct net_device *dev, ...@@ -191,13 +236,11 @@ void rate_control_get_rate(struct net_device *dev,
if (sband->bitrates[sel->rate_idx].bitrate < rate->bitrate) if (sband->bitrates[sel->rate_idx].bitrate < rate->bitrate)
break; break;
if (rate_supported(sta, sband->band, i) && if (rate_supported(ista, sband->band, i) &&
!(rate->flags & IEEE80211_RATE_ERP_G)) !(rate->flags & IEEE80211_RATE_ERP_G))
sel->nonerp_idx = i; sel->nonerp_idx = i;
} }
} }
rcu_read_unlock();
} }
struct rate_control_ref *rate_control_get(struct rate_control_ref *ref) struct rate_control_ref *rate_control_get(struct rate_control_ref *ref)
......
...@@ -19,77 +19,48 @@ ...@@ -19,77 +19,48 @@
#include "ieee80211_i.h" #include "ieee80211_i.h"
#include "sta_info.h" #include "sta_info.h"
/**
* struct rate_selection - rate selection for rate control algos
* @rate: selected transmission rate index
* @nonerp: Non-ERP rate to use instead if ERP cannot be used
* @probe: rate for probing (or -1)
*
*/
struct rate_selection {
s8 rate_idx, nonerp_idx, probe_idx;
};
struct rate_control_ops {
struct module *module;
const char *name;
void (*tx_status)(void *priv, struct net_device *dev,
struct sk_buff *skb);
void (*get_rate)(void *priv, struct net_device *dev,
struct ieee80211_supported_band *band,
struct sk_buff *skb,
struct rate_selection *sel);
void (*rate_init)(void *priv, void *priv_sta,
struct ieee80211_local *local, struct sta_info *sta);
void (*clear)(void *priv);
void *(*alloc)(struct ieee80211_local *local);
void (*free)(void *priv);
void *(*alloc_sta)(void *priv, gfp_t gfp);
void (*free_sta)(void *priv, void *priv_sta);
int (*add_attrs)(void *priv, struct kobject *kobj);
void (*remove_attrs)(void *priv, struct kobject *kobj);
void (*add_sta_debugfs)(void *priv, void *priv_sta,
struct dentry *dir);
void (*remove_sta_debugfs)(void *priv, void *priv_sta);
};
struct rate_control_ref { struct rate_control_ref {
struct ieee80211_local *local;
struct rate_control_ops *ops; struct rate_control_ops *ops;
void *priv; void *priv;
struct kref kref; struct kref kref;
}; };
int ieee80211_rate_control_register(struct rate_control_ops *ops);
void ieee80211_rate_control_unregister(struct rate_control_ops *ops);
/* Get a reference to the rate control algorithm. If `name' is NULL, get the /* Get a reference to the rate control algorithm. If `name' is NULL, get the
* first available algorithm. */ * first available algorithm. */
struct rate_control_ref *rate_control_alloc(const char *name, struct rate_control_ref *rate_control_alloc(const char *name,
struct ieee80211_local *local); struct ieee80211_local *local);
void rate_control_get_rate(struct net_device *dev, void rate_control_get_rate(struct ieee80211_sub_if_data *sdata,
struct ieee80211_supported_band *sband, struct ieee80211_supported_band *sband,
struct sk_buff *skb, struct sta_info *sta, struct sk_buff *skb,
struct rate_selection *sel); struct rate_selection *sel);
struct rate_control_ref *rate_control_get(struct rate_control_ref *ref); struct rate_control_ref *rate_control_get(struct rate_control_ref *ref);
void rate_control_put(struct rate_control_ref *ref); void rate_control_put(struct rate_control_ref *ref);
static inline void rate_control_tx_status(struct net_device *dev, static inline void rate_control_tx_status(struct ieee80211_local *local,
struct ieee80211_supported_band *sband,
struct sta_info *sta,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr);
struct rate_control_ref *ref = local->rate_ctrl; struct rate_control_ref *ref = local->rate_ctrl;
struct ieee80211_sta *ista = &sta->sta;
void *priv_sta = sta->rate_ctrl_priv;
ref->ops->tx_status(ref->priv, dev, skb); ref->ops->tx_status(ref->priv, sband, ista, priv_sta, skb);
} }
static inline void rate_control_rate_init(struct sta_info *sta, static inline void rate_control_rate_init(struct sta_info *sta)
struct ieee80211_local *local)
{ {
struct ieee80211_local *local = sta->sdata->local;
struct rate_control_ref *ref = sta->rate_ctrl; struct rate_control_ref *ref = sta->rate_ctrl;
ref->ops->rate_init(ref->priv, sta->rate_ctrl_priv, local, sta); struct ieee80211_sta *ista = &sta->sta;
void *priv_sta = sta->rate_ctrl_priv;
struct ieee80211_supported_band *sband;
sband = local->hw.wiphy->bands[local->hw.conf.channel->band];
ref->ops->rate_init(ref->priv, sband, ista, priv_sta);
} }
...@@ -100,15 +71,19 @@ static inline void rate_control_clear(struct ieee80211_local *local) ...@@ -100,15 +71,19 @@ static inline void rate_control_clear(struct ieee80211_local *local)
} }
static inline void *rate_control_alloc_sta(struct rate_control_ref *ref, static inline void *rate_control_alloc_sta(struct rate_control_ref *ref,
struct ieee80211_sta *sta,
gfp_t gfp) gfp_t gfp)
{ {
return ref->ops->alloc_sta(ref->priv, gfp); return ref->ops->alloc_sta(ref->priv, sta, gfp);
} }
static inline void rate_control_free_sta(struct rate_control_ref *ref, static inline void rate_control_free_sta(struct sta_info *sta)
void *priv)
{ {
ref->ops->free_sta(ref->priv, priv); struct rate_control_ref *ref = sta->rate_ctrl;
struct ieee80211_sta *ista = &sta->sta;
void *priv_sta = sta->rate_ctrl_priv;
ref->ops->free_sta(ref->priv, ista, priv_sta);
} }
static inline void rate_control_add_sta_debugfs(struct sta_info *sta) static inline void rate_control_add_sta_debugfs(struct sta_info *sta)
...@@ -130,31 +105,6 @@ static inline void rate_control_remove_sta_debugfs(struct sta_info *sta) ...@@ -130,31 +105,6 @@ static inline void rate_control_remove_sta_debugfs(struct sta_info *sta)
#endif #endif
} }
static inline int rate_supported(struct sta_info *sta,
enum ieee80211_band band,
int index)
{
return (sta == NULL || sta->sta.supp_rates[band] & BIT(index));
}
static inline s8
rate_lowest_index(struct ieee80211_local *local,
struct ieee80211_supported_band *sband,
struct sta_info *sta)
{
int i;
for (i = 0; i < sband->n_bitrates; i++)
if (rate_supported(sta, sband->band, i))
return i;
/* warn when we cannot find a rate. */
WARN_ON(1);
return 0;
}
/* functions for rate control related to a device */ /* functions for rate control related to a device */
int ieee80211_init_rate_ctrl_alg(struct ieee80211_local *local, int ieee80211_init_rate_ctrl_alg(struct ieee80211_local *local,
const char *name); const char *name);
......
...@@ -124,7 +124,6 @@ struct rc_pid_events_file_info { ...@@ -124,7 +124,6 @@ struct rc_pid_events_file_info {
* struct rc_pid_debugfs_entries - tunable parameters * struct rc_pid_debugfs_entries - tunable parameters
* *
* Algorithm parameters, tunable via debugfs. * Algorithm parameters, tunable via debugfs.
* @dir: the debugfs directory for a specific phy
* @target: target percentage for failed frames * @target: target percentage for failed frames
* @sampling_period: error sampling interval in milliseconds * @sampling_period: error sampling interval in milliseconds
* @coeff_p: absolute value of the proportional coefficient * @coeff_p: absolute value of the proportional coefficient
...@@ -143,7 +142,6 @@ struct rc_pid_events_file_info { ...@@ -143,7 +142,6 @@ struct rc_pid_events_file_info {
* ordering of rates) * ordering of rates)
*/ */
struct rc_pid_debugfs_entries { struct rc_pid_debugfs_entries {
struct dentry *dir;
struct dentry *target; struct dentry *target;
struct dentry *sampling_period; struct dentry *sampling_period;
struct dentry *coeff_p; struct dentry *coeff_p;
......
This diff is collapsed.
...@@ -93,8 +93,7 @@ static int sta_info_hash_del(struct ieee80211_local *local, ...@@ -93,8 +93,7 @@ static int sta_info_hash_del(struct ieee80211_local *local,
} }
/* protected by RCU */ /* protected by RCU */
static struct sta_info *__sta_info_find(struct ieee80211_local *local, struct sta_info *sta_info_get(struct ieee80211_local *local, const u8 *addr)
const u8 *addr)
{ {
struct sta_info *sta; struct sta_info *sta;
...@@ -107,12 +106,6 @@ static struct sta_info *__sta_info_find(struct ieee80211_local *local, ...@@ -107,12 +106,6 @@ static struct sta_info *__sta_info_find(struct ieee80211_local *local,
return sta; return sta;
} }
struct sta_info *sta_info_get(struct ieee80211_local *local, u8 *addr)
{
return __sta_info_find(local, addr);
}
EXPORT_SYMBOL(sta_info_get);
struct sta_info *sta_info_get_by_idx(struct ieee80211_local *local, int idx, struct sta_info *sta_info_get_by_idx(struct ieee80211_local *local, int idx,
struct net_device *dev) struct net_device *dev)
{ {
...@@ -146,7 +139,7 @@ static void __sta_info_free(struct ieee80211_local *local, ...@@ -146,7 +139,7 @@ static void __sta_info_free(struct ieee80211_local *local,
{ {
DECLARE_MAC_BUF(mbuf); DECLARE_MAC_BUF(mbuf);
rate_control_free_sta(sta->rate_ctrl, sta->rate_ctrl_priv); rate_control_free_sta(sta);
rate_control_put(sta->rate_ctrl); rate_control_put(sta->rate_ctrl);
#ifdef CONFIG_MAC80211_VERBOSE_DEBUG #ifdef CONFIG_MAC80211_VERBOSE_DEBUG
...@@ -244,7 +237,7 @@ struct sta_info *sta_info_alloc(struct ieee80211_sub_if_data *sdata, ...@@ -244,7 +237,7 @@ struct sta_info *sta_info_alloc(struct ieee80211_sub_if_data *sdata,
sta->rate_ctrl = rate_control_get(local->rate_ctrl); sta->rate_ctrl = rate_control_get(local->rate_ctrl);
sta->rate_ctrl_priv = rate_control_alloc_sta(sta->rate_ctrl, sta->rate_ctrl_priv = rate_control_alloc_sta(sta->rate_ctrl,
gfp); &sta->sta, gfp);
if (!sta->rate_ctrl_priv) { if (!sta->rate_ctrl_priv) {
rate_control_put(sta->rate_ctrl); rate_control_put(sta->rate_ctrl);
kfree(sta); kfree(sta);
...@@ -308,7 +301,7 @@ int sta_info_insert(struct sta_info *sta) ...@@ -308,7 +301,7 @@ int sta_info_insert(struct sta_info *sta)
spin_lock_irqsave(&local->sta_lock, flags); spin_lock_irqsave(&local->sta_lock, flags);
/* check if STA exists already */ /* check if STA exists already */
if (__sta_info_find(local, sta->sta.addr)) { if (sta_info_get(local, sta->sta.addr)) {
spin_unlock_irqrestore(&local->sta_lock, flags); spin_unlock_irqrestore(&local->sta_lock, flags);
err = -EEXIST; err = -EEXIST;
goto out_free; goto out_free;
...@@ -834,7 +827,7 @@ void ieee80211_sta_expire(struct ieee80211_sub_if_data *sdata, ...@@ -834,7 +827,7 @@ void ieee80211_sta_expire(struct ieee80211_sub_if_data *sdata,
struct ieee80211_sta *ieee80211_find_sta(struct ieee80211_hw *hw, struct ieee80211_sta *ieee80211_find_sta(struct ieee80211_hw *hw,
const u8 *addr) const u8 *addr)
{ {
struct sta_info *sta = __sta_info_find(hw_to_local(hw), addr); struct sta_info *sta = sta_info_get(hw_to_local(hw), addr);
if (!sta) if (!sta)
return NULL; return NULL;
......
...@@ -416,7 +416,7 @@ static inline u32 get_sta_flags(struct sta_info *sta) ...@@ -416,7 +416,7 @@ static inline u32 get_sta_flags(struct sta_info *sta)
/* /*
* Get a STA info, must have be under RCU read lock. * Get a STA info, must have be under RCU read lock.
*/ */
struct sta_info *sta_info_get(struct ieee80211_local *local, u8 *addr); struct sta_info *sta_info_get(struct ieee80211_local *local, const u8 *addr);
/* /*
* Get STA info by index, BROKEN! * Get STA info by index, BROKEN!
*/ */
......
...@@ -446,7 +446,8 @@ ieee80211_tx_h_rate_ctrl(struct ieee80211_tx_data *tx) ...@@ -446,7 +446,8 @@ ieee80211_tx_h_rate_ctrl(struct ieee80211_tx_data *tx)
sband = tx->local->hw.wiphy->bands[tx->channel->band]; sband = tx->local->hw.wiphy->bands[tx->channel->band];
if (likely(tx->rate_idx < 0)) { if (likely(tx->rate_idx < 0)) {
rate_control_get_rate(tx->dev, sband, tx->skb, &rsel); rate_control_get_rate(tx->sdata, sband, tx->sta,
tx->skb, &rsel);
if (tx->sta) if (tx->sta)
tx->sta->last_txrate_idx = rsel.rate_idx; tx->sta->last_txrate_idx = rsel.rate_idx;
tx->rate_idx = rsel.rate_idx; tx->rate_idx = rsel.rate_idx;
...@@ -1955,7 +1956,7 @@ struct sk_buff *ieee80211_beacon_get(struct ieee80211_hw *hw, ...@@ -1955,7 +1956,7 @@ struct sk_buff *ieee80211_beacon_get(struct ieee80211_hw *hw,
skb->do_not_encrypt = 1; skb->do_not_encrypt = 1;
info->band = band; info->band = band;
rate_control_get_rate(local->mdev, sband, skb, &rsel); rate_control_get_rate(sdata, sband, NULL, skb, &rsel);
if (unlikely(rsel.rate_idx < 0)) { if (unlikely(rsel.rate_idx < 0)) {
if (net_ratelimit()) { if (net_ratelimit()) {
......
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