c0eb09aa7e
This was added to mitigate the effects of too much sampling on devices that use a static global fallback table instead of configurable multi-rate retry. Now that the sampling algorithm is improved, this code path no longer performs any better than the standard probing on affected devices. Signed-off-by: Felix Fietkau <nbd@nbd.name> Link: https://lore.kernel.org/r/20210127055735.78599-6-nbd@nbd.name Signed-off-by: Johannes Berg <johannes.berg@intel.com>
204 lines
5.0 KiB
C
204 lines
5.0 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
/*
|
|
* Copyright (C) 2010 Felix Fietkau <nbd@openwrt.org>
|
|
*/
|
|
|
|
#ifndef __RC_MINSTREL_HT_H
|
|
#define __RC_MINSTREL_HT_H
|
|
|
|
#include <linux/bitfield.h>
|
|
|
|
/* number of highest throughput rates to consider*/
|
|
#define MAX_THR_RATES 4
|
|
#define SAMPLE_COLUMNS 10 /* number of columns in sample table */
|
|
|
|
/* scaled fraction values */
|
|
#define MINSTREL_SCALE 12
|
|
#define MINSTREL_FRAC(val, div) (((val) << MINSTREL_SCALE) / div)
|
|
#define MINSTREL_TRUNC(val) ((val) >> MINSTREL_SCALE)
|
|
|
|
#define EWMA_LEVEL 96 /* ewma weighting factor [/EWMA_DIV] */
|
|
#define EWMA_DIV 128
|
|
|
|
/*
|
|
* Coefficients for moving average with noise filter (period=16),
|
|
* scaled by 10 bits
|
|
*
|
|
* a1 = exp(-pi * sqrt(2) / period)
|
|
* coeff2 = 2 * a1 * cos(sqrt(2) * 2 * pi / period)
|
|
* coeff3 = -sqr(a1)
|
|
* coeff1 = 1 - coeff2 - coeff3
|
|
*/
|
|
#define MINSTREL_AVG_COEFF1 (MINSTREL_FRAC(1, 1) - \
|
|
MINSTREL_AVG_COEFF2 - \
|
|
MINSTREL_AVG_COEFF3)
|
|
#define MINSTREL_AVG_COEFF2 0x00001499
|
|
#define MINSTREL_AVG_COEFF3 -0x0000092e
|
|
|
|
/*
|
|
* The number of streams can be changed to 2 to reduce code
|
|
* size and memory footprint.
|
|
*/
|
|
#define MINSTREL_MAX_STREAMS 4
|
|
#define MINSTREL_HT_STREAM_GROUPS 4 /* BW(=2) * SGI(=2) */
|
|
#define MINSTREL_VHT_STREAM_GROUPS 6 /* BW(=3) * SGI(=2) */
|
|
|
|
#define MINSTREL_HT_GROUPS_NB (MINSTREL_MAX_STREAMS * \
|
|
MINSTREL_HT_STREAM_GROUPS)
|
|
#define MINSTREL_VHT_GROUPS_NB (MINSTREL_MAX_STREAMS * \
|
|
MINSTREL_VHT_STREAM_GROUPS)
|
|
#define MINSTREL_LEGACY_GROUPS_NB 2
|
|
#define MINSTREL_GROUPS_NB (MINSTREL_HT_GROUPS_NB + \
|
|
MINSTREL_VHT_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_OFDM_GROUP (MINSTREL_CCK_GROUP + 1)
|
|
#define MINSTREL_VHT_GROUP_0 (MINSTREL_OFDM_GROUP + 1)
|
|
|
|
#define MCS_GROUP_RATES 10
|
|
|
|
#define MI_RATE_IDX_MASK GENMASK(3, 0)
|
|
#define MI_RATE_GROUP_MASK GENMASK(15, 4)
|
|
|
|
#define MI_RATE(_group, _idx) \
|
|
(FIELD_PREP(MI_RATE_GROUP_MASK, _group) | \
|
|
FIELD_PREP(MI_RATE_IDX_MASK, _idx))
|
|
|
|
#define MI_RATE_IDX(_rate) FIELD_GET(MI_RATE_IDX_MASK, _rate)
|
|
#define MI_RATE_GROUP(_rate) FIELD_GET(MI_RATE_GROUP_MASK, _rate)
|
|
|
|
#define MINSTREL_SAMPLE_RATES 5 /* rates per sample type */
|
|
#define MINSTREL_SAMPLE_INTERVAL (HZ / 50)
|
|
|
|
struct minstrel_priv {
|
|
struct ieee80211_hw *hw;
|
|
bool has_mrr;
|
|
unsigned int cw_min;
|
|
unsigned int cw_max;
|
|
unsigned int max_retry;
|
|
unsigned int segment_size;
|
|
unsigned int update_interval;
|
|
|
|
u8 cck_rates[4];
|
|
u8 ofdm_rates[NUM_NL80211_BANDS][8];
|
|
|
|
#ifdef CONFIG_MAC80211_DEBUGFS
|
|
/*
|
|
* enable fixed rate processing per RC
|
|
* - write static index to debugfs:ieee80211/phyX/rc/fixed_rate_idx
|
|
* - write -1 to enable RC processing again
|
|
* - setting will be applied on next update
|
|
*/
|
|
u32 fixed_rate_idx;
|
|
#endif
|
|
};
|
|
|
|
|
|
struct mcs_group {
|
|
u16 flags;
|
|
u8 streams;
|
|
u8 shift;
|
|
u8 bw;
|
|
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_rate_stats {
|
|
/* current / last sampling period attempts/success counters */
|
|
u16 attempts, last_attempts;
|
|
u16 success, last_success;
|
|
|
|
/* total attempts/success counters */
|
|
u32 att_hist, succ_hist;
|
|
|
|
/* prob_avg - moving average of prob */
|
|
u16 prob_avg;
|
|
u16 prob_avg_1;
|
|
|
|
/* maximum retry counts */
|
|
u8 retry_count;
|
|
u8 retry_count_rtscts;
|
|
|
|
bool retry_updated;
|
|
};
|
|
|
|
enum minstrel_sample_type {
|
|
MINSTREL_SAMPLE_TYPE_INC,
|
|
MINSTREL_SAMPLE_TYPE_JUMP,
|
|
MINSTREL_SAMPLE_TYPE_SLOW,
|
|
__MINSTREL_SAMPLE_TYPE_MAX
|
|
};
|
|
|
|
struct minstrel_mcs_group_data {
|
|
u8 index;
|
|
u8 column;
|
|
|
|
/* sorted rate set within a MCS group*/
|
|
u16 max_group_tp_rate[MAX_THR_RATES];
|
|
u16 max_group_prob_rate;
|
|
|
|
/* MCS rate statistics */
|
|
struct minstrel_rate_stats rates[MCS_GROUP_RATES];
|
|
};
|
|
|
|
struct minstrel_sample_category {
|
|
u8 sample_group;
|
|
u16 sample_rates[MINSTREL_SAMPLE_RATES];
|
|
u16 cur_sample_rates[MINSTREL_SAMPLE_RATES];
|
|
};
|
|
|
|
struct minstrel_ht_sta {
|
|
struct ieee80211_sta *sta;
|
|
|
|
/* ampdu length (average, per sampling interval) */
|
|
unsigned int ampdu_len;
|
|
unsigned int ampdu_packets;
|
|
|
|
/* ampdu length (EWMA) */
|
|
unsigned int avg_ampdu_len;
|
|
|
|
/* overall sorted rate set */
|
|
u16 max_tp_rate[MAX_THR_RATES];
|
|
u16 max_prob_rate;
|
|
|
|
/* time of last status update */
|
|
unsigned long last_stats_update;
|
|
|
|
/* 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;
|
|
unsigned int sample_packets;
|
|
|
|
/* tx flags to add for frames for this sta */
|
|
u32 tx_flags;
|
|
|
|
u8 band;
|
|
|
|
u8 sample_seq;
|
|
u16 sample_rate;
|
|
|
|
unsigned long sample_time;
|
|
struct minstrel_sample_category sample[__MINSTREL_SAMPLE_TYPE_MAX];
|
|
|
|
/* Bitfield of supported MCS rates of all groups */
|
|
u16 supported[MINSTREL_GROUPS_NB];
|
|
|
|
/* MCS rate group info and statistics */
|
|
struct minstrel_mcs_group_data groups[MINSTREL_GROUPS_NB];
|
|
};
|
|
|
|
void minstrel_ht_add_sta_debugfs(void *priv, void *priv_sta, struct dentry *dir);
|
|
int minstrel_ht_get_tp_avg(struct minstrel_ht_sta *mi, int group, int rate,
|
|
int prob_avg);
|
|
|
|
#endif
|