blob: 4ae94860a1613129b50194629f02f0dcc4f93e0a [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>
Paul Stewart3117bbdb2012-03-13 07:46:18 -07006#include <net/cfg80211.h>
Johannes Bergf444de02010-05-05 15:25:02 +02007#include "ieee80211_i.h"
8
Johannes Berg368a07d2010-05-28 14:26:23 +02009static enum ieee80211_chan_mode
Johannes Bergf444de02010-05-05 15:25:02 +020010__ieee80211_get_channel_mode(struct ieee80211_local *local,
11 struct ieee80211_sub_if_data *ignore)
12{
13 struct ieee80211_sub_if_data *sdata;
14
Johannes Berg46a5eba2010-09-15 13:28:15 +020015 lockdep_assert_held(&local->iflist_mtx);
Johannes Bergf444de02010-05-05 15:25:02 +020016
17 list_for_each_entry(sdata, &local->interfaces, list) {
18 if (sdata == ignore)
19 continue;
20
21 if (!ieee80211_sdata_running(sdata))
22 continue;
23
Johannes Berge9980e62012-01-09 13:57:36 +010024 switch (sdata->vif.type) {
25 case NL80211_IFTYPE_MONITOR:
Johannes Bergf444de02010-05-05 15:25:02 +020026 continue;
Johannes Berge9980e62012-01-09 13:57:36 +010027 case NL80211_IFTYPE_STATION:
28 if (!sdata->u.mgd.associated)
29 continue;
30 break;
31 case NL80211_IFTYPE_ADHOC:
Johannes Bergf444de02010-05-05 15:25:02 +020032 if (!sdata->u.ibss.ssid_len)
33 continue;
34 if (!sdata->u.ibss.fixed_channel)
35 return CHAN_MODE_HOPPING;
Johannes Berge9980e62012-01-09 13:57:36 +010036 break;
37 case NL80211_IFTYPE_AP_VLAN:
38 /* will also have _AP interface */
Johannes Bergf444de02010-05-05 15:25:02 +020039 continue;
Johannes Berge9980e62012-01-09 13:57:36 +010040 case NL80211_IFTYPE_AP:
41 if (!sdata->u.ap.beacon)
42 continue;
43 break;
Thomas Pedersenbe0f4232012-05-13 22:24:08 -070044 case NL80211_IFTYPE_MESH_POINT:
45 if (!sdata->wdev.mesh_id_len)
46 continue;
47 break;
Johannes Berge9980e62012-01-09 13:57:36 +010048 default:
49 break;
50 }
Johannes Bergf444de02010-05-05 15:25:02 +020051
52 return CHAN_MODE_FIXED;
53 }
54
55 return CHAN_MODE_UNDEFINED;
56}
57
58enum ieee80211_chan_mode
59ieee80211_get_channel_mode(struct ieee80211_local *local,
60 struct ieee80211_sub_if_data *ignore)
61{
62 enum ieee80211_chan_mode mode;
63
64 mutex_lock(&local->iflist_mtx);
65 mode = __ieee80211_get_channel_mode(local, ignore);
66 mutex_unlock(&local->iflist_mtx);
67
68 return mode;
69}
Johannes Berg0aaffa92010-05-05 15:28:27 +020070
Michal Kazior23a85b452012-06-26 14:37:21 +020071static enum nl80211_channel_type
72ieee80211_get_superchan(struct ieee80211_local *local,
73 struct ieee80211_sub_if_data *sdata)
Johannes Berg0aaffa92010-05-05 15:28:27 +020074{
Johannes Berg0aaffa92010-05-05 15:28:27 +020075 enum nl80211_channel_type superchan = NL80211_CHAN_NO_HT;
Michal Kazior23a85b452012-06-26 14:37:21 +020076 struct ieee80211_sub_if_data *tmp;
Johannes Berg0aaffa92010-05-05 15:28:27 +020077
78 mutex_lock(&local->iflist_mtx);
Johannes Berg0aaffa92010-05-05 15:28:27 +020079 list_for_each_entry(tmp, &local->interfaces, list) {
80 if (tmp == sdata)
81 continue;
82
83 if (!ieee80211_sdata_running(tmp))
84 continue;
85
86 switch (tmp->vif.bss_conf.channel_type) {
87 case NL80211_CHAN_NO_HT:
88 case NL80211_CHAN_HT20:
Felix Fietkau9db372f2011-03-11 21:45:51 +010089 if (superchan > tmp->vif.bss_conf.channel_type)
90 break;
91
Johannes Berg0aaffa92010-05-05 15:28:27 +020092 superchan = tmp->vif.bss_conf.channel_type;
93 break;
94 case NL80211_CHAN_HT40PLUS:
95 WARN_ON(superchan == NL80211_CHAN_HT40MINUS);
96 superchan = NL80211_CHAN_HT40PLUS;
97 break;
98 case NL80211_CHAN_HT40MINUS:
99 WARN_ON(superchan == NL80211_CHAN_HT40PLUS);
100 superchan = NL80211_CHAN_HT40MINUS;
101 break;
102 }
103 }
Michal Kazior23a85b452012-06-26 14:37:21 +0200104 mutex_unlock(&local->iflist_mtx);
Johannes Berg0aaffa92010-05-05 15:28:27 +0200105
Michal Kazior23a85b452012-06-26 14:37:21 +0200106 return superchan;
107}
108
109static bool
110ieee80211_channel_types_are_compatible(enum nl80211_channel_type chantype1,
111 enum nl80211_channel_type chantype2,
112 enum nl80211_channel_type *compat)
113{
114 /*
115 * start out with chantype1 being the result,
116 * overwriting later if needed
117 */
118 if (compat)
119 *compat = chantype1;
120
121 switch (chantype1) {
Johannes Berg0aaffa92010-05-05 15:28:27 +0200122 case NL80211_CHAN_NO_HT:
Michal Kazior23a85b452012-06-26 14:37:21 +0200123 if (compat)
124 *compat = chantype2;
125 break;
Johannes Berg0aaffa92010-05-05 15:28:27 +0200126 case NL80211_CHAN_HT20:
127 /*
128 * allow any change that doesn't go to no-HT
129 * (if it already is no-HT no change is needed)
130 */
Michal Kazior23a85b452012-06-26 14:37:21 +0200131 if (chantype2 == NL80211_CHAN_NO_HT)
Johannes Berg0aaffa92010-05-05 15:28:27 +0200132 break;
Michal Kazior23a85b452012-06-26 14:37:21 +0200133 if (compat)
134 *compat = chantype2;
Johannes Berg0aaffa92010-05-05 15:28:27 +0200135 break;
136 case NL80211_CHAN_HT40PLUS:
137 case NL80211_CHAN_HT40MINUS:
138 /* allow smaller bandwidth and same */
Michal Kazior23a85b452012-06-26 14:37:21 +0200139 if (chantype2 == NL80211_CHAN_NO_HT)
Johannes Berg0aaffa92010-05-05 15:28:27 +0200140 break;
Michal Kazior23a85b452012-06-26 14:37:21 +0200141 if (chantype2 == NL80211_CHAN_HT20)
Johannes Berg0aaffa92010-05-05 15:28:27 +0200142 break;
Michal Kazior23a85b452012-06-26 14:37:21 +0200143 if (chantype2 == chantype1)
Johannes Berg0aaffa92010-05-05 15:28:27 +0200144 break;
Michal Kazior23a85b452012-06-26 14:37:21 +0200145 return false;
Johannes Berg0aaffa92010-05-05 15:28:27 +0200146 }
147
Michal Kazior23a85b452012-06-26 14:37:21 +0200148 return true;
149}
150
151bool ieee80211_set_channel_type(struct ieee80211_local *local,
152 struct ieee80211_sub_if_data *sdata,
153 enum nl80211_channel_type chantype)
154{
155 enum nl80211_channel_type superchan;
156 enum nl80211_channel_type compatchan;
157
158 superchan = ieee80211_get_superchan(local, sdata);
159 if (!ieee80211_channel_types_are_compatible(superchan, chantype,
160 &compatchan))
161 return false;
162
163 local->_oper_channel_type = compatchan;
Johannes Berg0aaffa92010-05-05 15:28:27 +0200164
165 if (sdata)
166 sdata->vif.bss_conf.channel_type = chantype;
167
Michal Kazior23a85b452012-06-26 14:37:21 +0200168 return true;
Johannes Berg0aaffa92010-05-05 15:28:27 +0200169
Johannes Berg0aaffa92010-05-05 15:28:27 +0200170}
Michal Kaziord01a1e62012-06-26 14:37:16 +0200171
172static struct ieee80211_chanctx *
173ieee80211_find_chanctx(struct ieee80211_local *local,
174 struct ieee80211_channel *channel,
175 enum nl80211_channel_type channel_type,
176 enum ieee80211_chanctx_mode mode)
177{
178 struct ieee80211_chanctx *ctx;
179
180 lockdep_assert_held(&local->chanctx_mtx);
181
182 if (mode == IEEE80211_CHANCTX_EXCLUSIVE)
183 return NULL;
184 if (WARN_ON(!channel))
185 return NULL;
186
187 list_for_each_entry(ctx, &local->chanctx_list, list) {
188 if (ctx->mode == IEEE80211_CHANCTX_EXCLUSIVE)
189 continue;
190 if (ctx->conf.channel != channel)
191 continue;
192 if (ctx->conf.channel_type != channel_type)
193 continue;
194
195 return ctx;
196 }
197
198 return NULL;
199}
200
201static struct ieee80211_chanctx *
202ieee80211_new_chanctx(struct ieee80211_local *local,
203 struct ieee80211_channel *channel,
204 enum nl80211_channel_type channel_type,
205 enum ieee80211_chanctx_mode mode)
206{
207 struct ieee80211_chanctx *ctx;
208
209 lockdep_assert_held(&local->chanctx_mtx);
210
211 ctx = kzalloc(sizeof(*ctx) + local->hw.chanctx_data_size, GFP_KERNEL);
212 if (!ctx)
213 return ERR_PTR(-ENOMEM);
214
215 ctx->conf.channel = channel;
216 ctx->conf.channel_type = channel_type;
217 ctx->mode = mode;
218
219 list_add(&ctx->list, &local->chanctx_list);
220
221 return ctx;
222}
223
224static void ieee80211_free_chanctx(struct ieee80211_local *local,
225 struct ieee80211_chanctx *ctx)
226{
227 lockdep_assert_held(&local->chanctx_mtx);
228
229 WARN_ON_ONCE(ctx->refcount != 0);
230
231 list_del(&ctx->list);
232 kfree_rcu(ctx, rcu_head);
233}
234
235static int ieee80211_assign_vif_chanctx(struct ieee80211_sub_if_data *sdata,
236 struct ieee80211_chanctx *ctx)
237{
238 struct ieee80211_local *local __maybe_unused = sdata->local;
239
240 lockdep_assert_held(&local->chanctx_mtx);
241
242 rcu_assign_pointer(sdata->vif.chanctx_conf, &ctx->conf);
243 ctx->refcount++;
244
245 return 0;
246}
247
248static void ieee80211_unassign_vif_chanctx(struct ieee80211_sub_if_data *sdata,
249 struct ieee80211_chanctx *ctx)
250{
251 struct ieee80211_local *local __maybe_unused = sdata->local;
252
253 lockdep_assert_held(&local->chanctx_mtx);
254
255 ctx->refcount--;
256 rcu_assign_pointer(sdata->vif.chanctx_conf, NULL);
257}
258
259static void __ieee80211_vif_release_channel(struct ieee80211_sub_if_data *sdata)
260{
261 struct ieee80211_local *local = sdata->local;
262 struct ieee80211_chanctx_conf *conf;
263 struct ieee80211_chanctx *ctx;
264
265 lockdep_assert_held(&local->chanctx_mtx);
266
267 conf = rcu_dereference_protected(sdata->vif.chanctx_conf,
268 lockdep_is_held(&local->chanctx_mtx));
269 if (!conf)
270 return;
271
272 ctx = container_of(conf, struct ieee80211_chanctx, conf);
273
274 ieee80211_unassign_vif_chanctx(sdata, ctx);
275 if (ctx->refcount == 0)
276 ieee80211_free_chanctx(local, ctx);
277}
278
279int ieee80211_vif_use_channel(struct ieee80211_sub_if_data *sdata,
280 struct ieee80211_channel *channel,
281 enum nl80211_channel_type channel_type,
282 enum ieee80211_chanctx_mode mode)
283{
284 struct ieee80211_local *local = sdata->local;
285 struct ieee80211_chanctx *ctx;
286 int ret;
287
288 mutex_lock(&local->chanctx_mtx);
289 __ieee80211_vif_release_channel(sdata);
290
291 ctx = ieee80211_find_chanctx(local, channel, channel_type, mode);
292 if (!ctx)
293 ctx = ieee80211_new_chanctx(local, channel, channel_type, mode);
294 if (IS_ERR(ctx)) {
295 ret = PTR_ERR(ctx);
296 goto out;
297 }
298
299 ret = ieee80211_assign_vif_chanctx(sdata, ctx);
300 if (ret) {
301 /* if assign fails refcount stays the same */
302 if (ctx->refcount == 0)
303 ieee80211_free_chanctx(local, ctx);
304 goto out;
305 }
306
307 out:
308 mutex_unlock(&local->chanctx_mtx);
309 return ret;
310}
311
312void ieee80211_vif_release_channel(struct ieee80211_sub_if_data *sdata)
313{
314 mutex_lock(&sdata->local->chanctx_mtx);
315 __ieee80211_vif_release_channel(sdata);
316 mutex_unlock(&sdata->local->chanctx_mtx);
317}