blob: fcac73cf82bc60c3f991fd054c42329297234eb3 [file] [log] [blame]
Emmanuel Grumbach6974e362008-04-14 21:16:06 -07001/******************************************************************************
2 *
Reinette Chatre1f447802010-01-15 13:43:41 -08003 * Copyright(c) 2003 - 2010 Intel Corporation. All rights reserved.
Emmanuel Grumbach6974e362008-04-14 21:16:06 -07004 *
5 * Portions of this file are derived from the ipw3945 project, as well
6 * as portions of the ieee80211 subsystem header files.
7 *
8 * This program is free software; you can redistribute it and/or modify it
9 * under the terms of version 2 of the GNU General Public License as
10 * published by the Free Software Foundation.
11 *
12 * This program is distributed in the hope that it will be useful, but WITHOUT
13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15 * more details.
16 *
17 * You should have received a copy of the GNU General Public License along with
18 * this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA
20 *
21 * The full GNU General Public License is included in this distribution in the
22 * file called LICENSE.
23 *
24 * Contact Information:
Winkler, Tomas759ef892008-12-09 11:28:58 -080025 * Intel Linux Wireless <ilw@linux.intel.com>
Emmanuel Grumbach6974e362008-04-14 21:16:06 -070026 * Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
27 *
28 *****************************************************************************/
29
30#include <net/mac80211.h>
Tomas Winkler947b13a2008-04-16 16:34:48 -070031#include <linux/etherdevice.h>
Emmanuel Grumbach6974e362008-04-14 21:16:06 -070032
Tomas Winkler3e0d4cb2008-04-24 11:55:38 -070033#include "iwl-dev.h"
Emmanuel Grumbach6974e362008-04-14 21:16:06 -070034#include "iwl-core.h"
35#include "iwl-sta.h"
Tomas Winkler7a999bf2008-05-29 16:35:02 +080036
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +080037#define IWL_STA_DRIVER_ACTIVE BIT(0) /* driver entry is active */
38#define IWL_STA_UCODE_ACTIVE BIT(1) /* ucode entry is active */
Tomas Winkler7a999bf2008-05-29 16:35:02 +080039
Tomas Winkler947b13a2008-04-16 16:34:48 -070040u8 iwl_find_station(struct iwl_priv *priv, const u8 *addr)
41{
42 int i;
43 int start = 0;
44 int ret = IWL_INVALID_STATION;
45 unsigned long flags;
Tomas Winkler947b13a2008-04-16 16:34:48 -070046
Johannes Berg05c914f2008-09-11 00:01:58 +020047 if ((priv->iw_mode == NL80211_IFTYPE_ADHOC) ||
48 (priv->iw_mode == NL80211_IFTYPE_AP))
Tomas Winkler947b13a2008-04-16 16:34:48 -070049 start = IWL_STA_ID;
50
51 if (is_broadcast_ether_addr(addr))
52 return priv->hw_params.bcast_sta_id;
53
54 spin_lock_irqsave(&priv->sta_lock, flags);
55 for (i = start; i < priv->hw_params.max_stations; i++)
56 if (priv->stations[i].used &&
57 (!compare_ether_addr(priv->stations[i].sta.sta.addr,
58 addr))) {
59 ret = i;
60 goto out;
61 }
62
Tomas Winklere1623442009-01-27 14:27:56 -080063 IWL_DEBUG_ASSOC_LIMIT(priv, "can not find STA %pM total %d\n",
Johannes Berge1749612008-10-27 15:59:26 -070064 addr, priv->num_stations);
Tomas Winkler947b13a2008-04-16 16:34:48 -070065
66 out:
67 spin_unlock_irqrestore(&priv->sta_lock, flags);
68 return ret;
69}
70EXPORT_SYMBOL(iwl_find_station);
71
Emmanuel Grumbachbe1f3ab62008-06-12 09:47:18 +080072int iwl_get_ra_sta_id(struct iwl_priv *priv, struct ieee80211_hdr *hdr)
73{
Johannes Berg05c914f2008-09-11 00:01:58 +020074 if (priv->iw_mode == NL80211_IFTYPE_STATION) {
Emmanuel Grumbachbe1f3ab62008-06-12 09:47:18 +080075 return IWL_AP_ID;
76 } else {
77 u8 *da = ieee80211_get_DA(hdr);
Tomas Winklerc587de02009-06-03 11:44:07 -070078 return iwl_find_station(priv, da);
Emmanuel Grumbachbe1f3ab62008-06-12 09:47:18 +080079 }
80}
81EXPORT_SYMBOL(iwl_get_ra_sta_id);
82
Reinette Chatre1fa97aa2010-01-22 14:22:48 -080083/* priv->sta_lock must be held */
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +080084static void iwl_sta_ucode_activate(struct iwl_priv *priv, u8 sta_id)
85{
Reinette Chatre1fa97aa2010-01-22 14:22:48 -080086
87 if (!(priv->stations[sta_id].used & IWL_STA_DRIVER_ACTIVE))
88 IWL_ERR(priv, "ACTIVATE a non DRIVER active station id %u addr %pM\n",
89 sta_id, priv->stations[sta_id].sta.sta.addr);
90
91 if (priv->stations[sta_id].used & IWL_STA_UCODE_ACTIVE) {
92 IWL_DEBUG_ASSOC(priv,
93 "STA id %u addr %pM already present in uCode (according to driver)\n",
94 sta_id, priv->stations[sta_id].sta.sta.addr);
95 } else {
96 priv->stations[sta_id].used |= IWL_STA_UCODE_ACTIVE;
97 IWL_DEBUG_ASSOC(priv, "Added STA id %u addr %pM to uCode\n",
98 sta_id, priv->stations[sta_id].sta.sta.addr);
99 }
100}
101
102static void iwl_process_add_sta_resp(struct iwl_priv *priv,
103 struct iwl_addsta_cmd *addsta,
104 struct iwl_rx_packet *pkt,
105 bool sync)
106{
107 u8 sta_id = addsta->sta.sta_id;
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800108 unsigned long flags;
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800109
Reinette Chatre1fa97aa2010-01-22 14:22:48 -0800110 if (pkt->hdr.flags & IWL_CMD_FAILED_MSK) {
111 IWL_ERR(priv, "Bad return from REPLY_ADD_STA (0x%08X)\n",
112 pkt->hdr.flags);
113 return;
114 }
115
116 IWL_DEBUG_INFO(priv, "Processing response for adding station %u\n",
117 sta_id);
118
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800119 spin_lock_irqsave(&priv->sta_lock, flags);
120
Reinette Chatre1fa97aa2010-01-22 14:22:48 -0800121 switch (pkt->u.add_sta.status) {
122 case ADD_STA_SUCCESS_MSK:
123 IWL_DEBUG_INFO(priv, "REPLY_ADD_STA PASSED\n");
124 iwl_sta_ucode_activate(priv, sta_id);
125 break;
126 case ADD_STA_NO_ROOM_IN_TABLE:
127 IWL_ERR(priv, "Adding station %d failed, no room in table.\n",
Winkler, Tomas15b16872008-12-19 10:37:33 +0800128 sta_id);
Reinette Chatre1fa97aa2010-01-22 14:22:48 -0800129 break;
130 case ADD_STA_NO_BLOCK_ACK_RESOURCE:
131 IWL_ERR(priv, "Adding station %d failed, no block ack resource.\n",
132 sta_id);
133 break;
134 case ADD_STA_MODIFY_NON_EXIST_STA:
135 IWL_ERR(priv, "Attempting to modify non-existing station %d \n",
136 sta_id);
137 break;
138 default:
139 IWL_DEBUG_ASSOC(priv, "Received REPLY_ADD_STA:(0x%08X)\n",
140 pkt->u.add_sta.status);
141 break;
142 }
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800143
Reinette Chatre1fa97aa2010-01-22 14:22:48 -0800144 IWL_DEBUG_INFO(priv, "%s station id %u addr %pM\n",
145 priv->stations[sta_id].sta.mode ==
146 STA_CONTROL_MODIFY_MSK ? "Modified" : "Added",
147 sta_id, priv->stations[sta_id].sta.sta.addr);
148
149 /*
150 * XXX: The MAC address in the command buffer is often changed from
151 * the original sent to the device. That is, the MAC address
152 * written to the command buffer often is not the same MAC adress
153 * read from the command buffer when the command returns. This
154 * issue has not yet been resolved and this debugging is left to
155 * observe the problem.
156 */
157 IWL_DEBUG_INFO(priv, "%s station according to cmd buffer %pM\n",
158 priv->stations[sta_id].sta.mode ==
159 STA_CONTROL_MODIFY_MSK ? "Modified" : "Added",
160 addsta->sta.addr);
161
162 /*
163 * Determine if we wanted to modify or add a station,
164 * if adding a station succeeded we have some more initialization
165 * to do when using station notification. TODO
166 */
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800167
168 spin_unlock_irqrestore(&priv->sta_lock, flags);
169}
170
Johannes Berg5696aea2009-07-24 11:13:06 -0700171static void iwl_add_sta_callback(struct iwl_priv *priv,
172 struct iwl_device_cmd *cmd,
Zhu Yi2f301222009-10-09 17:19:45 +0800173 struct iwl_rx_packet *pkt)
Tomas Winkler42132bc2008-05-29 16:35:03 +0800174{
Tomas Winkler3257e5d2008-10-14 12:32:43 -0700175 struct iwl_addsta_cmd *addsta =
176 (struct iwl_addsta_cmd *)cmd->cmd.payload;
Tomas Winkler42132bc2008-05-29 16:35:03 +0800177
Reinette Chatre1fa97aa2010-01-22 14:22:48 -0800178 iwl_process_add_sta_resp(priv, addsta, pkt, false);
Tomas Winkler42132bc2008-05-29 16:35:03 +0800179
Tomas Winkler42132bc2008-05-29 16:35:03 +0800180}
181
Samuel Ortiz17f841c2009-01-23 13:45:20 -0800182int iwl_send_add_sta(struct iwl_priv *priv,
Tomas Winkler133636d2008-05-05 10:22:34 +0800183 struct iwl_addsta_cmd *sta, u8 flags)
184{
Zhu Yi2f301222009-10-09 17:19:45 +0800185 struct iwl_rx_packet *pkt = NULL;
Tomas Winkler133636d2008-05-05 10:22:34 +0800186 int ret = 0;
187 u8 data[sizeof(*sta)];
188 struct iwl_host_cmd cmd = {
189 .id = REPLY_ADD_STA,
Johannes Bergc2acea82009-07-24 11:13:05 -0700190 .flags = flags,
Tomas Winkler133636d2008-05-05 10:22:34 +0800191 .data = data,
192 };
193
Tomas Winkler42132bc2008-05-29 16:35:03 +0800194 if (flags & CMD_ASYNC)
Johannes Bergc2acea82009-07-24 11:13:05 -0700195 cmd.callback = iwl_add_sta_callback;
Tomas Winkler42132bc2008-05-29 16:35:03 +0800196 else
Johannes Bergc2acea82009-07-24 11:13:05 -0700197 cmd.flags |= CMD_WANT_SKB;
Tomas Winkler133636d2008-05-05 10:22:34 +0800198
199 cmd.len = priv->cfg->ops->utils->build_addsta_hcmd(sta, data);
200 ret = iwl_send_cmd(priv, &cmd);
201
202 if (ret || (flags & CMD_ASYNC))
203 return ret;
204
Tomas Winkler133636d2008-05-05 10:22:34 +0800205 if (ret == 0) {
Reinette Chatre1fa97aa2010-01-22 14:22:48 -0800206 pkt = (struct iwl_rx_packet *)cmd.reply_page;
207 iwl_process_add_sta_resp(priv, sta, pkt, true);
Tomas Winkler133636d2008-05-05 10:22:34 +0800208 }
Zhu Yi64a76b52009-12-10 14:37:21 -0800209 iwl_free_pages(priv, cmd.reply_page);
Tomas Winkler133636d2008-05-05 10:22:34 +0800210
211 return ret;
212}
Samuel Ortiz17f841c2009-01-23 13:45:20 -0800213EXPORT_SYMBOL(iwl_send_add_sta);
Tomas Winkler947b13a2008-04-16 16:34:48 -0700214
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800215static void iwl_set_ht_add_station(struct iwl_priv *priv, u8 index,
Johannes Bergd9fe60d2008-10-09 12:13:49 +0200216 struct ieee80211_sta_ht_cap *sta_ht_inf)
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800217{
218 __le32 sta_flags;
219 u8 mimo_ps_mode;
220
221 if (!sta_ht_inf || !sta_ht_inf->ht_supported)
222 goto done;
223
Tomas Winkler00c5ae22008-09-03 11:26:42 +0800224 mimo_ps_mode = (sta_ht_inf->cap & IEEE80211_HT_CAP_SM_PS) >> 2;
Wey-Yi Guy3f3e0372009-10-30 14:36:17 -0700225 IWL_DEBUG_ASSOC(priv, "spatial multiplexing power save mode: %s\n",
226 (mimo_ps_mode == WLAN_HT_CAP_SM_PS_STATIC) ?
227 "static" :
228 (mimo_ps_mode == WLAN_HT_CAP_SM_PS_DYNAMIC) ?
229 "dynamic" : "disabled");
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800230
231 sta_flags = priv->stations[index].sta.station_flags;
232
233 sta_flags &= ~(STA_FLG_RTS_MIMO_PROT_MSK | STA_FLG_MIMO_DIS_MSK);
234
235 switch (mimo_ps_mode) {
Tomas Winkler00c5ae22008-09-03 11:26:42 +0800236 case WLAN_HT_CAP_SM_PS_STATIC:
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800237 sta_flags |= STA_FLG_MIMO_DIS_MSK;
238 break;
Tomas Winkler00c5ae22008-09-03 11:26:42 +0800239 case WLAN_HT_CAP_SM_PS_DYNAMIC:
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800240 sta_flags |= STA_FLG_RTS_MIMO_PROT_MSK;
241 break;
Tomas Winkler00c5ae22008-09-03 11:26:42 +0800242 case WLAN_HT_CAP_SM_PS_DISABLED:
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800243 break;
244 default:
Winkler, Tomas39aadf82008-12-19 10:37:32 +0800245 IWL_WARN(priv, "Invalid MIMO PS mode %d\n", mimo_ps_mode);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800246 break;
247 }
248
249 sta_flags |= cpu_to_le32(
250 (u32)sta_ht_inf->ampdu_factor << STA_FLG_MAX_AGG_SIZE_POS);
251
252 sta_flags |= cpu_to_le32(
253 (u32)sta_ht_inf->ampdu_density << STA_FLG_AGG_MPDU_DENSITY_POS);
254
Wey-Yi Guy7aafef12009-08-07 15:41:38 -0700255 if (iwl_is_ht40_tx_allowed(priv, sta_ht_inf))
256 sta_flags |= STA_FLG_HT40_EN_MSK;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800257 else
Wey-Yi Guy7aafef12009-08-07 15:41:38 -0700258 sta_flags &= ~STA_FLG_HT40_EN_MSK;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800259
260 priv->stations[index].sta.station_flags = sta_flags;
261 done:
262 return;
263}
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800264
265/**
Tomas Winklerc587de02009-06-03 11:44:07 -0700266 * iwl_add_station - Add station to tables in driver and device
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800267 */
Tomas Winklerc587de02009-06-03 11:44:07 -0700268u8 iwl_add_station(struct iwl_priv *priv, const u8 *addr, bool is_ap, u8 flags,
269 struct ieee80211_sta_ht_cap *ht_info)
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800270{
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800271 struct iwl_station_entry *station;
272 unsigned long flags_spin;
Tomas Winklerc587de02009-06-03 11:44:07 -0700273 int i;
274 int sta_id = IWL_INVALID_STATION;
275 u16 rate;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800276
277 spin_lock_irqsave(&priv->sta_lock, flags_spin);
278 if (is_ap)
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800279 sta_id = IWL_AP_ID;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800280 else if (is_broadcast_ether_addr(addr))
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800281 sta_id = priv->hw_params.bcast_sta_id;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800282 else
283 for (i = IWL_STA_ID; i < priv->hw_params.max_stations; i++) {
284 if (!compare_ether_addr(priv->stations[i].sta.sta.addr,
285 addr)) {
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800286 sta_id = i;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800287 break;
288 }
289
290 if (!priv->stations[i].used &&
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800291 sta_id == IWL_INVALID_STATION)
292 sta_id = i;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800293 }
294
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800295 /* These two conditions have the same outcome, but keep them separate
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800296 since they have different meanings */
297 if (unlikely(sta_id == IWL_INVALID_STATION)) {
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800298 spin_unlock_irqrestore(&priv->sta_lock, flags_spin);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800299 return sta_id;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800300 }
301
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800302 if (priv->stations[sta_id].used &&
303 !compare_ether_addr(priv->stations[sta_id].sta.sta.addr, addr)) {
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800304 spin_unlock_irqrestore(&priv->sta_lock, flags_spin);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800305 return sta_id;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800306 }
307
308
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800309 station = &priv->stations[sta_id];
310 station->used = IWL_STA_DRIVER_ACTIVE;
Tomas Winklere1623442009-01-27 14:27:56 -0800311 IWL_DEBUG_ASSOC(priv, "Add STA to driver ID %d: %pM\n",
Johannes Berge1749612008-10-27 15:59:26 -0700312 sta_id, addr);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800313 priv->num_stations++;
314
315 /* Set up the REPLY_ADD_STA command to send to device */
316 memset(&station->sta, 0, sizeof(struct iwl_addsta_cmd));
317 memcpy(station->sta.sta.addr, addr, ETH_ALEN);
318 station->sta.mode = 0;
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800319 station->sta.sta.sta_id = sta_id;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800320 station->sta.station_flags = 0;
321
322 /* BCAST station and IBSS stations do not work in HT mode */
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800323 if (sta_id != priv->hw_params.bcast_sta_id &&
Johannes Berg05c914f2008-09-11 00:01:58 +0200324 priv->iw_mode != NL80211_IFTYPE_ADHOC)
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800325 iwl_set_ht_add_station(priv, sta_id, ht_info);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800326
Tomas Winklerc587de02009-06-03 11:44:07 -0700327 /* 3945 only */
328 rate = (priv->band == IEEE80211_BAND_5GHZ) ?
329 IWL_RATE_6M_PLCP : IWL_RATE_1M_PLCP;
330 /* Turn on both antennas for the station... */
331 station->sta.rate_n_flags = cpu_to_le16(rate | RATE_MCS_ANT_AB_MSK);
332
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800333 spin_unlock_irqrestore(&priv->sta_lock, flags_spin);
334
335 /* Add station to device's station table */
336 iwl_send_add_sta(priv, &station->sta, flags);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800337 return sta_id;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800338
339}
Tomas Winklerc587de02009-06-03 11:44:07 -0700340EXPORT_SYMBOL(iwl_add_station);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800341
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800342static void iwl_sta_ucode_deactivate(struct iwl_priv *priv, const char *addr)
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800343{
344 unsigned long flags;
Tomas Winklerc587de02009-06-03 11:44:07 -0700345 u8 sta_id = iwl_find_station(priv, addr);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800346
347 BUG_ON(sta_id == IWL_INVALID_STATION);
348
Tomas Winklere1623442009-01-27 14:27:56 -0800349 IWL_DEBUG_ASSOC(priv, "Removed STA from Ucode: %pM\n", addr);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800350
351 spin_lock_irqsave(&priv->sta_lock, flags);
352
353 /* Ucode must be active and driver must be non active */
354 if (priv->stations[sta_id].used != IWL_STA_UCODE_ACTIVE)
Winkler, Tomas15b16872008-12-19 10:37:33 +0800355 IWL_ERR(priv, "removed non active STA %d\n", sta_id);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800356
357 priv->stations[sta_id].used &= ~IWL_STA_UCODE_ACTIVE;
358
359 memset(&priv->stations[sta_id], 0, sizeof(struct iwl_station_entry));
360 spin_unlock_irqrestore(&priv->sta_lock, flags);
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800361}
362
Johannes Berg5696aea2009-07-24 11:13:06 -0700363static void iwl_remove_sta_callback(struct iwl_priv *priv,
364 struct iwl_device_cmd *cmd,
Zhu Yi2f301222009-10-09 17:19:45 +0800365 struct iwl_rx_packet *pkt)
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800366{
Tomas Winkler3257e5d2008-10-14 12:32:43 -0700367 struct iwl_rem_sta_cmd *rm_sta =
Zhu Yi2f301222009-10-09 17:19:45 +0800368 (struct iwl_rem_sta_cmd *)cmd->cmd.payload;
Tomas Winkler3257e5d2008-10-14 12:32:43 -0700369 const char *addr = rm_sta->addr;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800370
Zhu Yi2f301222009-10-09 17:19:45 +0800371 if (pkt->hdr.flags & IWL_CMD_FAILED_MSK) {
Winkler, Tomas15b16872008-12-19 10:37:33 +0800372 IWL_ERR(priv, "Bad return from REPLY_REMOVE_STA (0x%08X)\n",
Zhu Yi2f301222009-10-09 17:19:45 +0800373 pkt->hdr.flags);
Johannes Berg5696aea2009-07-24 11:13:06 -0700374 return;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800375 }
376
Zhu Yi2f301222009-10-09 17:19:45 +0800377 switch (pkt->u.rem_sta.status) {
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800378 case REM_STA_SUCCESS_MSK:
379 iwl_sta_ucode_deactivate(priv, addr);
380 break;
381 default:
Winkler, Tomas15b16872008-12-19 10:37:33 +0800382 IWL_ERR(priv, "REPLY_REMOVE_STA failed\n");
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800383 break;
384 }
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800385}
386
387static int iwl_send_remove_station(struct iwl_priv *priv, const u8 *addr,
388 u8 flags)
389{
Zhu Yi2f301222009-10-09 17:19:45 +0800390 struct iwl_rx_packet *pkt;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800391 int ret;
392
393 struct iwl_rem_sta_cmd rm_sta_cmd;
394
395 struct iwl_host_cmd cmd = {
396 .id = REPLY_REMOVE_STA,
397 .len = sizeof(struct iwl_rem_sta_cmd),
Johannes Bergc2acea82009-07-24 11:13:05 -0700398 .flags = flags,
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800399 .data = &rm_sta_cmd,
400 };
401
402 memset(&rm_sta_cmd, 0, sizeof(rm_sta_cmd));
403 rm_sta_cmd.num_sta = 1;
404 memcpy(&rm_sta_cmd.addr, addr , ETH_ALEN);
405
406 if (flags & CMD_ASYNC)
Johannes Bergc2acea82009-07-24 11:13:05 -0700407 cmd.callback = iwl_remove_sta_callback;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800408 else
Johannes Bergc2acea82009-07-24 11:13:05 -0700409 cmd.flags |= CMD_WANT_SKB;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800410 ret = iwl_send_cmd(priv, &cmd);
411
412 if (ret || (flags & CMD_ASYNC))
413 return ret;
414
Zhu Yi2f301222009-10-09 17:19:45 +0800415 pkt = (struct iwl_rx_packet *)cmd.reply_page;
416 if (pkt->hdr.flags & IWL_CMD_FAILED_MSK) {
Winkler, Tomas15b16872008-12-19 10:37:33 +0800417 IWL_ERR(priv, "Bad return from REPLY_REMOVE_STA (0x%08X)\n",
Zhu Yi2f301222009-10-09 17:19:45 +0800418 pkt->hdr.flags);
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800419 ret = -EIO;
420 }
421
422 if (!ret) {
Zhu Yi2f301222009-10-09 17:19:45 +0800423 switch (pkt->u.rem_sta.status) {
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800424 case REM_STA_SUCCESS_MSK:
425 iwl_sta_ucode_deactivate(priv, addr);
Tomas Winklere1623442009-01-27 14:27:56 -0800426 IWL_DEBUG_ASSOC(priv, "REPLY_REMOVE_STA PASSED\n");
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800427 break;
428 default:
429 ret = -EIO;
Winkler, Tomas15b16872008-12-19 10:37:33 +0800430 IWL_ERR(priv, "REPLY_REMOVE_STA failed\n");
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800431 break;
432 }
433 }
Zhu Yi64a76b52009-12-10 14:37:21 -0800434 iwl_free_pages(priv, cmd.reply_page);
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800435
436 return ret;
437}
Emmanuel Grumbachbe1f3ab62008-06-12 09:47:18 +0800438
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800439/**
440 * iwl_remove_station - Remove driver's knowledge of station.
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800441 */
Tomas Winklerc587de02009-06-03 11:44:07 -0700442int iwl_remove_station(struct iwl_priv *priv, const u8 *addr, bool is_ap)
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800443{
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800444 int sta_id = IWL_INVALID_STATION;
445 int i, ret = -EINVAL;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800446 unsigned long flags;
447
448 spin_lock_irqsave(&priv->sta_lock, flags);
449
450 if (is_ap)
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800451 sta_id = IWL_AP_ID;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800452 else if (is_broadcast_ether_addr(addr))
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800453 sta_id = priv->hw_params.bcast_sta_id;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800454 else
455 for (i = IWL_STA_ID; i < priv->hw_params.max_stations; i++)
456 if (priv->stations[i].used &&
457 !compare_ether_addr(priv->stations[i].sta.sta.addr,
458 addr)) {
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800459 sta_id = i;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800460 break;
461 }
462
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800463 if (unlikely(sta_id == IWL_INVALID_STATION))
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800464 goto out;
465
Tomas Winklere1623442009-01-27 14:27:56 -0800466 IWL_DEBUG_ASSOC(priv, "Removing STA from driver:%d %pM\n",
Johannes Berge1749612008-10-27 15:59:26 -0700467 sta_id, addr);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800468
469 if (!(priv->stations[sta_id].used & IWL_STA_DRIVER_ACTIVE)) {
Winkler, Tomas15b16872008-12-19 10:37:33 +0800470 IWL_ERR(priv, "Removing %pM but non DRIVER active\n",
Johannes Berge1749612008-10-27 15:59:26 -0700471 addr);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800472 goto out;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800473 }
474
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800475 if (!(priv->stations[sta_id].used & IWL_STA_UCODE_ACTIVE)) {
Winkler, Tomas15b16872008-12-19 10:37:33 +0800476 IWL_ERR(priv, "Removing %pM but non UCODE active\n",
Johannes Berge1749612008-10-27 15:59:26 -0700477 addr);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800478 goto out;
479 }
480
481
482 priv->stations[sta_id].used &= ~IWL_STA_DRIVER_ACTIVE;
483
484 priv->num_stations--;
485
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800486 BUG_ON(priv->num_stations < 0);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800487
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800488 spin_unlock_irqrestore(&priv->sta_lock, flags);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800489
490 ret = iwl_send_remove_station(priv, addr, CMD_ASYNC);
491 return ret;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800492out:
493 spin_unlock_irqrestore(&priv->sta_lock, flags);
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800494 return ret;
Tomas Winkler7a999bf2008-05-29 16:35:02 +0800495}
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +0800496
Winkler, Tomas83dde8c2008-11-19 15:32:23 -0800497/**
498 * iwl_clear_stations_table - Clear the driver's station table
499 *
500 * NOTE: This does not clear or otherwise alter the device's station table.
501 */
502void iwl_clear_stations_table(struct iwl_priv *priv)
503{
504 unsigned long flags;
Mohamed Abbas48676eb2009-03-11 11:17:59 -0700505 int i;
Winkler, Tomas83dde8c2008-11-19 15:32:23 -0800506
507 spin_lock_irqsave(&priv->sta_lock, flags);
508
509 if (iwl_is_alive(priv) &&
510 !test_bit(STATUS_EXIT_PENDING, &priv->status) &&
511 iwl_send_cmd_pdu_async(priv, REPLY_REMOVE_ALL_STA, 0, NULL, NULL))
Winkler, Tomas15b16872008-12-19 10:37:33 +0800512 IWL_ERR(priv, "Couldn't clear the station table\n");
Winkler, Tomas83dde8c2008-11-19 15:32:23 -0800513
514 priv->num_stations = 0;
515 memset(priv->stations, 0, sizeof(priv->stations));
516
Reinette Chatre5e468822009-01-28 09:38:30 -0800517 /* clean ucode key table bit map */
518 priv->ucode_key_table = 0;
519
Mohamed Abbas48676eb2009-03-11 11:17:59 -0700520 /* keep track of static keys */
521 for (i = 0; i < WEP_KEYS_MAX ; i++) {
522 if (priv->wep_keys[i].key_size)
Tomas Winkler4e8e2c82009-04-30 13:56:29 -0700523 set_bit(i, &priv->ucode_key_table);
Mohamed Abbas48676eb2009-03-11 11:17:59 -0700524 }
525
Winkler, Tomas83dde8c2008-11-19 15:32:23 -0800526 spin_unlock_irqrestore(&priv->sta_lock, flags);
527}
528EXPORT_SYMBOL(iwl_clear_stations_table);
529
Abhijeet Kolekar6e21f152009-02-27 16:21:21 -0800530int iwl_get_free_ucode_key_index(struct iwl_priv *priv)
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700531{
532 int i;
533
534 for (i = 0; i < STA_KEY_MAX_NUM; i++)
Emmanuel Grumbach77bab602008-04-15 16:01:44 -0700535 if (!test_and_set_bit(i, &priv->ucode_key_table))
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700536 return i;
537
Tomas Winkler40a9a822008-11-25 23:29:03 +0200538 return WEP_INVALID_OFFSET;
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700539}
Abhijeet Kolekar6e21f152009-02-27 16:21:21 -0800540EXPORT_SYMBOL(iwl_get_free_ucode_key_index);
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700541
542int iwl_send_static_wepkey_cmd(struct iwl_priv *priv, u8 send_if_empty)
543{
544 int i, not_empty = 0;
545 u8 buff[sizeof(struct iwl_wep_cmd) +
546 sizeof(struct iwl_wep_key) * WEP_KEYS_MAX];
547 struct iwl_wep_cmd *wep_cmd = (struct iwl_wep_cmd *)buff;
548 size_t cmd_size = sizeof(struct iwl_wep_cmd);
549 struct iwl_host_cmd cmd = {
550 .id = REPLY_WEPKEY,
551 .data = wep_cmd,
Reinette Chatrebba98872009-09-17 10:43:55 -0700552 .flags = CMD_ASYNC,
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700553 };
554
555 memset(wep_cmd, 0, cmd_size +
556 (sizeof(struct iwl_wep_key) * WEP_KEYS_MAX));
557
558 for (i = 0; i < WEP_KEYS_MAX ; i++) {
559 wep_cmd->key[i].key_index = i;
560 if (priv->wep_keys[i].key_size) {
561 wep_cmd->key[i].key_offset = i;
562 not_empty = 1;
563 } else {
564 wep_cmd->key[i].key_offset = WEP_INVALID_OFFSET;
565 }
566
567 wep_cmd->key[i].key_size = priv->wep_keys[i].key_size;
568 memcpy(&wep_cmd->key[i].key[3], priv->wep_keys[i].key,
569 priv->wep_keys[i].key_size);
570 }
571
572 wep_cmd->global_key_type = WEP_KEY_WEP_TYPE;
573 wep_cmd->num_keys = WEP_KEYS_MAX;
574
575 cmd_size += sizeof(struct iwl_wep_key) * WEP_KEYS_MAX;
576
577 cmd.len = cmd_size;
578
579 if (not_empty || send_if_empty)
580 return iwl_send_cmd(priv, &cmd);
581 else
582 return 0;
583}
Tomas Winkler27aaba02008-05-05 10:22:44 +0800584EXPORT_SYMBOL(iwl_send_static_wepkey_cmd);
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700585
586int iwl_remove_default_wep_key(struct iwl_priv *priv,
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700587 struct ieee80211_key_conf *keyconf)
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700588{
589 int ret;
590 unsigned long flags;
591
592 spin_lock_irqsave(&priv->sta_lock, flags);
Reinette Chatre2d1bb9e2009-07-17 09:30:18 -0700593 IWL_DEBUG_WEP(priv, "Removing default WEP key: idx=%d\n",
594 keyconf->keyidx);
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700595
596 if (!test_and_clear_bit(keyconf->keyidx, &priv->ucode_key_table))
Winkler, Tomas15b16872008-12-19 10:37:33 +0800597 IWL_ERR(priv, "index %d not used in uCode key table.\n",
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700598 keyconf->keyidx);
599
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700600 priv->default_wep_key--;
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700601 memset(&priv->wep_keys[keyconf->keyidx], 0, sizeof(priv->wep_keys[0]));
Reinette Chatre2d1bb9e2009-07-17 09:30:18 -0700602 if (iwl_is_rfkill(priv)) {
603 IWL_DEBUG_WEP(priv, "Not sending REPLY_WEPKEY command due to RFKILL.\n");
604 spin_unlock_irqrestore(&priv->sta_lock, flags);
605 return 0;
606 }
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700607 ret = iwl_send_static_wepkey_cmd(priv, 1);
Tomas Winklere1623442009-01-27 14:27:56 -0800608 IWL_DEBUG_WEP(priv, "Remove default WEP key: idx=%d ret=%d\n",
Emmanuel Grumbach4564ce82008-06-12 09:47:09 +0800609 keyconf->keyidx, ret);
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700610 spin_unlock_irqrestore(&priv->sta_lock, flags);
611
612 return ret;
613}
Tomas Winkler27aaba02008-05-05 10:22:44 +0800614EXPORT_SYMBOL(iwl_remove_default_wep_key);
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700615
616int iwl_set_default_wep_key(struct iwl_priv *priv,
617 struct ieee80211_key_conf *keyconf)
618{
619 int ret;
620 unsigned long flags;
621
Emmanuel Grumbach4564ce82008-06-12 09:47:09 +0800622 if (keyconf->keylen != WEP_KEY_LEN_128 &&
623 keyconf->keylen != WEP_KEY_LEN_64) {
Tomas Winklere1623442009-01-27 14:27:56 -0800624 IWL_DEBUG_WEP(priv, "Bad WEP key length %d\n", keyconf->keylen);
Emmanuel Grumbach4564ce82008-06-12 09:47:09 +0800625 return -EINVAL;
626 }
627
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700628 keyconf->flags &= ~IEEE80211_KEY_FLAG_GENERATE_IV;
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800629 keyconf->hw_key_idx = HW_KEY_DEFAULT;
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700630 priv->stations[IWL_AP_ID].keyinfo.alg = ALG_WEP;
631
632 spin_lock_irqsave(&priv->sta_lock, flags);
633 priv->default_wep_key++;
634
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700635 if (test_and_set_bit(keyconf->keyidx, &priv->ucode_key_table))
Winkler, Tomas15b16872008-12-19 10:37:33 +0800636 IWL_ERR(priv, "index %d already used in uCode key table.\n",
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700637 keyconf->keyidx);
638
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700639 priv->wep_keys[keyconf->keyidx].key_size = keyconf->keylen;
640 memcpy(&priv->wep_keys[keyconf->keyidx].key, &keyconf->key,
641 keyconf->keylen);
642
643 ret = iwl_send_static_wepkey_cmd(priv, 0);
Tomas Winklere1623442009-01-27 14:27:56 -0800644 IWL_DEBUG_WEP(priv, "Set default WEP key: len=%d idx=%d ret=%d\n",
Emmanuel Grumbach4564ce82008-06-12 09:47:09 +0800645 keyconf->keylen, keyconf->keyidx, ret);
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700646 spin_unlock_irqrestore(&priv->sta_lock, flags);
647
648 return ret;
649}
Tomas Winkler27aaba02008-05-05 10:22:44 +0800650EXPORT_SYMBOL(iwl_set_default_wep_key);
Emmanuel Grumbach6974e362008-04-14 21:16:06 -0700651
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700652static int iwl_set_wep_dynamic_key_info(struct iwl_priv *priv,
Emmanuel Grumbach0211ddd2008-04-14 21:16:07 -0700653 struct ieee80211_key_conf *keyconf,
654 u8 sta_id)
655{
656 unsigned long flags;
657 __le16 key_flags = 0;
658 int ret;
659
660 keyconf->flags &= ~IEEE80211_KEY_FLAG_GENERATE_IV;
Emmanuel Grumbach0211ddd2008-04-14 21:16:07 -0700661
662 key_flags |= (STA_KEY_FLG_WEP | STA_KEY_FLG_MAP_KEY_MSK);
663 key_flags |= cpu_to_le16(keyconf->keyidx << STA_KEY_FLG_KEYID_POS);
664 key_flags &= ~STA_KEY_FLG_INVALID;
665
666 if (keyconf->keylen == WEP_KEY_LEN_128)
667 key_flags |= STA_KEY_FLG_KEY_SIZE_MSK;
668
Tomas Winkler5425e492008-04-15 16:01:38 -0700669 if (sta_id == priv->hw_params.bcast_sta_id)
Emmanuel Grumbach0211ddd2008-04-14 21:16:07 -0700670 key_flags |= STA_KEY_MULTICAST_MSK;
671
672 spin_lock_irqsave(&priv->sta_lock, flags);
673
674 priv->stations[sta_id].keyinfo.alg = keyconf->alg;
675 priv->stations[sta_id].keyinfo.keylen = keyconf->keylen;
676 priv->stations[sta_id].keyinfo.keyidx = keyconf->keyidx;
677
678 memcpy(priv->stations[sta_id].keyinfo.key,
679 keyconf->key, keyconf->keylen);
680
681 memcpy(&priv->stations[sta_id].sta.key.key[3],
682 keyconf->key, keyconf->keylen);
683
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700684 if ((priv->stations[sta_id].sta.key.key_flags & STA_KEY_FLG_ENCRYPT_MSK)
685 == STA_KEY_FLG_NO_ENC)
686 priv->stations[sta_id].sta.key.key_offset =
Emmanuel Grumbach80fb47a2008-04-14 21:16:08 -0700687 iwl_get_free_ucode_key_index(priv);
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700688 /* else, we are overriding an existing key => no need to allocated room
689 * in uCode. */
Emmanuel Grumbach0211ddd2008-04-14 21:16:07 -0700690
Tomas Winkler40a9a822008-11-25 23:29:03 +0200691 WARN(priv->stations[sta_id].sta.key.key_offset == WEP_INVALID_OFFSET,
Jiri Slabye724b8f2009-01-05 17:06:06 +0100692 "no space for a new key");
Tomas Winkler40a9a822008-11-25 23:29:03 +0200693
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700694 priv->stations[sta_id].sta.key.key_flags = key_flags;
Emmanuel Grumbach0211ddd2008-04-14 21:16:07 -0700695 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_KEY_MASK;
696 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
697
Tomas Winkler133636d2008-05-05 10:22:34 +0800698 ret = iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
Emmanuel Grumbach0211ddd2008-04-14 21:16:07 -0700699
700 spin_unlock_irqrestore(&priv->sta_lock, flags);
701
702 return ret;
703}
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700704
705static int iwl_set_ccmp_dynamic_key_info(struct iwl_priv *priv,
706 struct ieee80211_key_conf *keyconf,
707 u8 sta_id)
708{
709 unsigned long flags;
710 __le16 key_flags = 0;
Tomas Winkler40a9a822008-11-25 23:29:03 +0200711 int ret;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700712
713 key_flags |= (STA_KEY_FLG_CCMP | STA_KEY_FLG_MAP_KEY_MSK);
714 key_flags |= cpu_to_le16(keyconf->keyidx << STA_KEY_FLG_KEYID_POS);
715 key_flags &= ~STA_KEY_FLG_INVALID;
716
Tomas Winkler5425e492008-04-15 16:01:38 -0700717 if (sta_id == priv->hw_params.bcast_sta_id)
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700718 key_flags |= STA_KEY_MULTICAST_MSK;
719
720 keyconf->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700721
722 spin_lock_irqsave(&priv->sta_lock, flags);
723 priv->stations[sta_id].keyinfo.alg = keyconf->alg;
724 priv->stations[sta_id].keyinfo.keylen = keyconf->keylen;
725
726 memcpy(priv->stations[sta_id].keyinfo.key, keyconf->key,
727 keyconf->keylen);
728
729 memcpy(priv->stations[sta_id].sta.key.key, keyconf->key,
730 keyconf->keylen);
731
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700732 if ((priv->stations[sta_id].sta.key.key_flags & STA_KEY_FLG_ENCRYPT_MSK)
733 == STA_KEY_FLG_NO_ENC)
734 priv->stations[sta_id].sta.key.key_offset =
735 iwl_get_free_ucode_key_index(priv);
736 /* else, we are overriding an existing key => no need to allocated room
737 * in uCode. */
738
Tomas Winkler40a9a822008-11-25 23:29:03 +0200739 WARN(priv->stations[sta_id].sta.key.key_offset == WEP_INVALID_OFFSET,
Jiri Slabye724b8f2009-01-05 17:06:06 +0100740 "no space for a new key");
Tomas Winkler40a9a822008-11-25 23:29:03 +0200741
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700742 priv->stations[sta_id].sta.key.key_flags = key_flags;
743 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_KEY_MASK;
744 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
745
Tomas Winkler40a9a822008-11-25 23:29:03 +0200746 ret = iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
747
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700748 spin_unlock_irqrestore(&priv->sta_lock, flags);
749
Tomas Winkler40a9a822008-11-25 23:29:03 +0200750 return ret;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700751}
752
753static int iwl_set_tkip_dynamic_key_info(struct iwl_priv *priv,
754 struct ieee80211_key_conf *keyconf,
755 u8 sta_id)
756{
757 unsigned long flags;
758 int ret = 0;
Reinette Chatre299f5462009-04-30 13:56:31 -0700759 __le16 key_flags = 0;
760
761 key_flags |= (STA_KEY_FLG_TKIP | STA_KEY_FLG_MAP_KEY_MSK);
762 key_flags |= cpu_to_le16(keyconf->keyidx << STA_KEY_FLG_KEYID_POS);
763 key_flags &= ~STA_KEY_FLG_INVALID;
764
765 if (sta_id == priv->hw_params.bcast_sta_id)
766 key_flags |= STA_KEY_MULTICAST_MSK;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700767
768 keyconf->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
769 keyconf->flags |= IEEE80211_KEY_FLAG_GENERATE_MMIC;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700770
771 spin_lock_irqsave(&priv->sta_lock, flags);
772
773 priv->stations[sta_id].keyinfo.alg = keyconf->alg;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700774 priv->stations[sta_id].keyinfo.keylen = 16;
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700775
776 if ((priv->stations[sta_id].sta.key.key_flags & STA_KEY_FLG_ENCRYPT_MSK)
777 == STA_KEY_FLG_NO_ENC)
778 priv->stations[sta_id].sta.key.key_offset =
Emmanuel Grumbach77bab602008-04-15 16:01:44 -0700779 iwl_get_free_ucode_key_index(priv);
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700780 /* else, we are overriding an existing key => no need to allocated room
781 * in uCode. */
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700782
Tomas Winkler40a9a822008-11-25 23:29:03 +0200783 WARN(priv->stations[sta_id].sta.key.key_offset == WEP_INVALID_OFFSET,
Jiri Slabye724b8f2009-01-05 17:06:06 +0100784 "no space for a new key");
Tomas Winkler40a9a822008-11-25 23:29:03 +0200785
Reinette Chatre299f5462009-04-30 13:56:31 -0700786 priv->stations[sta_id].sta.key.key_flags = key_flags;
787
788
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700789 /* This copy is acutally not needed: we get the key with each TX */
790 memcpy(priv->stations[sta_id].keyinfo.key, keyconf->key, 16);
791
792 memcpy(priv->stations[sta_id].sta.key.key, keyconf->key, 16);
793
794 spin_unlock_irqrestore(&priv->sta_lock, flags);
795
796 return ret;
797}
798
Tomas Winkler9f586712008-11-12 13:14:05 -0800799void iwl_update_tkip_key(struct iwl_priv *priv,
800 struct ieee80211_key_conf *keyconf,
801 const u8 *addr, u32 iv32, u16 *phase1key)
802{
803 u8 sta_id = IWL_INVALID_STATION;
804 unsigned long flags;
Tomas Winkler9f586712008-11-12 13:14:05 -0800805 int i;
Tomas Winkler9f586712008-11-12 13:14:05 -0800806
Tomas Winklerc587de02009-06-03 11:44:07 -0700807 sta_id = iwl_find_station(priv, addr);
Tomas Winkler9f586712008-11-12 13:14:05 -0800808 if (sta_id == IWL_INVALID_STATION) {
Tomas Winklere1623442009-01-27 14:27:56 -0800809 IWL_DEBUG_MAC80211(priv, "leave - %pM not in station map.\n",
Tomas Winkler9f586712008-11-12 13:14:05 -0800810 addr);
811 return;
812 }
813
814 if (iwl_scan_cancel(priv)) {
815 /* cancel scan failed, just live w/ bad key and rely
816 briefly on SW decryption */
817 return;
818 }
819
Tomas Winkler9f586712008-11-12 13:14:05 -0800820 spin_lock_irqsave(&priv->sta_lock, flags);
821
Tomas Winkler9f586712008-11-12 13:14:05 -0800822 priv->stations[sta_id].sta.key.tkip_rx_tsc_byte2 = (u8) iv32;
823
824 for (i = 0; i < 5; i++)
825 priv->stations[sta_id].sta.key.tkip_rx_ttak[i] =
826 cpu_to_le16(phase1key[i]);
827
828 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_KEY_MASK;
829 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
830
831 iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
832
833 spin_unlock_irqrestore(&priv->sta_lock, flags);
834
835}
836EXPORT_SYMBOL(iwl_update_tkip_key);
837
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700838int iwl_remove_dynamic_key(struct iwl_priv *priv,
839 struct ieee80211_key_conf *keyconf,
840 u8 sta_id)
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700841{
842 unsigned long flags;
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700843 int ret = 0;
844 u16 key_flags;
845 u8 keyidx;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700846
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800847 priv->key_mapping_key--;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700848
849 spin_lock_irqsave(&priv->sta_lock, flags);
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700850 key_flags = le16_to_cpu(priv->stations[sta_id].sta.key.key_flags);
851 keyidx = (key_flags >> STA_KEY_FLG_KEYID_POS) & 0x3;
852
Tomas Winklere1623442009-01-27 14:27:56 -0800853 IWL_DEBUG_WEP(priv, "Remove dynamic key: idx=%d sta=%d\n",
Emmanuel Grumbach4564ce82008-06-12 09:47:09 +0800854 keyconf->keyidx, sta_id);
855
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700856 if (keyconf->keyidx != keyidx) {
857 /* We need to remove a key with index different that the one
858 * in the uCode. This means that the key we need to remove has
859 * been replaced by another one with different index.
860 * Don't do anything and return ok
861 */
862 spin_unlock_irqrestore(&priv->sta_lock, flags);
863 return 0;
864 }
865
Tomas Winkler40a9a822008-11-25 23:29:03 +0200866 if (priv->stations[sta_id].sta.key.key_offset == WEP_INVALID_OFFSET) {
Winkler, Tomas39aadf82008-12-19 10:37:32 +0800867 IWL_WARN(priv, "Removing wrong key %d 0x%x\n",
Tomas Winkler40a9a822008-11-25 23:29:03 +0200868 keyconf->keyidx, key_flags);
869 spin_unlock_irqrestore(&priv->sta_lock, flags);
870 return 0;
871 }
872
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700873 if (!test_and_clear_bit(priv->stations[sta_id].sta.key.key_offset,
874 &priv->ucode_key_table))
Winkler, Tomas15b16872008-12-19 10:37:33 +0800875 IWL_ERR(priv, "index %d not used in uCode key table.\n",
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700876 priv->stations[sta_id].sta.key.key_offset);
877 memset(&priv->stations[sta_id].keyinfo, 0,
Tomas Winkler6def9762008-05-05 10:22:31 +0800878 sizeof(struct iwl_hw_key));
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700879 memset(&priv->stations[sta_id].sta.key, 0,
880 sizeof(struct iwl4965_keyinfo));
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700881 priv->stations[sta_id].sta.key.key_flags =
882 STA_KEY_FLG_NO_ENC | STA_KEY_FLG_INVALID;
883 priv->stations[sta_id].sta.key.key_offset = WEP_INVALID_OFFSET;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700884 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_KEY_MASK;
885 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700886
Reinette Chatre2d1bb9e2009-07-17 09:30:18 -0700887 if (iwl_is_rfkill(priv)) {
888 IWL_DEBUG_WEP(priv, "Not sending REPLY_ADD_STA command because RFKILL enabled. \n");
889 spin_unlock_irqrestore(&priv->sta_lock, flags);
890 return 0;
891 }
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800892 ret = iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
Emmanuel Grumbach3ec47732008-04-17 16:03:36 -0700893 spin_unlock_irqrestore(&priv->sta_lock, flags);
894 return ret;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700895}
Tomas Winkler27aaba02008-05-05 10:22:44 +0800896EXPORT_SYMBOL(iwl_remove_dynamic_key);
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700897
898int iwl_set_dynamic_key(struct iwl_priv *priv,
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800899 struct ieee80211_key_conf *keyconf, u8 sta_id)
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700900{
901 int ret;
902
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800903 priv->key_mapping_key++;
904 keyconf->hw_key_idx = HW_KEY_DYNAMIC;
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700905
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800906 switch (keyconf->alg) {
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700907 case ALG_CCMP:
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800908 ret = iwl_set_ccmp_dynamic_key_info(priv, keyconf, sta_id);
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700909 break;
910 case ALG_TKIP:
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800911 ret = iwl_set_tkip_dynamic_key_info(priv, keyconf, sta_id);
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700912 break;
913 case ALG_WEP:
Emmanuel Grumbachccc038a2008-05-15 13:54:09 +0800914 ret = iwl_set_wep_dynamic_key_info(priv, keyconf, sta_id);
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700915 break;
916 default:
Winkler, Tomas15b16872008-12-19 10:37:33 +0800917 IWL_ERR(priv,
918 "Unknown alg: %s alg = %d\n", __func__, keyconf->alg);
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700919 ret = -EINVAL;
920 }
921
Tomas Winklere1623442009-01-27 14:27:56 -0800922 IWL_DEBUG_WEP(priv, "Set dynamic key: alg= %d len=%d idx=%d sta=%d ret=%d\n",
Emmanuel Grumbach4564ce82008-06-12 09:47:09 +0800923 keyconf->alg, keyconf->keylen, keyconf->keyidx,
924 sta_id, ret);
925
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700926 return ret;
927}
Tomas Winkler27aaba02008-05-05 10:22:44 +0800928EXPORT_SYMBOL(iwl_set_dynamic_key);
Emmanuel Grumbach74805132008-04-14 21:16:09 -0700929
Tomas Winkler66c73db2008-04-15 16:01:40 -0700930#ifdef CONFIG_IWLWIFI_DEBUG
931static void iwl_dump_lq_cmd(struct iwl_priv *priv,
932 struct iwl_link_quality_cmd *lq)
933{
934 int i;
Tomas Winklere1623442009-01-27 14:27:56 -0800935 IWL_DEBUG_RATE(priv, "lq station id 0x%x\n", lq->sta_id);
936 IWL_DEBUG_RATE(priv, "lq ant 0x%X 0x%X\n",
Tomas Winkler66c73db2008-04-15 16:01:40 -0700937 lq->general_params.single_stream_ant_msk,
938 lq->general_params.dual_stream_ant_msk);
939
940 for (i = 0; i < LINK_QUAL_MAX_RETRY_NUM; i++)
Tomas Winklere1623442009-01-27 14:27:56 -0800941 IWL_DEBUG_RATE(priv, "lq index %d 0x%X\n",
Tomas Winkler66c73db2008-04-15 16:01:40 -0700942 i, lq->rs_table[i].rate_n_flags);
943}
944#else
945static inline void iwl_dump_lq_cmd(struct iwl_priv *priv,
946 struct iwl_link_quality_cmd *lq)
947{
948}
949#endif
950
951int iwl_send_lq_cmd(struct iwl_priv *priv,
952 struct iwl_link_quality_cmd *lq, u8 flags)
953{
954 struct iwl_host_cmd cmd = {
955 .id = REPLY_TX_LINK_QUALITY_CMD,
956 .len = sizeof(struct iwl_link_quality_cmd),
Johannes Bergc2acea82009-07-24 11:13:05 -0700957 .flags = flags,
Tomas Winkler66c73db2008-04-15 16:01:40 -0700958 .data = lq,
959 };
960
961 if ((lq->sta_id == 0xFF) &&
Johannes Berg05c914f2008-09-11 00:01:58 +0200962 (priv->iw_mode == NL80211_IFTYPE_ADHOC))
Tomas Winkler66c73db2008-04-15 16:01:40 -0700963 return -EINVAL;
964
965 if (lq->sta_id == 0xFF)
966 lq->sta_id = IWL_AP_ID;
967
Tomas Winkler3ac7f142008-07-21 02:40:14 +0300968 iwl_dump_lq_cmd(priv, lq);
Tomas Winkler66c73db2008-04-15 16:01:40 -0700969
Ron Rindjunsky36e1f162008-06-30 17:23:13 +0800970 if (iwl_is_associated(priv) && priv->assoc_station_added)
Tomas Winkler66c73db2008-04-15 16:01:40 -0700971 return iwl_send_cmd(priv, &cmd);
972
973 return 0;
974}
975EXPORT_SYMBOL(iwl_send_lq_cmd);
976
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800977/**
978 * iwl_sta_init_lq - Initialize a station's hardware rate table
979 *
980 * The uCode's station table contains a table of fallback rates
981 * for automatic fallback during transmission.
982 *
983 * NOTE: This sets up a default set of values. These will be replaced later
Tomas Winklere227cea2008-07-18 13:53:05 +0800984 * if the driver's iwl-agn-rs rate scaling algorithm is used, instead of
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800985 * rc80211_simple.
986 *
987 * NOTE: Run REPLY_ADD_STA command to set up station table entry, before
988 * calling this function (which runs REPLY_TX_LINK_QUALITY_CMD,
989 * which requires station table entry to exist).
990 */
Tomas Winklerc587de02009-06-03 11:44:07 -0700991static void iwl_sta_init_lq(struct iwl_priv *priv, const u8 *addr, bool is_ap)
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800992{
993 int i, r;
994 struct iwl_link_quality_cmd link_cmd = {
995 .reserved1 = 0,
996 };
Tomas Winkler5d664a42008-10-08 09:37:29 +0800997 u32 rate_flags;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +0800998
999 /* Set up the rate scaling to start at selected rate, fall back
1000 * all the way down to 1M in IEEE order, and then spin on 1M */
1001 if (is_ap)
1002 r = IWL_RATE_54M_INDEX;
1003 else if (priv->band == IEEE80211_BAND_5GHZ)
1004 r = IWL_RATE_6M_INDEX;
1005 else
1006 r = IWL_RATE_1M_INDEX;
1007
1008 for (i = 0; i < LINK_QUAL_MAX_RETRY_NUM; i++) {
1009 rate_flags = 0;
1010 if (r >= IWL_FIRST_CCK_RATE && r <= IWL_LAST_CCK_RATE)
1011 rate_flags |= RATE_MCS_CCK_MSK;
1012
Tomas Winkler5d664a42008-10-08 09:37:29 +08001013 rate_flags |= first_antenna(priv->hw_params.valid_tx_ant) <<
1014 RATE_MCS_ANT_POS;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001015
1016 link_cmd.rs_table[i].rate_n_flags =
Tomas Winklere7d326ac2008-06-12 09:47:11 +08001017 iwl_hw_set_rate_n_flags(iwl_rates[r].plcp, rate_flags);
Winkler, Tomaseb4779c2008-10-29 14:05:44 -07001018 r = iwl_get_prev_ieee_rate(r);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001019 }
1020
Tomas Winkler5d664a42008-10-08 09:37:29 +08001021 link_cmd.general_params.single_stream_ant_msk =
1022 first_antenna(priv->hw_params.valid_tx_ant);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001023 link_cmd.general_params.dual_stream_ant_msk = 3;
Wey-Yi Guy13c33a02009-06-03 11:44:11 -07001024 link_cmd.agg_params.agg_dis_start_th = LINK_QUAL_AGG_DISABLE_START_DEF;
1025 link_cmd.agg_params.agg_time_limit =
1026 cpu_to_le16(LINK_QUAL_AGG_TIME_LIMIT_DEF);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001027
1028 /* Update the rate scaling for control frame Tx to AP */
1029 link_cmd.sta_id = is_ap ? IWL_AP_ID : priv->hw_params.bcast_sta_id;
1030
1031 iwl_send_cmd_pdu_async(priv, REPLY_TX_LINK_QUALITY_CMD,
1032 sizeof(link_cmd), &link_cmd, NULL);
1033}
Emmanuel Grumbach24e5c402008-06-30 17:23:03 +08001034
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001035/**
1036 * iwl_rxon_add_station - add station into station table.
1037 *
1038 * there is only one AP station with id= IWL_AP_ID
Tomas Winklera96a27f2008-10-23 23:48:56 -07001039 * NOTE: mutex must be held before calling this function
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001040 */
Tomas Winklerc587de02009-06-03 11:44:07 -07001041int iwl_rxon_add_station(struct iwl_priv *priv, const u8 *addr, bool is_ap)
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001042{
Johannes Bergae5eb022008-10-14 16:58:37 +02001043 struct ieee80211_sta *sta;
1044 struct ieee80211_sta_ht_cap ht_config;
1045 struct ieee80211_sta_ht_cap *cur_ht_config = NULL;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001046 u8 sta_id;
1047
1048 /* Add station to device's station table */
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001049
Johannes Bergae5eb022008-10-14 16:58:37 +02001050 /*
1051 * XXX: This check is definitely not correct, if we're an AP
1052 * it'll always be false which is not what we want, but
1053 * it doesn't look like iwlagn is prepared to be an HT
1054 * AP anyway.
1055 */
1056 if (priv->current_ht_config.is_ht) {
1057 rcu_read_lock();
Johannes Berg5ed176e2009-11-04 14:42:28 +01001058 sta = ieee80211_find_sta(priv->vif, addr);
Johannes Bergae5eb022008-10-14 16:58:37 +02001059 if (sta) {
1060 memcpy(&ht_config, &sta->ht_cap, sizeof(ht_config));
1061 cur_ht_config = &ht_config;
1062 }
1063 rcu_read_unlock();
1064 }
1065
Tomas Winklerc587de02009-06-03 11:44:07 -07001066 sta_id = iwl_add_station(priv, addr, is_ap, CMD_SYNC, cur_ht_config);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001067
1068 /* Set up default rate scaling table in device's station table */
1069 iwl_sta_init_lq(priv, addr, is_ap);
1070
1071 return sta_id;
1072}
1073EXPORT_SYMBOL(iwl_rxon_add_station);
1074
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001075/**
Reinette Chatre9a9ca652009-10-30 14:36:12 -07001076 * iwl_sta_init_bcast_lq - Initialize a bcast station's hardware rate table
1077 *
1078 * NOTE: Run REPLY_ADD_STA command to set up station table entry, before
1079 * calling this function (which runs REPLY_TX_LINK_QUALITY_CMD,
1080 * which requires station table entry to exist).
1081 */
1082static void iwl_sta_init_bcast_lq(struct iwl_priv *priv)
1083{
1084 int i, r;
1085 struct iwl_link_quality_cmd link_cmd = {
1086 .reserved1 = 0,
1087 };
1088 u32 rate_flags;
1089
1090 /* Set up the rate scaling to start at selected rate, fall back
1091 * all the way down to 1M in IEEE order, and then spin on 1M */
1092 if (priv->band == IEEE80211_BAND_5GHZ)
1093 r = IWL_RATE_6M_INDEX;
1094 else
1095 r = IWL_RATE_1M_INDEX;
1096
1097 for (i = 0; i < LINK_QUAL_MAX_RETRY_NUM; i++) {
1098 rate_flags = 0;
1099 if (r >= IWL_FIRST_CCK_RATE && r <= IWL_LAST_CCK_RATE)
1100 rate_flags |= RATE_MCS_CCK_MSK;
1101
1102 rate_flags |= first_antenna(priv->hw_params.valid_tx_ant) <<
1103 RATE_MCS_ANT_POS;
1104
1105 link_cmd.rs_table[i].rate_n_flags =
1106 iwl_hw_set_rate_n_flags(iwl_rates[r].plcp, rate_flags);
1107 r = iwl_get_prev_ieee_rate(r);
1108 }
1109
1110 link_cmd.general_params.single_stream_ant_msk =
1111 first_antenna(priv->hw_params.valid_tx_ant);
1112 link_cmd.general_params.dual_stream_ant_msk = 3;
1113 link_cmd.agg_params.agg_dis_start_th = LINK_QUAL_AGG_DISABLE_START_DEF;
1114 link_cmd.agg_params.agg_time_limit =
1115 cpu_to_le16(LINK_QUAL_AGG_TIME_LIMIT_DEF);
1116
1117 /* Update the rate scaling for control frame Tx to AP */
1118 link_cmd.sta_id = priv->hw_params.bcast_sta_id;
1119
1120 iwl_send_cmd_pdu_async(priv, REPLY_TX_LINK_QUALITY_CMD,
1121 sizeof(link_cmd), &link_cmd, NULL);
1122}
1123
1124
1125/**
1126 * iwl_add_bcast_station - add broadcast station into station table.
1127 */
1128void iwl_add_bcast_station(struct iwl_priv *priv)
1129{
Reinette Chatre3459ab52010-01-22 14:22:49 -08001130 IWL_DEBUG_INFO(priv, "Adding broadcast station to station table\n");
Reinette Chatre9a9ca652009-10-30 14:36:12 -07001131 iwl_add_station(priv, iwl_bcast_addr, false, CMD_SYNC, NULL);
1132
1133 /* Set up default rate scaling table in device's station table */
1134 iwl_sta_init_bcast_lq(priv);
1135}
1136EXPORT_SYMBOL(iwl_add_bcast_station);
1137
1138/**
Reinette Chatre3459ab52010-01-22 14:22:49 -08001139 * iwl3945_add_bcast_station - add broadcast station into station table.
1140 */
1141void iwl3945_add_bcast_station(struct iwl_priv *priv)
1142{
1143 IWL_DEBUG_INFO(priv, "Adding broadcast station to station table\n");
1144 iwl_add_station(priv, iwl_bcast_addr, false, CMD_SYNC, NULL);
1145}
1146EXPORT_SYMBOL(iwl3945_add_bcast_station);
1147
1148/**
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001149 * iwl_get_sta_id - Find station's index within station table
1150 *
1151 * If new IBSS station, create new entry in station table
1152 */
1153int iwl_get_sta_id(struct iwl_priv *priv, struct ieee80211_hdr *hdr)
1154{
1155 int sta_id;
Luis R. Rodriguez943ab702009-07-14 20:14:07 -04001156 __le16 fc = hdr->frame_control;
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001157
1158 /* If this frame is broadcast or management, use broadcast station id */
Luis R. Rodriguez943ab702009-07-14 20:14:07 -04001159 if (!ieee80211_is_data(fc) || is_multicast_ether_addr(hdr->addr1))
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001160 return priv->hw_params.bcast_sta_id;
1161
1162 switch (priv->iw_mode) {
1163
1164 /* If we are a client station in a BSS network, use the special
1165 * AP station entry (that's the only station we communicate with) */
Johannes Berg05c914f2008-09-11 00:01:58 +02001166 case NL80211_IFTYPE_STATION:
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001167 return IWL_AP_ID;
1168
1169 /* If we are an AP, then find the station, or use BCAST */
Johannes Berg05c914f2008-09-11 00:01:58 +02001170 case NL80211_IFTYPE_AP:
Tomas Winklerc587de02009-06-03 11:44:07 -07001171 sta_id = iwl_find_station(priv, hdr->addr1);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001172 if (sta_id != IWL_INVALID_STATION)
1173 return sta_id;
1174 return priv->hw_params.bcast_sta_id;
1175
1176 /* If this frame is going out to an IBSS network, find the station,
1177 * or create a new station table entry */
Johannes Berg05c914f2008-09-11 00:01:58 +02001178 case NL80211_IFTYPE_ADHOC:
Tomas Winklerc587de02009-06-03 11:44:07 -07001179 sta_id = iwl_find_station(priv, hdr->addr1);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001180 if (sta_id != IWL_INVALID_STATION)
1181 return sta_id;
1182
1183 /* Create new station table entry */
Tomas Winklerc587de02009-06-03 11:44:07 -07001184 sta_id = iwl_add_station(priv, hdr->addr1, false,
1185 CMD_ASYNC, NULL);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001186
1187 if (sta_id != IWL_INVALID_STATION)
1188 return sta_id;
1189
Tomas Winklere1623442009-01-27 14:27:56 -08001190 IWL_DEBUG_DROP(priv, "Station %pM not in station map. "
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001191 "Defaulting to broadcast...\n",
Johannes Berge1749612008-10-27 15:59:26 -07001192 hdr->addr1);
Reinette Chatre3d816c72009-08-07 15:41:37 -07001193 iwl_print_hex_dump(priv, IWL_DL_DROP, (u8 *) hdr, sizeof(*hdr));
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001194 return priv->hw_params.bcast_sta_id;
1195
1196 default:
Winkler, Tomas39aadf82008-12-19 10:37:32 +08001197 IWL_WARN(priv, "Unknown mode of operation: %d\n",
1198 priv->iw_mode);
Tomas Winkler4f40e4d2008-05-15 13:54:04 +08001199 return priv->hw_params.bcast_sta_id;
1200 }
1201}
1202EXPORT_SYMBOL(iwl_get_sta_id);
1203
Tomas Winkler5083e562008-05-29 16:35:15 +08001204/**
Tomas Winkler9f586712008-11-12 13:14:05 -08001205 * iwl_sta_tx_modify_enable_tid - Enable Tx for this TID in station table
Tomas Winkler5083e562008-05-29 16:35:15 +08001206 */
Tomas Winkler9f586712008-11-12 13:14:05 -08001207void iwl_sta_tx_modify_enable_tid(struct iwl_priv *priv, int sta_id, int tid)
Tomas Winkler5083e562008-05-29 16:35:15 +08001208{
1209 unsigned long flags;
1210
1211 /* Remove "disable" flag, to enable Tx for this TID */
1212 spin_lock_irqsave(&priv->sta_lock, flags);
1213 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_TID_DISABLE_TX;
1214 priv->stations[sta_id].sta.tid_disable_tx &= cpu_to_le16(~(1 << tid));
1215 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
1216 spin_unlock_irqrestore(&priv->sta_lock, flags);
1217
1218 iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
1219}
Tomas Winkler9f586712008-11-12 13:14:05 -08001220EXPORT_SYMBOL(iwl_sta_tx_modify_enable_tid);
1221
1222int iwl_sta_rx_agg_start(struct iwl_priv *priv,
1223 const u8 *addr, int tid, u16 ssn)
1224{
1225 unsigned long flags;
1226 int sta_id;
1227
Tomas Winklerc587de02009-06-03 11:44:07 -07001228 sta_id = iwl_find_station(priv, addr);
Tomas Winkler9f586712008-11-12 13:14:05 -08001229 if (sta_id == IWL_INVALID_STATION)
1230 return -ENXIO;
1231
1232 spin_lock_irqsave(&priv->sta_lock, flags);
1233 priv->stations[sta_id].sta.station_flags_msk = 0;
1234 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_ADDBA_TID_MSK;
1235 priv->stations[sta_id].sta.add_immediate_ba_tid = (u8)tid;
1236 priv->stations[sta_id].sta.add_immediate_ba_ssn = cpu_to_le16(ssn);
1237 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
1238 spin_unlock_irqrestore(&priv->sta_lock, flags);
1239
1240 return iwl_send_add_sta(priv, &priv->stations[sta_id].sta,
1241 CMD_ASYNC);
1242}
1243EXPORT_SYMBOL(iwl_sta_rx_agg_start);
1244
1245int iwl_sta_rx_agg_stop(struct iwl_priv *priv, const u8 *addr, int tid)
1246{
1247 unsigned long flags;
1248 int sta_id;
1249
Tomas Winklerc587de02009-06-03 11:44:07 -07001250 sta_id = iwl_find_station(priv, addr);
Wey-Yi Guya2f1cbe2009-03-17 21:51:52 -07001251 if (sta_id == IWL_INVALID_STATION) {
1252 IWL_ERR(priv, "Invalid station for AGG tid %d\n", tid);
Tomas Winkler9f586712008-11-12 13:14:05 -08001253 return -ENXIO;
Wey-Yi Guya2f1cbe2009-03-17 21:51:52 -07001254 }
Tomas Winkler9f586712008-11-12 13:14:05 -08001255
1256 spin_lock_irqsave(&priv->sta_lock, flags);
1257 priv->stations[sta_id].sta.station_flags_msk = 0;
1258 priv->stations[sta_id].sta.sta.modify_mask = STA_MODIFY_DELBA_TID_MSK;
1259 priv->stations[sta_id].sta.remove_immediate_ba_tid = (u8)tid;
1260 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
1261 spin_unlock_irqrestore(&priv->sta_lock, flags);
1262
1263 return iwl_send_add_sta(priv, &priv->stations[sta_id].sta,
1264 CMD_ASYNC);
1265}
1266EXPORT_SYMBOL(iwl_sta_rx_agg_stop);
1267
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001268void iwl_sta_modify_ps_wake(struct iwl_priv *priv, int sta_id)
Tomas Winkler9f586712008-11-12 13:14:05 -08001269{
1270 unsigned long flags;
1271
1272 spin_lock_irqsave(&priv->sta_lock, flags);
1273 priv->stations[sta_id].sta.station_flags &= ~STA_FLG_PWR_SAVE_MSK;
1274 priv->stations[sta_id].sta.station_flags_msk = STA_FLG_PWR_SAVE_MSK;
1275 priv->stations[sta_id].sta.sta.modify_mask = 0;
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001276 priv->stations[sta_id].sta.sleep_tx_count = 0;
Tomas Winkler9f586712008-11-12 13:14:05 -08001277 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
1278 spin_unlock_irqrestore(&priv->sta_lock, flags);
1279
1280 iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
1281}
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001282EXPORT_SYMBOL(iwl_sta_modify_ps_wake);
Tomas Winkler9f586712008-11-12 13:14:05 -08001283
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001284void iwl_sta_modify_sleep_tx_count(struct iwl_priv *priv, int sta_id, int cnt)
Tomas Winkler9f586712008-11-12 13:14:05 -08001285{
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001286 unsigned long flags;
Tomas Winkler9f586712008-11-12 13:14:05 -08001287
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001288 spin_lock_irqsave(&priv->sta_lock, flags);
1289 priv->stations[sta_id].sta.station_flags |= STA_FLG_PWR_SAVE_MSK;
1290 priv->stations[sta_id].sta.station_flags_msk = STA_FLG_PWR_SAVE_MSK;
1291 priv->stations[sta_id].sta.sta.modify_mask =
1292 STA_MODIFY_SLEEP_TX_COUNT_MSK;
1293 priv->stations[sta_id].sta.sleep_tx_count = cpu_to_le16(cnt);
1294 priv->stations[sta_id].sta.mode = STA_CONTROL_MODIFY_MSK;
1295 spin_unlock_irqrestore(&priv->sta_lock, flags);
Tomas Winkler9f586712008-11-12 13:14:05 -08001296
Johannes Berg6ab10ff2009-11-13 11:56:37 -08001297 iwl_send_add_sta(priv, &priv->stations[sta_id].sta, CMD_ASYNC);
Tomas Winkler9f586712008-11-12 13:14:05 -08001298}