mirror of
https://github.com/openwrt/openwrt.git
synced 2025-02-20 17:32:57 +00:00
mac80211: replace legacy minstrel with minstrel_ht, improve rate selection
Legacy minstrel is essentially unmaintained and was showing poor performance Replace it with minstrel_ht and improve rate selection and sampling behavior Signed-off-by: Felix Fietkau <nbd@nbd.name>
This commit is contained in:
parent
55e23f2c02
commit
3a12c6679e
@ -0,0 +1,166 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Fri, 25 Dec 2020 16:22:52 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: clean up CCK code
|
||||
|
||||
- move ack overhead out of rate duration table
|
||||
- remove cck_supported, cck_supported_short
|
||||
|
||||
Preparation for adding OFDM legacy rates support
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -136,20 +136,16 @@
|
||||
__VHT_GROUP(_streams, _sgi, _bw, \
|
||||
VHT_GROUP_SHIFT(_streams, _sgi, _bw))
|
||||
|
||||
-#define CCK_DURATION(_bitrate, _short, _len) \
|
||||
+#define CCK_DURATION(_bitrate, _short) \
|
||||
(1000 * (10 /* SIFS */ + \
|
||||
(_short ? 72 + 24 : 144 + 48) + \
|
||||
- (8 * (_len + 4) * 10) / (_bitrate)))
|
||||
-
|
||||
-#define CCK_ACK_DURATION(_bitrate, _short) \
|
||||
- (CCK_DURATION((_bitrate > 10 ? 20 : 10), false, 60) + \
|
||||
- CCK_DURATION(_bitrate, _short, AVG_PKT_SIZE))
|
||||
+ (8 * (AVG_PKT_SIZE + 4) * 10) / (_bitrate)))
|
||||
|
||||
#define CCK_DURATION_LIST(_short, _s) \
|
||||
- CCK_ACK_DURATION(10, _short) >> _s, \
|
||||
- CCK_ACK_DURATION(20, _short) >> _s, \
|
||||
- CCK_ACK_DURATION(55, _short) >> _s, \
|
||||
- CCK_ACK_DURATION(110, _short) >> _s
|
||||
+ CCK_DURATION(10, _short) >> _s, \
|
||||
+ CCK_DURATION(20, _short) >> _s, \
|
||||
+ CCK_DURATION(55, _short) >> _s, \
|
||||
+ CCK_DURATION(110, _short) >> _s
|
||||
|
||||
#define __CCK_GROUP(_s) \
|
||||
[MINSTREL_CCK_GROUP] = { \
|
||||
@@ -163,7 +159,7 @@
|
||||
}
|
||||
|
||||
#define CCK_GROUP_SHIFT \
|
||||
- GROUP_SHIFT(CCK_ACK_DURATION(10, false))
|
||||
+ GROUP_SHIFT(CCK_DURATION(10, false))
|
||||
|
||||
#define CCK_GROUP __CCK_GROUP(CCK_GROUP_SHIFT)
|
||||
|
||||
@@ -349,15 +345,19 @@ int
|
||||
minstrel_ht_get_tp_avg(struct minstrel_ht_sta *mi, int group, int rate,
|
||||
int prob_avg)
|
||||
{
|
||||
- unsigned int nsecs = 0;
|
||||
+ unsigned int nsecs = 0, overhead = mi->overhead;
|
||||
+ unsigned int ampdu_len = 1;
|
||||
|
||||
/* do not account throughput if sucess prob is below 10% */
|
||||
if (prob_avg < MINSTREL_FRAC(10, 100))
|
||||
return 0;
|
||||
|
||||
- if (group != MINSTREL_CCK_GROUP)
|
||||
- nsecs = 1000 * mi->overhead / minstrel_ht_avg_ampdu_len(mi);
|
||||
+ if (group == MINSTREL_CCK_GROUP)
|
||||
+ overhead = mi->overhead_legacy;
|
||||
+ else
|
||||
+ ampdu_len = minstrel_ht_avg_ampdu_len(mi);
|
||||
|
||||
+ nsecs = 1000 * overhead / ampdu_len;
|
||||
nsecs += minstrel_mcs_groups[group].duration[rate] <<
|
||||
minstrel_mcs_groups[group].shift;
|
||||
|
||||
@@ -1031,7 +1031,10 @@ minstrel_calc_retransmit(struct minstrel
|
||||
ctime += (t_slot * cw) >> 1;
|
||||
cw = min((cw << 1) | 1, mp->cw_max);
|
||||
|
||||
- if (index / MCS_GROUP_RATES != MINSTREL_CCK_GROUP) {
|
||||
+ if (index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP) {
|
||||
+ overhead = mi->overhead_legacy;
|
||||
+ overhead_rtscts = mi->overhead_legacy_rtscts;
|
||||
+ } else {
|
||||
overhead = mi->overhead;
|
||||
overhead_rtscts = mi->overhead_rtscts;
|
||||
}
|
||||
@@ -1369,18 +1372,14 @@ minstrel_ht_update_cck(struct minstrel_p
|
||||
if (!ieee80211_hw_check(mp->hw, SUPPORTS_HT_CCK_RATES))
|
||||
return;
|
||||
|
||||
- mi->cck_supported = 0;
|
||||
- mi->cck_supported_short = 0;
|
||||
for (i = 0; i < 4; i++) {
|
||||
if (!rate_supported(sta, sband->band, mp->cck_rates[i]))
|
||||
continue;
|
||||
|
||||
- mi->cck_supported |= BIT(i);
|
||||
+ mi->supported[MINSTREL_CCK_GROUP] |= BIT(i);
|
||||
if (sband->bitrates[i].flags & IEEE80211_RATE_SHORT_PREAMBLE)
|
||||
- mi->cck_supported_short |= BIT(i);
|
||||
+ mi->supported[MINSTREL_CCK_GROUP] |= BIT(i + 4);
|
||||
}
|
||||
-
|
||||
- mi->supported[MINSTREL_CCK_GROUP] = mi->cck_supported;
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -1394,12 +1393,13 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
struct ieee80211_mcs_info *mcs = &sta->ht_cap.mcs;
|
||||
u16 ht_cap = sta->ht_cap.cap;
|
||||
struct ieee80211_sta_vht_cap *vht_cap = &sta->vht_cap;
|
||||
+ const struct ieee80211_rate *ctl_rate;
|
||||
+ bool ldpc, erp;
|
||||
int use_vht;
|
||||
int n_supported = 0;
|
||||
int ack_dur;
|
||||
int stbc;
|
||||
int i;
|
||||
- bool ldpc;
|
||||
|
||||
/* fall back to the old minstrel for legacy stations */
|
||||
if (!sta->ht_cap.ht_supported)
|
||||
@@ -1423,6 +1423,14 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
mi->overhead += ack_dur;
|
||||
mi->overhead_rtscts = mi->overhead + 2 * ack_dur;
|
||||
|
||||
+ ctl_rate = &sband->bitrates[rate_lowest_index(sband, sta)];
|
||||
+ erp = ctl_rate->flags & IEEE80211_RATE_ERP_G;
|
||||
+ ack_dur = ieee80211_frame_duration(sband->band, 10,
|
||||
+ ctl_rate->bitrate, erp, 1,
|
||||
+ ieee80211_chandef_get_shift(chandef));
|
||||
+ mi->overhead_legacy = ack_dur;
|
||||
+ mi->overhead_legacy_rtscts = mi->overhead_legacy + 2 * ack_dur;
|
||||
+
|
||||
mi->avg_ampdu_len = MINSTREL_FRAC(1, 1);
|
||||
|
||||
/* When using MRR, sample more on the first attempt, without delay */
|
||||
@@ -1523,8 +1531,6 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
if (!n_supported)
|
||||
goto use_legacy;
|
||||
|
||||
- mi->supported[MINSTREL_CCK_GROUP] |= mi->cck_supported_short << 4;
|
||||
-
|
||||
/* create an initial rate table with the lowest supported rates */
|
||||
minstrel_ht_update_stats(mp, mi, true);
|
||||
minstrel_ht_update_rates(mp, mi);
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.h
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.h
|
||||
@@ -77,6 +77,8 @@ struct minstrel_ht_sta {
|
||||
/* overhead time in usec for each frame */
|
||||
unsigned int overhead;
|
||||
unsigned int overhead_rtscts;
|
||||
+ unsigned int overhead_legacy;
|
||||
+ unsigned int overhead_legacy_rtscts;
|
||||
|
||||
unsigned int total_packets_last;
|
||||
unsigned int total_packets_cur;
|
||||
@@ -97,9 +99,6 @@ struct minstrel_ht_sta {
|
||||
/* current MCS group to be sampled */
|
||||
u8 sample_group;
|
||||
|
||||
- u8 cck_supported;
|
||||
- u8 cck_supported_short;
|
||||
-
|
||||
/* Bitfield of supported MCS rates of all groups */
|
||||
u16 supported[MINSTREL_GROUPS_NB];
|
||||
|
@ -0,0 +1,762 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Sat, 26 Dec 2020 13:56:42 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: add support for OFDM rates on
|
||||
non-HT clients
|
||||
|
||||
The legacy minstrel code is essentially unmaintained and receives only very
|
||||
little testing. In order to bring the significant algorithm improvements from
|
||||
minstrel_ht to legacy clients, this patch adds support for OFDM rates to
|
||||
minstrel_ht and removes the fallback to the legacy codepath.
|
||||
This also makes it work much better on hardware with rate selection constraints,
|
||||
e.g. mt76.
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel.h
|
||||
+++ b/net/mac80211/rc80211_minstrel.h
|
||||
@@ -152,6 +152,7 @@ struct minstrel_priv {
|
||||
unsigned int lookaround_rate_mrr;
|
||||
|
||||
u8 cck_rates[4];
|
||||
+ u8 ofdm_rates[NUM_NL80211_BANDS][8];
|
||||
|
||||
#ifdef CPTCFG_MAC80211_DEBUGFS
|
||||
/*
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -163,6 +163,38 @@
|
||||
|
||||
#define CCK_GROUP __CCK_GROUP(CCK_GROUP_SHIFT)
|
||||
|
||||
+#define OFDM_DURATION(_bitrate) \
|
||||
+ (1000 * (16 /* SIFS + signal ext */ + \
|
||||
+ 16 /* T_PREAMBLE */ + \
|
||||
+ 4 /* T_SIGNAL */ + \
|
||||
+ 4 * (((16 + 80 * (AVG_PKT_SIZE + 4) + 6) / \
|
||||
+ ((_bitrate) * 4)))))
|
||||
+
|
||||
+#define OFDM_DURATION_LIST(_s) \
|
||||
+ OFDM_DURATION(60) >> _s, \
|
||||
+ OFDM_DURATION(90) >> _s, \
|
||||
+ OFDM_DURATION(120) >> _s, \
|
||||
+ OFDM_DURATION(180) >> _s, \
|
||||
+ OFDM_DURATION(240) >> _s, \
|
||||
+ OFDM_DURATION(360) >> _s, \
|
||||
+ OFDM_DURATION(480) >> _s, \
|
||||
+ OFDM_DURATION(540) >> _s
|
||||
+
|
||||
+#define __OFDM_GROUP(_s) \
|
||||
+ [MINSTREL_OFDM_GROUP] = { \
|
||||
+ .streams = 1, \
|
||||
+ .flags = 0, \
|
||||
+ .shift = _s, \
|
||||
+ .duration = { \
|
||||
+ OFDM_DURATION_LIST(_s), \
|
||||
+ } \
|
||||
+ }
|
||||
+
|
||||
+#define OFDM_GROUP_SHIFT \
|
||||
+ GROUP_SHIFT(OFDM_DURATION(60))
|
||||
+
|
||||
+#define OFDM_GROUP __OFDM_GROUP(OFDM_GROUP_SHIFT)
|
||||
+
|
||||
|
||||
static bool minstrel_vht_only = true;
|
||||
module_param(minstrel_vht_only, bool, 0644);
|
||||
@@ -199,6 +231,7 @@ const struct mcs_group minstrel_mcs_grou
|
||||
MCS_GROUP(4, 1, BW_40),
|
||||
|
||||
CCK_GROUP,
|
||||
+ OFDM_GROUP,
|
||||
|
||||
VHT_GROUP(1, 0, BW_20),
|
||||
VHT_GROUP(2, 0, BW_20),
|
||||
@@ -231,6 +264,8 @@ const struct mcs_group minstrel_mcs_grou
|
||||
VHT_GROUP(4, 1, BW_80),
|
||||
};
|
||||
|
||||
+const s16 minstrel_cck_bitrates[4] = { 10, 20, 55, 110 };
|
||||
+const s16 minstrel_ofdm_bitrates[8] = { 60, 90, 120, 180, 240, 360, 480, 540 };
|
||||
static u8 sample_table[SAMPLE_COLUMNS][MCS_GROUP_RATES] __read_mostly;
|
||||
|
||||
static void
|
||||
@@ -275,6 +310,13 @@ minstrel_get_valid_vht_rates(int bw, int
|
||||
return 0x3ff & ~mask;
|
||||
}
|
||||
|
||||
+static bool
|
||||
+minstrel_ht_is_legacy_group(int group)
|
||||
+{
|
||||
+ return group == MINSTREL_CCK_GROUP ||
|
||||
+ group == MINSTREL_OFDM_GROUP;
|
||||
+}
|
||||
+
|
||||
/*
|
||||
* Look up an MCS group index based on mac80211 rate information
|
||||
*/
|
||||
@@ -304,21 +346,34 @@ minstrel_ht_get_stats(struct minstrel_pr
|
||||
if (rate->flags & IEEE80211_TX_RC_MCS) {
|
||||
group = minstrel_ht_get_group_idx(rate);
|
||||
idx = rate->idx % 8;
|
||||
- } else if (rate->flags & IEEE80211_TX_RC_VHT_MCS) {
|
||||
+ goto out;
|
||||
+ }
|
||||
+
|
||||
+ if (rate->flags & IEEE80211_TX_RC_VHT_MCS) {
|
||||
group = minstrel_vht_get_group_idx(rate);
|
||||
idx = ieee80211_rate_get_vht_mcs(rate);
|
||||
- } else {
|
||||
- group = MINSTREL_CCK_GROUP;
|
||||
+ goto out;
|
||||
+ }
|
||||
|
||||
- for (idx = 0; idx < ARRAY_SIZE(mp->cck_rates); idx++)
|
||||
- if (rate->idx == mp->cck_rates[idx])
|
||||
- break;
|
||||
+ group = MINSTREL_CCK_GROUP;
|
||||
+ for (idx = 0; idx < ARRAY_SIZE(mp->cck_rates); idx++) {
|
||||
+ if (rate->idx != mp->cck_rates[idx])
|
||||
+ continue;
|
||||
|
||||
/* short preamble */
|
||||
if ((mi->supported[group] & BIT(idx + 4)) &&
|
||||
(rate->flags & IEEE80211_TX_RC_USE_SHORT_PREAMBLE))
|
||||
- idx += 4;
|
||||
+ idx += 4;
|
||||
+ goto out;
|
||||
}
|
||||
+
|
||||
+ group = MINSTREL_OFDM_GROUP;
|
||||
+ for (idx = 0; idx < ARRAY_SIZE(mp->ofdm_rates[0]); idx++)
|
||||
+ if (rate->idx == mp->ofdm_rates[mi->band][idx])
|
||||
+ goto out;
|
||||
+
|
||||
+ idx = 0;
|
||||
+out:
|
||||
return &mi->groups[group].rates[idx];
|
||||
}
|
||||
|
||||
@@ -352,7 +407,7 @@ minstrel_ht_get_tp_avg(struct minstrel_h
|
||||
if (prob_avg < MINSTREL_FRAC(10, 100))
|
||||
return 0;
|
||||
|
||||
- if (group == MINSTREL_CCK_GROUP)
|
||||
+ if (minstrel_ht_is_legacy_group(group))
|
||||
overhead = mi->overhead_legacy;
|
||||
else
|
||||
ampdu_len = minstrel_ht_avg_ampdu_len(mi);
|
||||
@@ -439,8 +494,8 @@ minstrel_ht_set_best_prob_rate(struct mi
|
||||
/* if max_tp_rate[0] is from MCS_GROUP max_prob_rate get selected from
|
||||
* MCS_GROUP as well as CCK_GROUP rates do not allow aggregation */
|
||||
max_tp_group = mi->max_tp_rate[0] / MCS_GROUP_RATES;
|
||||
- if((index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP) &&
|
||||
- (max_tp_group != MINSTREL_CCK_GROUP))
|
||||
+ if (minstrel_ht_is_legacy_group(index / MCS_GROUP_RATES) &&
|
||||
+ !minstrel_ht_is_legacy_group(max_tp_group))
|
||||
return;
|
||||
|
||||
max_gpr_group = mg->max_group_prob_rate / MCS_GROUP_RATES;
|
||||
@@ -476,13 +531,13 @@ minstrel_ht_set_best_prob_rate(struct mi
|
||||
static void
|
||||
minstrel_ht_assign_best_tp_rates(struct minstrel_ht_sta *mi,
|
||||
u16 tmp_mcs_tp_rate[MAX_THR_RATES],
|
||||
- u16 tmp_cck_tp_rate[MAX_THR_RATES])
|
||||
+ u16 tmp_legacy_tp_rate[MAX_THR_RATES])
|
||||
{
|
||||
unsigned int tmp_group, tmp_idx, tmp_cck_tp, tmp_mcs_tp, tmp_prob;
|
||||
int i;
|
||||
|
||||
- tmp_group = tmp_cck_tp_rate[0] / MCS_GROUP_RATES;
|
||||
- tmp_idx = tmp_cck_tp_rate[0] % MCS_GROUP_RATES;
|
||||
+ tmp_group = tmp_legacy_tp_rate[0] / MCS_GROUP_RATES;
|
||||
+ tmp_idx = tmp_legacy_tp_rate[0] % MCS_GROUP_RATES;
|
||||
tmp_prob = mi->groups[tmp_group].rates[tmp_idx].prob_avg;
|
||||
tmp_cck_tp = minstrel_ht_get_tp_avg(mi, tmp_group, tmp_idx, tmp_prob);
|
||||
|
||||
@@ -493,7 +548,7 @@ minstrel_ht_assign_best_tp_rates(struct
|
||||
|
||||
if (tmp_cck_tp > tmp_mcs_tp) {
|
||||
for(i = 0; i < MAX_THR_RATES; i++) {
|
||||
- minstrel_ht_sort_best_tp_rates(mi, tmp_cck_tp_rate[i],
|
||||
+ minstrel_ht_sort_best_tp_rates(mi, tmp_legacy_tp_rate[i],
|
||||
tmp_mcs_tp_rate);
|
||||
}
|
||||
}
|
||||
@@ -511,6 +566,9 @@ minstrel_ht_prob_rate_reduce_streams(str
|
||||
int tmp_max_streams, group, tmp_idx, tmp_prob;
|
||||
int tmp_tp = 0;
|
||||
|
||||
+ if (!mi->sta->ht_cap.ht_supported)
|
||||
+ return;
|
||||
+
|
||||
tmp_max_streams = minstrel_mcs_groups[mi->max_tp_rate[0] /
|
||||
MCS_GROUP_RATES].streams;
|
||||
for (group = 0; group < ARRAY_SIZE(minstrel_mcs_groups); group++) {
|
||||
@@ -675,7 +733,8 @@ minstrel_ht_update_stats(struct minstrel
|
||||
struct minstrel_rate_stats *mrs;
|
||||
int group, i, j, cur_prob;
|
||||
u16 tmp_mcs_tp_rate[MAX_THR_RATES], tmp_group_tp_rate[MAX_THR_RATES];
|
||||
- u16 tmp_cck_tp_rate[MAX_THR_RATES], index;
|
||||
+ u16 tmp_legacy_tp_rate[MAX_THR_RATES], index;
|
||||
+ bool ht_supported = mi->sta->ht_cap.ht_supported;
|
||||
|
||||
mi->sample_mode = MINSTREL_SAMPLE_IDLE;
|
||||
|
||||
@@ -704,21 +763,29 @@ minstrel_ht_update_stats(struct minstrel
|
||||
mi->sample_count = 0;
|
||||
|
||||
memset(tmp_mcs_tp_rate, 0, sizeof(tmp_mcs_tp_rate));
|
||||
- memset(tmp_cck_tp_rate, 0, sizeof(tmp_cck_tp_rate));
|
||||
+ memset(tmp_legacy_tp_rate, 0, sizeof(tmp_legacy_tp_rate));
|
||||
if (mi->supported[MINSTREL_CCK_GROUP])
|
||||
- for (j = 0; j < ARRAY_SIZE(tmp_cck_tp_rate); j++)
|
||||
- tmp_cck_tp_rate[j] = MINSTREL_CCK_GROUP * MCS_GROUP_RATES;
|
||||
+ for (j = 0; j < ARRAY_SIZE(tmp_legacy_tp_rate); j++)
|
||||
+ tmp_legacy_tp_rate[j] = MINSTREL_CCK_GROUP * MCS_GROUP_RATES;
|
||||
+ else if (mi->supported[MINSTREL_OFDM_GROUP])
|
||||
+ for (j = 0; j < ARRAY_SIZE(tmp_legacy_tp_rate); j++)
|
||||
+ tmp_legacy_tp_rate[j] = MINSTREL_OFDM_GROUP * MCS_GROUP_RATES;
|
||||
|
||||
if (mi->supported[MINSTREL_VHT_GROUP_0])
|
||||
index = MINSTREL_VHT_GROUP_0 * MCS_GROUP_RATES;
|
||||
- else
|
||||
+ else if (ht_supported)
|
||||
index = MINSTREL_HT_GROUP_0 * MCS_GROUP_RATES;
|
||||
+ else if (mi->supported[MINSTREL_CCK_GROUP])
|
||||
+ index = MINSTREL_CCK_GROUP * MCS_GROUP_RATES;
|
||||
+ else
|
||||
+ index = MINSTREL_OFDM_GROUP * MCS_GROUP_RATES;
|
||||
|
||||
for (j = 0; j < ARRAY_SIZE(tmp_mcs_tp_rate); j++)
|
||||
tmp_mcs_tp_rate[j] = index;
|
||||
|
||||
/* Find best rate sets within all MCS groups*/
|
||||
for (group = 0; group < ARRAY_SIZE(minstrel_mcs_groups); group++) {
|
||||
+ u16 *tp_rate = tmp_mcs_tp_rate;
|
||||
|
||||
mg = &mi->groups[group];
|
||||
if (!mi->supported[group])
|
||||
@@ -730,6 +797,9 @@ minstrel_ht_update_stats(struct minstrel
|
||||
for(j = 0; j < MAX_THR_RATES; j++)
|
||||
tmp_group_tp_rate[j] = MCS_GROUP_RATES * group;
|
||||
|
||||
+ if (group == MINSTREL_CCK_GROUP && ht_supported)
|
||||
+ tp_rate = tmp_legacy_tp_rate;
|
||||
+
|
||||
for (i = 0; i < MCS_GROUP_RATES; i++) {
|
||||
if (!(mi->supported[group] & BIT(i)))
|
||||
continue;
|
||||
@@ -745,13 +815,7 @@ minstrel_ht_update_stats(struct minstrel
|
||||
continue;
|
||||
|
||||
/* Find max throughput rate set */
|
||||
- if (group != MINSTREL_CCK_GROUP) {
|
||||
- minstrel_ht_sort_best_tp_rates(mi, index,
|
||||
- tmp_mcs_tp_rate);
|
||||
- } else if (group == MINSTREL_CCK_GROUP) {
|
||||
- minstrel_ht_sort_best_tp_rates(mi, index,
|
||||
- tmp_cck_tp_rate);
|
||||
- }
|
||||
+ minstrel_ht_sort_best_tp_rates(mi, index, tp_rate);
|
||||
|
||||
/* Find max throughput rate set within a group */
|
||||
minstrel_ht_sort_best_tp_rates(mi, index,
|
||||
@@ -766,7 +830,8 @@ minstrel_ht_update_stats(struct minstrel
|
||||
}
|
||||
|
||||
/* Assign new rate set per sta */
|
||||
- minstrel_ht_assign_best_tp_rates(mi, tmp_mcs_tp_rate, tmp_cck_tp_rate);
|
||||
+ minstrel_ht_assign_best_tp_rates(mi, tmp_mcs_tp_rate,
|
||||
+ tmp_legacy_tp_rate);
|
||||
memcpy(mi->max_tp_rate, tmp_mcs_tp_rate, sizeof(mi->max_tp_rate));
|
||||
|
||||
/* Try to increase robustness of max_prob_rate*/
|
||||
@@ -795,8 +860,11 @@ minstrel_ht_update_stats(struct minstrel
|
||||
}
|
||||
|
||||
static bool
|
||||
-minstrel_ht_txstat_valid(struct minstrel_priv *mp, struct ieee80211_tx_rate *rate)
|
||||
+minstrel_ht_txstat_valid(struct minstrel_priv *mp, struct minstrel_ht_sta *mi,
|
||||
+ struct ieee80211_tx_rate *rate)
|
||||
{
|
||||
+ int i;
|
||||
+
|
||||
if (rate->idx < 0)
|
||||
return false;
|
||||
|
||||
@@ -807,10 +875,15 @@ minstrel_ht_txstat_valid(struct minstrel
|
||||
rate->flags & IEEE80211_TX_RC_VHT_MCS)
|
||||
return true;
|
||||
|
||||
- return rate->idx == mp->cck_rates[0] ||
|
||||
- rate->idx == mp->cck_rates[1] ||
|
||||
- rate->idx == mp->cck_rates[2] ||
|
||||
- rate->idx == mp->cck_rates[3];
|
||||
+ for (i = 0; i < ARRAY_SIZE(mp->cck_rates); i++)
|
||||
+ if (rate->idx == mp->cck_rates[i])
|
||||
+ return true;
|
||||
+
|
||||
+ for (i = 0; i < ARRAY_SIZE(mp->ofdm_rates[0]); i++)
|
||||
+ if (rate->idx == mp->ofdm_rates[mi->band][i])
|
||||
+ return true;
|
||||
+
|
||||
+ return false;
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -897,11 +970,6 @@ minstrel_ht_tx_status(void *priv, struct
|
||||
bool sample_status = false;
|
||||
int i;
|
||||
|
||||
- if (!msp->is_ht)
|
||||
- return mac80211_minstrel.tx_status_ext(priv, sband,
|
||||
- &msp->legacy, st);
|
||||
-
|
||||
-
|
||||
/* This packet was aggregated but doesn't carry status info */
|
||||
if ((info->flags & IEEE80211_TX_CTL_AMPDU) &&
|
||||
!(info->flags & IEEE80211_TX_STAT_AMPDU))
|
||||
@@ -930,10 +998,10 @@ minstrel_ht_tx_status(void *priv, struct
|
||||
if (mi->sample_mode != MINSTREL_SAMPLE_IDLE)
|
||||
rate_sample = minstrel_get_ratestats(mi, mi->sample_rate);
|
||||
|
||||
- last = !minstrel_ht_txstat_valid(mp, &ar[0]);
|
||||
+ last = !minstrel_ht_txstat_valid(mp, mi, &ar[0]);
|
||||
for (i = 0; !last; i++) {
|
||||
last = (i == IEEE80211_TX_MAX_RATES - 1) ||
|
||||
- !minstrel_ht_txstat_valid(mp, &ar[i + 1]);
|
||||
+ !minstrel_ht_txstat_valid(mp, mi, &ar[i + 1]);
|
||||
|
||||
rate = minstrel_ht_get_stats(mp, mi, &ar[i]);
|
||||
if (rate == rate_sample)
|
||||
@@ -1031,7 +1099,7 @@ minstrel_calc_retransmit(struct minstrel
|
||||
ctime += (t_slot * cw) >> 1;
|
||||
cw = min((cw << 1) | 1, mp->cw_max);
|
||||
|
||||
- if (index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP) {
|
||||
+ if (minstrel_ht_is_legacy_group(index / MCS_GROUP_RATES)) {
|
||||
overhead = mi->overhead_legacy;
|
||||
overhead_rtscts = mi->overhead_legacy_rtscts;
|
||||
} else {
|
||||
@@ -1064,7 +1132,8 @@ static void
|
||||
minstrel_ht_set_rate(struct minstrel_priv *mp, struct minstrel_ht_sta *mi,
|
||||
struct ieee80211_sta_rates *ratetbl, int offset, int index)
|
||||
{
|
||||
- const struct mcs_group *group = &minstrel_mcs_groups[index / MCS_GROUP_RATES];
|
||||
+ int group_idx = index / MCS_GROUP_RATES;
|
||||
+ const struct mcs_group *group = &minstrel_mcs_groups[group_idx];
|
||||
struct minstrel_rate_stats *mrs;
|
||||
u8 idx;
|
||||
u16 flags = group->flags;
|
||||
@@ -1083,13 +1152,17 @@ minstrel_ht_set_rate(struct minstrel_pri
|
||||
ratetbl->rate[offset].count_rts = mrs->retry_count_rtscts;
|
||||
}
|
||||
|
||||
- if (index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP)
|
||||
+ index %= MCS_GROUP_RATES;
|
||||
+ if (group_idx == MINSTREL_CCK_GROUP)
|
||||
idx = mp->cck_rates[index % ARRAY_SIZE(mp->cck_rates)];
|
||||
+ else if (group_idx == MINSTREL_OFDM_GROUP)
|
||||
+ idx = mp->ofdm_rates[mi->band][index %
|
||||
+ ARRAY_SIZE(mp->ofdm_rates[0])];
|
||||
else if (flags & IEEE80211_TX_RC_VHT_MCS)
|
||||
idx = ((group->streams - 1) << 4) |
|
||||
- ((index % MCS_GROUP_RATES) & 0xF);
|
||||
+ (index & 0xF);
|
||||
else
|
||||
- idx = index % MCS_GROUP_RATES + (group->streams - 1) * 8;
|
||||
+ idx = index + (group->streams - 1) * 8;
|
||||
|
||||
/* enable RTS/CTS if needed:
|
||||
* - if station is in dynamic SMPS (and streams > 1)
|
||||
@@ -1304,11 +1377,8 @@ minstrel_ht_get_rate(void *priv, struct
|
||||
struct minstrel_priv *mp = priv;
|
||||
int sample_idx;
|
||||
|
||||
- if (!msp->is_ht)
|
||||
- return mac80211_minstrel.get_rate(priv, sta, &msp->legacy, txrc);
|
||||
-
|
||||
if (!(info->flags & IEEE80211_TX_CTL_AMPDU) &&
|
||||
- mi->max_prob_rate / MCS_GROUP_RATES != MINSTREL_CCK_GROUP)
|
||||
+ !minstrel_ht_is_legacy_group(mi->max_prob_rate / MCS_GROUP_RATES))
|
||||
minstrel_aggr_check(sta, txrc->skb);
|
||||
|
||||
info->flags |= mi->tx_flags;
|
||||
@@ -1349,6 +1419,9 @@ minstrel_ht_get_rate(void *priv, struct
|
||||
if (sample_group == &minstrel_mcs_groups[MINSTREL_CCK_GROUP]) {
|
||||
int idx = sample_idx % ARRAY_SIZE(mp->cck_rates);
|
||||
rate->idx = mp->cck_rates[idx];
|
||||
+ } else if (sample_group == &minstrel_mcs_groups[MINSTREL_OFDM_GROUP]) {
|
||||
+ int idx = sample_idx % ARRAY_SIZE(mp->ofdm_rates[0]);
|
||||
+ rate->idx = mp->ofdm_rates[mi->band][idx];
|
||||
} else if (sample_group->flags & IEEE80211_TX_RC_VHT_MCS) {
|
||||
ieee80211_rate_set_vht(rate, sample_idx % MCS_GROUP_RATES,
|
||||
sample_group->streams);
|
||||
@@ -1369,11 +1442,13 @@ minstrel_ht_update_cck(struct minstrel_p
|
||||
if (sband->band != NL80211_BAND_2GHZ)
|
||||
return;
|
||||
|
||||
- if (!ieee80211_hw_check(mp->hw, SUPPORTS_HT_CCK_RATES))
|
||||
+ if (sta->ht_cap.ht_supported &&
|
||||
+ !ieee80211_hw_check(mp->hw, SUPPORTS_HT_CCK_RATES))
|
||||
return;
|
||||
|
||||
for (i = 0; i < 4; i++) {
|
||||
- if (!rate_supported(sta, sband->band, mp->cck_rates[i]))
|
||||
+ if (mp->cck_rates[i] == 0xff ||
|
||||
+ !rate_supported(sta, sband->band, mp->cck_rates[i]))
|
||||
continue;
|
||||
|
||||
mi->supported[MINSTREL_CCK_GROUP] |= BIT(i);
|
||||
@@ -1383,9 +1458,30 @@ minstrel_ht_update_cck(struct minstrel_p
|
||||
}
|
||||
|
||||
static void
|
||||
+minstrel_ht_update_ofdm(struct minstrel_priv *mp, struct minstrel_ht_sta *mi,
|
||||
+ struct ieee80211_supported_band *sband,
|
||||
+ struct ieee80211_sta *sta)
|
||||
+{
|
||||
+ const u8 *rates;
|
||||
+ int i;
|
||||
+
|
||||
+ if (sta->ht_cap.ht_supported)
|
||||
+ return;
|
||||
+
|
||||
+ rates = mp->ofdm_rates[sband->band];
|
||||
+ for (i = 0; i < ARRAY_SIZE(mp->ofdm_rates[0]); i++) {
|
||||
+ if (rates[i] == 0xff ||
|
||||
+ !rate_supported(sta, sband->band, rates[i]))
|
||||
+ continue;
|
||||
+
|
||||
+ mi->supported[MINSTREL_OFDM_GROUP] |= BIT(i);
|
||||
+ }
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
minstrel_ht_update_caps(void *priv, struct ieee80211_supported_band *sband,
|
||||
struct cfg80211_chan_def *chandef,
|
||||
- struct ieee80211_sta *sta, void *priv_sta)
|
||||
+ struct ieee80211_sta *sta, void *priv_sta)
|
||||
{
|
||||
struct minstrel_priv *mp = priv;
|
||||
struct minstrel_ht_sta_priv *msp = priv_sta;
|
||||
@@ -1401,10 +1497,6 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
int stbc;
|
||||
int i;
|
||||
|
||||
- /* fall back to the old minstrel for legacy stations */
|
||||
- if (!sta->ht_cap.ht_supported)
|
||||
- goto use_legacy;
|
||||
-
|
||||
BUILD_BUG_ON(ARRAY_SIZE(minstrel_mcs_groups) != MINSTREL_GROUPS_NB);
|
||||
|
||||
if (vht_cap->vht_supported)
|
||||
@@ -1412,10 +1504,10 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
else
|
||||
use_vht = 0;
|
||||
|
||||
- msp->is_ht = true;
|
||||
memset(mi, 0, sizeof(*mi));
|
||||
|
||||
mi->sta = sta;
|
||||
+ mi->band = sband->band;
|
||||
mi->last_stats_update = jiffies;
|
||||
|
||||
ack_dur = ieee80211_frame_duration(sband->band, 10, 60, 1, 1, 0);
|
||||
@@ -1464,10 +1556,8 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
int bw, nss;
|
||||
|
||||
mi->supported[i] = 0;
|
||||
- if (i == MINSTREL_CCK_GROUP) {
|
||||
- minstrel_ht_update_cck(mp, mi, sband, sta);
|
||||
+ if (minstrel_ht_is_legacy_group(i))
|
||||
continue;
|
||||
- }
|
||||
|
||||
if (gflags & IEEE80211_TX_RC_SHORT_GI) {
|
||||
if (gflags & IEEE80211_TX_RC_40_MHZ_WIDTH) {
|
||||
@@ -1528,22 +1618,12 @@ minstrel_ht_update_caps(void *priv, stru
|
||||
n_supported++;
|
||||
}
|
||||
|
||||
- if (!n_supported)
|
||||
- goto use_legacy;
|
||||
+ minstrel_ht_update_cck(mp, mi, sband, sta);
|
||||
+ minstrel_ht_update_ofdm(mp, mi, sband, sta);
|
||||
|
||||
/* create an initial rate table with the lowest supported rates */
|
||||
minstrel_ht_update_stats(mp, mi, true);
|
||||
minstrel_ht_update_rates(mp, mi);
|
||||
-
|
||||
- return;
|
||||
-
|
||||
-use_legacy:
|
||||
- msp->is_ht = false;
|
||||
- memset(&msp->legacy, 0, sizeof(msp->legacy));
|
||||
- msp->legacy.r = msp->ratelist;
|
||||
- msp->legacy.sample_table = msp->sample_table;
|
||||
- return mac80211_minstrel.rate_init(priv, sband, chandef, sta,
|
||||
- &msp->legacy);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -1611,40 +1691,70 @@ minstrel_ht_free_sta(void *priv, struct
|
||||
}
|
||||
|
||||
static void
|
||||
-minstrel_ht_init_cck_rates(struct minstrel_priv *mp)
|
||||
+minstrel_ht_fill_rate_array(u8 *dest, struct ieee80211_supported_band *sband,
|
||||
+ const s16 *bitrates, int n_rates, u32 rate_flags)
|
||||
{
|
||||
- static const int bitrates[4] = { 10, 20, 55, 110 };
|
||||
- struct ieee80211_supported_band *sband;
|
||||
- u32 rate_flags = ieee80211_chandef_rate_flags(&mp->hw->conf.chandef);
|
||||
int i, j;
|
||||
|
||||
- sband = mp->hw->wiphy->bands[NL80211_BAND_2GHZ];
|
||||
- if (!sband)
|
||||
- return;
|
||||
-
|
||||
for (i = 0; i < sband->n_bitrates; i++) {
|
||||
struct ieee80211_rate *rate = &sband->bitrates[i];
|
||||
|
||||
- if (rate->flags & IEEE80211_RATE_ERP_G)
|
||||
- continue;
|
||||
-
|
||||
if ((rate_flags & sband->bitrates[i].flags) != rate_flags)
|
||||
continue;
|
||||
|
||||
- for (j = 0; j < ARRAY_SIZE(bitrates); j++) {
|
||||
+ for (j = 0; j < n_rates; j++) {
|
||||
if (rate->bitrate != bitrates[j])
|
||||
continue;
|
||||
|
||||
- mp->cck_rates[j] = i;
|
||||
+ dest[j] = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+static void
|
||||
+minstrel_ht_init_cck_rates(struct minstrel_priv *mp)
|
||||
+{
|
||||
+ static const s16 bitrates[4] = { 10, 20, 55, 110 };
|
||||
+ struct ieee80211_supported_band *sband;
|
||||
+ u32 rate_flags = ieee80211_chandef_rate_flags(&mp->hw->conf.chandef);
|
||||
+
|
||||
+ memset(mp->cck_rates, 0xff, sizeof(mp->cck_rates));
|
||||
+ sband = mp->hw->wiphy->bands[NL80211_BAND_2GHZ];
|
||||
+ if (!sband)
|
||||
+ return;
|
||||
+
|
||||
+ BUILD_BUG_ON(ARRAY_SIZE(mp->cck_rates) != ARRAY_SIZE(bitrates));
|
||||
+ minstrel_ht_fill_rate_array(mp->cck_rates, sband,
|
||||
+ minstrel_cck_bitrates,
|
||||
+ ARRAY_SIZE(minstrel_cck_bitrates),
|
||||
+ rate_flags);
|
||||
+}
|
||||
+
|
||||
+static void
|
||||
+minstrel_ht_init_ofdm_rates(struct minstrel_priv *mp, enum nl80211_band band)
|
||||
+{
|
||||
+ static const s16 bitrates[8] = { 60, 90, 120, 180, 240, 360, 480, 540 };
|
||||
+ struct ieee80211_supported_band *sband;
|
||||
+ u32 rate_flags = ieee80211_chandef_rate_flags(&mp->hw->conf.chandef);
|
||||
+
|
||||
+ memset(mp->ofdm_rates[band], 0xff, sizeof(mp->ofdm_rates[band]));
|
||||
+ sband = mp->hw->wiphy->bands[band];
|
||||
+ if (!sband)
|
||||
+ return;
|
||||
+
|
||||
+ BUILD_BUG_ON(ARRAY_SIZE(mp->ofdm_rates[band]) != ARRAY_SIZE(bitrates));
|
||||
+ minstrel_ht_fill_rate_array(mp->ofdm_rates[band], sband,
|
||||
+ minstrel_ofdm_bitrates,
|
||||
+ ARRAY_SIZE(minstrel_ofdm_bitrates),
|
||||
+ rate_flags);
|
||||
+}
|
||||
+
|
||||
static void *
|
||||
minstrel_ht_alloc(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct minstrel_priv *mp;
|
||||
+ int i;
|
||||
|
||||
mp = kzalloc(sizeof(struct minstrel_priv), GFP_ATOMIC);
|
||||
if (!mp)
|
||||
@@ -1681,6 +1791,8 @@ minstrel_ht_alloc(struct ieee80211_hw *h
|
||||
mp->new_avg = true;
|
||||
|
||||
minstrel_ht_init_cck_rates(mp);
|
||||
+ for (i = 0; i < ARRAY_SIZE(mp->hw->wiphy->bands); i++)
|
||||
+ minstrel_ht_init_ofdm_rates(mp, i);
|
||||
|
||||
return mp;
|
||||
}
|
||||
@@ -1713,9 +1825,6 @@ static u32 minstrel_ht_get_expected_thro
|
||||
struct minstrel_ht_sta *mi = &msp->ht;
|
||||
int i, j, prob, tp_avg;
|
||||
|
||||
- if (!msp->is_ht)
|
||||
- return mac80211_minstrel.get_expected_throughput(priv_sta);
|
||||
-
|
||||
i = mi->max_tp_rate[0] / MCS_GROUP_RATES;
|
||||
j = mi->max_tp_rate[0] % MCS_GROUP_RATES;
|
||||
prob = mi->groups[i].rates[j].prob_avg;
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.h
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.h
|
||||
@@ -18,14 +18,15 @@
|
||||
MINSTREL_HT_STREAM_GROUPS)
|
||||
#define MINSTREL_VHT_GROUPS_NB (MINSTREL_MAX_STREAMS * \
|
||||
MINSTREL_VHT_STREAM_GROUPS)
|
||||
-#define MINSTREL_CCK_GROUPS_NB 1
|
||||
+#define MINSTREL_LEGACY_GROUPS_NB 2
|
||||
#define MINSTREL_GROUPS_NB (MINSTREL_HT_GROUPS_NB + \
|
||||
MINSTREL_VHT_GROUPS_NB + \
|
||||
- MINSTREL_CCK_GROUPS_NB)
|
||||
+ MINSTREL_LEGACY_GROUPS_NB)
|
||||
|
||||
#define MINSTREL_HT_GROUP_0 0
|
||||
#define MINSTREL_CCK_GROUP (MINSTREL_HT_GROUP_0 + MINSTREL_HT_GROUPS_NB)
|
||||
-#define MINSTREL_VHT_GROUP_0 (MINSTREL_CCK_GROUP + 1)
|
||||
+#define MINSTREL_OFDM_GROUP (MINSTREL_CCK_GROUP + 1)
|
||||
+#define MINSTREL_VHT_GROUP_0 (MINSTREL_OFDM_GROUP + 1)
|
||||
|
||||
#define MCS_GROUP_RATES 10
|
||||
|
||||
@@ -37,6 +38,8 @@ struct mcs_group {
|
||||
u16 duration[MCS_GROUP_RATES];
|
||||
};
|
||||
|
||||
+extern const s16 minstrel_cck_bitrates[4];
|
||||
+extern const s16 minstrel_ofdm_bitrates[8];
|
||||
extern const struct mcs_group minstrel_mcs_groups[];
|
||||
|
||||
struct minstrel_mcs_group_data {
|
||||
@@ -99,6 +102,8 @@ struct minstrel_ht_sta {
|
||||
/* current MCS group to be sampled */
|
||||
u8 sample_group;
|
||||
|
||||
+ u8 band;
|
||||
+
|
||||
/* Bitfield of supported MCS rates of all groups */
|
||||
u16 supported[MINSTREL_GROUPS_NB];
|
||||
|
||||
@@ -107,13 +112,9 @@ struct minstrel_ht_sta {
|
||||
};
|
||||
|
||||
struct minstrel_ht_sta_priv {
|
||||
- union {
|
||||
- struct minstrel_ht_sta ht;
|
||||
- struct minstrel_sta_info legacy;
|
||||
- };
|
||||
+ struct minstrel_ht_sta ht;
|
||||
void *ratelist;
|
||||
void *sample_table;
|
||||
- bool is_ht;
|
||||
};
|
||||
|
||||
void minstrel_ht_add_sta_debugfs(void *priv, void *priv_sta, struct dentry *dir);
|
||||
--- a/net/mac80211/rc80211_minstrel_ht_debugfs.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht_debugfs.c
|
||||
@@ -52,7 +52,6 @@ minstrel_ht_stats_dump(struct minstrel_h
|
||||
|
||||
for (j = 0; j < MCS_GROUP_RATES; j++) {
|
||||
struct minstrel_rate_stats *mrs = &mi->groups[i].rates[j];
|
||||
- static const int bitrates[4] = { 10, 20, 55, 110 };
|
||||
int idx = i * MCS_GROUP_RATES + j;
|
||||
unsigned int duration;
|
||||
|
||||
@@ -67,6 +66,9 @@ minstrel_ht_stats_dump(struct minstrel_h
|
||||
p += sprintf(p, "VHT%c0 ", htmode);
|
||||
p += sprintf(p, "%cGI ", gimode);
|
||||
p += sprintf(p, "%d ", mg->streams);
|
||||
+ } else if (i == MINSTREL_OFDM_GROUP) {
|
||||
+ p += sprintf(p, "OFDM ");
|
||||
+ p += sprintf(p, "1 ");
|
||||
} else {
|
||||
p += sprintf(p, "CCK ");
|
||||
p += sprintf(p, "%cP ", j < 4 ? 'L' : 'S');
|
||||
@@ -84,7 +86,12 @@ minstrel_ht_stats_dump(struct minstrel_h
|
||||
} else if (gflags & IEEE80211_TX_RC_VHT_MCS) {
|
||||
p += sprintf(p, " MCS%-1u/%1u", j, mg->streams);
|
||||
} else {
|
||||
- int r = bitrates[j % 4];
|
||||
+ int r;
|
||||
+
|
||||
+ if (i == MINSTREL_OFDM_GROUP)
|
||||
+ r = minstrel_ofdm_bitrates[j % 8];
|
||||
+ else
|
||||
+ r = minstrel_cck_bitrates[j % 4];
|
||||
|
||||
p += sprintf(p, " %2u.%1uM", r / 10, r % 10);
|
||||
}
|
||||
@@ -124,16 +131,8 @@ minstrel_ht_stats_open(struct inode *ino
|
||||
struct minstrel_ht_sta *mi = &msp->ht;
|
||||
struct minstrel_debugfs_info *ms;
|
||||
unsigned int i;
|
||||
- int ret;
|
||||
char *p;
|
||||
|
||||
- if (!msp->is_ht) {
|
||||
- inode->i_private = &msp->legacy;
|
||||
- ret = minstrel_stats_open(inode, file);
|
||||
- inode->i_private = msp;
|
||||
- return ret;
|
||||
- }
|
||||
-
|
||||
ms = kmalloc(32768, GFP_KERNEL);
|
||||
if (!ms)
|
||||
return -ENOMEM;
|
||||
@@ -199,7 +198,6 @@ minstrel_ht_stats_csv_dump(struct minstr
|
||||
|
||||
for (j = 0; j < MCS_GROUP_RATES; j++) {
|
||||
struct minstrel_rate_stats *mrs = &mi->groups[i].rates[j];
|
||||
- static const int bitrates[4] = { 10, 20, 55, 110 };
|
||||
int idx = i * MCS_GROUP_RATES + j;
|
||||
unsigned int duration;
|
||||
|
||||
@@ -214,6 +212,8 @@ minstrel_ht_stats_csv_dump(struct minstr
|
||||
p += sprintf(p, "VHT%c0,", htmode);
|
||||
p += sprintf(p, "%cGI,", gimode);
|
||||
p += sprintf(p, "%d,", mg->streams);
|
||||
+ } else if (i == MINSTREL_OFDM_GROUP) {
|
||||
+ p += sprintf(p, "OFDM,,1,");
|
||||
} else {
|
||||
p += sprintf(p, "CCK,");
|
||||
p += sprintf(p, "%cP,", j < 4 ? 'L' : 'S');
|
||||
@@ -231,7 +231,13 @@ minstrel_ht_stats_csv_dump(struct minstr
|
||||
} else if (gflags & IEEE80211_TX_RC_VHT_MCS) {
|
||||
p += sprintf(p, ",MCS%-1u/%1u,", j, mg->streams);
|
||||
} else {
|
||||
- int r = bitrates[j % 4];
|
||||
+ int r;
|
||||
+
|
||||
+ if (i == MINSTREL_OFDM_GROUP)
|
||||
+ r = minstrel_ofdm_bitrates[j % 8];
|
||||
+ else
|
||||
+ r = minstrel_cck_bitrates[j % 4];
|
||||
+
|
||||
p += sprintf(p, ",%2u.%1uM,", r / 10, r % 10);
|
||||
}
|
||||
|
||||
@@ -274,18 +280,9 @@ minstrel_ht_stats_csv_open(struct inode
|
||||
struct minstrel_ht_sta *mi = &msp->ht;
|
||||
struct minstrel_debugfs_info *ms;
|
||||
unsigned int i;
|
||||
- int ret;
|
||||
char *p;
|
||||
|
||||
- if (!msp->is_ht) {
|
||||
- inode->i_private = &msp->legacy;
|
||||
- ret = minstrel_stats_csv_open(inode, file);
|
||||
- inode->i_private = msp;
|
||||
- return ret;
|
||||
- }
|
||||
-
|
||||
ms = kmalloc(32768, GFP_KERNEL);
|
||||
-
|
||||
if (!ms)
|
||||
return -ENOMEM;
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,96 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Sat, 26 Dec 2020 14:34:30 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: remove old ewma based rate average
|
||||
code
|
||||
|
||||
The new noise filter has been the default for a while now with no reported
|
||||
downside and significant improvement compared to the old code.
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -769,17 +769,8 @@ minstrel_ht_calc_rate_stats(struct minst
|
||||
if (unlikely(mrs->attempts > 0)) {
|
||||
mrs->sample_skipped = 0;
|
||||
cur_prob = MINSTREL_FRAC(mrs->success, mrs->attempts);
|
||||
- if (mp->new_avg) {
|
||||
- minstrel_filter_avg_add(&mrs->prob_avg,
|
||||
- &mrs->prob_avg_1, cur_prob);
|
||||
- } else if (unlikely(!mrs->att_hist)) {
|
||||
- mrs->prob_avg = cur_prob;
|
||||
- } else {
|
||||
- /*update exponential weighted moving avarage */
|
||||
- mrs->prob_avg = minstrel_ewma(mrs->prob_avg,
|
||||
- cur_prob,
|
||||
- EWMA_LEVEL);
|
||||
- }
|
||||
+ minstrel_filter_avg_add(&mrs->prob_avg,
|
||||
+ &mrs->prob_avg_1, cur_prob);
|
||||
mrs->att_hist += mrs->attempts;
|
||||
mrs->succ_hist += mrs->success;
|
||||
} else {
|
||||
@@ -913,10 +904,8 @@ minstrel_ht_update_stats(struct minstrel
|
||||
/* Try to increase robustness of max_prob_rate*/
|
||||
minstrel_ht_prob_rate_reduce_streams(mi);
|
||||
|
||||
- /* try to sample all available rates during each interval */
|
||||
- mi->sample_count *= 8;
|
||||
- if (mp->new_avg)
|
||||
- mi->sample_count /= 2;
|
||||
+ /* try to sample half of all available rates during each interval */
|
||||
+ mi->sample_count *= 4;
|
||||
|
||||
if (sample)
|
||||
minstrel_ht_rate_sample_switch(mp, mi);
|
||||
@@ -1040,7 +1029,7 @@ minstrel_ht_tx_status(void *priv, struct
|
||||
struct ieee80211_tx_rate *ar = info->status.rates;
|
||||
struct minstrel_rate_stats *rate, *rate2, *rate_sample = NULL;
|
||||
struct minstrel_priv *mp = priv;
|
||||
- u32 update_interval = mp->update_interval / 2;
|
||||
+ u32 update_interval = mp->update_interval;
|
||||
bool last, update = false;
|
||||
bool sample_status = false;
|
||||
int i;
|
||||
@@ -1090,9 +1079,8 @@ minstrel_ht_tx_status(void *priv, struct
|
||||
|
||||
switch (mi->sample_mode) {
|
||||
case MINSTREL_SAMPLE_IDLE:
|
||||
- if (mp->new_avg &&
|
||||
- (mp->hw->max_rates > 1 ||
|
||||
- mi->total_packets_cur < SAMPLE_SWITCH_THR))
|
||||
+ if (mp->hw->max_rates > 1 ||
|
||||
+ mi->total_packets_cur < SAMPLE_SWITCH_THR)
|
||||
update_interval /= 2;
|
||||
break;
|
||||
|
||||
@@ -1832,8 +1820,7 @@ minstrel_ht_alloc(struct ieee80211_hw *h
|
||||
mp->has_mrr = true;
|
||||
|
||||
mp->hw = hw;
|
||||
- mp->update_interval = HZ / 10;
|
||||
- mp->new_avg = true;
|
||||
+ mp->update_interval = HZ / 20;
|
||||
|
||||
minstrel_ht_init_cck_rates(mp);
|
||||
for (i = 0; i < ARRAY_SIZE(mp->hw->wiphy->bands); i++)
|
||||
@@ -1853,8 +1840,6 @@ static void minstrel_ht_add_debugfs(stru
|
||||
&mp->fixed_rate_idx);
|
||||
debugfs_create_u32("sample_switch", S_IRUGO | S_IWUSR, debugfsdir,
|
||||
&mp->sample_switch);
|
||||
- debugfs_create_bool("new_avg", S_IRUGO | S_IWUSR, debugfsdir,
|
||||
- &mp->new_avg);
|
||||
}
|
||||
#endif
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.h
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.h
|
||||
@@ -60,7 +60,6 @@
|
||||
struct minstrel_priv {
|
||||
struct ieee80211_hw *hw;
|
||||
bool has_mrr;
|
||||
- bool new_avg;
|
||||
u32 sample_switch;
|
||||
unsigned int cw_min;
|
||||
unsigned int cw_max;
|
@ -0,0 +1,67 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Sat, 26 Dec 2020 19:08:19 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: improve ampdu length estimation
|
||||
|
||||
If the driver does not report A-MPDU length, estimate it based on the rate.
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -382,13 +382,37 @@ minstrel_get_ratestats(struct minstrel_h
|
||||
return &mi->groups[index / MCS_GROUP_RATES].rates[index % MCS_GROUP_RATES];
|
||||
}
|
||||
|
||||
+static inline int
|
||||
+minstrel_get_duration(int index)
|
||||
+{
|
||||
+ const struct mcs_group *group = &minstrel_mcs_groups[index / MCS_GROUP_RATES];
|
||||
+ unsigned int duration = group->duration[index % MCS_GROUP_RATES];
|
||||
+ return duration << group->shift;
|
||||
+}
|
||||
+
|
||||
static unsigned int
|
||||
minstrel_ht_avg_ampdu_len(struct minstrel_ht_sta *mi)
|
||||
{
|
||||
- if (!mi->avg_ampdu_len)
|
||||
- return AVG_AMPDU_SIZE;
|
||||
+ int duration;
|
||||
|
||||
- return MINSTREL_TRUNC(mi->avg_ampdu_len);
|
||||
+ if (mi->avg_ampdu_len)
|
||||
+ return MINSTREL_TRUNC(mi->avg_ampdu_len);
|
||||
+
|
||||
+ if (minstrel_ht_is_legacy_group(mi->max_tp_rate[0] / MCS_GROUP_RATES))
|
||||
+ return 1;
|
||||
+
|
||||
+ duration = minstrel_get_duration(mi->max_tp_rate[0]);
|
||||
+
|
||||
+ if (duration > 400 * 1000)
|
||||
+ return 2;
|
||||
+
|
||||
+ if (duration > 250 * 1000)
|
||||
+ return 4;
|
||||
+
|
||||
+ if (duration > 150 * 1000)
|
||||
+ return 8;
|
||||
+
|
||||
+ return 16;
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -588,14 +612,6 @@ minstrel_ht_prob_rate_reduce_streams(str
|
||||
}
|
||||
}
|
||||
|
||||
-static inline int
|
||||
-minstrel_get_duration(int index)
|
||||
-{
|
||||
- const struct mcs_group *group = &minstrel_mcs_groups[index / MCS_GROUP_RATES];
|
||||
- unsigned int duration = group->duration[index % MCS_GROUP_RATES];
|
||||
- return duration << group->shift;
|
||||
-}
|
||||
-
|
||||
static bool
|
||||
minstrel_ht_probe_group(struct minstrel_ht_sta *mi, const struct mcs_group *tp_group,
|
||||
int tp_idx, const struct mcs_group *group)
|
@ -0,0 +1,31 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Sat, 26 Dec 2020 19:12:22 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: improve sample rate selection
|
||||
|
||||
Always allow sampling of rates faster than the primary max throughput rate.
|
||||
When the second max_tp_rate is higher than the first one, sample attempts were
|
||||
previously skipped, potentially causing rate control to get stuck at a slightly
|
||||
lower rate
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -1379,13 +1379,13 @@ minstrel_get_sample_rate(struct minstrel
|
||||
mrs = &mg->rates[sample_idx];
|
||||
sample_idx += sample_group * MCS_GROUP_RATES;
|
||||
|
||||
- /* Set tp_rate1, tp_rate2 to the highest / second highest max_tp_rate */
|
||||
+ tp_rate1 = mi->max_tp_rate[0];
|
||||
+
|
||||
+ /* Set tp_rate2 to the second highest max_tp_rate */
|
||||
if (minstrel_get_duration(mi->max_tp_rate[0]) >
|
||||
minstrel_get_duration(mi->max_tp_rate[1])) {
|
||||
- tp_rate1 = mi->max_tp_rate[1];
|
||||
tp_rate2 = mi->max_tp_rate[0];
|
||||
} else {
|
||||
- tp_rate1 = mi->max_tp_rate[0];
|
||||
tp_rate2 = mi->max_tp_rate[1];
|
||||
}
|
||||
|
@ -0,0 +1,116 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Sat, 26 Dec 2020 19:09:08 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: fix max probability rate selection
|
||||
|
||||
- do not select rates faster than the max throughput rate if probability is lower
|
||||
- reset previous rate before sorting again
|
||||
|
||||
This ensures that the max prob rate gets set to a more reliable rate
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -495,12 +495,13 @@ minstrel_ht_sort_best_tp_rates(struct mi
|
||||
* Find and set the topmost probability rate per sta and per group
|
||||
*/
|
||||
static void
|
||||
-minstrel_ht_set_best_prob_rate(struct minstrel_ht_sta *mi, u16 index)
|
||||
+minstrel_ht_set_best_prob_rate(struct minstrel_ht_sta *mi, u16 *dest, u16 index)
|
||||
{
|
||||
struct minstrel_mcs_group_data *mg;
|
||||
struct minstrel_rate_stats *mrs;
|
||||
int tmp_group, tmp_idx, tmp_tp_avg, tmp_prob;
|
||||
- int max_tp_group, cur_tp_avg, cur_group, cur_idx;
|
||||
+ int max_tp_group, max_tp_idx, max_tp_prob;
|
||||
+ int cur_tp_avg, cur_group, cur_idx;
|
||||
int max_gpr_group, max_gpr_idx;
|
||||
int max_gpr_tp_avg, max_gpr_prob;
|
||||
|
||||
@@ -509,18 +510,26 @@ minstrel_ht_set_best_prob_rate(struct mi
|
||||
mg = &mi->groups[index / MCS_GROUP_RATES];
|
||||
mrs = &mg->rates[index % MCS_GROUP_RATES];
|
||||
|
||||
- tmp_group = mi->max_prob_rate / MCS_GROUP_RATES;
|
||||
- tmp_idx = mi->max_prob_rate % MCS_GROUP_RATES;
|
||||
+ tmp_group = *dest / MCS_GROUP_RATES;
|
||||
+ tmp_idx = *dest % MCS_GROUP_RATES;
|
||||
tmp_prob = mi->groups[tmp_group].rates[tmp_idx].prob_avg;
|
||||
tmp_tp_avg = minstrel_ht_get_tp_avg(mi, tmp_group, tmp_idx, tmp_prob);
|
||||
|
||||
/* if max_tp_rate[0] is from MCS_GROUP max_prob_rate get selected from
|
||||
* MCS_GROUP as well as CCK_GROUP rates do not allow aggregation */
|
||||
max_tp_group = mi->max_tp_rate[0] / MCS_GROUP_RATES;
|
||||
+ max_tp_idx = mi->max_tp_rate[0] % MCS_GROUP_RATES;
|
||||
+ max_tp_prob = mi->groups[max_tp_group].rates[max_tp_idx].prob_avg;
|
||||
+
|
||||
if (minstrel_ht_is_legacy_group(index / MCS_GROUP_RATES) &&
|
||||
!minstrel_ht_is_legacy_group(max_tp_group))
|
||||
return;
|
||||
|
||||
+ /* skip rates faster than max tp rate with lower prob */
|
||||
+ if (minstrel_get_duration(mi->max_tp_rate[0]) > minstrel_get_duration(index) &&
|
||||
+ mrs->prob_avg < max_tp_prob)
|
||||
+ return;
|
||||
+
|
||||
max_gpr_group = mg->max_group_prob_rate / MCS_GROUP_RATES;
|
||||
max_gpr_idx = mg->max_group_prob_rate % MCS_GROUP_RATES;
|
||||
max_gpr_prob = mi->groups[max_gpr_group].rates[max_gpr_idx].prob_avg;
|
||||
@@ -538,7 +547,7 @@ minstrel_ht_set_best_prob_rate(struct mi
|
||||
mg->max_group_prob_rate = index;
|
||||
} else {
|
||||
if (mrs->prob_avg > tmp_prob)
|
||||
- mi->max_prob_rate = index;
|
||||
+ *dest = index;
|
||||
if (mrs->prob_avg > max_gpr_prob)
|
||||
mg->max_group_prob_rate = index;
|
||||
}
|
||||
@@ -816,7 +825,8 @@ minstrel_ht_update_stats(struct minstrel
|
||||
struct minstrel_rate_stats *mrs;
|
||||
int group, i, j, cur_prob;
|
||||
u16 tmp_mcs_tp_rate[MAX_THR_RATES], tmp_group_tp_rate[MAX_THR_RATES];
|
||||
- u16 tmp_legacy_tp_rate[MAX_THR_RATES], index;
|
||||
+ u16 tmp_legacy_tp_rate[MAX_THR_RATES], tmp_max_prob_rate;
|
||||
+ u16 index;
|
||||
bool ht_supported = mi->sta->ht_cap.ht_supported;
|
||||
|
||||
mi->sample_mode = MINSTREL_SAMPLE_IDLE;
|
||||
@@ -903,9 +913,6 @@ minstrel_ht_update_stats(struct minstrel
|
||||
/* Find max throughput rate set within a group */
|
||||
minstrel_ht_sort_best_tp_rates(mi, index,
|
||||
tmp_group_tp_rate);
|
||||
-
|
||||
- /* Find max probability rate per group and global */
|
||||
- minstrel_ht_set_best_prob_rate(mi, index);
|
||||
}
|
||||
|
||||
memcpy(mg->max_group_tp_rate, tmp_group_tp_rate,
|
||||
@@ -917,6 +924,27 @@ minstrel_ht_update_stats(struct minstrel
|
||||
tmp_legacy_tp_rate);
|
||||
memcpy(mi->max_tp_rate, tmp_mcs_tp_rate, sizeof(mi->max_tp_rate));
|
||||
|
||||
+ for (group = 0; group < ARRAY_SIZE(minstrel_mcs_groups); group++) {
|
||||
+ if (!mi->supported[group])
|
||||
+ continue;
|
||||
+
|
||||
+ mg = &mi->groups[group];
|
||||
+ mg->max_group_prob_rate = MCS_GROUP_RATES * group;
|
||||
+
|
||||
+ for (i = 0; i < MCS_GROUP_RATES; i++) {
|
||||
+ if (!(mi->supported[group] & BIT(i)))
|
||||
+ continue;
|
||||
+
|
||||
+ index = MCS_GROUP_RATES * group + i;
|
||||
+
|
||||
+ /* Find max probability rate per group and global */
|
||||
+ minstrel_ht_set_best_prob_rate(mi, &tmp_max_prob_rate,
|
||||
+ index);
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ mi->max_prob_rate = tmp_max_prob_rate;
|
||||
+
|
||||
/* Try to increase robustness of max_prob_rate*/
|
||||
minstrel_ht_prob_rate_reduce_streams(mi);
|
||||
|
@ -0,0 +1,20 @@
|
||||
From: Felix Fietkau <nbd@nbd.name>
|
||||
Date: Sat, 26 Dec 2020 19:14:58 +0100
|
||||
Subject: [PATCH] mac80211: minstrel_ht: increase stats update interval
|
||||
|
||||
The shorter interval was leading to too many frames being used for probing
|
||||
|
||||
Signed-off-by: Felix Fietkau <nbd@nbd.name>
|
||||
---
|
||||
|
||||
--- a/net/mac80211/rc80211_minstrel_ht.c
|
||||
+++ b/net/mac80211/rc80211_minstrel_ht.c
|
||||
@@ -1864,7 +1864,7 @@ minstrel_ht_alloc(struct ieee80211_hw *h
|
||||
mp->has_mrr = true;
|
||||
|
||||
mp->hw = hw;
|
||||
- mp->update_interval = HZ / 20;
|
||||
+ mp->update_interval = HZ / 10;
|
||||
|
||||
minstrel_ht_init_cck_rates(mp);
|
||||
for (i = 0; i < ARRAY_SIZE(mp->hw->wiphy->bands); i++)
|
Loading…
x
Reference in New Issue
Block a user