1 // SPDX-License-Identifier: ISC
3 * Copyright (C) 2019 Felix Fietkau <nbd@nbd.name>
4 * Copyright (C) 2021-2022 Intel Corporation
7 #include <net/mac80211.h>
8 #include "ieee80211_i.h"
11 #define AVG_PKT_SIZE 1024
13 /* Number of bits for an average sized packet */
14 #define MCS_NBITS (AVG_PKT_SIZE << 3)
16 /* Number of kilo-symbols (symbols * 1024) for a packet with (bps) bits per
17 * symbol. We use k-symbols to avoid rounding in the _TIME macros below.
19 #define MCS_N_KSYMS(bps) DIV_ROUND_UP(MCS_NBITS << 10, (bps))
21 /* Transmission time (in 1024 * usec) for a packet containing (ksyms) * 1024
24 #define MCS_SYMBOL_TIME(sgi, ksyms) \
26 ((ksyms) * 4 * 18) / 20 : /* 3.6 us per sym */ \
27 ((ksyms) * 4) /* 4.0 us per sym */ \
30 /* Transmit duration for the raw data part of an average sized packet */
31 #define MCS_DURATION(streams, sgi, bps) \
32 ((u32)MCS_SYMBOL_TIME(sgi, MCS_N_KSYMS((streams) * (bps))))
34 #define MCS_DURATION_S(shift, streams, sgi, bps) \
35 ((u16)((MCS_DURATION(streams, sgi, bps) >> shift)))
37 /* These should match the values in enum nl80211_he_gi */
42 /* Transmission time (1024 usec) for a packet containing (ksyms) * k-symbols */
43 #define HE_SYMBOL_TIME(gi, ksyms) \
45 ((ksyms) * 16 * 17) / 20 : /* 13.6 us per sym */ \
47 ((ksyms) * 16 * 18) / 20 : /* 14.4 us per sym */ \
48 ((ksyms) * 16) /* 16.0 us per sym */ \
51 /* Transmit duration for the raw data part of an average sized packet */
52 #define HE_DURATION(streams, gi, bps) \
53 ((u32)HE_SYMBOL_TIME(gi, MCS_N_KSYMS((streams) * (bps))))
55 #define HE_DURATION_S(shift, streams, gi, bps) \
56 (HE_DURATION(streams, gi, bps) >> shift)
58 /* gi in HE/EHT is identical. It matches enum nl80211_eht_gi as well */
59 #define EHT_GI_08 HE_GI_08
60 #define EHT_GI_16 HE_GI_16
61 #define EHT_GI_32 HE_GI_32
63 #define EHT_DURATION(streams, gi, bps) \
64 HE_DURATION(streams, gi, bps)
65 #define EHT_DURATION_S(shift, streams, gi, bps) \
66 HE_DURATION_S(shift, streams, gi, bps)
75 * Define group sort order: HT40 -> SGI -> #streams
77 #define IEEE80211_MAX_STREAMS 4
78 #define IEEE80211_HT_STREAM_GROUPS 4 /* BW(=2) * SGI(=2) */
79 #define IEEE80211_VHT_STREAM_GROUPS 8 /* BW(=4) * SGI(=2) */
81 #define IEEE80211_HE_MAX_STREAMS 8
82 #define IEEE80211_HE_STREAM_GROUPS 12 /* BW(=4) * GI(=3) */
84 #define IEEE80211_EHT_MAX_STREAMS 8
85 #define IEEE80211_EHT_STREAM_GROUPS 15 /* BW(=5) * GI(=3) */
87 #define IEEE80211_HT_GROUPS_NB (IEEE80211_MAX_STREAMS * \
88 IEEE80211_HT_STREAM_GROUPS)
89 #define IEEE80211_VHT_GROUPS_NB (IEEE80211_MAX_STREAMS * \
90 IEEE80211_VHT_STREAM_GROUPS)
91 #define IEEE80211_HE_GROUPS_NB (IEEE80211_HE_MAX_STREAMS * \
92 IEEE80211_HE_STREAM_GROUPS)
93 #define IEEE80211_EHT_GROUPS_NB (IEEE80211_EHT_MAX_STREAMS * \
94 IEEE80211_EHT_STREAM_GROUPS)
96 #define IEEE80211_HT_GROUP_0 0
97 #define IEEE80211_VHT_GROUP_0 (IEEE80211_HT_GROUP_0 + IEEE80211_HT_GROUPS_NB)
98 #define IEEE80211_HE_GROUP_0 (IEEE80211_VHT_GROUP_0 + IEEE80211_VHT_GROUPS_NB)
99 #define IEEE80211_EHT_GROUP_0 (IEEE80211_HE_GROUP_0 + IEEE80211_HE_GROUPS_NB)
101 #define MCS_GROUP_RATES 14
103 #define HT_GROUP_IDX(_streams, _sgi, _ht40) \
104 IEEE80211_HT_GROUP_0 + \
105 IEEE80211_MAX_STREAMS * 2 * _ht40 + \
106 IEEE80211_MAX_STREAMS * _sgi + \
109 #define _MAX(a, b) (((a)>(b))?(a):(b))
111 #define GROUP_SHIFT(duration) \
112 _MAX(0, 16 - __builtin_clz(duration))
114 /* MCS rate information for an MCS group */
115 #define __MCS_GROUP(_streams, _sgi, _ht40, _s) \
116 [HT_GROUP_IDX(_streams, _sgi, _ht40)] = { \
119 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 54 : 26), \
120 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 108 : 52), \
121 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 162 : 78), \
122 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 216 : 104), \
123 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 324 : 156), \
124 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 432 : 208), \
125 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 486 : 234), \
126 MCS_DURATION_S(_s, _streams, _sgi, _ht40 ? 540 : 260) \
130 #define MCS_GROUP_SHIFT(_streams, _sgi, _ht40) \
131 GROUP_SHIFT(MCS_DURATION(_streams, _sgi, _ht40 ? 54 : 26))
133 #define MCS_GROUP(_streams, _sgi, _ht40) \
134 __MCS_GROUP(_streams, _sgi, _ht40, \
135 MCS_GROUP_SHIFT(_streams, _sgi, _ht40))
137 #define VHT_GROUP_IDX(_streams, _sgi, _bw) \
138 (IEEE80211_VHT_GROUP_0 + \
139 IEEE80211_MAX_STREAMS * 2 * (_bw) + \
140 IEEE80211_MAX_STREAMS * (_sgi) + \
143 #define BW2VBPS(_bw, r4, r3, r2, r1) \
144 (_bw == BW_160 ? r4 : _bw == BW_80 ? r3 : _bw == BW_40 ? r2 : r1)
146 #define __VHT_GROUP(_streams, _sgi, _bw, _s) \
147 [VHT_GROUP_IDX(_streams, _sgi, _bw)] = { \
150 MCS_DURATION_S(_s, _streams, _sgi, \
151 BW2VBPS(_bw, 234, 117, 54, 26)), \
152 MCS_DURATION_S(_s, _streams, _sgi, \
153 BW2VBPS(_bw, 468, 234, 108, 52)), \
154 MCS_DURATION_S(_s, _streams, _sgi, \
155 BW2VBPS(_bw, 702, 351, 162, 78)), \
156 MCS_DURATION_S(_s, _streams, _sgi, \
157 BW2VBPS(_bw, 936, 468, 216, 104)), \
158 MCS_DURATION_S(_s, _streams, _sgi, \
159 BW2VBPS(_bw, 1404, 702, 324, 156)), \
160 MCS_DURATION_S(_s, _streams, _sgi, \
161 BW2VBPS(_bw, 1872, 936, 432, 208)), \
162 MCS_DURATION_S(_s, _streams, _sgi, \
163 BW2VBPS(_bw, 2106, 1053, 486, 234)), \
164 MCS_DURATION_S(_s, _streams, _sgi, \
165 BW2VBPS(_bw, 2340, 1170, 540, 260)), \
166 MCS_DURATION_S(_s, _streams, _sgi, \
167 BW2VBPS(_bw, 2808, 1404, 648, 312)), \
168 MCS_DURATION_S(_s, _streams, _sgi, \
169 BW2VBPS(_bw, 3120, 1560, 720, 346)) \
173 #define VHT_GROUP_SHIFT(_streams, _sgi, _bw) \
174 GROUP_SHIFT(MCS_DURATION(_streams, _sgi, \
175 BW2VBPS(_bw, 243, 117, 54, 26)))
177 #define VHT_GROUP(_streams, _sgi, _bw) \
178 __VHT_GROUP(_streams, _sgi, _bw, \
179 VHT_GROUP_SHIFT(_streams, _sgi, _bw))
182 #define HE_GROUP_IDX(_streams, _gi, _bw) \
183 (IEEE80211_HE_GROUP_0 + \
184 IEEE80211_HE_MAX_STREAMS * 3 * (_bw) + \
185 IEEE80211_HE_MAX_STREAMS * (_gi) + \
188 #define __HE_GROUP(_streams, _gi, _bw, _s) \
189 [HE_GROUP_IDX(_streams, _gi, _bw)] = { \
192 HE_DURATION_S(_s, _streams, _gi, \
193 BW2VBPS(_bw, 979, 489, 230, 115)), \
194 HE_DURATION_S(_s, _streams, _gi, \
195 BW2VBPS(_bw, 1958, 979, 475, 230)), \
196 HE_DURATION_S(_s, _streams, _gi, \
197 BW2VBPS(_bw, 2937, 1468, 705, 345)), \
198 HE_DURATION_S(_s, _streams, _gi, \
199 BW2VBPS(_bw, 3916, 1958, 936, 475)), \
200 HE_DURATION_S(_s, _streams, _gi, \
201 BW2VBPS(_bw, 5875, 2937, 1411, 705)), \
202 HE_DURATION_S(_s, _streams, _gi, \
203 BW2VBPS(_bw, 7833, 3916, 1872, 936)), \
204 HE_DURATION_S(_s, _streams, _gi, \
205 BW2VBPS(_bw, 8827, 4406, 2102, 1051)), \
206 HE_DURATION_S(_s, _streams, _gi, \
207 BW2VBPS(_bw, 9806, 4896, 2347, 1166)), \
208 HE_DURATION_S(_s, _streams, _gi, \
209 BW2VBPS(_bw, 11764, 5875, 2808, 1411)), \
210 HE_DURATION_S(_s, _streams, _gi, \
211 BW2VBPS(_bw, 13060, 6523, 3124, 1555)), \
212 HE_DURATION_S(_s, _streams, _gi, \
213 BW2VBPS(_bw, 14702, 7344, 3513, 1756)), \
214 HE_DURATION_S(_s, _streams, _gi, \
215 BW2VBPS(_bw, 16329, 8164, 3902, 1944)) \
219 #define HE_GROUP_SHIFT(_streams, _gi, _bw) \
220 GROUP_SHIFT(HE_DURATION(_streams, _gi, \
221 BW2VBPS(_bw, 979, 489, 230, 115)))
223 #define HE_GROUP(_streams, _gi, _bw) \
224 __HE_GROUP(_streams, _gi, _bw, \
225 HE_GROUP_SHIFT(_streams, _gi, _bw))
227 #define EHT_BW2VBPS(_bw, r5, r4, r3, r2, r1) \
228 ((_bw) == BW_320 ? r5 : BW2VBPS(_bw, r4, r3, r2, r1))
230 #define EHT_GROUP_IDX(_streams, _gi, _bw) \
231 (IEEE80211_EHT_GROUP_0 + \
232 IEEE80211_EHT_MAX_STREAMS * 3 * (_bw) + \
233 IEEE80211_EHT_MAX_STREAMS * (_gi) + \
236 #define __EHT_GROUP(_streams, _gi, _bw, _s) \
237 [EHT_GROUP_IDX(_streams, _gi, _bw)] = { \
240 EHT_DURATION_S(_s, _streams, _gi, \
241 EHT_BW2VBPS(_bw, 1960, 980, 490, 234, 117)), \
242 EHT_DURATION_S(_s, _streams, _gi, \
243 EHT_BW2VBPS(_bw, 3920, 1960, 980, 468, 234)), \
244 EHT_DURATION_S(_s, _streams, _gi, \
245 EHT_BW2VBPS(_bw, 5880, 2937, 1470, 702, 351)), \
246 EHT_DURATION_S(_s, _streams, _gi, \
247 EHT_BW2VBPS(_bw, 7840, 3920, 1960, 936, 468)), \
248 EHT_DURATION_S(_s, _streams, _gi, \
249 EHT_BW2VBPS(_bw, 11760, 5880, 2940, 1404, 702)), \
250 EHT_DURATION_S(_s, _streams, _gi, \
251 EHT_BW2VBPS(_bw, 15680, 7840, 3920, 1872, 936)), \
252 EHT_DURATION_S(_s, _streams, _gi, \
253 EHT_BW2VBPS(_bw, 17640, 8820, 4410, 2106, 1053)), \
254 EHT_DURATION_S(_s, _streams, _gi, \
255 EHT_BW2VBPS(_bw, 19600, 9800, 4900, 2340, 1170)), \
256 EHT_DURATION_S(_s, _streams, _gi, \
257 EHT_BW2VBPS(_bw, 23520, 11760, 5880, 2808, 1404)), \
258 EHT_DURATION_S(_s, _streams, _gi, \
259 EHT_BW2VBPS(_bw, 26133, 13066, 6533, 3120, 1560)), \
260 EHT_DURATION_S(_s, _streams, _gi, \
261 EHT_BW2VBPS(_bw, 29400, 14700, 7350, 3510, 1755)), \
262 EHT_DURATION_S(_s, _streams, _gi, \
263 EHT_BW2VBPS(_bw, 32666, 16333, 8166, 3900, 1950)), \
264 EHT_DURATION_S(_s, _streams, _gi, \
265 EHT_BW2VBPS(_bw, 35280, 17640, 8820, 4212, 2106)), \
266 EHT_DURATION_S(_s, _streams, _gi, \
267 EHT_BW2VBPS(_bw, 39200, 19600, 9800, 4680, 2340)) \
271 #define EHT_GROUP_SHIFT(_streams, _gi, _bw) \
272 GROUP_SHIFT(EHT_DURATION(_streams, _gi, \
273 EHT_BW2VBPS(_bw, 1960, 980, 490, 234, 117)))
275 #define EHT_GROUP(_streams, _gi, _bw) \
276 __EHT_GROUP(_streams, _gi, _bw, \
277 EHT_GROUP_SHIFT(_streams, _gi, _bw))
279 #define EHT_GROUP_RANGE(_gi, _bw) \
280 EHT_GROUP(1, _gi, _bw), \
281 EHT_GROUP(2, _gi, _bw), \
282 EHT_GROUP(3, _gi, _bw), \
283 EHT_GROUP(4, _gi, _bw), \
284 EHT_GROUP(5, _gi, _bw), \
285 EHT_GROUP(6, _gi, _bw), \
286 EHT_GROUP(7, _gi, _bw), \
287 EHT_GROUP(8, _gi, _bw)
291 u16 duration
[MCS_GROUP_RATES
];
294 static const struct mcs_group airtime_mcs_groups
[] = {
295 MCS_GROUP(1, 0, BW_20
),
296 MCS_GROUP(2, 0, BW_20
),
297 MCS_GROUP(3, 0, BW_20
),
298 MCS_GROUP(4, 0, BW_20
),
300 MCS_GROUP(1, 1, BW_20
),
301 MCS_GROUP(2, 1, BW_20
),
302 MCS_GROUP(3, 1, BW_20
),
303 MCS_GROUP(4, 1, BW_20
),
305 MCS_GROUP(1, 0, BW_40
),
306 MCS_GROUP(2, 0, BW_40
),
307 MCS_GROUP(3, 0, BW_40
),
308 MCS_GROUP(4, 0, BW_40
),
310 MCS_GROUP(1, 1, BW_40
),
311 MCS_GROUP(2, 1, BW_40
),
312 MCS_GROUP(3, 1, BW_40
),
313 MCS_GROUP(4, 1, BW_40
),
315 VHT_GROUP(1, 0, BW_20
),
316 VHT_GROUP(2, 0, BW_20
),
317 VHT_GROUP(3, 0, BW_20
),
318 VHT_GROUP(4, 0, BW_20
),
320 VHT_GROUP(1, 1, BW_20
),
321 VHT_GROUP(2, 1, BW_20
),
322 VHT_GROUP(3, 1, BW_20
),
323 VHT_GROUP(4, 1, BW_20
),
325 VHT_GROUP(1, 0, BW_40
),
326 VHT_GROUP(2, 0, BW_40
),
327 VHT_GROUP(3, 0, BW_40
),
328 VHT_GROUP(4, 0, BW_40
),
330 VHT_GROUP(1, 1, BW_40
),
331 VHT_GROUP(2, 1, BW_40
),
332 VHT_GROUP(3, 1, BW_40
),
333 VHT_GROUP(4, 1, BW_40
),
335 VHT_GROUP(1, 0, BW_80
),
336 VHT_GROUP(2, 0, BW_80
),
337 VHT_GROUP(3, 0, BW_80
),
338 VHT_GROUP(4, 0, BW_80
),
340 VHT_GROUP(1, 1, BW_80
),
341 VHT_GROUP(2, 1, BW_80
),
342 VHT_GROUP(3, 1, BW_80
),
343 VHT_GROUP(4, 1, BW_80
),
345 VHT_GROUP(1, 0, BW_160
),
346 VHT_GROUP(2, 0, BW_160
),
347 VHT_GROUP(3, 0, BW_160
),
348 VHT_GROUP(4, 0, BW_160
),
350 VHT_GROUP(1, 1, BW_160
),
351 VHT_GROUP(2, 1, BW_160
),
352 VHT_GROUP(3, 1, BW_160
),
353 VHT_GROUP(4, 1, BW_160
),
355 HE_GROUP(1, HE_GI_08
, BW_20
),
356 HE_GROUP(2, HE_GI_08
, BW_20
),
357 HE_GROUP(3, HE_GI_08
, BW_20
),
358 HE_GROUP(4, HE_GI_08
, BW_20
),
359 HE_GROUP(5, HE_GI_08
, BW_20
),
360 HE_GROUP(6, HE_GI_08
, BW_20
),
361 HE_GROUP(7, HE_GI_08
, BW_20
),
362 HE_GROUP(8, HE_GI_08
, BW_20
),
364 HE_GROUP(1, HE_GI_16
, BW_20
),
365 HE_GROUP(2, HE_GI_16
, BW_20
),
366 HE_GROUP(3, HE_GI_16
, BW_20
),
367 HE_GROUP(4, HE_GI_16
, BW_20
),
368 HE_GROUP(5, HE_GI_16
, BW_20
),
369 HE_GROUP(6, HE_GI_16
, BW_20
),
370 HE_GROUP(7, HE_GI_16
, BW_20
),
371 HE_GROUP(8, HE_GI_16
, BW_20
),
373 HE_GROUP(1, HE_GI_32
, BW_20
),
374 HE_GROUP(2, HE_GI_32
, BW_20
),
375 HE_GROUP(3, HE_GI_32
, BW_20
),
376 HE_GROUP(4, HE_GI_32
, BW_20
),
377 HE_GROUP(5, HE_GI_32
, BW_20
),
378 HE_GROUP(6, HE_GI_32
, BW_20
),
379 HE_GROUP(7, HE_GI_32
, BW_20
),
380 HE_GROUP(8, HE_GI_32
, BW_20
),
382 HE_GROUP(1, HE_GI_08
, BW_40
),
383 HE_GROUP(2, HE_GI_08
, BW_40
),
384 HE_GROUP(3, HE_GI_08
, BW_40
),
385 HE_GROUP(4, HE_GI_08
, BW_40
),
386 HE_GROUP(5, HE_GI_08
, BW_40
),
387 HE_GROUP(6, HE_GI_08
, BW_40
),
388 HE_GROUP(7, HE_GI_08
, BW_40
),
389 HE_GROUP(8, HE_GI_08
, BW_40
),
391 HE_GROUP(1, HE_GI_16
, BW_40
),
392 HE_GROUP(2, HE_GI_16
, BW_40
),
393 HE_GROUP(3, HE_GI_16
, BW_40
),
394 HE_GROUP(4, HE_GI_16
, BW_40
),
395 HE_GROUP(5, HE_GI_16
, BW_40
),
396 HE_GROUP(6, HE_GI_16
, BW_40
),
397 HE_GROUP(7, HE_GI_16
, BW_40
),
398 HE_GROUP(8, HE_GI_16
, BW_40
),
400 HE_GROUP(1, HE_GI_32
, BW_40
),
401 HE_GROUP(2, HE_GI_32
, BW_40
),
402 HE_GROUP(3, HE_GI_32
, BW_40
),
403 HE_GROUP(4, HE_GI_32
, BW_40
),
404 HE_GROUP(5, HE_GI_32
, BW_40
),
405 HE_GROUP(6, HE_GI_32
, BW_40
),
406 HE_GROUP(7, HE_GI_32
, BW_40
),
407 HE_GROUP(8, HE_GI_32
, BW_40
),
409 HE_GROUP(1, HE_GI_08
, BW_80
),
410 HE_GROUP(2, HE_GI_08
, BW_80
),
411 HE_GROUP(3, HE_GI_08
, BW_80
),
412 HE_GROUP(4, HE_GI_08
, BW_80
),
413 HE_GROUP(5, HE_GI_08
, BW_80
),
414 HE_GROUP(6, HE_GI_08
, BW_80
),
415 HE_GROUP(7, HE_GI_08
, BW_80
),
416 HE_GROUP(8, HE_GI_08
, BW_80
),
418 HE_GROUP(1, HE_GI_16
, BW_80
),
419 HE_GROUP(2, HE_GI_16
, BW_80
),
420 HE_GROUP(3, HE_GI_16
, BW_80
),
421 HE_GROUP(4, HE_GI_16
, BW_80
),
422 HE_GROUP(5, HE_GI_16
, BW_80
),
423 HE_GROUP(6, HE_GI_16
, BW_80
),
424 HE_GROUP(7, HE_GI_16
, BW_80
),
425 HE_GROUP(8, HE_GI_16
, BW_80
),
427 HE_GROUP(1, HE_GI_32
, BW_80
),
428 HE_GROUP(2, HE_GI_32
, BW_80
),
429 HE_GROUP(3, HE_GI_32
, BW_80
),
430 HE_GROUP(4, HE_GI_32
, BW_80
),
431 HE_GROUP(5, HE_GI_32
, BW_80
),
432 HE_GROUP(6, HE_GI_32
, BW_80
),
433 HE_GROUP(7, HE_GI_32
, BW_80
),
434 HE_GROUP(8, HE_GI_32
, BW_80
),
436 HE_GROUP(1, HE_GI_08
, BW_160
),
437 HE_GROUP(2, HE_GI_08
, BW_160
),
438 HE_GROUP(3, HE_GI_08
, BW_160
),
439 HE_GROUP(4, HE_GI_08
, BW_160
),
440 HE_GROUP(5, HE_GI_08
, BW_160
),
441 HE_GROUP(6, HE_GI_08
, BW_160
),
442 HE_GROUP(7, HE_GI_08
, BW_160
),
443 HE_GROUP(8, HE_GI_08
, BW_160
),
445 HE_GROUP(1, HE_GI_16
, BW_160
),
446 HE_GROUP(2, HE_GI_16
, BW_160
),
447 HE_GROUP(3, HE_GI_16
, BW_160
),
448 HE_GROUP(4, HE_GI_16
, BW_160
),
449 HE_GROUP(5, HE_GI_16
, BW_160
),
450 HE_GROUP(6, HE_GI_16
, BW_160
),
451 HE_GROUP(7, HE_GI_16
, BW_160
),
452 HE_GROUP(8, HE_GI_16
, BW_160
),
454 HE_GROUP(1, HE_GI_32
, BW_160
),
455 HE_GROUP(2, HE_GI_32
, BW_160
),
456 HE_GROUP(3, HE_GI_32
, BW_160
),
457 HE_GROUP(4, HE_GI_32
, BW_160
),
458 HE_GROUP(5, HE_GI_32
, BW_160
),
459 HE_GROUP(6, HE_GI_32
, BW_160
),
460 HE_GROUP(7, HE_GI_32
, BW_160
),
461 HE_GROUP(8, HE_GI_32
, BW_160
),
463 EHT_GROUP_RANGE(EHT_GI_08
, BW_20
),
464 EHT_GROUP_RANGE(EHT_GI_16
, BW_20
),
465 EHT_GROUP_RANGE(EHT_GI_32
, BW_20
),
467 EHT_GROUP_RANGE(EHT_GI_08
, BW_40
),
468 EHT_GROUP_RANGE(EHT_GI_16
, BW_40
),
469 EHT_GROUP_RANGE(EHT_GI_32
, BW_40
),
471 EHT_GROUP_RANGE(EHT_GI_08
, BW_80
),
472 EHT_GROUP_RANGE(EHT_GI_16
, BW_80
),
473 EHT_GROUP_RANGE(EHT_GI_32
, BW_80
),
475 EHT_GROUP_RANGE(EHT_GI_08
, BW_160
),
476 EHT_GROUP_RANGE(EHT_GI_16
, BW_160
),
477 EHT_GROUP_RANGE(EHT_GI_32
, BW_160
),
479 EHT_GROUP_RANGE(EHT_GI_08
, BW_320
),
480 EHT_GROUP_RANGE(EHT_GI_16
, BW_320
),
481 EHT_GROUP_RANGE(EHT_GI_32
, BW_320
),
485 ieee80211_calc_legacy_rate_duration(u16 bitrate
, bool short_pre
,
491 duration
= 144 + 48; /* preamble + PLCP */
495 duration
+= 10; /* SIFS */
497 duration
= 20 + 16; /* premable + SIFS */
501 duration
+= (len
* 10) / bitrate
;
506 static u32
ieee80211_get_rate_duration(struct ieee80211_hw
*hw
,
507 struct ieee80211_rx_status
*status
,
510 bool sgi
= status
->enc_flags
& RX_ENC_FLAG_SHORT_GI
;
515 switch (status
->bw
) {
516 case RATE_INFO_BW_20
:
519 case RATE_INFO_BW_40
:
522 case RATE_INFO_BW_80
:
525 case RATE_INFO_BW_160
:
528 case RATE_INFO_BW_320
:
536 switch (status
->encoding
) {
538 streams
= status
->nss
;
539 idx
= status
->rate_idx
;
540 group
= VHT_GROUP_IDX(streams
, sgi
, bw
);
543 streams
= ((status
->rate_idx
>> 3) & 3) + 1;
544 idx
= status
->rate_idx
& 7;
545 group
= HT_GROUP_IDX(streams
, sgi
, bw
);
548 streams
= status
->nss
;
549 idx
= status
->rate_idx
;
550 group
= HE_GROUP_IDX(streams
, status
->he_gi
, bw
);
553 streams
= status
->nss
;
554 idx
= status
->rate_idx
;
555 group
= EHT_GROUP_IDX(streams
, status
->eht
.gi
, bw
);
562 switch (status
->encoding
) {
565 if (WARN_ON_ONCE(streams
> 8))
569 if (WARN_ON_ONCE(streams
> 4))
574 if (idx
>= MCS_GROUP_RATES
)
577 duration
= airtime_mcs_groups
[group
].duration
[idx
];
578 duration
<<= airtime_mcs_groups
[group
].shift
;
579 *overhead
= 36 + (streams
<< 2);
585 u32
ieee80211_calc_rx_airtime(struct ieee80211_hw
*hw
,
586 struct ieee80211_rx_status
*status
,
589 struct ieee80211_supported_band
*sband
;
590 u32 duration
, overhead
= 0;
592 if (status
->encoding
== RX_ENC_LEGACY
) {
593 const struct ieee80211_rate
*rate
;
594 bool sp
= status
->enc_flags
& RX_ENC_FLAG_SHORTPRE
;
597 /* on 60GHz or sub-1GHz band, there are no legacy rates */
598 if (WARN_ON_ONCE(status
->band
== NL80211_BAND_60GHZ
||
599 status
->band
== NL80211_BAND_S1GHZ
))
602 sband
= hw
->wiphy
->bands
[status
->band
];
603 if (!sband
|| status
->rate_idx
>= sband
->n_bitrates
)
606 rate
= &sband
->bitrates
[status
->rate_idx
];
607 cck
= rate
->flags
& IEEE80211_RATE_MANDATORY_B
;
609 return ieee80211_calc_legacy_rate_duration(rate
->bitrate
, sp
,
613 duration
= ieee80211_get_rate_duration(hw
, status
, &overhead
);
618 duration
/= AVG_PKT_SIZE
;
621 return duration
+ overhead
;
623 EXPORT_SYMBOL_GPL(ieee80211_calc_rx_airtime
);
625 static bool ieee80211_fill_rate_info(struct ieee80211_hw
*hw
,
626 struct ieee80211_rx_status
*stat
, u8 band
,
627 struct rate_info
*ri
)
629 struct ieee80211_supported_band
*sband
= hw
->wiphy
->bands
[band
];
637 stat
->rate_idx
= ri
->mcs
;
639 if (ri
->flags
& RATE_INFO_FLAGS_EHT_MCS
)
640 stat
->encoding
= RX_ENC_EHT
;
641 else if (ri
->flags
& RATE_INFO_FLAGS_HE_MCS
)
642 stat
->encoding
= RX_ENC_HE
;
643 else if (ri
->flags
& RATE_INFO_FLAGS_VHT_MCS
)
644 stat
->encoding
= RX_ENC_VHT
;
645 else if (ri
->flags
& RATE_INFO_FLAGS_MCS
)
646 stat
->encoding
= RX_ENC_HT
;
648 stat
->encoding
= RX_ENC_LEGACY
;
650 if (ri
->flags
& RATE_INFO_FLAGS_SHORT_GI
)
651 stat
->enc_flags
|= RX_ENC_FLAG_SHORT_GI
;
653 switch (stat
->encoding
) {
655 stat
->eht
.gi
= ri
->eht_gi
;
658 stat
->he_gi
= ri
->he_gi
;
662 if (stat
->encoding
!= RX_ENC_LEGACY
)
666 for (i
= 0; i
< sband
->n_bitrates
; i
++) {
667 if (ri
->legacy
!= sband
->bitrates
[i
].bitrate
)
677 static int ieee80211_fill_rx_status(struct ieee80211_rx_status
*stat
,
678 struct ieee80211_hw
*hw
,
679 struct ieee80211_tx_rate
*rate
,
680 struct rate_info
*ri
, u8 band
, int len
)
682 memset(stat
, 0, sizeof(*stat
));
685 if (ieee80211_fill_rate_info(hw
, stat
, band
, ri
))
688 if (!ieee80211_rate_valid(rate
))
691 if (rate
->flags
& IEEE80211_TX_RC_160_MHZ_WIDTH
)
692 stat
->bw
= RATE_INFO_BW_160
;
693 else if (rate
->flags
& IEEE80211_TX_RC_80_MHZ_WIDTH
)
694 stat
->bw
= RATE_INFO_BW_80
;
695 else if (rate
->flags
& IEEE80211_TX_RC_40_MHZ_WIDTH
)
696 stat
->bw
= RATE_INFO_BW_40
;
698 stat
->bw
= RATE_INFO_BW_20
;
701 if (rate
->flags
& IEEE80211_TX_RC_USE_SHORT_PREAMBLE
)
702 stat
->enc_flags
|= RX_ENC_FLAG_SHORTPRE
;
703 if (rate
->flags
& IEEE80211_TX_RC_SHORT_GI
)
704 stat
->enc_flags
|= RX_ENC_FLAG_SHORT_GI
;
706 stat
->rate_idx
= rate
->idx
;
707 if (rate
->flags
& IEEE80211_TX_RC_VHT_MCS
) {
708 stat
->encoding
= RX_ENC_VHT
;
709 stat
->rate_idx
= ieee80211_rate_get_vht_mcs(rate
);
710 stat
->nss
= ieee80211_rate_get_vht_nss(rate
);
711 } else if (rate
->flags
& IEEE80211_TX_RC_MCS
) {
712 stat
->encoding
= RX_ENC_HT
;
714 stat
->encoding
= RX_ENC_LEGACY
;
720 static u32
ieee80211_calc_tx_airtime_rate(struct ieee80211_hw
*hw
,
721 struct ieee80211_tx_rate
*rate
,
722 struct rate_info
*ri
,
725 struct ieee80211_rx_status stat
;
727 if (ieee80211_fill_rx_status(&stat
, hw
, rate
, ri
, band
, len
))
730 return ieee80211_calc_rx_airtime(hw
, &stat
, len
);
733 u32
ieee80211_calc_tx_airtime(struct ieee80211_hw
*hw
,
734 struct ieee80211_tx_info
*info
,
740 for (i
= 0; i
< ARRAY_SIZE(info
->status
.rates
); i
++) {
741 struct ieee80211_tx_rate
*rate
= &info
->status
.rates
[i
];
744 cur_duration
= ieee80211_calc_tx_airtime_rate(hw
, rate
, NULL
,
749 duration
+= cur_duration
* rate
->count
;
754 EXPORT_SYMBOL_GPL(ieee80211_calc_tx_airtime
);
756 u32
ieee80211_calc_expected_tx_airtime(struct ieee80211_hw
*hw
,
757 struct ieee80211_vif
*vif
,
758 struct ieee80211_sta
*pubsta
,
761 struct ieee80211_supported_band
*sband
;
762 struct ieee80211_chanctx_conf
*conf
;
764 bool cck
, short_pream
;
769 len
+= 38; /* Ethernet header length */
771 conf
= rcu_dereference(vif
->bss_conf
.chanctx_conf
);
773 band
= conf
->def
.chan
->band
;
776 struct sta_info
*sta
= container_of(pubsta
, struct sta_info
,
778 struct ieee80211_rx_status stat
;
779 struct ieee80211_tx_rate
*tx_rate
= &sta
->deflink
.tx_stats
.last_rate
;
780 struct rate_info
*ri
= &sta
->deflink
.tx_stats
.last_rate_info
;
781 u32 duration
, overhead
;
784 if (ieee80211_fill_rx_status(&stat
, hw
, tx_rate
, ri
, band
, len
))
787 if (stat
.encoding
== RX_ENC_LEGACY
|| !ampdu
)
788 return ieee80211_calc_rx_airtime(hw
, &stat
, len
);
790 duration
= ieee80211_get_rate_duration(hw
, &stat
, &overhead
);
792 * Assume that HT/VHT transmission on any AC except VO will
793 * use aggregation. Since we don't have reliable reporting
794 * of aggregation length, assume an average size based on the
796 * This will not be very accurate, but much better than simply
797 * assuming un-aggregated tx in all cases.
799 if (duration
> 400 * 1024) /* <= VHT20 MCS2 1S */
801 else if (duration
> 250 * 1024) /* <= VHT20 MCS3 1S or MCS1 2S */
803 else if (duration
> 150 * 1024) /* <= VHT20 MCS5 1S or MCS2 2S */
805 else if (duration
> 70 * 1024) /* <= VHT20 MCS5 2S */
807 else if (stat
.encoding
!= RX_ENC_HE
||
808 duration
> 20 * 1024) /* <= HE40 MCS6 2S */
814 duration
/= AVG_PKT_SIZE
;
816 duration
+= (overhead
>> agg_shift
);
818 return max_t(u32
, duration
, 4);
824 /* No station to get latest rate from, so calculate the worst-case
825 * duration using the lowest configured basic rate.
827 sband
= hw
->wiphy
->bands
[band
];
829 basic_rates
= vif
->bss_conf
.basic_rates
;
830 short_pream
= vif
->bss_conf
.use_short_preamble
;
832 rateidx
= basic_rates
? ffs(basic_rates
) - 1 : 0;
833 rate
= sband
->bitrates
[rateidx
].bitrate
;
834 cck
= sband
->bitrates
[rateidx
].flags
& IEEE80211_RATE_MANDATORY_B
;
836 return ieee80211_calc_legacy_rate_duration(rate
, short_pream
, cck
, len
);