blob: d60e50a3b910ccbad7b6d42b221813b9e6fd9932 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001// SPDX-License-Identifier: GPL-2.0
Johannes Berg59bbb6f2009-08-07 17:22:35 +02002/*
3 * This file contains helper code to handle channel
4 * settings and keeping track of what is possible at
5 * any point in time.
6 *
7 * Copyright 2009 Johannes Berg <johannes@sipsolutions.net>
Johannes Berg2740f0c2014-09-03 15:24:58 +03008 * Copyright 2013-2014 Intel Mobile Communications GmbH
Luca Coelho925b5972018-12-15 11:03:21 +02009 * Copyright 2018 Intel Corporation
Johannes Berg59bbb6f2009-08-07 17:22:35 +020010 */
11
Alexander Simon54858ee5b2011-11-30 16:56:32 +010012#include <linux/export.h>
Johannes Berg59bbb6f2009-08-07 17:22:35 +020013#include <net/cfg80211.h>
14#include "core.h"
Hila Gonene35e4d22012-06-27 17:19:42 +030015#include "rdev-ops.h"
Johannes Berg59bbb6f2009-08-07 17:22:35 +020016
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +030017static bool cfg80211_valid_60g_freq(u32 freq)
18{
19 return freq >= 58320 && freq <= 70200;
20}
21
Johannes Berg3d9d1d62012-11-08 23:14:50 +010022void cfg80211_chandef_create(struct cfg80211_chan_def *chandef,
23 struct ieee80211_channel *chan,
24 enum nl80211_channel_type chan_type)
25{
26 if (WARN_ON(!chan))
27 return;
28
29 chandef->chan = chan;
Thomas Pedersen934f4c72020-04-01 18:18:03 -070030 chandef->freq1_offset = chan->freq_offset;
Johannes Berg3d9d1d62012-11-08 23:14:50 +010031 chandef->center_freq2 = 0;
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +030032 chandef->edmg.bw_config = 0;
33 chandef->edmg.channels = 0;
Johannes Berg3d9d1d62012-11-08 23:14:50 +010034
35 switch (chan_type) {
36 case NL80211_CHAN_NO_HT:
37 chandef->width = NL80211_CHAN_WIDTH_20_NOHT;
38 chandef->center_freq1 = chan->center_freq;
39 break;
40 case NL80211_CHAN_HT20:
41 chandef->width = NL80211_CHAN_WIDTH_20;
42 chandef->center_freq1 = chan->center_freq;
43 break;
44 case NL80211_CHAN_HT40PLUS:
45 chandef->width = NL80211_CHAN_WIDTH_40;
46 chandef->center_freq1 = chan->center_freq + 10;
47 break;
48 case NL80211_CHAN_HT40MINUS:
49 chandef->width = NL80211_CHAN_WIDTH_40;
50 chandef->center_freq1 = chan->center_freq - 10;
51 break;
52 default:
53 WARN_ON(1);
54 }
55}
56EXPORT_SYMBOL(cfg80211_chandef_create);
57
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +030058static bool cfg80211_edmg_chandef_valid(const struct cfg80211_chan_def *chandef)
59{
60 int max_contiguous = 0;
61 int num_of_enabled = 0;
62 int contiguous = 0;
63 int i;
64
65 if (!chandef->edmg.channels || !chandef->edmg.bw_config)
66 return false;
67
68 if (!cfg80211_valid_60g_freq(chandef->chan->center_freq))
69 return false;
70
71 for (i = 0; i < 6; i++) {
72 if (chandef->edmg.channels & BIT(i)) {
73 contiguous++;
74 num_of_enabled++;
75 } else {
76 contiguous = 0;
77 }
78
79 max_contiguous = max(contiguous, max_contiguous);
80 }
81 /* basic verification of edmg configuration according to
82 * IEEE P802.11ay/D4.0 section 9.4.2.251
83 */
84 /* check bw_config against contiguous edmg channels */
85 switch (chandef->edmg.bw_config) {
86 case IEEE80211_EDMG_BW_CONFIG_4:
87 case IEEE80211_EDMG_BW_CONFIG_8:
88 case IEEE80211_EDMG_BW_CONFIG_12:
89 if (max_contiguous < 1)
90 return false;
91 break;
92 case IEEE80211_EDMG_BW_CONFIG_5:
93 case IEEE80211_EDMG_BW_CONFIG_9:
94 case IEEE80211_EDMG_BW_CONFIG_13:
95 if (max_contiguous < 2)
96 return false;
97 break;
98 case IEEE80211_EDMG_BW_CONFIG_6:
99 case IEEE80211_EDMG_BW_CONFIG_10:
100 case IEEE80211_EDMG_BW_CONFIG_14:
101 if (max_contiguous < 3)
102 return false;
103 break;
104 case IEEE80211_EDMG_BW_CONFIG_7:
105 case IEEE80211_EDMG_BW_CONFIG_11:
106 case IEEE80211_EDMG_BW_CONFIG_15:
107 if (max_contiguous < 4)
108 return false;
109 break;
110
111 default:
112 return false;
113 }
114
115 /* check bw_config against aggregated (non contiguous) edmg channels */
116 switch (chandef->edmg.bw_config) {
117 case IEEE80211_EDMG_BW_CONFIG_4:
118 case IEEE80211_EDMG_BW_CONFIG_5:
119 case IEEE80211_EDMG_BW_CONFIG_6:
120 case IEEE80211_EDMG_BW_CONFIG_7:
121 break;
122 case IEEE80211_EDMG_BW_CONFIG_8:
123 case IEEE80211_EDMG_BW_CONFIG_9:
124 case IEEE80211_EDMG_BW_CONFIG_10:
125 case IEEE80211_EDMG_BW_CONFIG_11:
126 if (num_of_enabled < 2)
127 return false;
128 break;
129 case IEEE80211_EDMG_BW_CONFIG_12:
130 case IEEE80211_EDMG_BW_CONFIG_13:
131 case IEEE80211_EDMG_BW_CONFIG_14:
132 case IEEE80211_EDMG_BW_CONFIG_15:
133 if (num_of_enabled < 4 || max_contiguous < 2)
134 return false;
135 break;
136 default:
137 return false;
138 }
139
140 return true;
141}
142
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100143bool cfg80211_chandef_valid(const struct cfg80211_chan_def *chandef)
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100144{
145 u32 control_freq;
146
147 if (!chandef->chan)
148 return false;
149
150 control_freq = chandef->chan->center_freq;
151
152 switch (chandef->width) {
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200153 case NL80211_CHAN_WIDTH_5:
154 case NL80211_CHAN_WIDTH_10:
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100155 case NL80211_CHAN_WIDTH_20:
156 case NL80211_CHAN_WIDTH_20_NOHT:
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700157 if (ieee80211_chandef_to_khz(chandef) !=
158 ieee80211_channel_to_khz(chandef->chan))
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100159 return false;
160 if (chandef->center_freq2)
161 return false;
162 break;
163 case NL80211_CHAN_WIDTH_40:
164 if (chandef->center_freq1 != control_freq + 10 &&
165 chandef->center_freq1 != control_freq - 10)
166 return false;
167 if (chandef->center_freq2)
168 return false;
169 break;
170 case NL80211_CHAN_WIDTH_80P80:
171 if (chandef->center_freq1 != control_freq + 30 &&
172 chandef->center_freq1 != control_freq + 10 &&
173 chandef->center_freq1 != control_freq - 10 &&
174 chandef->center_freq1 != control_freq - 30)
175 return false;
176 if (!chandef->center_freq2)
177 return false;
Johannes Berg9cab3152012-12-14 00:19:08 +0100178 /* adjacent is not allowed -- that's a 160 MHz channel */
179 if (chandef->center_freq1 - chandef->center_freq2 == 80 ||
180 chandef->center_freq2 - chandef->center_freq1 == 80)
181 return false;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100182 break;
183 case NL80211_CHAN_WIDTH_80:
184 if (chandef->center_freq1 != control_freq + 30 &&
185 chandef->center_freq1 != control_freq + 10 &&
186 chandef->center_freq1 != control_freq - 10 &&
187 chandef->center_freq1 != control_freq - 30)
188 return false;
189 if (chandef->center_freq2)
190 return false;
191 break;
192 case NL80211_CHAN_WIDTH_160:
193 if (chandef->center_freq1 != control_freq + 70 &&
194 chandef->center_freq1 != control_freq + 50 &&
195 chandef->center_freq1 != control_freq + 30 &&
196 chandef->center_freq1 != control_freq + 10 &&
197 chandef->center_freq1 != control_freq - 10 &&
198 chandef->center_freq1 != control_freq - 30 &&
199 chandef->center_freq1 != control_freq - 50 &&
200 chandef->center_freq1 != control_freq - 70)
201 return false;
202 if (chandef->center_freq2)
203 return false;
204 break;
205 default:
206 return false;
207 }
208
Masashi Honmaec649fe2019-10-21 16:50:45 +0900209 /* channel 14 is only for IEEE 802.11b */
210 if (chandef->center_freq1 == 2484 &&
211 chandef->width != NL80211_CHAN_WIDTH_20_NOHT)
212 return false;
213
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +0300214 if (cfg80211_chandef_is_edmg(chandef) &&
215 !cfg80211_edmg_chandef_valid(chandef))
216 return false;
217
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100218 return true;
219}
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100220EXPORT_SYMBOL(cfg80211_chandef_valid);
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100221
222static void chandef_primary_freqs(const struct cfg80211_chan_def *c,
Johannes Bergfc1f48f2014-10-29 17:05:39 +0100223 u32 *pri40, u32 *pri80)
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100224{
225 int tmp;
226
227 switch (c->width) {
228 case NL80211_CHAN_WIDTH_40:
229 *pri40 = c->center_freq1;
230 *pri80 = 0;
231 break;
232 case NL80211_CHAN_WIDTH_80:
233 case NL80211_CHAN_WIDTH_80P80:
234 *pri80 = c->center_freq1;
235 /* n_P20 */
236 tmp = (30 + c->chan->center_freq - c->center_freq1)/20;
237 /* n_P40 */
238 tmp /= 2;
239 /* freq_P40 */
240 *pri40 = c->center_freq1 - 20 + 40 * tmp;
241 break;
242 case NL80211_CHAN_WIDTH_160:
243 /* n_P20 */
244 tmp = (70 + c->chan->center_freq - c->center_freq1)/20;
245 /* n_P40 */
246 tmp /= 2;
247 /* freq_P40 */
248 *pri40 = c->center_freq1 - 60 + 40 * tmp;
249 /* n_P80 */
250 tmp /= 2;
251 *pri80 = c->center_freq1 - 40 + 80 * tmp;
252 break;
253 default:
254 WARN_ON_ONCE(1);
255 }
256}
257
Simon Wunderlich04f39042013-02-08 18:16:19 +0100258static int cfg80211_chandef_get_width(const struct cfg80211_chan_def *c)
259{
260 int width;
261
262 switch (c->width) {
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200263 case NL80211_CHAN_WIDTH_5:
264 width = 5;
265 break;
266 case NL80211_CHAN_WIDTH_10:
267 width = 10;
268 break;
Simon Wunderlich04f39042013-02-08 18:16:19 +0100269 case NL80211_CHAN_WIDTH_20:
270 case NL80211_CHAN_WIDTH_20_NOHT:
271 width = 20;
272 break;
273 case NL80211_CHAN_WIDTH_40:
274 width = 40;
275 break;
276 case NL80211_CHAN_WIDTH_80P80:
277 case NL80211_CHAN_WIDTH_80:
278 width = 80;
279 break;
280 case NL80211_CHAN_WIDTH_160:
281 width = 160;
282 break;
283 default:
284 WARN_ON_ONCE(1);
285 return -1;
286 }
287 return width;
288}
289
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100290const struct cfg80211_chan_def *
291cfg80211_chandef_compatible(const struct cfg80211_chan_def *c1,
292 const struct cfg80211_chan_def *c2)
293{
294 u32 c1_pri40, c1_pri80, c2_pri40, c2_pri80;
295
296 /* If they are identical, return */
297 if (cfg80211_chandef_identical(c1, c2))
298 return c1;
299
300 /* otherwise, must have same control channel */
301 if (c1->chan != c2->chan)
302 return NULL;
303
304 /*
305 * If they have the same width, but aren't identical,
306 * then they can't be compatible.
307 */
308 if (c1->width == c2->width)
309 return NULL;
310
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200311 /*
312 * can't be compatible if one of them is 5 or 10 MHz,
313 * but they don't have the same width.
314 */
315 if (c1->width == NL80211_CHAN_WIDTH_5 ||
316 c1->width == NL80211_CHAN_WIDTH_10 ||
317 c2->width == NL80211_CHAN_WIDTH_5 ||
318 c2->width == NL80211_CHAN_WIDTH_10)
319 return NULL;
320
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100321 if (c1->width == NL80211_CHAN_WIDTH_20_NOHT ||
322 c1->width == NL80211_CHAN_WIDTH_20)
323 return c2;
324
325 if (c2->width == NL80211_CHAN_WIDTH_20_NOHT ||
326 c2->width == NL80211_CHAN_WIDTH_20)
327 return c1;
328
329 chandef_primary_freqs(c1, &c1_pri40, &c1_pri80);
330 chandef_primary_freqs(c2, &c2_pri40, &c2_pri80);
331
332 if (c1_pri40 != c2_pri40)
333 return NULL;
334
335 WARN_ON(!c1_pri80 && !c2_pri80);
336 if (c1_pri80 && c2_pri80 && c1_pri80 != c2_pri80)
337 return NULL;
338
339 if (c1->width > c2->width)
340 return c1;
341 return c2;
342}
343EXPORT_SYMBOL(cfg80211_chandef_compatible);
344
Simon Wunderlich04f39042013-02-08 18:16:19 +0100345static void cfg80211_set_chans_dfs_state(struct wiphy *wiphy, u32 center_freq,
346 u32 bandwidth,
347 enum nl80211_dfs_state dfs_state)
348{
349 struct ieee80211_channel *c;
350 u32 freq;
351
352 for (freq = center_freq - bandwidth/2 + 10;
353 freq <= center_freq + bandwidth/2 - 10;
354 freq += 20) {
355 c = ieee80211_get_channel(wiphy, freq);
356 if (!c || !(c->flags & IEEE80211_CHAN_RADAR))
357 continue;
358
359 c->dfs_state = dfs_state;
360 c->dfs_state_entered = jiffies;
361 }
362}
363
364void cfg80211_set_dfs_state(struct wiphy *wiphy,
365 const struct cfg80211_chan_def *chandef,
366 enum nl80211_dfs_state dfs_state)
367{
368 int width;
369
370 if (WARN_ON(!cfg80211_chandef_valid(chandef)))
371 return;
372
373 width = cfg80211_chandef_get_width(chandef);
374 if (width < 0)
375 return;
376
377 cfg80211_set_chans_dfs_state(wiphy, chandef->center_freq1,
378 width, dfs_state);
379
380 if (!chandef->center_freq2)
381 return;
382 cfg80211_set_chans_dfs_state(wiphy, chandef->center_freq2,
383 width, dfs_state);
384}
385
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100386static u32 cfg80211_get_start_freq(u32 center_freq,
387 u32 bandwidth)
388{
389 u32 start_freq;
390
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700391 bandwidth = MHZ_TO_KHZ(bandwidth);
392 if (bandwidth <= MHZ_TO_KHZ(20))
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100393 start_freq = center_freq;
394 else
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700395 start_freq = center_freq - bandwidth / 2 + MHZ_TO_KHZ(10);
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100396
397 return start_freq;
398}
399
400static u32 cfg80211_get_end_freq(u32 center_freq,
401 u32 bandwidth)
402{
403 u32 end_freq;
404
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700405 bandwidth = MHZ_TO_KHZ(bandwidth);
406 if (bandwidth <= MHZ_TO_KHZ(20))
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100407 end_freq = center_freq;
408 else
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700409 end_freq = center_freq + bandwidth / 2 - MHZ_TO_KHZ(10);
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100410
411 return end_freq;
412}
413
Simon Wunderlich04f39042013-02-08 18:16:19 +0100414static int cfg80211_get_chans_dfs_required(struct wiphy *wiphy,
415 u32 center_freq,
416 u32 bandwidth)
417{
418 struct ieee80211_channel *c;
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200419 u32 freq, start_freq, end_freq;
Simon Wunderlich04f39042013-02-08 18:16:19 +0100420
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100421 start_freq = cfg80211_get_start_freq(center_freq, bandwidth);
422 end_freq = cfg80211_get_end_freq(center_freq, bandwidth);
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200423
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700424 for (freq = start_freq; freq <= end_freq; freq += MHZ_TO_KHZ(20)) {
425 c = ieee80211_get_channel_khz(wiphy, freq);
Simon Wunderlich04f39042013-02-08 18:16:19 +0100426 if (!c)
427 return -EINVAL;
428
429 if (c->flags & IEEE80211_CHAN_RADAR)
430 return 1;
431 }
432 return 0;
433}
434
435
436int cfg80211_chandef_dfs_required(struct wiphy *wiphy,
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200437 const struct cfg80211_chan_def *chandef,
438 enum nl80211_iftype iftype)
Simon Wunderlich04f39042013-02-08 18:16:19 +0100439{
440 int width;
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200441 int ret;
Simon Wunderlich04f39042013-02-08 18:16:19 +0100442
443 if (WARN_ON(!cfg80211_chandef_valid(chandef)))
444 return -EINVAL;
445
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200446 switch (iftype) {
447 case NL80211_IFTYPE_ADHOC:
448 case NL80211_IFTYPE_AP:
449 case NL80211_IFTYPE_P2P_GO:
450 case NL80211_IFTYPE_MESH_POINT:
451 width = cfg80211_chandef_get_width(chandef);
452 if (width < 0)
453 return -EINVAL;
Simon Wunderlich04f39042013-02-08 18:16:19 +0100454
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200455 ret = cfg80211_get_chans_dfs_required(wiphy,
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700456 ieee80211_chandef_to_khz(chandef),
457 width);
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200458 if (ret < 0)
459 return ret;
460 else if (ret > 0)
461 return BIT(chandef->width);
Simon Wunderlich04f39042013-02-08 18:16:19 +0100462
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200463 if (!chandef->center_freq2)
464 return 0;
Simon Wunderlich04f39042013-02-08 18:16:19 +0100465
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200466 ret = cfg80211_get_chans_dfs_required(wiphy,
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700467 MHZ_TO_KHZ(chandef->center_freq2),
468 width);
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200469 if (ret < 0)
470 return ret;
471 else if (ret > 0)
472 return BIT(chandef->width);
473
474 break;
475 case NL80211_IFTYPE_STATION:
Rostislav Lisovy6e0bd6c2014-11-03 10:33:18 +0100476 case NL80211_IFTYPE_OCB:
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200477 case NL80211_IFTYPE_P2P_CLIENT:
478 case NL80211_IFTYPE_MONITOR:
479 case NL80211_IFTYPE_AP_VLAN:
480 case NL80211_IFTYPE_WDS:
481 case NL80211_IFTYPE_P2P_DEVICE:
Ayala Bekercb3b7d82016-09-20 17:31:13 +0300482 case NL80211_IFTYPE_NAN:
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200483 break;
Luciano Coelho00ec75f2014-05-15 13:05:39 +0300484 case NL80211_IFTYPE_UNSPECIFIED:
Luciano Coelho2beb6dab2014-02-18 11:40:36 +0200485 case NUM_NL80211_IFTYPES:
486 WARN_ON(1);
487 }
488
489 return 0;
Simon Wunderlich04f39042013-02-08 18:16:19 +0100490}
Simon Wunderlich774f0732013-08-28 13:41:28 +0200491EXPORT_SYMBOL(cfg80211_chandef_dfs_required);
Simon Wunderlich04f39042013-02-08 18:16:19 +0100492
Janusz Dziedzicfe7c3a12013-11-05 14:48:48 +0100493static int cfg80211_get_chans_dfs_usable(struct wiphy *wiphy,
494 u32 center_freq,
495 u32 bandwidth)
496{
497 struct ieee80211_channel *c;
498 u32 freq, start_freq, end_freq;
499 int count = 0;
500
501 start_freq = cfg80211_get_start_freq(center_freq, bandwidth);
502 end_freq = cfg80211_get_end_freq(center_freq, bandwidth);
503
504 /*
505 * Check entire range of channels for the bandwidth.
506 * Check all channels are DFS channels (DFS_USABLE or
507 * DFS_AVAILABLE). Return number of usable channels
508 * (require CAC). Allow DFS and non-DFS channel mix.
509 */
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700510 for (freq = start_freq; freq <= end_freq; freq += MHZ_TO_KHZ(20)) {
511 c = ieee80211_get_channel_khz(wiphy, freq);
Janusz Dziedzicfe7c3a12013-11-05 14:48:48 +0100512 if (!c)
513 return -EINVAL;
514
515 if (c->flags & IEEE80211_CHAN_DISABLED)
516 return -EINVAL;
517
518 if (c->flags & IEEE80211_CHAN_RADAR) {
519 if (c->dfs_state == NL80211_DFS_UNAVAILABLE)
520 return -EINVAL;
521
522 if (c->dfs_state == NL80211_DFS_USABLE)
523 count++;
524 }
525 }
526
527 return count;
528}
529
530bool cfg80211_chandef_dfs_usable(struct wiphy *wiphy,
531 const struct cfg80211_chan_def *chandef)
532{
533 int width;
534 int r1, r2 = 0;
535
536 if (WARN_ON(!cfg80211_chandef_valid(chandef)))
537 return false;
538
539 width = cfg80211_chandef_get_width(chandef);
540 if (width < 0)
541 return false;
542
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700543 r1 = cfg80211_get_chans_dfs_usable(wiphy,
544 MHZ_TO_KHZ(chandef->center_freq1),
545 width);
Janusz Dziedzicfe7c3a12013-11-05 14:48:48 +0100546
547 if (r1 < 0)
548 return false;
549
550 switch (chandef->width) {
551 case NL80211_CHAN_WIDTH_80P80:
552 WARN_ON(!chandef->center_freq2);
553 r2 = cfg80211_get_chans_dfs_usable(wiphy,
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700554 MHZ_TO_KHZ(chandef->center_freq2),
555 width);
Janusz Dziedzicfe7c3a12013-11-05 14:48:48 +0100556 if (r2 < 0)
557 return false;
558 break;
559 default:
560 WARN_ON(chandef->center_freq2);
561 break;
562 }
563
564 return (r1 + r2 > 0);
565}
566
Vasanthakumar Thiagarajanb35a51c2017-02-27 17:04:33 +0530567/*
568 * Checks if center frequency of chan falls with in the bandwidth
569 * range of chandef.
570 */
571bool cfg80211_is_sub_chan(struct cfg80211_chan_def *chandef,
572 struct ieee80211_channel *chan)
573{
574 int width;
Johannes Berga67a4892017-10-12 11:23:04 +0200575 u32 freq;
Vasanthakumar Thiagarajanb35a51c2017-02-27 17:04:33 +0530576
577 if (chandef->chan->center_freq == chan->center_freq)
578 return true;
579
580 width = cfg80211_chandef_get_width(chandef);
581 if (width <= 20)
582 return false;
583
Vasanthakumar Thiagarajanb35a51c2017-02-27 17:04:33 +0530584 for (freq = chandef->center_freq1 - width / 2 + 10;
585 freq <= chandef->center_freq1 + width / 2 - 10; freq += 20) {
586 if (chan->center_freq == freq)
587 return true;
588 }
589
590 if (!chandef->center_freq2)
591 return false;
592
593 for (freq = chandef->center_freq2 - width / 2 + 10;
594 freq <= chandef->center_freq2 + width / 2 - 10; freq += 20) {
595 if (chan->center_freq == freq)
596 return true;
597 }
598
599 return false;
600}
601
602bool cfg80211_beaconing_iface_active(struct wireless_dev *wdev)
603{
604 bool active = false;
605
606 ASSERT_WDEV_LOCK(wdev);
607
608 if (!wdev->chandef.chan)
609 return false;
610
611 switch (wdev->iftype) {
612 case NL80211_IFTYPE_AP:
613 case NL80211_IFTYPE_P2P_GO:
614 active = wdev->beacon_interval != 0;
615 break;
616 case NL80211_IFTYPE_ADHOC:
617 active = wdev->ssid_len != 0;
618 break;
619 case NL80211_IFTYPE_MESH_POINT:
620 active = wdev->mesh_id_len != 0;
621 break;
622 case NL80211_IFTYPE_STATION:
623 case NL80211_IFTYPE_OCB:
624 case NL80211_IFTYPE_P2P_CLIENT:
625 case NL80211_IFTYPE_MONITOR:
626 case NL80211_IFTYPE_AP_VLAN:
627 case NL80211_IFTYPE_WDS:
628 case NL80211_IFTYPE_P2P_DEVICE:
629 /* Can NAN type be considered as beaconing interface? */
630 case NL80211_IFTYPE_NAN:
631 break;
632 case NL80211_IFTYPE_UNSPECIFIED:
633 case NUM_NL80211_IFTYPES:
634 WARN_ON(1);
635 }
636
637 return active;
638}
639
Vasanthakumar Thiagarajan89766722017-02-27 17:04:35 +0530640static bool cfg80211_is_wiphy_oper_chan(struct wiphy *wiphy,
641 struct ieee80211_channel *chan)
Vasanthakumar Thiagarajanb35a51c2017-02-27 17:04:33 +0530642{
643 struct wireless_dev *wdev;
644
Vasanthakumar Thiagarajanb35a51c2017-02-27 17:04:33 +0530645 list_for_each_entry(wdev, &wiphy->wdev_list, list) {
646 wdev_lock(wdev);
647 if (!cfg80211_beaconing_iface_active(wdev)) {
648 wdev_unlock(wdev);
649 continue;
650 }
651
652 if (cfg80211_is_sub_chan(&wdev->chandef, chan)) {
653 wdev_unlock(wdev);
654 return true;
655 }
656 wdev_unlock(wdev);
657 }
658
659 return false;
660}
Janusz Dziedzicfe7c3a12013-11-05 14:48:48 +0100661
Vasanthakumar Thiagarajan89766722017-02-27 17:04:35 +0530662bool cfg80211_any_wiphy_oper_chan(struct wiphy *wiphy,
663 struct ieee80211_channel *chan)
664{
665 struct cfg80211_registered_device *rdev;
666
667 ASSERT_RTNL();
668
669 if (!(chan->flags & IEEE80211_CHAN_RADAR))
670 return false;
671
672 list_for_each_entry(rdev, &cfg80211_rdev_list, list) {
673 if (!reg_dfs_domain_same(wiphy, &rdev->wiphy))
674 continue;
675
676 if (cfg80211_is_wiphy_oper_chan(&rdev->wiphy, chan))
677 return true;
678 }
679
680 return false;
681}
682
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100683static bool cfg80211_get_chans_dfs_available(struct wiphy *wiphy,
684 u32 center_freq,
685 u32 bandwidth)
686{
687 struct ieee80211_channel *c;
688 u32 freq, start_freq, end_freq;
Dmitry Lebed2c390e42018-03-26 16:36:32 +0300689 bool dfs_offload;
690
691 dfs_offload = wiphy_ext_feature_isset(wiphy,
692 NL80211_EXT_FEATURE_DFS_OFFLOAD);
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100693
694 start_freq = cfg80211_get_start_freq(center_freq, bandwidth);
695 end_freq = cfg80211_get_end_freq(center_freq, bandwidth);
696
697 /*
698 * Check entire range of channels for the bandwidth.
699 * If any channel in between is disabled or has not
700 * had gone through CAC return false
701 */
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700702 for (freq = start_freq; freq <= end_freq; freq += MHZ_TO_KHZ(20)) {
703 c = ieee80211_get_channel_khz(wiphy, freq);
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100704 if (!c)
705 return false;
706
707 if (c->flags & IEEE80211_CHAN_DISABLED)
708 return false;
709
Dmitry Lebed2c390e42018-03-26 16:36:32 +0300710 if ((c->flags & IEEE80211_CHAN_RADAR) &&
711 (c->dfs_state != NL80211_DFS_AVAILABLE) &&
712 !(c->dfs_state == NL80211_DFS_USABLE && dfs_offload))
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100713 return false;
714 }
715
716 return true;
717}
718
719static bool cfg80211_chandef_dfs_available(struct wiphy *wiphy,
720 const struct cfg80211_chan_def *chandef)
721{
722 int width;
723 int r;
724
725 if (WARN_ON(!cfg80211_chandef_valid(chandef)))
726 return false;
727
728 width = cfg80211_chandef_get_width(chandef);
729 if (width < 0)
730 return false;
731
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700732 r = cfg80211_get_chans_dfs_available(wiphy,
733 MHZ_TO_KHZ(chandef->center_freq1),
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100734 width);
735
736 /* If any of channels unavailable for cf1 just return */
737 if (!r)
738 return r;
739
740 switch (chandef->width) {
741 case NL80211_CHAN_WIDTH_80P80:
742 WARN_ON(!chandef->center_freq2);
743 r = cfg80211_get_chans_dfs_available(wiphy,
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700744 MHZ_TO_KHZ(chandef->center_freq2),
745 width);
Colin Ian King680682d2016-07-17 19:55:27 +0100746 break;
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100747 default:
748 WARN_ON(chandef->center_freq2);
749 break;
750 }
751
752 return r;
753}
754
Janusz Dziedzic31559f32014-02-21 19:46:13 +0100755static unsigned int cfg80211_get_chans_dfs_cac_time(struct wiphy *wiphy,
756 u32 center_freq,
757 u32 bandwidth)
758{
759 struct ieee80211_channel *c;
760 u32 start_freq, end_freq, freq;
761 unsigned int dfs_cac_ms = 0;
762
763 start_freq = cfg80211_get_start_freq(center_freq, bandwidth);
764 end_freq = cfg80211_get_end_freq(center_freq, bandwidth);
765
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700766 for (freq = start_freq; freq <= end_freq; freq += MHZ_TO_KHZ(20)) {
767 c = ieee80211_get_channel_khz(wiphy, freq);
Janusz Dziedzic31559f32014-02-21 19:46:13 +0100768 if (!c)
769 return 0;
770
771 if (c->flags & IEEE80211_CHAN_DISABLED)
772 return 0;
773
774 if (!(c->flags & IEEE80211_CHAN_RADAR))
775 continue;
776
777 if (c->dfs_cac_ms > dfs_cac_ms)
778 dfs_cac_ms = c->dfs_cac_ms;
779 }
780
781 return dfs_cac_ms;
782}
783
784unsigned int
785cfg80211_chandef_dfs_cac_time(struct wiphy *wiphy,
786 const struct cfg80211_chan_def *chandef)
787{
788 int width;
789 unsigned int t1 = 0, t2 = 0;
790
791 if (WARN_ON(!cfg80211_chandef_valid(chandef)))
792 return 0;
793
794 width = cfg80211_chandef_get_width(chandef);
795 if (width < 0)
796 return 0;
797
798 t1 = cfg80211_get_chans_dfs_cac_time(wiphy,
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700799 MHZ_TO_KHZ(chandef->center_freq1),
Janusz Dziedzic31559f32014-02-21 19:46:13 +0100800 width);
801
802 if (!chandef->center_freq2)
803 return t1;
804
805 t2 = cfg80211_get_chans_dfs_cac_time(wiphy,
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700806 MHZ_TO_KHZ(chandef->center_freq2),
Janusz Dziedzic31559f32014-02-21 19:46:13 +0100807 width);
808
809 return max(t1, t2);
810}
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100811
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100812static bool cfg80211_secondary_chans_ok(struct wiphy *wiphy,
813 u32 center_freq, u32 bandwidth,
814 u32 prohibited_flags)
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100815{
816 struct ieee80211_channel *c;
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200817 u32 freq, start_freq, end_freq;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100818
Janusz Dziedzic40d1ba62013-11-05 14:48:47 +0100819 start_freq = cfg80211_get_start_freq(center_freq, bandwidth);
820 end_freq = cfg80211_get_end_freq(center_freq, bandwidth);
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200821
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700822 for (freq = start_freq; freq <= end_freq; freq += MHZ_TO_KHZ(20)) {
823 c = ieee80211_get_channel_khz(wiphy, freq);
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +0100824 if (!c || c->flags & prohibited_flags)
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100825 return false;
826 }
827
828 return true;
829}
830
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +0300831/* check if the operating channels are valid and supported */
832static bool cfg80211_edmg_usable(struct wiphy *wiphy, u8 edmg_channels,
833 enum ieee80211_edmg_bw_config edmg_bw_config,
834 int primary_channel,
835 struct ieee80211_edmg *edmg_cap)
836{
837 struct ieee80211_channel *chan;
838 int i, freq;
839 int channels_counter = 0;
840
841 if (!edmg_channels && !edmg_bw_config)
842 return true;
843
844 if ((!edmg_channels && edmg_bw_config) ||
845 (edmg_channels && !edmg_bw_config))
846 return false;
847
848 if (!(edmg_channels & BIT(primary_channel - 1)))
849 return false;
850
851 /* 60GHz channels 1..6 */
852 for (i = 0; i < 6; i++) {
853 if (!(edmg_channels & BIT(i)))
854 continue;
855
856 if (!(edmg_cap->channels & BIT(i)))
857 return false;
858
859 channels_counter++;
860
861 freq = ieee80211_channel_to_frequency(i + 1,
862 NL80211_BAND_60GHZ);
863 chan = ieee80211_get_channel(wiphy, freq);
864 if (!chan || chan->flags & IEEE80211_CHAN_DISABLED)
865 return false;
866 }
867
868 /* IEEE802.11 allows max 4 channels */
869 if (channels_counter > 4)
870 return false;
871
872 /* check bw_config is a subset of what driver supports
873 * (see IEEE P802.11ay/D4.0 section 9.4.2.251, Table 13)
874 */
875 if ((edmg_bw_config % 4) > (edmg_cap->bw_config % 4))
876 return false;
877
878 if (edmg_bw_config > edmg_cap->bw_config)
879 return false;
880
881 return true;
882}
883
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100884bool cfg80211_chandef_usable(struct wiphy *wiphy,
885 const struct cfg80211_chan_def *chandef,
886 u32 prohibited_flags)
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100887{
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100888 struct ieee80211_sta_ht_cap *ht_cap;
889 struct ieee80211_sta_vht_cap *vht_cap;
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +0300890 struct ieee80211_edmg *edmg_cap;
Jouni Malinen08f6f142014-12-11 23:48:55 +0200891 u32 width, control_freq, cap;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100892
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100893 if (WARN_ON(!cfg80211_chandef_valid(chandef)))
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100894 return false;
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100895
896 ht_cap = &wiphy->bands[chandef->chan->band]->ht_cap;
897 vht_cap = &wiphy->bands[chandef->chan->band]->vht_cap;
Alexei Avshalom Lazar2a380752019-08-18 17:35:17 +0300898 edmg_cap = &wiphy->bands[chandef->chan->band]->edmg_cap;
899
900 if (edmg_cap->channels &&
901 !cfg80211_edmg_usable(wiphy,
902 chandef->edmg.channels,
903 chandef->edmg.bw_config,
904 chandef->chan->hw_value,
905 edmg_cap))
906 return false;
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100907
908 control_freq = chandef->chan->center_freq;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100909
910 switch (chandef->width) {
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200911 case NL80211_CHAN_WIDTH_5:
912 width = 5;
913 break;
914 case NL80211_CHAN_WIDTH_10:
Rostislav Lisovyea077c12014-04-15 14:37:55 +0200915 prohibited_flags |= IEEE80211_CHAN_NO_10MHZ;
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200916 width = 10;
917 break;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100918 case NL80211_CHAN_WIDTH_20:
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100919 if (!ht_cap->ht_supported)
920 return false;
Luca Coelho925b5972018-12-15 11:03:21 +0200921 /* fall through */
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100922 case NL80211_CHAN_WIDTH_20_NOHT:
Rostislav Lisovyea077c12014-04-15 14:37:55 +0200923 prohibited_flags |= IEEE80211_CHAN_NO_20MHZ;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100924 width = 20;
Mark Mentovai09a02fd2010-11-17 16:34:37 -0500925 break;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100926 case NL80211_CHAN_WIDTH_40:
927 width = 40;
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100928 if (!ht_cap->ht_supported)
929 return false;
930 if (!(ht_cap->cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40) ||
931 ht_cap->cap & IEEE80211_HT_CAP_40MHZ_INTOLERANT)
932 return false;
933 if (chandef->center_freq1 < control_freq &&
934 chandef->chan->flags & IEEE80211_CHAN_NO_HT40MINUS)
935 return false;
936 if (chandef->center_freq1 > control_freq &&
937 chandef->chan->flags & IEEE80211_CHAN_NO_HT40PLUS)
938 return false;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100939 break;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100940 case NL80211_CHAN_WIDTH_80P80:
Jouni Malinen08f6f142014-12-11 23:48:55 +0200941 cap = vht_cap->cap & IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_MASK;
942 if (cap != IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ)
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100943 return false;
Luca Coelho925b5972018-12-15 11:03:21 +0200944 /* fall through */
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100945 case NL80211_CHAN_WIDTH_80:
946 if (!vht_cap->vht_supported)
947 return false;
Johannes Bergc7a6ee22012-12-12 17:50:39 +0100948 prohibited_flags |= IEEE80211_CHAN_NO_80MHZ;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100949 width = 80;
950 break;
951 case NL80211_CHAN_WIDTH_160:
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100952 if (!vht_cap->vht_supported)
953 return false;
Jouni Malinen08f6f142014-12-11 23:48:55 +0200954 cap = vht_cap->cap & IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_MASK;
955 if (cap != IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160MHZ &&
956 cap != IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ)
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100957 return false;
Johannes Bergc7a6ee22012-12-12 17:50:39 +0100958 prohibited_flags |= IEEE80211_CHAN_NO_160MHZ;
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100959 width = 160;
Mark Mentovai09a02fd2010-11-17 16:34:37 -0500960 break;
Luis R. Rodriguez9236d832010-11-12 16:31:23 -0800961 default:
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100962 WARN_ON_ONCE(1);
Luis R. Rodriguez9236d832010-11-12 16:31:23 -0800963 return false;
Beni Lev4ee3e062012-08-27 12:49:39 +0300964 }
Luis R. Rodriguez9236d832010-11-12 16:31:23 -0800965
Johannes Bergc7a6ee22012-12-12 17:50:39 +0100966 /*
967 * TODO: What if there are only certain 80/160/80+80 MHz channels
968 * allowed by the driver, or only certain combinations?
969 * For 40 MHz the driver can set the NO_HT40 flags, but for
970 * 80/160 MHz and in particular 80+80 MHz this isn't really
971 * feasible and we only have NO_80MHZ/NO_160MHZ so far but
972 * no way to cover 80+80 MHz or more complex restrictions.
973 * Note that such restrictions also need to be advertised to
974 * userspace, for example for P2P channel selection.
975 */
Johannes Berg3d9d1d62012-11-08 23:14:50 +0100976
Johannes Berga6662db2012-12-04 20:49:42 +0100977 if (width > 20)
978 prohibited_flags |= IEEE80211_CHAN_NO_OFDM;
979
Simon Wunderlich2f301ab2013-05-16 13:00:28 +0200980 /* 5 and 10 MHz are only defined for the OFDM PHY */
981 if (width < 20)
982 prohibited_flags |= IEEE80211_CHAN_NO_OFDM;
983
984
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700985 if (!cfg80211_secondary_chans_ok(wiphy,
986 ieee80211_chandef_to_khz(chandef),
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100987 width, prohibited_flags))
988 return false;
989
990 if (!chandef->center_freq2)
991 return true;
Thomas Pedersen934f4c72020-04-01 18:18:03 -0700992 return cfg80211_secondary_chans_ok(wiphy,
993 MHZ_TO_KHZ(chandef->center_freq2),
Johannes Berg9f5e8f62012-11-22 16:59:45 +0100994 width, prohibited_flags);
995}
996EXPORT_SYMBOL(cfg80211_chandef_usable);
997
Ilan Peer174e0cd2014-02-23 09:13:01 +0200998/*
Arik Nemtsov06f207f2015-05-06 16:28:31 +0300999 * Check if the channel can be used under permissive conditions mandated by
1000 * some regulatory bodies, i.e., the channel is marked with
1001 * IEEE80211_CHAN_IR_CONCURRENT and there is an additional station interface
Ilan Peer174e0cd2014-02-23 09:13:01 +02001002 * associated to an AP on the same channel or on the same UNII band
1003 * (assuming that the AP is an authorized master).
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001004 * In addition allow operation on a channel on which indoor operation is
Ilan Peerc8866e52014-02-23 09:13:03 +02001005 * allowed, iff we are currently operating in an indoor environment.
Ilan Peer174e0cd2014-02-23 09:13:01 +02001006 */
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001007static bool cfg80211_ir_permissive_chan(struct wiphy *wiphy,
1008 enum nl80211_iftype iftype,
Ilan Peer174e0cd2014-02-23 09:13:01 +02001009 struct ieee80211_channel *chan)
Johannes Berg9f5e8f62012-11-22 16:59:45 +01001010{
Avraham Sternbe69c242015-04-27 16:52:16 +03001011 struct wireless_dev *wdev;
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001012 struct cfg80211_registered_device *rdev = wiphy_to_rdev(wiphy);
Ilan Peer174e0cd2014-02-23 09:13:01 +02001013
1014 ASSERT_RTNL();
1015
Masahiro Yamada97f26452016-08-03 13:45:50 -07001016 if (!IS_ENABLED(CONFIG_CFG80211_REG_RELAX_NO_IR) ||
Ilan Peerc8866e52014-02-23 09:13:03 +02001017 !(wiphy->regulatory_flags & REGULATORY_ENABLE_RELAX_NO_IR))
1018 return false;
1019
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001020 /* only valid for GO and TDLS off-channel (station/p2p-CL) */
1021 if (iftype != NL80211_IFTYPE_P2P_GO &&
1022 iftype != NL80211_IFTYPE_STATION &&
1023 iftype != NL80211_IFTYPE_P2P_CLIENT)
1024 return false;
1025
Ilan Peerc8866e52014-02-23 09:13:03 +02001026 if (regulatory_indoor_allowed() &&
1027 (chan->flags & IEEE80211_CHAN_INDOOR_ONLY))
1028 return true;
1029
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001030 if (!(chan->flags & IEEE80211_CHAN_IR_CONCURRENT))
Ilan Peer174e0cd2014-02-23 09:13:01 +02001031 return false;
1032
1033 /*
1034 * Generally, it is possible to rely on another device/driver to allow
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001035 * the IR concurrent relaxation, however, since the device can further
Ilan Peer174e0cd2014-02-23 09:13:01 +02001036 * enforce the relaxation (by doing a similar verifications as this),
1037 * and thus fail the GO instantiation, consider only the interfaces of
1038 * the current registered device.
1039 */
Johannes Berg53873f12016-05-03 16:52:04 +03001040 list_for_each_entry(wdev, &rdev->wiphy.wdev_list, list) {
Ilan Peer174e0cd2014-02-23 09:13:01 +02001041 struct ieee80211_channel *other_chan = NULL;
1042 int r1, r2;
1043
Avraham Sternbe69c242015-04-27 16:52:16 +03001044 wdev_lock(wdev);
1045 if (wdev->iftype == NL80211_IFTYPE_STATION &&
1046 wdev->current_bss)
1047 other_chan = wdev->current_bss->pub.channel;
Ilan Peer174e0cd2014-02-23 09:13:01 +02001048
Avraham Sternbe69c242015-04-27 16:52:16 +03001049 /*
1050 * If a GO already operates on the same GO_CONCURRENT channel,
1051 * this one (maybe the same one) can beacon as well. We allow
1052 * the operation even if the station we relied on with
1053 * GO_CONCURRENT is disconnected now. But then we must make sure
1054 * we're not outdoor on an indoor-only channel.
1055 */
Arik Nemtsov06f207f2015-05-06 16:28:31 +03001056 if (iftype == NL80211_IFTYPE_P2P_GO &&
1057 wdev->iftype == NL80211_IFTYPE_P2P_GO &&
Avraham Sternbe69c242015-04-27 16:52:16 +03001058 wdev->beacon_interval &&
1059 !(chan->flags & IEEE80211_CHAN_INDOOR_ONLY))
1060 other_chan = wdev->chandef.chan;
1061 wdev_unlock(wdev);
Ilan Peer174e0cd2014-02-23 09:13:01 +02001062
1063 if (!other_chan)
1064 continue;
1065
1066 if (chan == other_chan)
1067 return true;
1068
Arend van Spriel0816e6b2019-08-02 13:31:03 +02001069 if (chan->band != NL80211_BAND_5GHZ &&
1070 chan->band != NL80211_BAND_6GHZ)
Ilan Peer174e0cd2014-02-23 09:13:01 +02001071 continue;
1072
1073 r1 = cfg80211_get_unii(chan->center_freq);
1074 r2 = cfg80211_get_unii(other_chan->center_freq);
1075
Ilan Peer46d53722014-04-23 09:22:58 +03001076 if (r1 != -EINVAL && r1 == r2) {
1077 /*
1078 * At some locations channels 149-165 are considered a
1079 * bundle, but at other locations, e.g., Indonesia,
1080 * channels 149-161 are considered a bundle while
1081 * channel 165 is left out and considered to be in a
1082 * different bundle. Thus, in case that there is a
1083 * station interface connected to an AP on channel 165,
1084 * it is assumed that channels 149-161 are allowed for
1085 * GO operations. However, having a station interface
1086 * connected to an AP on channels 149-161, does not
1087 * allow GO operation on channel 165.
1088 */
1089 if (chan->center_freq == 5825 &&
1090 other_chan->center_freq != 5825)
1091 continue;
Ilan Peer174e0cd2014-02-23 09:13:01 +02001092 return true;
Ilan Peer46d53722014-04-23 09:22:58 +03001093 }
Ilan Peer174e0cd2014-02-23 09:13:01 +02001094 }
1095
1096 return false;
1097}
1098
Arik Nemtsov923b3522015-07-08 15:41:44 +03001099static bool _cfg80211_reg_can_beacon(struct wiphy *wiphy,
1100 struct cfg80211_chan_def *chandef,
1101 enum nl80211_iftype iftype,
1102 bool check_no_ir)
Ilan Peer174e0cd2014-02-23 09:13:01 +02001103{
Johannes Berg9f5e8f62012-11-22 16:59:45 +01001104 bool res;
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +01001105 u32 prohibited_flags = IEEE80211_CHAN_DISABLED |
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +01001106 IEEE80211_CHAN_RADAR;
Johannes Berg9f5e8f62012-11-22 16:59:45 +01001107
Arik Nemtsov923b3522015-07-08 15:41:44 +03001108 trace_cfg80211_reg_can_beacon(wiphy, chandef, iftype, check_no_ir);
Ilan Peer174e0cd2014-02-23 09:13:01 +02001109
Arik Nemtsov923b3522015-07-08 15:41:44 +03001110 if (check_no_ir)
Ilan Peer174e0cd2014-02-23 09:13:01 +02001111 prohibited_flags |= IEEE80211_CHAN_NO_IR;
Johannes Berg9f5e8f62012-11-22 16:59:45 +01001112
Luciano Coelho00ec75f2014-05-15 13:05:39 +03001113 if (cfg80211_chandef_dfs_required(wiphy, chandef, iftype) > 0 &&
Janusz Dziedzic6bc54fb2013-11-06 13:55:53 +01001114 cfg80211_chandef_dfs_available(wiphy, chandef)) {
1115 /* We can skip IEEE80211_CHAN_NO_IR if chandef dfs available */
1116 prohibited_flags = IEEE80211_CHAN_DISABLED;
1117 }
1118
1119 res = cfg80211_chandef_usable(wiphy, chandef, prohibited_flags);
Johannes Berg3d9d1d62012-11-08 23:14:50 +01001120
1121 trace_cfg80211_return_bool(res);
1122 return res;
Luis R. Rodriguez9236d832010-11-12 16:31:23 -08001123}
Arik Nemtsov923b3522015-07-08 15:41:44 +03001124
1125bool cfg80211_reg_can_beacon(struct wiphy *wiphy,
1126 struct cfg80211_chan_def *chandef,
1127 enum nl80211_iftype iftype)
1128{
1129 return _cfg80211_reg_can_beacon(wiphy, chandef, iftype, true);
1130}
Johannes Berg683b6d32012-11-08 21:25:48 +01001131EXPORT_SYMBOL(cfg80211_reg_can_beacon);
Luis R. Rodriguez9236d832010-11-12 16:31:23 -08001132
Arik Nemtsov923b3522015-07-08 15:41:44 +03001133bool cfg80211_reg_can_beacon_relax(struct wiphy *wiphy,
1134 struct cfg80211_chan_def *chandef,
1135 enum nl80211_iftype iftype)
1136{
1137 bool check_no_ir;
1138
1139 ASSERT_RTNL();
1140
1141 /*
1142 * Under certain conditions suggested by some regulatory bodies a
1143 * GO/STA can IR on channels marked with IEEE80211_NO_IR. Set this flag
1144 * only if such relaxations are not enabled and the conditions are not
1145 * met.
1146 */
1147 check_no_ir = !cfg80211_ir_permissive_chan(wiphy, iftype,
1148 chandef->chan);
1149
1150 return _cfg80211_reg_can_beacon(wiphy, chandef, iftype, check_no_ir);
1151}
1152EXPORT_SYMBOL(cfg80211_reg_can_beacon_relax);
1153
Johannes Berge8c9bd52012-06-06 08:18:22 +02001154int cfg80211_set_monitor_channel(struct cfg80211_registered_device *rdev,
Johannes Berg683b6d32012-11-08 21:25:48 +01001155 struct cfg80211_chan_def *chandef)
Johannes Berg59bbb6f2009-08-07 17:22:35 +02001156{
Johannes Berge8c9bd52012-06-06 08:18:22 +02001157 if (!rdev->ops->set_monitor_channel)
Johannes Berg59bbb6f2009-08-07 17:22:35 +02001158 return -EOPNOTSUPP;
Michal Kazior4f03c1e2012-06-29 12:47:03 +02001159 if (!cfg80211_has_monitors_only(rdev))
1160 return -EBUSY;
Johannes Berg59bbb6f2009-08-07 17:22:35 +02001161
Johannes Berg683b6d32012-11-08 21:25:48 +01001162 return rdev_set_monitor_channel(rdev, chandef);
Johannes Berg59bbb6f2009-08-07 17:22:35 +02001163}
Michal Kazior26ab9a02012-06-29 12:47:00 +02001164
1165void
Johannes Berg8e95ea42012-07-10 19:39:02 +02001166cfg80211_get_chan_state(struct wireless_dev *wdev,
Michal Kazior26ab9a02012-06-29 12:47:00 +02001167 struct ieee80211_channel **chan,
Michal Kazior9e0e2962014-01-29 14:22:27 +01001168 enum cfg80211_chan_mode *chanmode,
1169 u8 *radar_detect)
Michal Kazior26ab9a02012-06-29 12:47:00 +02001170{
Luciano Coelho2beb6dab2014-02-18 11:40:36 +02001171 int ret;
1172
Michal Kazior26ab9a02012-06-29 12:47:00 +02001173 *chan = NULL;
1174 *chanmode = CHAN_MODE_UNDEFINED;
1175
Michal Kazior26ab9a02012-06-29 12:47:00 +02001176 ASSERT_WDEV_LOCK(wdev);
1177
Johannes Berg98104fde2012-06-16 00:19:54 +02001178 if (wdev->netdev && !netif_running(wdev->netdev))
Michal Kazior26ab9a02012-06-29 12:47:00 +02001179 return;
1180
1181 switch (wdev->iftype) {
1182 case NL80211_IFTYPE_ADHOC:
1183 if (wdev->current_bss) {
1184 *chan = wdev->current_bss->pub.channel;
Simon Wunderlich5336fa82013-10-07 18:41:05 +02001185 *chanmode = (wdev->ibss_fixed &&
1186 !wdev->ibss_dfs_possible)
Michal Kazior26ab9a02012-06-29 12:47:00 +02001187 ? CHAN_MODE_SHARED
1188 : CHAN_MODE_EXCLUSIVE;
Michal Kazior9e0e2962014-01-29 14:22:27 +01001189
1190 /* consider worst-case - IBSS can try to return to the
1191 * original user-specified channel as creator */
1192 if (wdev->ibss_dfs_possible)
1193 *radar_detect |= BIT(wdev->chandef.width);
Michal Kazior26ab9a02012-06-29 12:47:00 +02001194 return;
1195 }
Johannes Berg0f0094b2013-10-25 12:46:44 +02001196 break;
Michal Kazior26ab9a02012-06-29 12:47:00 +02001197 case NL80211_IFTYPE_STATION:
1198 case NL80211_IFTYPE_P2P_CLIENT:
1199 if (wdev->current_bss) {
1200 *chan = wdev->current_bss->pub.channel;
1201 *chanmode = CHAN_MODE_SHARED;
1202 return;
1203 }
1204 break;
1205 case NL80211_IFTYPE_AP:
1206 case NL80211_IFTYPE_P2P_GO:
Simon Wunderlich04f39042013-02-08 18:16:19 +01001207 if (wdev->cac_started) {
Michal Kazior9e0e2962014-01-29 14:22:27 +01001208 *chan = wdev->chandef.chan;
Simon Wunderlich04f39042013-02-08 18:16:19 +01001209 *chanmode = CHAN_MODE_SHARED;
Michal Kazior9e0e2962014-01-29 14:22:27 +01001210 *radar_detect |= BIT(wdev->chandef.width);
Simon Wunderlich04f39042013-02-08 18:16:19 +01001211 } else if (wdev->beacon_interval) {
Michal Kazior9e0e2962014-01-29 14:22:27 +01001212 *chan = wdev->chandef.chan;
Felix Fietkauf53594a2012-07-12 16:10:02 +02001213 *chanmode = CHAN_MODE_SHARED;
Michal Kazior9e0e2962014-01-29 14:22:27 +01001214
Luciano Coelho2beb6dab2014-02-18 11:40:36 +02001215 ret = cfg80211_chandef_dfs_required(wdev->wiphy,
1216 &wdev->chandef,
1217 wdev->iftype);
1218 WARN_ON(ret < 0);
1219 if (ret > 0)
Michal Kazior9e0e2962014-01-29 14:22:27 +01001220 *radar_detect |= BIT(wdev->chandef.width);
Felix Fietkauf53594a2012-07-12 16:10:02 +02001221 }
1222 return;
Michal Kazior26ab9a02012-06-29 12:47:00 +02001223 case NL80211_IFTYPE_MESH_POINT:
Felix Fietkauf53594a2012-07-12 16:10:02 +02001224 if (wdev->mesh_id_len) {
Michal Kazior9e0e2962014-01-29 14:22:27 +01001225 *chan = wdev->chandef.chan;
Felix Fietkauf53594a2012-07-12 16:10:02 +02001226 *chanmode = CHAN_MODE_SHARED;
Michal Kazior9e0e2962014-01-29 14:22:27 +01001227
Luciano Coelho2beb6dab2014-02-18 11:40:36 +02001228 ret = cfg80211_chandef_dfs_required(wdev->wiphy,
1229 &wdev->chandef,
1230 wdev->iftype);
1231 WARN_ON(ret < 0);
1232 if (ret > 0)
Michal Kazior9e0e2962014-01-29 14:22:27 +01001233 *radar_detect |= BIT(wdev->chandef.width);
Felix Fietkauf53594a2012-07-12 16:10:02 +02001234 }
Michal Kazior26ab9a02012-06-29 12:47:00 +02001235 return;
Rostislav Lisovy6e0bd6c2014-11-03 10:33:18 +01001236 case NL80211_IFTYPE_OCB:
1237 if (wdev->chandef.chan) {
1238 *chan = wdev->chandef.chan;
1239 *chanmode = CHAN_MODE_SHARED;
1240 return;
1241 }
1242 break;
Michal Kazior26ab9a02012-06-29 12:47:00 +02001243 case NL80211_IFTYPE_MONITOR:
1244 case NL80211_IFTYPE_AP_VLAN:
1245 case NL80211_IFTYPE_WDS:
Johannes Berg98104fde2012-06-16 00:19:54 +02001246 case NL80211_IFTYPE_P2P_DEVICE:
Ayala Bekercb3b7d82016-09-20 17:31:13 +03001247 case NL80211_IFTYPE_NAN:
Johannes Berge7aceef2014-02-12 14:21:15 +01001248 /* these interface types don't really have a channel */
Johannes Berg98104fde2012-06-16 00:19:54 +02001249 return;
Michal Kazior26ab9a02012-06-29 12:47:00 +02001250 case NL80211_IFTYPE_UNSPECIFIED:
1251 case NUM_NL80211_IFTYPES:
1252 WARN_ON(1);
1253 }
Michal Kazior26ab9a02012-06-29 12:47:00 +02001254}