aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net')
-rw-r--r--drivers/net/wireless/iwlwifi/iwl4965-base.c182
1 files changed, 101 insertions, 81 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl4965-base.c b/drivers/net/wireless/iwlwifi/iwl4965-base.c
index 028523af0104..4bfc670e8784 100644
--- a/drivers/net/wireless/iwlwifi/iwl4965-base.c
+++ b/drivers/net/wireless/iwlwifi/iwl4965-base.c
@@ -99,7 +99,7 @@ __le16 *ieee80211_get_qos_ctrl(struct ieee80211_hdr *hdr)
99 return NULL; 99 return NULL;
100} 100}
101 101
102static const struct ieee80211_supported_band *iwl4965_get_hw_mode( 102static const struct ieee80211_supported_band *iwl_get_hw_mode(
103 struct iwl_priv *priv, enum ieee80211_band band) 103 struct iwl_priv *priv, enum ieee80211_band band)
104{ 104{
105 return priv->hw->wiphy->bands[band]; 105 return priv->hw->wiphy->bands[band];
@@ -1166,6 +1166,91 @@ static u16 iwl4965_supported_rate_to_ie(u8 *ie, u16 supported_rate,
1166 return ret_rates; 1166 return ret_rates;
1167} 1167}
1168 1168
1169#ifdef CONFIG_IWL4965_HT
1170static void iwl4965_ht_conf(struct iwl_priv *priv,
1171 struct ieee80211_bss_conf *bss_conf)
1172{
1173 struct ieee80211_ht_info *ht_conf = bss_conf->ht_conf;
1174 struct ieee80211_ht_bss_info *ht_bss_conf = bss_conf->ht_bss_conf;
1175 struct iwl_ht_info *iwl_conf = &priv->current_ht_config;
1176
1177 IWL_DEBUG_MAC80211("enter: \n");
1178
1179 iwl_conf->is_ht = bss_conf->assoc_ht;
1180
1181 if (!iwl_conf->is_ht)
1182 return;
1183
1184 priv->ps_mode = (u8)((ht_conf->cap & IEEE80211_HT_CAP_MIMO_PS) >> 2);
1185
1186 if (ht_conf->cap & IEEE80211_HT_CAP_SGI_20)
1187 iwl_conf->sgf |= 0x1;
1188 if (ht_conf->cap & IEEE80211_HT_CAP_SGI_40)
1189 iwl_conf->sgf |= 0x2;
1190
1191 iwl_conf->is_green_field = !!(ht_conf->cap & IEEE80211_HT_CAP_GRN_FLD);
1192 iwl_conf->max_amsdu_size =
1193 !!(ht_conf->cap & IEEE80211_HT_CAP_MAX_AMSDU);
1194
1195 iwl_conf->supported_chan_width =
1196 !!(ht_conf->cap & IEEE80211_HT_CAP_SUP_WIDTH);
1197 iwl_conf->extension_chan_offset =
1198 ht_bss_conf->bss_cap & IEEE80211_HT_IE_CHA_SEC_OFFSET;
1199 /* If no above or below channel supplied disable FAT channel */
1200 if (iwl_conf->extension_chan_offset != IWL_EXT_CHANNEL_OFFSET_ABOVE &&
1201 iwl_conf->extension_chan_offset != IWL_EXT_CHANNEL_OFFSET_BELOW)
1202 iwl_conf->supported_chan_width = 0;
1203
1204 iwl_conf->tx_mimo_ps_mode =
1205 (u8)((ht_conf->cap & IEEE80211_HT_CAP_MIMO_PS) >> 2);
1206 memcpy(iwl_conf->supp_mcs_set, ht_conf->supp_mcs_set, 16);
1207
1208 iwl_conf->control_channel = ht_bss_conf->primary_channel;
1209 iwl_conf->tx_chan_width =
1210 !!(ht_bss_conf->bss_cap & IEEE80211_HT_IE_CHA_WIDTH);
1211 iwl_conf->ht_protection =
1212 ht_bss_conf->bss_op_mode & IEEE80211_HT_IE_HT_PROTECTION;
1213 iwl_conf->non_GF_STA_present =
1214 !!(ht_bss_conf->bss_op_mode & IEEE80211_HT_IE_NON_GF_STA_PRSNT);
1215
1216 IWL_DEBUG_MAC80211("control channel %d\n", iwl_conf->control_channel);
1217 IWL_DEBUG_MAC80211("leave\n");
1218}
1219
1220static void iwl_ht_cap_to_ie(const struct ieee80211_supported_band *sband,
1221 u8 *pos, int *left)
1222{
1223 struct ieee80211_ht_cap *ht_cap;
1224
1225 if (!sband || !sband->ht_info.ht_supported)
1226 return;
1227
1228 if (*left < sizeof(struct ieee80211_ht_cap))
1229 return;
1230
1231 *pos++ = sizeof(struct ieee80211_ht_cap);
1232 ht_cap = (struct ieee80211_ht_cap *) pos;
1233
1234 ht_cap->cap_info = cpu_to_le16(sband->ht_info.cap);
1235 memcpy(ht_cap->supp_mcs_set, sband->ht_info.supp_mcs_set, 16);
1236 ht_cap->ampdu_params_info =
1237 (sband->ht_info.ampdu_factor & IEEE80211_HT_CAP_AMPDU_FACTOR) |
1238 ((sband->ht_info.ampdu_density << 2) &
1239 IEEE80211_HT_CAP_AMPDU_DENSITY);
1240 *left -= sizeof(struct ieee80211_ht_cap);
1241}
1242#else
1243static inline void iwl4965_ht_conf(struct iwl_priv *priv,
1244 struct ieee80211_bss_conf *bss_conf)
1245{
1246}
1247static void iwl_ht_cap_to_ie(const struct ieee80211_supported_band *sband,
1248 u8 *pos, int *left)
1249{
1250}
1251#endif
1252
1253
1169/** 1254/**
1170 * iwl4965_fill_probe_req - fill in all required fields and IE for probe request 1255 * iwl4965_fill_probe_req - fill in all required fields and IE for probe request
1171 */ 1256 */
@@ -1177,10 +1262,8 @@ static u16 iwl4965_fill_probe_req(struct iwl_priv *priv,
1177 int len = 0; 1262 int len = 0;
1178 u8 *pos = NULL; 1263 u8 *pos = NULL;
1179 u16 active_rates, ret_rates, cck_rates, active_rate_basic; 1264 u16 active_rates, ret_rates, cck_rates, active_rate_basic;
1180#ifdef CONFIG_IWL4965_HT
1181 const struct ieee80211_supported_band *sband = 1265 const struct ieee80211_supported_band *sband =
1182 iwl4965_get_hw_mode(priv, band); 1266 iwl_get_hw_mode(priv, band);
1183#endif /* CONFIG_IWL4965_HT */
1184 1267
1185 /* Make sure there is enough space for the probe request, 1268 /* Make sure there is enough space for the probe request,
1186 * two mandatory IEs and the data */ 1269 * two mandatory IEs and the data */
@@ -1263,24 +1346,19 @@ static u16 iwl4965_fill_probe_req(struct iwl_priv *priv,
1263 if (*pos > 0) 1346 if (*pos > 0)
1264 len += 2 + *pos; 1347 len += 2 + *pos;
1265 1348
1266#ifdef CONFIG_IWL4965_HT
1267 if (sband && sband->ht_info.ht_supported) {
1268 struct ieee80211_ht_cap *ht_cap;
1269 pos += (*pos) + 1;
1270 *pos++ = WLAN_EID_HT_CAPABILITY;
1271 *pos++ = sizeof(struct ieee80211_ht_cap);
1272 ht_cap = (struct ieee80211_ht_cap *)pos;
1273 ht_cap->cap_info = cpu_to_le16(sband->ht_info.cap);
1274 memcpy(ht_cap->supp_mcs_set, sband->ht_info.supp_mcs_set, 16);
1275 ht_cap->ampdu_params_info =(sband->ht_info.ampdu_factor &
1276 IEEE80211_HT_CAP_AMPDU_FACTOR) |
1277 ((sband->ht_info.ampdu_density << 2) &
1278 IEEE80211_HT_CAP_AMPDU_DENSITY);
1279 len += 2 + sizeof(struct ieee80211_ht_cap);
1280 }
1281#endif /*CONFIG_IWL4965_HT */
1282
1283 fill_end: 1349 fill_end:
1350 /* fill in HT IE */
1351 left -= 2;
1352 if (left < 0)
1353 return 0;
1354
1355 *pos++ = WLAN_EID_HT_CAPABILITY;
1356 *pos = 0;
1357
1358 iwl_ht_cap_to_ie(sband, pos, &left);
1359
1360 if (*pos > 0)
1361 len += 2 + *pos;
1284 return (u16)len; 1362 return (u16)len;
1285} 1363}
1286 1364
@@ -2146,7 +2224,7 @@ static void iwl4965_set_rate(struct iwl_priv *priv)
2146 struct ieee80211_rate *rate; 2224 struct ieee80211_rate *rate;
2147 int i; 2225 int i;
2148 2226
2149 hw = iwl4965_get_hw_mode(priv, priv->band); 2227 hw = iwl_get_hw_mode(priv, priv->band);
2150 if (!hw) { 2228 if (!hw) {
2151 IWL_ERROR("Failed to set rate: unable to get hw mode\n"); 2229 IWL_ERROR("Failed to set rate: unable to get hw mode\n");
2152 return; 2230 return;
@@ -4360,7 +4438,7 @@ static int iwl4965_get_channels_for_scan(struct iwl_priv *priv,
4360 u16 active_dwell = 0; 4438 u16 active_dwell = 0;
4361 int added, i; 4439 int added, i;
4362 4440
4363 sband = iwl4965_get_hw_mode(priv, band); 4441 sband = iwl_get_hw_mode(priv, band);
4364 if (!sband) 4442 if (!sband)
4365 return 0; 4443 return 0;
4366 4444
@@ -6328,64 +6406,6 @@ static void iwl4965_mac_remove_interface(struct ieee80211_hw *hw,
6328 6406
6329} 6407}
6330 6408
6331
6332#ifdef CONFIG_IWL4965_HT
6333static void iwl4965_ht_conf(struct iwl_priv *priv,
6334 struct ieee80211_bss_conf *bss_conf)
6335{
6336 struct ieee80211_ht_info *ht_conf = bss_conf->ht_conf;
6337 struct ieee80211_ht_bss_info *ht_bss_conf = bss_conf->ht_bss_conf;
6338 struct iwl_ht_info *iwl_conf = &priv->current_ht_config;
6339
6340 IWL_DEBUG_MAC80211("enter: \n");
6341
6342 iwl_conf->is_ht = bss_conf->assoc_ht;
6343
6344 if (!iwl_conf->is_ht)
6345 return;
6346
6347 priv->ps_mode = (u8)((ht_conf->cap & IEEE80211_HT_CAP_MIMO_PS) >> 2);
6348
6349 if (ht_conf->cap & IEEE80211_HT_CAP_SGI_20)
6350 iwl_conf->sgf |= 0x1;
6351 if (ht_conf->cap & IEEE80211_HT_CAP_SGI_40)
6352 iwl_conf->sgf |= 0x2;
6353
6354 iwl_conf->is_green_field = !!(ht_conf->cap & IEEE80211_HT_CAP_GRN_FLD);
6355 iwl_conf->max_amsdu_size =
6356 !!(ht_conf->cap & IEEE80211_HT_CAP_MAX_AMSDU);
6357
6358 iwl_conf->supported_chan_width =
6359 !!(ht_conf->cap & IEEE80211_HT_CAP_SUP_WIDTH);
6360 iwl_conf->extension_chan_offset =
6361 ht_bss_conf->bss_cap & IEEE80211_HT_IE_CHA_SEC_OFFSET;
6362 /* If no above or below channel supplied disable FAT channel */
6363 if (iwl_conf->extension_chan_offset != IWL_EXT_CHANNEL_OFFSET_ABOVE &&
6364 iwl_conf->extension_chan_offset != IWL_EXT_CHANNEL_OFFSET_BELOW)
6365 iwl_conf->supported_chan_width = 0;
6366
6367 iwl_conf->tx_mimo_ps_mode =
6368 (u8)((ht_conf->cap & IEEE80211_HT_CAP_MIMO_PS) >> 2);
6369 memcpy(iwl_conf->supp_mcs_set, ht_conf->supp_mcs_set, 16);
6370
6371 iwl_conf->control_channel = ht_bss_conf->primary_channel;
6372 iwl_conf->tx_chan_width =
6373 !!(ht_bss_conf->bss_cap & IEEE80211_HT_IE_CHA_WIDTH);
6374 iwl_conf->ht_protection =
6375 ht_bss_conf->bss_op_mode & IEEE80211_HT_IE_HT_PROTECTION;
6376 iwl_conf->non_GF_STA_present =
6377 !!(ht_bss_conf->bss_op_mode & IEEE80211_HT_IE_NON_GF_STA_PRSNT);
6378
6379 IWL_DEBUG_MAC80211("control channel %d\n", iwl_conf->control_channel);
6380 IWL_DEBUG_MAC80211("leave\n");
6381}
6382#else
6383static inline void iwl4965_ht_conf(struct iwl_priv *priv,
6384 struct ieee80211_bss_conf *bss_conf)
6385{
6386}
6387#endif
6388
6389#define IWL_DELAY_NEXT_SCAN_AFTER_ASSOC (HZ*6) 6409#define IWL_DELAY_NEXT_SCAN_AFTER_ASSOC (HZ*6)
6390static void iwl4965_bss_info_changed(struct ieee80211_hw *hw, 6410static void iwl4965_bss_info_changed(struct ieee80211_hw *hw,
6391 struct ieee80211_vif *vif, 6411 struct ieee80211_vif *vif,