aboutsummaryrefslogtreecommitdiffstats
path: root/net/mac80211/ibss.c
diff options
context:
space:
mode:
authorAlexander Simon <an.alexsimon@googlemail.com>2011-11-30 10:56:34 -0500
committerJohn W. Linville <linville@tuxdriver.com>2011-12-06 16:05:25 -0500
commit13c40c54682ffe62977f670681268a26d500d6fa (patch)
tree4aa8897bc4054f13a637f2f29a1555c3501b3e97 /net/mac80211/ibss.c
parentff3cc5f40f36db1a60a8f1051be7fbc92233419b (diff)
mac80211: Add HT operation modes for IBSS
The HT mode is set by iw (previous patchsets). The interface is set into the specified HT mode. HT mode and capabilities are announced in beacons. If we add a station that uses HT also, the fastest matching HT mode will be used for transmission. That means if we are using HT40+ and we add a station running on HT40-, we would transfer at HT20. If we join an IBSS with HT40, but the secondary channel is not available, we will fall back into HT20 as well. Allow frame aggregation to start in IBSS mode. Signed-off-by: Alexander Simon <an.alexsimon@googlemail.com> [siwu@hrz.tu-chemnitz.de: Updates] * remove implicit channel_type enum assumptions * use rate_control_rate_init() if channel type changed * remove channel flags check * activate HT IBSS feature support * slightly reword commit message * rebase on wireless-testing Signed-off-by: Simon Wunderlich <siwu@hrz.tu-chemnitz.de> Signed-off-by: Mathias Kretschmer <mathias.kretschmer@fokus.fraunhofer.de> Reviewed-by: Johannes Berg <johannes@sipsolutions.net> Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'net/mac80211/ibss.c')
-rw-r--r--net/mac80211/ibss.c83
1 files changed, 74 insertions, 9 deletions
diff --git a/net/mac80211/ibss.c b/net/mac80211/ibss.c
index a82f6b4adb58..3f830ac159e5 100644
--- a/net/mac80211/ibss.c
+++ b/net/mac80211/ibss.c
@@ -77,6 +77,7 @@ static void __ieee80211_sta_join_ibss(struct ieee80211_sub_if_data *sdata,
77 struct cfg80211_bss *bss; 77 struct cfg80211_bss *bss;
78 u32 bss_change; 78 u32 bss_change;
79 u8 supp_rates[IEEE80211_MAX_SUPP_RATES]; 79 u8 supp_rates[IEEE80211_MAX_SUPP_RATES];
80 enum nl80211_channel_type channel_type;
80 81
81 lockdep_assert_held(&ifibss->mtx); 82 lockdep_assert_held(&ifibss->mtx);
82 83
@@ -105,8 +106,16 @@ static void __ieee80211_sta_join_ibss(struct ieee80211_sub_if_data *sdata,
105 106
106 sdata->drop_unencrypted = capability & WLAN_CAPABILITY_PRIVACY ? 1 : 0; 107 sdata->drop_unencrypted = capability & WLAN_CAPABILITY_PRIVACY ? 1 : 0;
107 108
108 local->oper_channel = chan; 109 channel_type = ifibss->channel_type;
109 WARN_ON(!ieee80211_set_channel_type(local, sdata, NL80211_CHAN_NO_HT)); 110 if (channel_type > NL80211_CHAN_HT20 &&
111 !cfg80211_can_beacon_sec_chan(local->hw.wiphy, chan, channel_type))
112 channel_type = NL80211_CHAN_HT20;
113 if (!ieee80211_set_channel_type(local, sdata, channel_type)) {
114 /* can only fail due to HT40+/- mismatch */
115 channel_type = NL80211_CHAN_HT20;
116 WARN_ON(!ieee80211_set_channel_type(local, sdata,
117 NL80211_CHAN_HT20));
118 }
110 ieee80211_hw_config(local, IEEE80211_CONF_CHANGE_CHANNEL); 119 ieee80211_hw_config(local, IEEE80211_CONF_CHANGE_CHANNEL);
111 120
112 sband = local->hw.wiphy->bands[chan->band]; 121 sband = local->hw.wiphy->bands[chan->band];
@@ -172,6 +181,19 @@ static void __ieee80211_sta_join_ibss(struct ieee80211_sub_if_data *sdata,
172 memcpy(skb_put(skb, ifibss->ie_len), 181 memcpy(skb_put(skb, ifibss->ie_len),
173 ifibss->ie, ifibss->ie_len); 182 ifibss->ie, ifibss->ie_len);
174 183
184 /* add HT capability and information IEs */
185 if (channel_type && sband->ht_cap.ht_supported) {
186 pos = skb_put(skb, 4 +
187 sizeof(struct ieee80211_ht_cap) +
188 sizeof(struct ieee80211_ht_info));
189 pos = ieee80211_ie_build_ht_cap(pos, &sband->ht_cap,
190 sband->ht_cap.cap);
191 pos = ieee80211_ie_build_ht_info(pos,
192 &sband->ht_cap,
193 chan,
194 channel_type);
195 }
196
175 if (local->hw.queues >= 4) { 197 if (local->hw.queues >= 4) {
176 pos = skb_put(skb, 9); 198 pos = skb_put(skb, 9);
177 *pos++ = WLAN_EID_VENDOR_SPECIFIC; 199 *pos++ = WLAN_EID_VENDOR_SPECIFIC;
@@ -195,6 +217,7 @@ static void __ieee80211_sta_join_ibss(struct ieee80211_sub_if_data *sdata,
195 bss_change |= BSS_CHANGED_BEACON; 217 bss_change |= BSS_CHANGED_BEACON;
196 bss_change |= BSS_CHANGED_BEACON_ENABLED; 218 bss_change |= BSS_CHANGED_BEACON_ENABLED;
197 bss_change |= BSS_CHANGED_BASIC_RATES; 219 bss_change |= BSS_CHANGED_BASIC_RATES;
220 bss_change |= BSS_CHANGED_HT;
198 bss_change |= BSS_CHANGED_IBSS; 221 bss_change |= BSS_CHANGED_IBSS;
199 sdata->vif.bss_conf.ibss_joined = true; 222 sdata->vif.bss_conf.ibss_joined = true;
200 ieee80211_bss_info_change_notify(sdata, bss_change); 223 ieee80211_bss_info_change_notify(sdata, bss_change);
@@ -268,6 +291,8 @@ static void ieee80211_rx_bss_info(struct ieee80211_sub_if_data *sdata,
268 u64 beacon_timestamp, rx_timestamp; 291 u64 beacon_timestamp, rx_timestamp;
269 u32 supp_rates = 0; 292 u32 supp_rates = 0;
270 enum ieee80211_band band = rx_status->band; 293 enum ieee80211_band band = rx_status->band;
294 struct ieee80211_supported_band *sband = local->hw.wiphy->bands[band];
295 bool rates_updated = false;
271 296
272 if (elems->ds_params && elems->ds_params_len == 1) 297 if (elems->ds_params && elems->ds_params_len == 1)
273 freq = ieee80211_channel_to_frequency(elems->ds_params[0], 298 freq = ieee80211_channel_to_frequency(elems->ds_params[0],
@@ -307,7 +332,7 @@ static void ieee80211_rx_bss_info(struct ieee80211_sub_if_data *sdata,
307 prev_rates, 332 prev_rates,
308 sta->sta.supp_rates[band]); 333 sta->sta.supp_rates[band]);
309#endif 334#endif
310 rate_control_rate_init(sta); 335 rates_updated = true;
311 } 336 }
312 } else 337 } else
313 sta = ieee80211_ibss_add_sta(sdata, mgmt->bssid, 338 sta = ieee80211_ibss_add_sta(sdata, mgmt->bssid,
@@ -318,6 +343,39 @@ static void ieee80211_rx_bss_info(struct ieee80211_sub_if_data *sdata,
318 if (sta && elems->wmm_info) 343 if (sta && elems->wmm_info)
319 set_sta_flag(sta, WLAN_STA_WME); 344 set_sta_flag(sta, WLAN_STA_WME);
320 345
346 if (sta && elems->ht_info_elem && elems->ht_cap_elem &&
347 sdata->u.ibss.channel_type != NL80211_CHAN_NO_HT) {
348 /* we both use HT */
349 struct ieee80211_sta_ht_cap sta_ht_cap_new;
350 enum nl80211_channel_type channel_type =
351 ieee80211_ht_info_to_channel_type(
352 elems->ht_info_elem);
353
354 ieee80211_ht_cap_ie_to_sta_ht_cap(sdata, sband,
355 elems->ht_cap_elem,
356 &sta_ht_cap_new);
357
358 /*
359 * fall back to HT20 if we don't use or use
360 * the other extension channel
361 */
362 if ((channel_type == NL80211_CHAN_HT40MINUS ||
363 channel_type == NL80211_CHAN_HT40PLUS) &&
364 channel_type != sdata->u.ibss.channel_type)
365 sta_ht_cap_new.cap &=
366 ~IEEE80211_HT_CAP_SUP_WIDTH_20_40;
367
368 if (memcmp(&sta->sta.ht_cap, &sta_ht_cap_new,
369 sizeof(sta_ht_cap_new))) {
370 memcpy(&sta->sta.ht_cap, &sta_ht_cap_new,
371 sizeof(sta_ht_cap_new));
372 rates_updated = true;
373 }
374 }
375
376 if (sta && rates_updated)
377 rate_control_rate_init(sta);
378
321 rcu_read_unlock(); 379 rcu_read_unlock();
322 } 380 }
323 381
@@ -899,10 +957,15 @@ int ieee80211_ibss_join(struct ieee80211_sub_if_data *sdata,
899 u32 changed = 0; 957 u32 changed = 0;
900 958
901 skb = dev_alloc_skb(sdata->local->hw.extra_tx_headroom + 959 skb = dev_alloc_skb(sdata->local->hw.extra_tx_headroom +
902 36 /* bitrates */ + 960 sizeof(struct ieee80211_hdr_3addr) +
903 34 /* SSID */ + 961 12 /* struct ieee80211_mgmt.u.beacon */ +
904 3 /* DS params */ + 962 2 + IEEE80211_MAX_SSID_LEN /* max SSID */ +
905 4 /* IBSS params */ + 963 2 + 8 /* max Supported Rates */ +
964 3 /* max DS params */ +
965 4 /* IBSS params */ +
966 2 + (IEEE80211_MAX_SUPP_RATES - 8) +
967 2 + sizeof(struct ieee80211_ht_cap) +
968 2 + sizeof(struct ieee80211_ht_info) +
906 params->ie_len); 969 params->ie_len);
907 if (!skb) 970 if (!skb)
908 return -ENOMEM; 971 return -ENOMEM;
@@ -923,13 +986,15 @@ int ieee80211_ibss_join(struct ieee80211_sub_if_data *sdata,
923 sdata->vif.bss_conf.beacon_int = params->beacon_interval; 986 sdata->vif.bss_conf.beacon_int = params->beacon_interval;
924 987
925 sdata->u.ibss.channel = params->channel; 988 sdata->u.ibss.channel = params->channel;
989 sdata->u.ibss.channel_type = params->channel_type;
926 sdata->u.ibss.fixed_channel = params->channel_fixed; 990 sdata->u.ibss.fixed_channel = params->channel_fixed;
927 991
928 /* fix ourselves to that channel now already */ 992 /* fix ourselves to that channel now already */
929 if (params->channel_fixed) { 993 if (params->channel_fixed) {
930 sdata->local->oper_channel = params->channel; 994 sdata->local->oper_channel = params->channel;
931 WARN_ON(!ieee80211_set_channel_type(sdata->local, sdata, 995 if (!ieee80211_set_channel_type(sdata->local, sdata,
932 NL80211_CHAN_NO_HT)); 996 params->channel_type))
997 return -EINVAL;
933 } 998 }
934 999
935 if (params->ie) { 1000 if (params->ie) {