blob: 03ba6b5c5373b373d47956518fafeaf1338b2ad0 [file] [log] [blame]
Johannes Bergf444de02010-05-05 15:25:02 +02001/*
2 * mac80211 - channel management
3 */
4
Johannes Berg0aaffa92010-05-05 15:28:27 +02005#include <linux/nl80211.h>
Johannes Berg3448c002012-09-11 17:57:42 +02006#include <linux/export.h>
Johannes Berg4d76d212012-12-11 20:38:41 +01007#include <linux/rtnetlink.h>
Paul Stewart3117bbdb2012-03-13 07:46:18 -07008#include <net/cfg80211.h>
Johannes Bergf444de02010-05-05 15:25:02 +02009#include "ieee80211_i.h"
Michal Kazior35f2fce2012-06-26 14:37:20 +020010#include "driver-ops.h"
Johannes Bergf444de02010-05-05 15:25:02 +020011
Johannes Berg18942d32013-02-07 21:30:37 +010012static void ieee80211_change_chanctx(struct ieee80211_local *local,
Johannes Berg4bf88532012-11-09 11:39:59 +010013 struct ieee80211_chanctx *ctx,
14 const struct cfg80211_chan_def *chandef)
Michal Kazior23a85b452012-06-26 14:37:21 +020015{
Johannes Berg4bf88532012-11-09 11:39:59 +010016 if (cfg80211_chandef_identical(&ctx->conf.def, chandef))
Michal Kaziore89a96f2012-06-26 14:37:22 +020017 return;
18
Johannes Berg4bf88532012-11-09 11:39:59 +010019 WARN_ON(!cfg80211_chandef_compatible(&ctx->conf.def, chandef));
20
21 ctx->conf.def = *chandef;
22 drv_change_chanctx(local, ctx, IEEE80211_CHANCTX_CHANGE_WIDTH);
Johannes Berg55de9082012-07-26 17:24:39 +020023
24 if (!local->use_chanctx) {
Karl Beldan675a0b02013-03-25 16:26:57 +010025 local->_oper_chandef = *chandef;
Johannes Berg55de9082012-07-26 17:24:39 +020026 ieee80211_hw_config(local, 0);
27 }
Johannes Berg0aaffa92010-05-05 15:28:27 +020028}
Michal Kaziord01a1e62012-06-26 14:37:16 +020029
30static struct ieee80211_chanctx *
31ieee80211_find_chanctx(struct ieee80211_local *local,
Johannes Berg4bf88532012-11-09 11:39:59 +010032 const struct cfg80211_chan_def *chandef,
Michal Kaziord01a1e62012-06-26 14:37:16 +020033 enum ieee80211_chanctx_mode mode)
34{
35 struct ieee80211_chanctx *ctx;
36
37 lockdep_assert_held(&local->chanctx_mtx);
38
39 if (mode == IEEE80211_CHANCTX_EXCLUSIVE)
40 return NULL;
Michal Kaziord01a1e62012-06-26 14:37:16 +020041
42 list_for_each_entry(ctx, &local->chanctx_list, list) {
Johannes Berg4bf88532012-11-09 11:39:59 +010043 const struct cfg80211_chan_def *compat;
Michal Kaziore89a96f2012-06-26 14:37:22 +020044
Michal Kaziord01a1e62012-06-26 14:37:16 +020045 if (ctx->mode == IEEE80211_CHANCTX_EXCLUSIVE)
46 continue;
Johannes Berg4bf88532012-11-09 11:39:59 +010047
48 compat = cfg80211_chandef_compatible(&ctx->conf.def, chandef);
49 if (!compat)
Michal Kaziord01a1e62012-06-26 14:37:16 +020050 continue;
51
Johannes Berg18942d32013-02-07 21:30:37 +010052 ieee80211_change_chanctx(local, ctx, compat);
Michal Kaziore89a96f2012-06-26 14:37:22 +020053
Michal Kaziord01a1e62012-06-26 14:37:16 +020054 return ctx;
55 }
56
57 return NULL;
58}
59
Simon Wunderliche4746852013-04-08 22:43:16 +020060static bool ieee80211_is_radar_required(struct ieee80211_local *local)
61{
62 struct ieee80211_sub_if_data *sdata;
63
64 rcu_read_lock();
65 list_for_each_entry_rcu(sdata, &local->interfaces, list) {
66 if (sdata->radar_required) {
67 rcu_read_unlock();
68 return true;
69 }
70 }
71 rcu_read_unlock();
72
73 return false;
74}
75
Michal Kaziord01a1e62012-06-26 14:37:16 +020076static struct ieee80211_chanctx *
77ieee80211_new_chanctx(struct ieee80211_local *local,
Johannes Berg4bf88532012-11-09 11:39:59 +010078 const struct cfg80211_chan_def *chandef,
Michal Kaziord01a1e62012-06-26 14:37:16 +020079 enum ieee80211_chanctx_mode mode)
80{
81 struct ieee80211_chanctx *ctx;
Johannes Berg382a1032013-03-22 22:30:09 +010082 u32 changed;
Michal Kazior35f2fce2012-06-26 14:37:20 +020083 int err;
Michal Kaziord01a1e62012-06-26 14:37:16 +020084
85 lockdep_assert_held(&local->chanctx_mtx);
86
87 ctx = kzalloc(sizeof(*ctx) + local->hw.chanctx_data_size, GFP_KERNEL);
88 if (!ctx)
89 return ERR_PTR(-ENOMEM);
90
Johannes Berg4bf88532012-11-09 11:39:59 +010091 ctx->conf.def = *chandef;
Johannes Berg04ecd252012-09-11 14:34:12 +020092 ctx->conf.rx_chains_static = 1;
93 ctx->conf.rx_chains_dynamic = 1;
Michal Kaziord01a1e62012-06-26 14:37:16 +020094 ctx->mode = mode;
Simon Wunderliche4746852013-04-08 22:43:16 +020095 ctx->conf.radar_enabled = ieee80211_is_radar_required(local);
96 if (!local->use_chanctx)
97 local->hw.conf.radar_enabled = ctx->conf.radar_enabled;
Michal Kaziord01a1e62012-06-26 14:37:16 +020098
Johannes Berg382a1032013-03-22 22:30:09 +010099 /* acquire mutex to prevent idle from changing */
100 mutex_lock(&local->mtx);
101 /* turn idle off *before* setting channel -- some drivers need that */
102 changed = ieee80211_idle_off(local);
103 if (changed)
104 ieee80211_hw_config(local, changed);
105
Johannes Berg55de9082012-07-26 17:24:39 +0200106 if (!local->use_chanctx) {
Karl Beldan675a0b02013-03-25 16:26:57 +0100107 local->_oper_chandef = *chandef;
Johannes Berg55de9082012-07-26 17:24:39 +0200108 ieee80211_hw_config(local, 0);
109 } else {
110 err = drv_add_chanctx(local, ctx);
111 if (err) {
112 kfree(ctx);
Johannes Berg382a1032013-03-22 22:30:09 +0100113 ctx = ERR_PTR(err);
114
115 ieee80211_recalc_idle(local);
116 goto out;
Johannes Berg55de9082012-07-26 17:24:39 +0200117 }
Michal Kazior35f2fce2012-06-26 14:37:20 +0200118 }
119
Johannes Berg382a1032013-03-22 22:30:09 +0100120 /* and keep the mutex held until the new chanctx is on the list */
Johannes Berg3448c002012-09-11 17:57:42 +0200121 list_add_rcu(&ctx->list, &local->chanctx_list);
Michal Kaziord01a1e62012-06-26 14:37:16 +0200122
Johannes Berg382a1032013-03-22 22:30:09 +0100123 out:
Johannes Bergfd0f9792013-02-07 00:14:51 +0100124 mutex_unlock(&local->mtx);
125
Michal Kaziord01a1e62012-06-26 14:37:16 +0200126 return ctx;
127}
128
129static void ieee80211_free_chanctx(struct ieee80211_local *local,
130 struct ieee80211_chanctx *ctx)
131{
Simon Wunderliche4746852013-04-08 22:43:16 +0200132 bool check_single_channel = false;
Michal Kaziord01a1e62012-06-26 14:37:16 +0200133 lockdep_assert_held(&local->chanctx_mtx);
134
135 WARN_ON_ONCE(ctx->refcount != 0);
136
Johannes Berg55de9082012-07-26 17:24:39 +0200137 if (!local->use_chanctx) {
Karl Beldan675a0b02013-03-25 16:26:57 +0100138 struct cfg80211_chan_def *chandef = &local->_oper_chandef;
139 chandef->width = NL80211_CHAN_WIDTH_20_NOHT;
140 chandef->center_freq1 = chandef->chan->center_freq;
141 chandef->center_freq2 = 0;
Simon Wunderliche4746852013-04-08 22:43:16 +0200142
143 /* NOTE: Disabling radar is only valid here for
144 * single channel context. To be sure, check it ...
145 */
146 if (local->hw.conf.radar_enabled)
147 check_single_channel = true;
148 local->hw.conf.radar_enabled = false;
149
Johannes Berg55de9082012-07-26 17:24:39 +0200150 ieee80211_hw_config(local, 0);
151 } else {
152 drv_remove_chanctx(local, ctx);
153 }
Michal Kazior35f2fce2012-06-26 14:37:20 +0200154
Johannes Berg3448c002012-09-11 17:57:42 +0200155 list_del_rcu(&ctx->list);
Michal Kaziord01a1e62012-06-26 14:37:16 +0200156 kfree_rcu(ctx, rcu_head);
Johannes Bergfd0f9792013-02-07 00:14:51 +0100157
Simon Wunderliche4746852013-04-08 22:43:16 +0200158 /* throw a warning if this wasn't the only channel context. */
159 WARN_ON(check_single_channel && !list_empty(&local->chanctx_list));
160
Johannes Bergfd0f9792013-02-07 00:14:51 +0100161 mutex_lock(&local->mtx);
162 ieee80211_recalc_idle(local);
163 mutex_unlock(&local->mtx);
Michal Kaziord01a1e62012-06-26 14:37:16 +0200164}
165
166static int ieee80211_assign_vif_chanctx(struct ieee80211_sub_if_data *sdata,
167 struct ieee80211_chanctx *ctx)
168{
Michal Kazior35f2fce2012-06-26 14:37:20 +0200169 struct ieee80211_local *local = sdata->local;
170 int ret;
Michal Kaziord01a1e62012-06-26 14:37:16 +0200171
172 lockdep_assert_held(&local->chanctx_mtx);
173
Michal Kazior35f2fce2012-06-26 14:37:20 +0200174 ret = drv_assign_vif_chanctx(local, sdata, ctx);
175 if (ret)
176 return ret;
177
Michal Kaziord01a1e62012-06-26 14:37:16 +0200178 rcu_assign_pointer(sdata->vif.chanctx_conf, &ctx->conf);
179 ctx->refcount++;
180
Johannes Berg1ea6f9c2012-10-24 10:59:25 +0200181 ieee80211_recalc_txpower(sdata);
Johannes Bergfd0f9792013-02-07 00:14:51 +0100182 sdata->vif.bss_conf.idle = false;
Johannes Berg5bbe754d2013-02-13 13:50:51 +0100183
184 if (sdata->vif.type != NL80211_IFTYPE_P2P_DEVICE &&
185 sdata->vif.type != NL80211_IFTYPE_MONITOR)
186 ieee80211_bss_info_change_notify(sdata, BSS_CHANGED_IDLE);
Johannes Berg1ea6f9c2012-10-24 10:59:25 +0200187
Michal Kaziord01a1e62012-06-26 14:37:16 +0200188 return 0;
189}
190
Johannes Berg4bf88532012-11-09 11:39:59 +0100191static void ieee80211_recalc_chanctx_chantype(struct ieee80211_local *local,
192 struct ieee80211_chanctx *ctx)
Michal Kaziore89a96f2012-06-26 14:37:22 +0200193{
194 struct ieee80211_chanctx_conf *conf = &ctx->conf;
195 struct ieee80211_sub_if_data *sdata;
Johannes Berg4bf88532012-11-09 11:39:59 +0100196 const struct cfg80211_chan_def *compat = NULL;
Michal Kaziore89a96f2012-06-26 14:37:22 +0200197
198 lockdep_assert_held(&local->chanctx_mtx);
199
200 rcu_read_lock();
201 list_for_each_entry_rcu(sdata, &local->interfaces, list) {
Johannes Berg4bf88532012-11-09 11:39:59 +0100202
Michal Kaziore89a96f2012-06-26 14:37:22 +0200203 if (!ieee80211_sdata_running(sdata))
204 continue;
205 if (rcu_access_pointer(sdata->vif.chanctx_conf) != conf)
206 continue;
207
Johannes Berg4bf88532012-11-09 11:39:59 +0100208 if (!compat)
209 compat = &sdata->vif.bss_conf.chandef;
210
211 compat = cfg80211_chandef_compatible(
212 &sdata->vif.bss_conf.chandef, compat);
213 if (!compat)
214 break;
Michal Kaziore89a96f2012-06-26 14:37:22 +0200215 }
216 rcu_read_unlock();
217
Johannes Berg4bf88532012-11-09 11:39:59 +0100218 if (WARN_ON_ONCE(!compat))
219 return;
Michal Kaziore89a96f2012-06-26 14:37:22 +0200220
Johannes Berg18942d32013-02-07 21:30:37 +0100221 ieee80211_change_chanctx(local, ctx, compat);
Michal Kaziore89a96f2012-06-26 14:37:22 +0200222}
223
Michal Kaziord01a1e62012-06-26 14:37:16 +0200224static void ieee80211_unassign_vif_chanctx(struct ieee80211_sub_if_data *sdata,
225 struct ieee80211_chanctx *ctx)
226{
Michal Kazior35f2fce2012-06-26 14:37:20 +0200227 struct ieee80211_local *local = sdata->local;
Michal Kaziord01a1e62012-06-26 14:37:16 +0200228
229 lockdep_assert_held(&local->chanctx_mtx);
230
231 ctx->refcount--;
232 rcu_assign_pointer(sdata->vif.chanctx_conf, NULL);
Michal Kazior35f2fce2012-06-26 14:37:20 +0200233
Johannes Bergfd0f9792013-02-07 00:14:51 +0100234 sdata->vif.bss_conf.idle = true;
Johannes Berg5bbe754d2013-02-13 13:50:51 +0100235
236 if (sdata->vif.type != NL80211_IFTYPE_P2P_DEVICE &&
237 sdata->vif.type != NL80211_IFTYPE_MONITOR)
238 ieee80211_bss_info_change_notify(sdata, BSS_CHANGED_IDLE);
Johannes Bergfd0f9792013-02-07 00:14:51 +0100239
Michal Kazior35f2fce2012-06-26 14:37:20 +0200240 drv_unassign_vif_chanctx(local, sdata, ctx);
Michal Kaziore89a96f2012-06-26 14:37:22 +0200241
Johannes Berg04ecd252012-09-11 14:34:12 +0200242 if (ctx->refcount > 0) {
Michal Kaziore89a96f2012-06-26 14:37:22 +0200243 ieee80211_recalc_chanctx_chantype(sdata->local, ctx);
Johannes Berg04ecd252012-09-11 14:34:12 +0200244 ieee80211_recalc_smps_chanctx(local, ctx);
Simon Wunderlich164eb022013-02-08 18:16:20 +0100245 ieee80211_recalc_radar_chanctx(local, ctx);
Johannes Berg04ecd252012-09-11 14:34:12 +0200246 }
Michal Kaziord01a1e62012-06-26 14:37:16 +0200247}
248
249static void __ieee80211_vif_release_channel(struct ieee80211_sub_if_data *sdata)
250{
251 struct ieee80211_local *local = sdata->local;
252 struct ieee80211_chanctx_conf *conf;
253 struct ieee80211_chanctx *ctx;
254
255 lockdep_assert_held(&local->chanctx_mtx);
256
257 conf = rcu_dereference_protected(sdata->vif.chanctx_conf,
258 lockdep_is_held(&local->chanctx_mtx));
259 if (!conf)
260 return;
261
262 ctx = container_of(conf, struct ieee80211_chanctx, conf);
263
264 ieee80211_unassign_vif_chanctx(sdata, ctx);
265 if (ctx->refcount == 0)
266 ieee80211_free_chanctx(local, ctx);
267}
268
Simon Wunderlich164eb022013-02-08 18:16:20 +0100269void ieee80211_recalc_radar_chanctx(struct ieee80211_local *local,
270 struct ieee80211_chanctx *chanctx)
271{
Simon Wunderliche4746852013-04-08 22:43:16 +0200272 bool radar_enabled;
Simon Wunderlich164eb022013-02-08 18:16:20 +0100273
274 lockdep_assert_held(&local->chanctx_mtx);
275
Simon Wunderliche4746852013-04-08 22:43:16 +0200276 radar_enabled = ieee80211_is_radar_required(local);
Simon Wunderlich164eb022013-02-08 18:16:20 +0100277
278 if (radar_enabled == chanctx->conf.radar_enabled)
279 return;
280
281 chanctx->conf.radar_enabled = radar_enabled;
282 local->radar_detect_enabled = chanctx->conf.radar_enabled;
283
284 if (!local->use_chanctx) {
285 local->hw.conf.radar_enabled = chanctx->conf.radar_enabled;
286 ieee80211_hw_config(local, IEEE80211_CONF_CHANGE_CHANNEL);
287 }
288
289 drv_change_chanctx(local, chanctx, IEEE80211_CHANCTX_CHANGE_RADAR);
290}
291
Johannes Berg04ecd252012-09-11 14:34:12 +0200292void ieee80211_recalc_smps_chanctx(struct ieee80211_local *local,
293 struct ieee80211_chanctx *chanctx)
294{
295 struct ieee80211_sub_if_data *sdata;
296 u8 rx_chains_static, rx_chains_dynamic;
297
298 lockdep_assert_held(&local->chanctx_mtx);
299
300 rx_chains_static = 1;
301 rx_chains_dynamic = 1;
302
303 rcu_read_lock();
304 list_for_each_entry_rcu(sdata, &local->interfaces, list) {
305 u8 needed_static, needed_dynamic;
306
307 if (!ieee80211_sdata_running(sdata))
308 continue;
309
310 if (rcu_access_pointer(sdata->vif.chanctx_conf) !=
311 &chanctx->conf)
312 continue;
313
314 switch (sdata->vif.type) {
315 case NL80211_IFTYPE_P2P_DEVICE:
316 continue;
317 case NL80211_IFTYPE_STATION:
318 if (!sdata->u.mgd.associated)
319 continue;
320 break;
321 case NL80211_IFTYPE_AP_VLAN:
322 continue;
323 case NL80211_IFTYPE_AP:
324 case NL80211_IFTYPE_ADHOC:
325 case NL80211_IFTYPE_WDS:
326 case NL80211_IFTYPE_MESH_POINT:
327 break;
328 default:
329 WARN_ON_ONCE(1);
330 }
331
332 switch (sdata->smps_mode) {
333 default:
334 WARN_ONCE(1, "Invalid SMPS mode %d\n",
335 sdata->smps_mode);
336 /* fall through */
337 case IEEE80211_SMPS_OFF:
338 needed_static = sdata->needed_rx_chains;
339 needed_dynamic = sdata->needed_rx_chains;
340 break;
341 case IEEE80211_SMPS_DYNAMIC:
342 needed_static = 1;
343 needed_dynamic = sdata->needed_rx_chains;
344 break;
345 case IEEE80211_SMPS_STATIC:
346 needed_static = 1;
347 needed_dynamic = 1;
348 break;
349 }
350
351 rx_chains_static = max(rx_chains_static, needed_static);
352 rx_chains_dynamic = max(rx_chains_dynamic, needed_dynamic);
353 }
354 rcu_read_unlock();
355
356 if (!local->use_chanctx) {
357 if (rx_chains_static > 1)
358 local->smps_mode = IEEE80211_SMPS_OFF;
359 else if (rx_chains_dynamic > 1)
360 local->smps_mode = IEEE80211_SMPS_DYNAMIC;
361 else
362 local->smps_mode = IEEE80211_SMPS_STATIC;
363 ieee80211_hw_config(local, 0);
364 }
365
366 if (rx_chains_static == chanctx->conf.rx_chains_static &&
367 rx_chains_dynamic == chanctx->conf.rx_chains_dynamic)
368 return;
369
370 chanctx->conf.rx_chains_static = rx_chains_static;
371 chanctx->conf.rx_chains_dynamic = rx_chains_dynamic;
372 drv_change_chanctx(local, chanctx, IEEE80211_CHANCTX_CHANGE_RX_CHAINS);
373}
374
Michal Kaziord01a1e62012-06-26 14:37:16 +0200375int ieee80211_vif_use_channel(struct ieee80211_sub_if_data *sdata,
Johannes Berg4bf88532012-11-09 11:39:59 +0100376 const struct cfg80211_chan_def *chandef,
Michal Kaziord01a1e62012-06-26 14:37:16 +0200377 enum ieee80211_chanctx_mode mode)
378{
379 struct ieee80211_local *local = sdata->local;
380 struct ieee80211_chanctx *ctx;
381 int ret;
382
Johannes Berg55de9082012-07-26 17:24:39 +0200383 WARN_ON(sdata->dev && netif_carrier_ok(sdata->dev));
384
Michal Kaziord01a1e62012-06-26 14:37:16 +0200385 mutex_lock(&local->chanctx_mtx);
386 __ieee80211_vif_release_channel(sdata);
387
Johannes Berg4bf88532012-11-09 11:39:59 +0100388 ctx = ieee80211_find_chanctx(local, chandef, mode);
Michal Kaziord01a1e62012-06-26 14:37:16 +0200389 if (!ctx)
Johannes Berg4bf88532012-11-09 11:39:59 +0100390 ctx = ieee80211_new_chanctx(local, chandef, mode);
Michal Kaziord01a1e62012-06-26 14:37:16 +0200391 if (IS_ERR(ctx)) {
392 ret = PTR_ERR(ctx);
393 goto out;
394 }
395
Johannes Berg4bf88532012-11-09 11:39:59 +0100396 sdata->vif.bss_conf.chandef = *chandef;
Johannes Berg55de9082012-07-26 17:24:39 +0200397
Michal Kaziord01a1e62012-06-26 14:37:16 +0200398 ret = ieee80211_assign_vif_chanctx(sdata, ctx);
399 if (ret) {
400 /* if assign fails refcount stays the same */
401 if (ctx->refcount == 0)
402 ieee80211_free_chanctx(local, ctx);
403 goto out;
404 }
405
Johannes Berg04ecd252012-09-11 14:34:12 +0200406 ieee80211_recalc_smps_chanctx(local, ctx);
Simon Wunderlich164eb022013-02-08 18:16:20 +0100407 ieee80211_recalc_radar_chanctx(local, ctx);
Michal Kaziord01a1e62012-06-26 14:37:16 +0200408 out:
409 mutex_unlock(&local->chanctx_mtx);
410 return ret;
411}
412
Simon Wunderlich73da7d52013-07-11 16:09:06 +0200413int ieee80211_vif_change_channel(struct ieee80211_sub_if_data *sdata,
414 const struct cfg80211_chan_def *chandef,
415 u32 *changed)
416{
417 struct ieee80211_local *local = sdata->local;
418 struct ieee80211_chanctx_conf *conf;
419 struct ieee80211_chanctx *ctx;
420 int ret;
421 u32 chanctx_changed = 0;
422
423 /* should never be called if not performing a channel switch. */
424 if (WARN_ON(!sdata->vif.csa_active))
425 return -EINVAL;
426
427 if (!cfg80211_chandef_usable(sdata->local->hw.wiphy, chandef,
428 IEEE80211_CHAN_DISABLED))
429 return -EINVAL;
430
431 mutex_lock(&local->chanctx_mtx);
432 conf = rcu_dereference_protected(sdata->vif.chanctx_conf,
433 lockdep_is_held(&local->chanctx_mtx));
434 if (!conf) {
435 ret = -EINVAL;
436 goto out;
437 }
438
439 ctx = container_of(conf, struct ieee80211_chanctx, conf);
440 if (ctx->refcount != 1) {
441 ret = -EINVAL;
442 goto out;
443 }
444
445 if (sdata->vif.bss_conf.chandef.width != chandef->width) {
446 chanctx_changed = IEEE80211_CHANCTX_CHANGE_WIDTH;
447 *changed |= BSS_CHANGED_BANDWIDTH;
448 }
449
450 sdata->vif.bss_conf.chandef = *chandef;
451 ctx->conf.def = *chandef;
452
453 chanctx_changed |= IEEE80211_CHANCTX_CHANGE_CHANNEL;
454 drv_change_chanctx(local, ctx, chanctx_changed);
455
Simon Wunderlich73da7d52013-07-11 16:09:06 +0200456 ieee80211_recalc_chanctx_chantype(local, ctx);
457 ieee80211_recalc_smps_chanctx(local, ctx);
458 ieee80211_recalc_radar_chanctx(local, ctx);
459
460 ret = 0;
461 out:
462 mutex_unlock(&local->chanctx_mtx);
463 return ret;
464}
465
Johannes Berg2c9b7352013-02-07 21:37:29 +0100466int ieee80211_vif_change_bandwidth(struct ieee80211_sub_if_data *sdata,
467 const struct cfg80211_chan_def *chandef,
468 u32 *changed)
469{
470 struct ieee80211_local *local = sdata->local;
471 struct ieee80211_chanctx_conf *conf;
472 struct ieee80211_chanctx *ctx;
473 int ret;
474
475 if (!cfg80211_chandef_usable(sdata->local->hw.wiphy, chandef,
476 IEEE80211_CHAN_DISABLED))
477 return -EINVAL;
478
479 mutex_lock(&local->chanctx_mtx);
480 if (cfg80211_chandef_identical(chandef, &sdata->vif.bss_conf.chandef)) {
481 ret = 0;
482 goto out;
483 }
484
485 if (chandef->width == NL80211_CHAN_WIDTH_20_NOHT ||
486 sdata->vif.bss_conf.chandef.width == NL80211_CHAN_WIDTH_20_NOHT) {
487 ret = -EINVAL;
488 goto out;
489 }
490
491 conf = rcu_dereference_protected(sdata->vif.chanctx_conf,
492 lockdep_is_held(&local->chanctx_mtx));
493 if (!conf) {
494 ret = -EINVAL;
495 goto out;
496 }
497
498 ctx = container_of(conf, struct ieee80211_chanctx, conf);
499 if (!cfg80211_chandef_compatible(&conf->def, chandef)) {
500 ret = -EINVAL;
501 goto out;
502 }
503
504 sdata->vif.bss_conf.chandef = *chandef;
505
506 ieee80211_recalc_chanctx_chantype(local, ctx);
507
508 *changed |= BSS_CHANGED_BANDWIDTH;
509 ret = 0;
510 out:
511 mutex_unlock(&local->chanctx_mtx);
512 return ret;
513}
514
Michal Kaziord01a1e62012-06-26 14:37:16 +0200515void ieee80211_vif_release_channel(struct ieee80211_sub_if_data *sdata)
516{
Johannes Berg55de9082012-07-26 17:24:39 +0200517 WARN_ON(sdata->dev && netif_carrier_ok(sdata->dev));
518
Michal Kaziord01a1e62012-06-26 14:37:16 +0200519 mutex_lock(&sdata->local->chanctx_mtx);
520 __ieee80211_vif_release_channel(sdata);
521 mutex_unlock(&sdata->local->chanctx_mtx);
522}
Johannes Berg3448c002012-09-11 17:57:42 +0200523
Johannes Berg4d76d212012-12-11 20:38:41 +0100524void ieee80211_vif_vlan_copy_chanctx(struct ieee80211_sub_if_data *sdata)
525{
526 struct ieee80211_local *local = sdata->local;
527 struct ieee80211_sub_if_data *ap;
528 struct ieee80211_chanctx_conf *conf;
529
530 if (WARN_ON(sdata->vif.type != NL80211_IFTYPE_AP_VLAN || !sdata->bss))
531 return;
532
533 ap = container_of(sdata->bss, struct ieee80211_sub_if_data, u.ap);
534
535 mutex_lock(&local->chanctx_mtx);
536
537 conf = rcu_dereference_protected(ap->vif.chanctx_conf,
538 lockdep_is_held(&local->chanctx_mtx));
539 rcu_assign_pointer(sdata->vif.chanctx_conf, conf);
540 mutex_unlock(&local->chanctx_mtx);
541}
542
Johannes Berg1f4ac5a2013-02-08 12:07:44 +0100543void ieee80211_vif_copy_chanctx_to_vlans(struct ieee80211_sub_if_data *sdata,
544 bool clear)
545{
546 struct ieee80211_local *local = sdata->local;
547 struct ieee80211_sub_if_data *vlan;
548 struct ieee80211_chanctx_conf *conf;
549
550 ASSERT_RTNL();
551
552 if (WARN_ON(sdata->vif.type != NL80211_IFTYPE_AP))
553 return;
554
555 mutex_lock(&local->chanctx_mtx);
556
557 /*
558 * Check that conf exists, even when clearing this function
559 * must be called with the AP's channel context still there
560 * as it would otherwise cause VLANs to have an invalid
561 * channel context pointer for a while, possibly pointing
562 * to a channel context that has already been freed.
563 */
564 conf = rcu_dereference_protected(sdata->vif.chanctx_conf,
565 lockdep_is_held(&local->chanctx_mtx));
566 WARN_ON(!conf);
567
568 if (clear)
569 conf = NULL;
570
571 list_for_each_entry(vlan, &sdata->u.ap.vlans, u.vlan.list)
572 rcu_assign_pointer(vlan->vif.chanctx_conf, conf);
573
574 mutex_unlock(&local->chanctx_mtx);
575}
576
Johannes Berg3448c002012-09-11 17:57:42 +0200577void ieee80211_iter_chan_contexts_atomic(
578 struct ieee80211_hw *hw,
579 void (*iter)(struct ieee80211_hw *hw,
580 struct ieee80211_chanctx_conf *chanctx_conf,
581 void *data),
582 void *iter_data)
583{
584 struct ieee80211_local *local = hw_to_local(hw);
585 struct ieee80211_chanctx *ctx;
586
587 rcu_read_lock();
588 list_for_each_entry_rcu(ctx, &local->chanctx_list, list)
Johannes Berg8a61af62012-12-13 17:42:30 +0100589 if (ctx->driver_present)
590 iter(hw, &ctx->conf, iter_data);
Johannes Berg3448c002012-09-11 17:57:42 +0200591 rcu_read_unlock();
592}
593EXPORT_SYMBOL_GPL(ieee80211_iter_chan_contexts_atomic);