diff options
Diffstat (limited to 'net/mac80211')
-rw-r--r-- | net/mac80211/agg-tx.c | 5 | ||||
-rw-r--r-- | net/mac80211/cfg.c | 65 | ||||
-rw-r--r-- | net/mac80211/debugfs.c | 94 | ||||
-rw-r--r-- | net/mac80211/debugfs_netdev.c | 44 | ||||
-rw-r--r-- | net/mac80211/driver-ops.h | 15 | ||||
-rw-r--r-- | net/mac80211/driver-trace.h | 23 | ||||
-rw-r--r-- | net/mac80211/ieee80211_i.h | 31 | ||||
-rw-r--r-- | net/mac80211/iface.c | 56 | ||||
-rw-r--r-- | net/mac80211/main.c | 7 | ||||
-rw-r--r-- | net/mac80211/mlme.c | 104 | ||||
-rw-r--r-- | net/mac80211/offchannel.c | 8 | ||||
-rw-r--r-- | net/mac80211/rate.c | 88 | ||||
-rw-r--r-- | net/mac80211/rate.h | 5 | ||||
-rw-r--r-- | net/mac80211/rx.c | 16 | ||||
-rw-r--r-- | net/mac80211/scan.c | 20 | ||||
-rw-r--r-- | net/mac80211/sta_info.c | 21 | ||||
-rw-r--r-- | net/mac80211/sta_info.h | 3 | ||||
-rw-r--r-- | net/mac80211/tx.c | 278 | ||||
-rw-r--r-- | net/mac80211/util.c | 60 | ||||
-rw-r--r-- | net/mac80211/wme.c | 96 | ||||
-rw-r--r-- | net/mac80211/wme.h | 8 | ||||
-rw-r--r-- | net/mac80211/work.c | 28 |
22 files changed, 834 insertions, 241 deletions
diff --git a/net/mac80211/agg-tx.c b/net/mac80211/agg-tx.c index ceda36618d3c..718fbcff84d2 100644 --- a/net/mac80211/agg-tx.c +++ b/net/mac80211/agg-tx.c | |||
@@ -179,7 +179,8 @@ static void sta_addba_resp_timer_expired(unsigned long data) | |||
179 | 179 | ||
180 | /* check if the TID waits for addBA response */ | 180 | /* check if the TID waits for addBA response */ |
181 | spin_lock_bh(&sta->lock); | 181 | spin_lock_bh(&sta->lock); |
182 | if ((*state & (HT_ADDBA_REQUESTED_MSK | HT_ADDBA_RECEIVED_MSK)) != | 182 | if ((*state & (HT_ADDBA_REQUESTED_MSK | HT_ADDBA_RECEIVED_MSK | |
183 | HT_AGG_STATE_REQ_STOP_BA_MSK)) != | ||
183 | HT_ADDBA_REQUESTED_MSK) { | 184 | HT_ADDBA_REQUESTED_MSK) { |
184 | spin_unlock_bh(&sta->lock); | 185 | spin_unlock_bh(&sta->lock); |
185 | *state = HT_AGG_STATE_IDLE; | 186 | *state = HT_AGG_STATE_IDLE; |
@@ -301,7 +302,7 @@ int ieee80211_start_tx_ba_session(struct ieee80211_sta *pubsta, u16 tid) | |||
301 | * call back right away, it must see that the flow has begun */ | 302 | * call back right away, it must see that the flow has begun */ |
302 | *state |= HT_ADDBA_REQUESTED_MSK; | 303 | *state |= HT_ADDBA_REQUESTED_MSK; |
303 | 304 | ||
304 | start_seq_num = sta->tid_seq[tid]; | 305 | start_seq_num = sta->tid_seq[tid] >> 4; |
305 | 306 | ||
306 | ret = drv_ampdu_action(local, sdata, IEEE80211_AMPDU_TX_START, | 307 | ret = drv_ampdu_action(local, sdata, IEEE80211_AMPDU_TX_START, |
307 | pubsta, tid, &start_seq_num); | 308 | pubsta, tid, &start_seq_num); |
diff --git a/net/mac80211/cfg.c b/net/mac80211/cfg.c index 2e5e841e9b7b..b0102c538b30 100644 --- a/net/mac80211/cfg.c +++ b/net/mac80211/cfg.c | |||
@@ -148,7 +148,7 @@ static int ieee80211_add_key(struct wiphy *wiphy, struct net_device *dev, | |||
148 | rcu_read_lock(); | 148 | rcu_read_lock(); |
149 | 149 | ||
150 | if (mac_addr) { | 150 | if (mac_addr) { |
151 | sta = sta_info_get(sdata, mac_addr); | 151 | sta = sta_info_get_bss(sdata, mac_addr); |
152 | if (!sta) { | 152 | if (!sta) { |
153 | ieee80211_key_free(key); | 153 | ieee80211_key_free(key); |
154 | err = -ENOENT; | 154 | err = -ENOENT; |
@@ -179,7 +179,7 @@ static int ieee80211_del_key(struct wiphy *wiphy, struct net_device *dev, | |||
179 | if (mac_addr) { | 179 | if (mac_addr) { |
180 | ret = -ENOENT; | 180 | ret = -ENOENT; |
181 | 181 | ||
182 | sta = sta_info_get(sdata, mac_addr); | 182 | sta = sta_info_get_bss(sdata, mac_addr); |
183 | if (!sta) | 183 | if (!sta) |
184 | goto out_unlock; | 184 | goto out_unlock; |
185 | 185 | ||
@@ -226,7 +226,7 @@ static int ieee80211_get_key(struct wiphy *wiphy, struct net_device *dev, | |||
226 | rcu_read_lock(); | 226 | rcu_read_lock(); |
227 | 227 | ||
228 | if (mac_addr) { | 228 | if (mac_addr) { |
229 | sta = sta_info_get(sdata, mac_addr); | 229 | sta = sta_info_get_bss(sdata, mac_addr); |
230 | if (!sta) | 230 | if (!sta) |
231 | goto out; | 231 | goto out; |
232 | 232 | ||
@@ -419,7 +419,7 @@ static int ieee80211_get_station(struct wiphy *wiphy, struct net_device *dev, | |||
419 | 419 | ||
420 | rcu_read_lock(); | 420 | rcu_read_lock(); |
421 | 421 | ||
422 | sta = sta_info_get(sdata, mac); | 422 | sta = sta_info_get_bss(sdata, mac); |
423 | if (sta) { | 423 | if (sta) { |
424 | ret = 0; | 424 | ret = 0; |
425 | sta_set_sinfo(sta, sinfo); | 425 | sta_set_sinfo(sta, sinfo); |
@@ -775,7 +775,7 @@ static int ieee80211_del_station(struct wiphy *wiphy, struct net_device *dev, | |||
775 | if (mac) { | 775 | if (mac) { |
776 | rcu_read_lock(); | 776 | rcu_read_lock(); |
777 | 777 | ||
778 | sta = sta_info_get(sdata, mac); | 778 | sta = sta_info_get_bss(sdata, mac); |
779 | if (!sta) { | 779 | if (!sta) { |
780 | rcu_read_unlock(); | 780 | rcu_read_unlock(); |
781 | return -ENOENT; | 781 | return -ENOENT; |
@@ -803,7 +803,7 @@ static int ieee80211_change_station(struct wiphy *wiphy, | |||
803 | 803 | ||
804 | rcu_read_lock(); | 804 | rcu_read_lock(); |
805 | 805 | ||
806 | sta = sta_info_get(sdata, mac); | 806 | sta = sta_info_get_bss(sdata, mac); |
807 | if (!sta) { | 807 | if (!sta) { |
808 | rcu_read_unlock(); | 808 | rcu_read_unlock(); |
809 | return -ENOENT; | 809 | return -ENOENT; |
@@ -1085,6 +1085,13 @@ static int ieee80211_change_bss(struct wiphy *wiphy, | |||
1085 | params->use_short_preamble; | 1085 | params->use_short_preamble; |
1086 | changed |= BSS_CHANGED_ERP_PREAMBLE; | 1086 | changed |= BSS_CHANGED_ERP_PREAMBLE; |
1087 | } | 1087 | } |
1088 | |||
1089 | if (!sdata->vif.bss_conf.use_short_slot && | ||
1090 | sdata->local->hw.conf.channel->band == IEEE80211_BAND_5GHZ) { | ||
1091 | sdata->vif.bss_conf.use_short_slot = true; | ||
1092 | changed |= BSS_CHANGED_ERP_SLOT; | ||
1093 | } | ||
1094 | |||
1088 | if (params->use_short_slot_time >= 0) { | 1095 | if (params->use_short_slot_time >= 0) { |
1089 | sdata->vif.bss_conf.use_short_slot = | 1096 | sdata->vif.bss_conf.use_short_slot = |
1090 | params->use_short_slot_time; | 1097 | params->use_short_slot_time; |
@@ -1128,6 +1135,13 @@ static int ieee80211_set_txq_params(struct wiphy *wiphy, | |||
1128 | p.cw_max = params->cwmax; | 1135 | p.cw_max = params->cwmax; |
1129 | p.cw_min = params->cwmin; | 1136 | p.cw_min = params->cwmin; |
1130 | p.txop = params->txop; | 1137 | p.txop = params->txop; |
1138 | |||
1139 | /* | ||
1140 | * Setting tx queue params disables u-apsd because it's only | ||
1141 | * called in master mode. | ||
1142 | */ | ||
1143 | p.uapsd = false; | ||
1144 | |||
1131 | if (drv_conf_tx(local, params->queue, &p)) { | 1145 | if (drv_conf_tx(local, params->queue, &p)) { |
1132 | printk(KERN_DEBUG "%s: failed to set TX queue " | 1146 | printk(KERN_DEBUG "%s: failed to set TX queue " |
1133 | "parameters for queue %d\n", | 1147 | "parameters for queue %d\n", |
@@ -1230,6 +1244,13 @@ static int ieee80211_set_wiphy_params(struct wiphy *wiphy, u32 changed) | |||
1230 | struct ieee80211_local *local = wiphy_priv(wiphy); | 1244 | struct ieee80211_local *local = wiphy_priv(wiphy); |
1231 | int err; | 1245 | int err; |
1232 | 1246 | ||
1247 | if (changed & WIPHY_PARAM_COVERAGE_CLASS) { | ||
1248 | err = drv_set_coverage_class(local, wiphy->coverage_class); | ||
1249 | |||
1250 | if (err) | ||
1251 | return err; | ||
1252 | } | ||
1253 | |||
1233 | if (changed & WIPHY_PARAM_RTS_THRESHOLD) { | 1254 | if (changed & WIPHY_PARAM_RTS_THRESHOLD) { |
1234 | err = drv_set_rts_threshold(local, wiphy->rts_threshold); | 1255 | err = drv_set_rts_threshold(local, wiphy->rts_threshold); |
1235 | 1256 | ||
@@ -1399,8 +1420,6 @@ static int ieee80211_set_bitrate_mask(struct wiphy *wiphy, | |||
1399 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); | 1420 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); |
1400 | struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); | 1421 | struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); |
1401 | int i; | 1422 | int i; |
1402 | u32 target_rate; | ||
1403 | struct ieee80211_supported_band *sband; | ||
1404 | 1423 | ||
1405 | /* | 1424 | /* |
1406 | * This _could_ be supported by providing a hook for | 1425 | * This _could_ be supported by providing a hook for |
@@ -1410,35 +1429,11 @@ static int ieee80211_set_bitrate_mask(struct wiphy *wiphy, | |||
1410 | if (local->hw.flags & IEEE80211_HW_HAS_RATE_CONTROL) | 1429 | if (local->hw.flags & IEEE80211_HW_HAS_RATE_CONTROL) |
1411 | return -EOPNOTSUPP; | 1430 | return -EOPNOTSUPP; |
1412 | 1431 | ||
1413 | sband = local->hw.wiphy->bands[local->hw.conf.channel->band]; | ||
1414 | |||
1415 | /* | ||
1416 | * target_rate = -1, rate->fixed = 0 means auto only, so use all rates | ||
1417 | * target_rate = X, rate->fixed = 1 means only rate X | ||
1418 | * target_rate = X, rate->fixed = 0 means all rates <= X | ||
1419 | */ | ||
1420 | sdata->max_ratectrl_rateidx = -1; | ||
1421 | sdata->force_unicast_rateidx = -1; | ||
1422 | |||
1423 | if (mask->fixed) | ||
1424 | target_rate = mask->fixed / 100; | ||
1425 | else if (mask->maxrate) | ||
1426 | target_rate = mask->maxrate / 100; | ||
1427 | else | ||
1428 | return 0; | ||
1429 | |||
1430 | for (i = 0; i< sband->n_bitrates; i++) { | ||
1431 | if (target_rate != sband->bitrates[i].bitrate) | ||
1432 | continue; | ||
1433 | 1432 | ||
1434 | /* requested bitrate found */ | 1433 | for (i = 0; i < IEEE80211_NUM_BANDS; i++) |
1435 | sdata->max_ratectrl_rateidx = i; | 1434 | sdata->rc_rateidx_mask[i] = mask->control[i].legacy; |
1436 | if (mask->fixed) | ||
1437 | sdata->force_unicast_rateidx = i; | ||
1438 | return 0; | ||
1439 | } | ||
1440 | 1435 | ||
1441 | return -EINVAL; | 1436 | return 0; |
1442 | } | 1437 | } |
1443 | 1438 | ||
1444 | static int ieee80211_remain_on_channel(struct wiphy *wiphy, | 1439 | static int ieee80211_remain_on_channel(struct wiphy *wiphy, |
diff --git a/net/mac80211/debugfs.c b/net/mac80211/debugfs.c index e4b54093d41b..b3bc32b62a5a 100644 --- a/net/mac80211/debugfs.c +++ b/net/mac80211/debugfs.c | |||
@@ -158,6 +158,98 @@ static const struct file_operations noack_ops = { | |||
158 | .open = mac80211_open_file_generic | 158 | .open = mac80211_open_file_generic |
159 | }; | 159 | }; |
160 | 160 | ||
161 | static ssize_t uapsd_queues_read(struct file *file, char __user *user_buf, | ||
162 | size_t count, loff_t *ppos) | ||
163 | { | ||
164 | struct ieee80211_local *local = file->private_data; | ||
165 | int res; | ||
166 | char buf[10]; | ||
167 | |||
168 | res = scnprintf(buf, sizeof(buf), "0x%x\n", local->uapsd_queues); | ||
169 | |||
170 | return simple_read_from_buffer(user_buf, count, ppos, buf, res); | ||
171 | } | ||
172 | |||
173 | static ssize_t uapsd_queues_write(struct file *file, | ||
174 | const char __user *user_buf, | ||
175 | size_t count, loff_t *ppos) | ||
176 | { | ||
177 | struct ieee80211_local *local = file->private_data; | ||
178 | unsigned long val; | ||
179 | char buf[10]; | ||
180 | size_t len; | ||
181 | int ret; | ||
182 | |||
183 | len = min(count, sizeof(buf) - 1); | ||
184 | if (copy_from_user(buf, user_buf, len)) | ||
185 | return -EFAULT; | ||
186 | buf[len] = '\0'; | ||
187 | |||
188 | ret = strict_strtoul(buf, 0, &val); | ||
189 | |||
190 | if (ret) | ||
191 | return -EINVAL; | ||
192 | |||
193 | if (val & ~IEEE80211_WMM_IE_STA_QOSINFO_AC_MASK) | ||
194 | return -ERANGE; | ||
195 | |||
196 | local->uapsd_queues = val; | ||
197 | |||
198 | return count; | ||
199 | } | ||
200 | |||
201 | static const struct file_operations uapsd_queues_ops = { | ||
202 | .read = uapsd_queues_read, | ||
203 | .write = uapsd_queues_write, | ||
204 | .open = mac80211_open_file_generic | ||
205 | }; | ||
206 | |||
207 | static ssize_t uapsd_max_sp_len_read(struct file *file, char __user *user_buf, | ||
208 | size_t count, loff_t *ppos) | ||
209 | { | ||
210 | struct ieee80211_local *local = file->private_data; | ||
211 | int res; | ||
212 | char buf[10]; | ||
213 | |||
214 | res = scnprintf(buf, sizeof(buf), "0x%x\n", local->uapsd_max_sp_len); | ||
215 | |||
216 | return simple_read_from_buffer(user_buf, count, ppos, buf, res); | ||
217 | } | ||
218 | |||
219 | static ssize_t uapsd_max_sp_len_write(struct file *file, | ||
220 | const char __user *user_buf, | ||
221 | size_t count, loff_t *ppos) | ||
222 | { | ||
223 | struct ieee80211_local *local = file->private_data; | ||
224 | unsigned long val; | ||
225 | char buf[10]; | ||
226 | size_t len; | ||
227 | int ret; | ||
228 | |||
229 | len = min(count, sizeof(buf) - 1); | ||
230 | if (copy_from_user(buf, user_buf, len)) | ||
231 | return -EFAULT; | ||
232 | buf[len] = '\0'; | ||
233 | |||
234 | ret = strict_strtoul(buf, 0, &val); | ||
235 | |||
236 | if (ret) | ||
237 | return -EINVAL; | ||
238 | |||
239 | if (val & ~IEEE80211_WMM_IE_STA_QOSINFO_SP_MASK) | ||
240 | return -ERANGE; | ||
241 | |||
242 | local->uapsd_max_sp_len = val; | ||
243 | |||
244 | return count; | ||
245 | } | ||
246 | |||
247 | static const struct file_operations uapsd_max_sp_len_ops = { | ||
248 | .read = uapsd_max_sp_len_read, | ||
249 | .write = uapsd_max_sp_len_write, | ||
250 | .open = mac80211_open_file_generic | ||
251 | }; | ||
252 | |||
161 | static ssize_t queues_read(struct file *file, char __user *user_buf, | 253 | static ssize_t queues_read(struct file *file, char __user *user_buf, |
162 | size_t count, loff_t *ppos) | 254 | size_t count, loff_t *ppos) |
163 | { | 255 | { |
@@ -314,6 +406,8 @@ void debugfs_hw_add(struct ieee80211_local *local) | |||
314 | DEBUGFS_ADD(queues); | 406 | DEBUGFS_ADD(queues); |
315 | DEBUGFS_ADD_MODE(reset, 0200); | 407 | DEBUGFS_ADD_MODE(reset, 0200); |
316 | DEBUGFS_ADD(noack); | 408 | DEBUGFS_ADD(noack); |
409 | DEBUGFS_ADD(uapsd_queues); | ||
410 | DEBUGFS_ADD(uapsd_max_sp_len); | ||
317 | 411 | ||
318 | statsd = debugfs_create_dir("statistics", phyd); | 412 | statsd = debugfs_create_dir("statistics", phyd); |
319 | 413 | ||
diff --git a/net/mac80211/debugfs_netdev.c b/net/mac80211/debugfs_netdev.c index 59f6e3bcbd09..9affe2cd185f 100644 --- a/net/mac80211/debugfs_netdev.c +++ b/net/mac80211/debugfs_netdev.c | |||
@@ -127,8 +127,10 @@ __IEEE80211_IF_FILE(name, ieee80211_if_write_##name) | |||
127 | 127 | ||
128 | /* common attributes */ | 128 | /* common attributes */ |
129 | IEEE80211_IF_FILE(drop_unencrypted, drop_unencrypted, DEC); | 129 | IEEE80211_IF_FILE(drop_unencrypted, drop_unencrypted, DEC); |
130 | IEEE80211_IF_FILE(force_unicast_rateidx, force_unicast_rateidx, DEC); | 130 | IEEE80211_IF_FILE(rc_rateidx_mask_2ghz, rc_rateidx_mask[IEEE80211_BAND_2GHZ], |
131 | IEEE80211_IF_FILE(max_ratectrl_rateidx, max_ratectrl_rateidx, DEC); | 131 | HEX); |
132 | IEEE80211_IF_FILE(rc_rateidx_mask_5ghz, rc_rateidx_mask[IEEE80211_BAND_5GHZ], | ||
133 | HEX); | ||
132 | 134 | ||
133 | /* STA attributes */ | 135 | /* STA attributes */ |
134 | IEEE80211_IF_FILE(bssid, u.mgd.bssid, MAC); | 136 | IEEE80211_IF_FILE(bssid, u.mgd.bssid, MAC); |
@@ -253,7 +255,7 @@ IEEE80211_IF_FILE(dot11MeshHWMPRootMode, | |||
253 | #endif | 255 | #endif |
254 | 256 | ||
255 | 257 | ||
256 | #define DEBUGFS_ADD(name, type) \ | 258 | #define DEBUGFS_ADD(name) \ |
257 | debugfs_create_file(#name, 0400, sdata->debugfs.dir, \ | 259 | debugfs_create_file(#name, 0400, sdata->debugfs.dir, \ |
258 | sdata, &name##_ops); | 260 | sdata, &name##_ops); |
259 | 261 | ||
@@ -263,40 +265,40 @@ IEEE80211_IF_FILE(dot11MeshHWMPRootMode, | |||
263 | 265 | ||
264 | static void add_sta_files(struct ieee80211_sub_if_data *sdata) | 266 | static void add_sta_files(struct ieee80211_sub_if_data *sdata) |
265 | { | 267 | { |
266 | DEBUGFS_ADD(drop_unencrypted, sta); | 268 | DEBUGFS_ADD(drop_unencrypted); |
267 | DEBUGFS_ADD(force_unicast_rateidx, sta); | 269 | DEBUGFS_ADD(rc_rateidx_mask_2ghz); |
268 | DEBUGFS_ADD(max_ratectrl_rateidx, sta); | 270 | DEBUGFS_ADD(rc_rateidx_mask_5ghz); |
269 | 271 | ||
270 | DEBUGFS_ADD(bssid, sta); | 272 | DEBUGFS_ADD(bssid); |
271 | DEBUGFS_ADD(aid, sta); | 273 | DEBUGFS_ADD(aid); |
272 | DEBUGFS_ADD_MODE(smps, 0600); | 274 | DEBUGFS_ADD_MODE(smps, 0600); |
273 | } | 275 | } |
274 | 276 | ||
275 | static void add_ap_files(struct ieee80211_sub_if_data *sdata) | 277 | static void add_ap_files(struct ieee80211_sub_if_data *sdata) |
276 | { | 278 | { |
277 | DEBUGFS_ADD(drop_unencrypted, ap); | 279 | DEBUGFS_ADD(drop_unencrypted); |
278 | DEBUGFS_ADD(force_unicast_rateidx, ap); | 280 | DEBUGFS_ADD(rc_rateidx_mask_2ghz); |
279 | DEBUGFS_ADD(max_ratectrl_rateidx, ap); | 281 | DEBUGFS_ADD(rc_rateidx_mask_5ghz); |
280 | 282 | ||
281 | DEBUGFS_ADD(num_sta_ps, ap); | 283 | DEBUGFS_ADD(num_sta_ps); |
282 | DEBUGFS_ADD(dtim_count, ap); | 284 | DEBUGFS_ADD(dtim_count); |
283 | DEBUGFS_ADD(num_buffered_multicast, ap); | 285 | DEBUGFS_ADD(num_buffered_multicast); |
284 | } | 286 | } |
285 | 287 | ||
286 | static void add_wds_files(struct ieee80211_sub_if_data *sdata) | 288 | static void add_wds_files(struct ieee80211_sub_if_data *sdata) |
287 | { | 289 | { |
288 | DEBUGFS_ADD(drop_unencrypted, wds); | 290 | DEBUGFS_ADD(drop_unencrypted); |
289 | DEBUGFS_ADD(force_unicast_rateidx, wds); | 291 | DEBUGFS_ADD(rc_rateidx_mask_2ghz); |
290 | DEBUGFS_ADD(max_ratectrl_rateidx, wds); | 292 | DEBUGFS_ADD(rc_rateidx_mask_5ghz); |
291 | 293 | ||
292 | DEBUGFS_ADD(peer, wds); | 294 | DEBUGFS_ADD(peer); |
293 | } | 295 | } |
294 | 296 | ||
295 | static void add_vlan_files(struct ieee80211_sub_if_data *sdata) | 297 | static void add_vlan_files(struct ieee80211_sub_if_data *sdata) |
296 | { | 298 | { |
297 | DEBUGFS_ADD(drop_unencrypted, vlan); | 299 | DEBUGFS_ADD(drop_unencrypted); |
298 | DEBUGFS_ADD(force_unicast_rateidx, vlan); | 300 | DEBUGFS_ADD(rc_rateidx_mask_2ghz); |
299 | DEBUGFS_ADD(max_ratectrl_rateidx, vlan); | 301 | DEBUGFS_ADD(rc_rateidx_mask_5ghz); |
300 | } | 302 | } |
301 | 303 | ||
302 | static void add_monitor_files(struct ieee80211_sub_if_data *sdata) | 304 | static void add_monitor_files(struct ieee80211_sub_if_data *sdata) |
diff --git a/net/mac80211/driver-ops.h b/net/mac80211/driver-ops.h index 8757ea73d544..de91d39e0276 100644 --- a/net/mac80211/driver-ops.h +++ b/net/mac80211/driver-ops.h | |||
@@ -214,6 +214,21 @@ static inline int drv_set_rts_threshold(struct ieee80211_local *local, | |||
214 | return ret; | 214 | return ret; |
215 | } | 215 | } |
216 | 216 | ||
217 | static inline int drv_set_coverage_class(struct ieee80211_local *local, | ||
218 | u8 value) | ||
219 | { | ||
220 | int ret = 0; | ||
221 | might_sleep(); | ||
222 | |||
223 | if (local->ops->set_coverage_class) | ||
224 | local->ops->set_coverage_class(&local->hw, value); | ||
225 | else | ||
226 | ret = -EOPNOTSUPP; | ||
227 | |||
228 | trace_drv_set_coverage_class(local, value, ret); | ||
229 | return ret; | ||
230 | } | ||
231 | |||
217 | static inline void drv_sta_notify(struct ieee80211_local *local, | 232 | static inline void drv_sta_notify(struct ieee80211_local *local, |
218 | struct ieee80211_sub_if_data *sdata, | 233 | struct ieee80211_sub_if_data *sdata, |
219 | enum sta_notify_cmd cmd, | 234 | enum sta_notify_cmd cmd, |
diff --git a/net/mac80211/driver-trace.h b/net/mac80211/driver-trace.h index 977cc7528bc6..0ea258123b8e 100644 --- a/net/mac80211/driver-trace.h +++ b/net/mac80211/driver-trace.h | |||
@@ -491,6 +491,29 @@ TRACE_EVENT(drv_set_rts_threshold, | |||
491 | ) | 491 | ) |
492 | ); | 492 | ); |
493 | 493 | ||
494 | TRACE_EVENT(drv_set_coverage_class, | ||
495 | TP_PROTO(struct ieee80211_local *local, u8 value, int ret), | ||
496 | |||
497 | TP_ARGS(local, value, ret), | ||
498 | |||
499 | TP_STRUCT__entry( | ||
500 | LOCAL_ENTRY | ||
501 | __field(u8, value) | ||
502 | __field(int, ret) | ||
503 | ), | ||
504 | |||
505 | TP_fast_assign( | ||
506 | LOCAL_ASSIGN; | ||
507 | __entry->ret = ret; | ||
508 | __entry->value = value; | ||
509 | ), | ||
510 | |||
511 | TP_printk( | ||
512 | LOCAL_PR_FMT " value:%d ret:%d", | ||
513 | LOCAL_PR_ARG, __entry->value, __entry->ret | ||
514 | ) | ||
515 | ); | ||
516 | |||
494 | TRACE_EVENT(drv_sta_notify, | 517 | TRACE_EVENT(drv_sta_notify, |
495 | TP_PROTO(struct ieee80211_local *local, | 518 | TP_PROTO(struct ieee80211_local *local, |
496 | struct ieee80211_sub_if_data *sdata, | 519 | struct ieee80211_sub_if_data *sdata, |
diff --git a/net/mac80211/ieee80211_i.h b/net/mac80211/ieee80211_i.h index a27921ee6e63..c18f576f1848 100644 --- a/net/mac80211/ieee80211_i.h +++ b/net/mac80211/ieee80211_i.h | |||
@@ -58,6 +58,15 @@ struct ieee80211_local; | |||
58 | 58 | ||
59 | #define TU_TO_EXP_TIME(x) (jiffies + usecs_to_jiffies((x) * 1024)) | 59 | #define TU_TO_EXP_TIME(x) (jiffies + usecs_to_jiffies((x) * 1024)) |
60 | 60 | ||
61 | #define IEEE80211_DEFAULT_UAPSD_QUEUES \ | ||
62 | (IEEE80211_WMM_IE_STA_QOSINFO_AC_BK | \ | ||
63 | IEEE80211_WMM_IE_STA_QOSINFO_AC_BE | \ | ||
64 | IEEE80211_WMM_IE_STA_QOSINFO_AC_VI | \ | ||
65 | IEEE80211_WMM_IE_STA_QOSINFO_AC_VO) | ||
66 | |||
67 | #define IEEE80211_DEFAULT_MAX_SP_LEN \ | ||
68 | IEEE80211_WMM_IE_STA_QOSINFO_SP_ALL | ||
69 | |||
61 | struct ieee80211_fragment_entry { | 70 | struct ieee80211_fragment_entry { |
62 | unsigned long first_frag_time; | 71 | unsigned long first_frag_time; |
63 | unsigned int seq; | 72 | unsigned int seq; |
@@ -78,6 +87,7 @@ struct ieee80211_bss { | |||
78 | u8 dtim_period; | 87 | u8 dtim_period; |
79 | 88 | ||
80 | bool wmm_used; | 89 | bool wmm_used; |
90 | bool uapsd_supported; | ||
81 | 91 | ||
82 | unsigned long last_probe_resp; | 92 | unsigned long last_probe_resp; |
83 | 93 | ||
@@ -285,7 +295,7 @@ struct ieee80211_work { | |||
285 | u8 ssid[IEEE80211_MAX_SSID_LEN]; | 295 | u8 ssid[IEEE80211_MAX_SSID_LEN]; |
286 | u8 ssid_len; | 296 | u8 ssid_len; |
287 | u8 supp_rates_len; | 297 | u8 supp_rates_len; |
288 | bool wmm_used, use_11n; | 298 | bool wmm_used, use_11n, uapsd_used; |
289 | } assoc; | 299 | } assoc; |
290 | struct { | 300 | struct { |
291 | u32 duration; | 301 | u32 duration; |
@@ -306,6 +316,7 @@ enum ieee80211_sta_flags { | |||
306 | IEEE80211_STA_DISABLE_11N = BIT(4), | 316 | IEEE80211_STA_DISABLE_11N = BIT(4), |
307 | IEEE80211_STA_CSA_RECEIVED = BIT(5), | 317 | IEEE80211_STA_CSA_RECEIVED = BIT(5), |
308 | IEEE80211_STA_MFP_ENABLED = BIT(6), | 318 | IEEE80211_STA_MFP_ENABLED = BIT(6), |
319 | IEEE80211_STA_UAPSD_ENABLED = BIT(7), | ||
309 | }; | 320 | }; |
310 | 321 | ||
311 | struct ieee80211_if_managed { | 322 | struct ieee80211_if_managed { |
@@ -494,8 +505,8 @@ struct ieee80211_sub_if_data { | |||
494 | */ | 505 | */ |
495 | struct ieee80211_if_ap *bss; | 506 | struct ieee80211_if_ap *bss; |
496 | 507 | ||
497 | int force_unicast_rateidx; /* forced TX rateidx for unicast frames */ | 508 | /* bitmap of allowed (non-MCS) rate indexes for rate control */ |
498 | int max_ratectrl_rateidx; /* max TX rateidx for rate control */ | 509 | u32 rc_rateidx_mask[IEEE80211_NUM_BANDS]; |
499 | 510 | ||
500 | union { | 511 | union { |
501 | struct ieee80211_if_ap ap; | 512 | struct ieee80211_if_ap ap; |
@@ -797,6 +808,20 @@ struct ieee80211_local { | |||
797 | int wifi_wme_noack_test; | 808 | int wifi_wme_noack_test; |
798 | unsigned int wmm_acm; /* bit field of ACM bits (BIT(802.1D tag)) */ | 809 | unsigned int wmm_acm; /* bit field of ACM bits (BIT(802.1D tag)) */ |
799 | 810 | ||
811 | /* | ||
812 | * Bitmask of enabled u-apsd queues, | ||
813 | * IEEE80211_WMM_IE_STA_QOSINFO_AC_BE & co. Needs a new association | ||
814 | * to take effect. | ||
815 | */ | ||
816 | unsigned int uapsd_queues; | ||
817 | |||
818 | /* | ||
819 | * Maximum number of buffered frames AP can deliver during a | ||
820 | * service period, IEEE80211_WMM_IE_STA_QOSINFO_SP_ALL or similar. | ||
821 | * Needs a new association to take effect. | ||
822 | */ | ||
823 | unsigned int uapsd_max_sp_len; | ||
824 | |||
800 | bool pspolling; | 825 | bool pspolling; |
801 | bool offchannel_ps_enabled; | 826 | bool offchannel_ps_enabled; |
802 | /* | 827 | /* |
diff --git a/net/mac80211/iface.c b/net/mac80211/iface.c index 00a1f4ccdaf1..fe140bf033f9 100644 --- a/net/mac80211/iface.c +++ b/net/mac80211/iface.c | |||
@@ -15,12 +15,14 @@ | |||
15 | #include <linux/netdevice.h> | 15 | #include <linux/netdevice.h> |
16 | #include <linux/rtnetlink.h> | 16 | #include <linux/rtnetlink.h> |
17 | #include <net/mac80211.h> | 17 | #include <net/mac80211.h> |
18 | #include <net/ieee80211_radiotap.h> | ||
18 | #include "ieee80211_i.h" | 19 | #include "ieee80211_i.h" |
19 | #include "sta_info.h" | 20 | #include "sta_info.h" |
20 | #include "debugfs_netdev.h" | 21 | #include "debugfs_netdev.h" |
21 | #include "mesh.h" | 22 | #include "mesh.h" |
22 | #include "led.h" | 23 | #include "led.h" |
23 | #include "driver-ops.h" | 24 | #include "driver-ops.h" |
25 | #include "wme.h" | ||
24 | 26 | ||
25 | /** | 27 | /** |
26 | * DOC: Interface list locking | 28 | * DOC: Interface list locking |
@@ -63,15 +65,16 @@ static int ieee80211_change_mtu(struct net_device *dev, int new_mtu) | |||
63 | static int ieee80211_change_mac(struct net_device *dev, void *addr) | 65 | static int ieee80211_change_mac(struct net_device *dev, void *addr) |
64 | { | 66 | { |
65 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); | 67 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); |
68 | struct sockaddr *sa = addr; | ||
66 | int ret; | 69 | int ret; |
67 | 70 | ||
68 | if (ieee80211_sdata_running(sdata)) | 71 | if (ieee80211_sdata_running(sdata)) |
69 | return -EBUSY; | 72 | return -EBUSY; |
70 | 73 | ||
71 | ret = eth_mac_addr(dev, addr); | 74 | ret = eth_mac_addr(dev, sa); |
72 | 75 | ||
73 | if (ret == 0) | 76 | if (ret == 0) |
74 | memcpy(sdata->vif.addr, addr, ETH_ALEN); | 77 | memcpy(sdata->vif.addr, sa->sa_data, ETH_ALEN); |
75 | 78 | ||
76 | return ret; | 79 | return ret; |
77 | } | 80 | } |
@@ -326,7 +329,7 @@ static int ieee80211_open(struct net_device *dev) | |||
326 | if (sdata->vif.type == NL80211_IFTYPE_STATION) | 329 | if (sdata->vif.type == NL80211_IFTYPE_STATION) |
327 | ieee80211_queue_work(&local->hw, &sdata->u.mgd.work); | 330 | ieee80211_queue_work(&local->hw, &sdata->u.mgd.work); |
328 | 331 | ||
329 | netif_start_queue(dev); | 332 | netif_tx_start_all_queues(dev); |
330 | 333 | ||
331 | return 0; | 334 | return 0; |
332 | err_del_interface: | 335 | err_del_interface: |
@@ -354,7 +357,7 @@ static int ieee80211_stop(struct net_device *dev) | |||
354 | /* | 357 | /* |
355 | * Stop TX on this interface first. | 358 | * Stop TX on this interface first. |
356 | */ | 359 | */ |
357 | netif_stop_queue(dev); | 360 | netif_tx_stop_all_queues(dev); |
358 | 361 | ||
359 | /* | 362 | /* |
360 | * Purge work for this interface. | 363 | * Purge work for this interface. |
@@ -657,6 +660,12 @@ static void ieee80211_teardown_sdata(struct net_device *dev) | |||
657 | WARN_ON(flushed); | 660 | WARN_ON(flushed); |
658 | } | 661 | } |
659 | 662 | ||
663 | static u16 ieee80211_netdev_select_queue(struct net_device *dev, | ||
664 | struct sk_buff *skb) | ||
665 | { | ||
666 | return ieee80211_select_queue(IEEE80211_DEV_TO_SUB_IF(dev), skb); | ||
667 | } | ||
668 | |||
660 | static const struct net_device_ops ieee80211_dataif_ops = { | 669 | static const struct net_device_ops ieee80211_dataif_ops = { |
661 | .ndo_open = ieee80211_open, | 670 | .ndo_open = ieee80211_open, |
662 | .ndo_stop = ieee80211_stop, | 671 | .ndo_stop = ieee80211_stop, |
@@ -665,8 +674,34 @@ static const struct net_device_ops ieee80211_dataif_ops = { | |||
665 | .ndo_set_multicast_list = ieee80211_set_multicast_list, | 674 | .ndo_set_multicast_list = ieee80211_set_multicast_list, |
666 | .ndo_change_mtu = ieee80211_change_mtu, | 675 | .ndo_change_mtu = ieee80211_change_mtu, |
667 | .ndo_set_mac_address = ieee80211_change_mac, | 676 | .ndo_set_mac_address = ieee80211_change_mac, |
677 | .ndo_select_queue = ieee80211_netdev_select_queue, | ||
668 | }; | 678 | }; |
669 | 679 | ||
680 | static u16 ieee80211_monitor_select_queue(struct net_device *dev, | ||
681 | struct sk_buff *skb) | ||
682 | { | ||
683 | struct ieee80211_sub_if_data *sdata = IEEE80211_DEV_TO_SUB_IF(dev); | ||
684 | struct ieee80211_local *local = sdata->local; | ||
685 | struct ieee80211_hdr *hdr; | ||
686 | struct ieee80211_radiotap_header *rtap = (void *)skb->data; | ||
687 | |||
688 | if (local->hw.queues < 4) | ||
689 | return 0; | ||
690 | |||
691 | if (skb->len < 4 || | ||
692 | skb->len < rtap->it_len + 2 /* frame control */) | ||
693 | return 0; /* doesn't matter, frame will be dropped */ | ||
694 | |||
695 | hdr = (void *)((u8 *)skb->data + rtap->it_len); | ||
696 | |||
697 | if (!ieee80211_is_data(hdr->frame_control)) { | ||
698 | skb->priority = 7; | ||
699 | return ieee802_1d_to_ac[skb->priority]; | ||
700 | } | ||
701 | |||
702 | return ieee80211_downgrade_queue(local, skb); | ||
703 | } | ||
704 | |||
670 | static const struct net_device_ops ieee80211_monitorif_ops = { | 705 | static const struct net_device_ops ieee80211_monitorif_ops = { |
671 | .ndo_open = ieee80211_open, | 706 | .ndo_open = ieee80211_open, |
672 | .ndo_stop = ieee80211_stop, | 707 | .ndo_stop = ieee80211_stop, |
@@ -675,6 +710,7 @@ static const struct net_device_ops ieee80211_monitorif_ops = { | |||
675 | .ndo_set_multicast_list = ieee80211_set_multicast_list, | 710 | .ndo_set_multicast_list = ieee80211_set_multicast_list, |
676 | .ndo_change_mtu = ieee80211_change_mtu, | 711 | .ndo_change_mtu = ieee80211_change_mtu, |
677 | .ndo_set_mac_address = eth_mac_addr, | 712 | .ndo_set_mac_address = eth_mac_addr, |
713 | .ndo_select_queue = ieee80211_monitor_select_queue, | ||
678 | }; | 714 | }; |
679 | 715 | ||
680 | static void ieee80211_if_setup(struct net_device *dev) | 716 | static void ieee80211_if_setup(struct net_device *dev) |
@@ -781,8 +817,8 @@ int ieee80211_if_add(struct ieee80211_local *local, const char *name, | |||
781 | 817 | ||
782 | ASSERT_RTNL(); | 818 | ASSERT_RTNL(); |
783 | 819 | ||
784 | ndev = alloc_netdev(sizeof(*sdata) + local->hw.vif_data_size, | 820 | ndev = alloc_netdev_mq(sizeof(*sdata) + local->hw.vif_data_size, |
785 | name, ieee80211_if_setup); | 821 | name, ieee80211_if_setup, local->hw.queues); |
786 | if (!ndev) | 822 | if (!ndev) |
787 | return -ENOMEM; | 823 | return -ENOMEM; |
788 | dev_net_set(ndev, wiphy_net(local->hw.wiphy)); | 824 | dev_net_set(ndev, wiphy_net(local->hw.wiphy)); |
@@ -820,8 +856,12 @@ int ieee80211_if_add(struct ieee80211_local *local, const char *name, | |||
820 | 856 | ||
821 | INIT_LIST_HEAD(&sdata->key_list); | 857 | INIT_LIST_HEAD(&sdata->key_list); |
822 | 858 | ||
823 | sdata->force_unicast_rateidx = -1; | 859 | for (i = 0; i < IEEE80211_NUM_BANDS; i++) { |
824 | sdata->max_ratectrl_rateidx = -1; | 860 | struct ieee80211_supported_band *sband; |
861 | sband = local->hw.wiphy->bands[i]; | ||
862 | sdata->rc_rateidx_mask[i] = | ||
863 | sband ? (1 << sband->n_bitrates) - 1 : 0; | ||
864 | } | ||
825 | 865 | ||
826 | /* setup type-dependent data */ | 866 | /* setup type-dependent data */ |
827 | ieee80211_setup_sdata(sdata, type); | 867 | ieee80211_setup_sdata(sdata, type); |
diff --git a/net/mac80211/main.c b/net/mac80211/main.c index d0a14d953f08..ec8f767ba95b 100644 --- a/net/mac80211/main.c +++ b/net/mac80211/main.c | |||
@@ -17,7 +17,6 @@ | |||
17 | #include <linux/skbuff.h> | 17 | #include <linux/skbuff.h> |
18 | #include <linux/etherdevice.h> | 18 | #include <linux/etherdevice.h> |
19 | #include <linux/if_arp.h> | 19 | #include <linux/if_arp.h> |
20 | #include <linux/wireless.h> | ||
21 | #include <linux/rtnetlink.h> | 20 | #include <linux/rtnetlink.h> |
22 | #include <linux/bitmap.h> | 21 | #include <linux/bitmap.h> |
23 | #include <linux/pm_qos_params.h> | 22 | #include <linux/pm_qos_params.h> |
@@ -385,6 +384,8 @@ struct ieee80211_hw *ieee80211_alloc_hw(size_t priv_data_len, | |||
385 | local->hw.conf.long_frame_max_tx_count = wiphy->retry_long; | 384 | local->hw.conf.long_frame_max_tx_count = wiphy->retry_long; |
386 | local->hw.conf.short_frame_max_tx_count = wiphy->retry_short; | 385 | local->hw.conf.short_frame_max_tx_count = wiphy->retry_short; |
387 | local->user_power_level = -1; | 386 | local->user_power_level = -1; |
387 | local->uapsd_queues = IEEE80211_DEFAULT_UAPSD_QUEUES; | ||
388 | local->uapsd_max_sp_len = IEEE80211_DEFAULT_MAX_SP_LEN; | ||
388 | 389 | ||
389 | INIT_LIST_HEAD(&local->interfaces); | 390 | INIT_LIST_HEAD(&local->interfaces); |
390 | mutex_init(&local->iflist_mtx); | 391 | mutex_init(&local->iflist_mtx); |
@@ -492,6 +493,10 @@ int ieee80211_register_hw(struct ieee80211_hw *hw) | |||
492 | else if (local->hw.flags & IEEE80211_HW_SIGNAL_UNSPEC) | 493 | else if (local->hw.flags & IEEE80211_HW_SIGNAL_UNSPEC) |
493 | local->hw.wiphy->signal_type = CFG80211_SIGNAL_TYPE_UNSPEC; | 494 | local->hw.wiphy->signal_type = CFG80211_SIGNAL_TYPE_UNSPEC; |
494 | 495 | ||
496 | WARN((local->hw.flags & IEEE80211_HW_SUPPORTS_UAPSD) | ||
497 | && (local->hw.flags & IEEE80211_HW_PS_NULLFUNC_STACK), | ||
498 | "U-APSD not supported with HW_PS_NULLFUNC_STACK\n"); | ||
499 | |||
495 | /* | 500 | /* |
496 | * Calculate scan IE length -- we need this to alloc | 501 | * Calculate scan IE length -- we need this to alloc |
497 | * memory and to subtract from the driver limit. It | 502 | * memory and to subtract from the driver limit. It |
diff --git a/net/mac80211/mlme.c b/net/mac80211/mlme.c index 72920ee07885..a82564e73d91 100644 --- a/net/mac80211/mlme.c +++ b/net/mac80211/mlme.c | |||
@@ -249,30 +249,15 @@ static void ieee80211_send_deauth_disassoc(struct ieee80211_sub_if_data *sdata, | |||
249 | void ieee80211_send_pspoll(struct ieee80211_local *local, | 249 | void ieee80211_send_pspoll(struct ieee80211_local *local, |
250 | struct ieee80211_sub_if_data *sdata) | 250 | struct ieee80211_sub_if_data *sdata) |
251 | { | 251 | { |
252 | struct ieee80211_if_managed *ifmgd = &sdata->u.mgd; | ||
253 | struct ieee80211_pspoll *pspoll; | 252 | struct ieee80211_pspoll *pspoll; |
254 | struct sk_buff *skb; | 253 | struct sk_buff *skb; |
255 | u16 fc; | ||
256 | 254 | ||
257 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + sizeof(*pspoll)); | 255 | skb = ieee80211_pspoll_get(&local->hw, &sdata->vif); |
258 | if (!skb) { | 256 | if (!skb) |
259 | printk(KERN_DEBUG "%s: failed to allocate buffer for " | ||
260 | "pspoll frame\n", sdata->name); | ||
261 | return; | 257 | return; |
262 | } | ||
263 | skb_reserve(skb, local->hw.extra_tx_headroom); | ||
264 | |||
265 | pspoll = (struct ieee80211_pspoll *) skb_put(skb, sizeof(*pspoll)); | ||
266 | memset(pspoll, 0, sizeof(*pspoll)); | ||
267 | fc = IEEE80211_FTYPE_CTL | IEEE80211_STYPE_PSPOLL | IEEE80211_FCTL_PM; | ||
268 | pspoll->frame_control = cpu_to_le16(fc); | ||
269 | pspoll->aid = cpu_to_le16(ifmgd->aid); | ||
270 | |||
271 | /* aid in PS-Poll has its two MSBs each set to 1 */ | ||
272 | pspoll->aid |= cpu_to_le16(1 << 15 | 1 << 14); | ||
273 | 258 | ||
274 | memcpy(pspoll->bssid, ifmgd->bssid, ETH_ALEN); | 259 | pspoll = (struct ieee80211_pspoll *) skb->data; |
275 | memcpy(pspoll->ta, sdata->vif.addr, ETH_ALEN); | 260 | pspoll->frame_control |= cpu_to_le16(IEEE80211_FCTL_PM); |
276 | 261 | ||
277 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; | 262 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; |
278 | ieee80211_tx_skb(sdata, skb); | 263 | ieee80211_tx_skb(sdata, skb); |
@@ -283,30 +268,47 @@ void ieee80211_send_nullfunc(struct ieee80211_local *local, | |||
283 | int powersave) | 268 | int powersave) |
284 | { | 269 | { |
285 | struct sk_buff *skb; | 270 | struct sk_buff *skb; |
271 | struct ieee80211_hdr_3addr *nullfunc; | ||
272 | |||
273 | skb = ieee80211_nullfunc_get(&local->hw, &sdata->vif); | ||
274 | if (!skb) | ||
275 | return; | ||
276 | |||
277 | nullfunc = (struct ieee80211_hdr_3addr *) skb->data; | ||
278 | if (powersave) | ||
279 | nullfunc->frame_control |= cpu_to_le16(IEEE80211_FCTL_PM); | ||
280 | |||
281 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; | ||
282 | ieee80211_tx_skb(sdata, skb); | ||
283 | } | ||
284 | |||
285 | static void ieee80211_send_4addr_nullfunc(struct ieee80211_local *local, | ||
286 | struct ieee80211_sub_if_data *sdata) | ||
287 | { | ||
288 | struct sk_buff *skb; | ||
286 | struct ieee80211_hdr *nullfunc; | 289 | struct ieee80211_hdr *nullfunc; |
287 | __le16 fc; | 290 | __le16 fc; |
288 | 291 | ||
289 | if (WARN_ON(sdata->vif.type != NL80211_IFTYPE_STATION)) | 292 | if (WARN_ON(sdata->vif.type != NL80211_IFTYPE_STATION)) |
290 | return; | 293 | return; |
291 | 294 | ||
292 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + 24); | 295 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + 30); |
293 | if (!skb) { | 296 | if (!skb) { |
294 | printk(KERN_DEBUG "%s: failed to allocate buffer for nullfunc " | 297 | printk(KERN_DEBUG "%s: failed to allocate buffer for 4addr " |
295 | "frame\n", sdata->name); | 298 | "nullfunc frame\n", sdata->name); |
296 | return; | 299 | return; |
297 | } | 300 | } |
298 | skb_reserve(skb, local->hw.extra_tx_headroom); | 301 | skb_reserve(skb, local->hw.extra_tx_headroom); |
299 | 302 | ||
300 | nullfunc = (struct ieee80211_hdr *) skb_put(skb, 24); | 303 | nullfunc = (struct ieee80211_hdr *) skb_put(skb, 30); |
301 | memset(nullfunc, 0, 24); | 304 | memset(nullfunc, 0, 30); |
302 | fc = cpu_to_le16(IEEE80211_FTYPE_DATA | IEEE80211_STYPE_NULLFUNC | | 305 | fc = cpu_to_le16(IEEE80211_FTYPE_DATA | IEEE80211_STYPE_NULLFUNC | |
303 | IEEE80211_FCTL_TODS); | 306 | IEEE80211_FCTL_FROMDS | IEEE80211_FCTL_TODS); |
304 | if (powersave) | ||
305 | fc |= cpu_to_le16(IEEE80211_FCTL_PM); | ||
306 | nullfunc->frame_control = fc; | 307 | nullfunc->frame_control = fc; |
307 | memcpy(nullfunc->addr1, sdata->u.mgd.bssid, ETH_ALEN); | 308 | memcpy(nullfunc->addr1, sdata->u.mgd.bssid, ETH_ALEN); |
308 | memcpy(nullfunc->addr2, sdata->vif.addr, ETH_ALEN); | 309 | memcpy(nullfunc->addr2, sdata->vif.addr, ETH_ALEN); |
309 | memcpy(nullfunc->addr3, sdata->u.mgd.bssid, ETH_ALEN); | 310 | memcpy(nullfunc->addr3, sdata->u.mgd.bssid, ETH_ALEN); |
311 | memcpy(nullfunc->addr4, sdata->vif.addr, ETH_ALEN); | ||
310 | 312 | ||
311 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; | 313 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; |
312 | ieee80211_tx_skb(sdata, skb); | 314 | ieee80211_tx_skb(sdata, skb); |
@@ -567,7 +569,7 @@ static void ieee80211_sta_wmm_params(struct ieee80211_local *local, | |||
567 | struct ieee80211_tx_queue_params params; | 569 | struct ieee80211_tx_queue_params params; |
568 | size_t left; | 570 | size_t left; |
569 | int count; | 571 | int count; |
570 | u8 *pos; | 572 | u8 *pos, uapsd_queues = 0; |
571 | 573 | ||
572 | if (local->hw.queues < 4) | 574 | if (local->hw.queues < 4) |
573 | return; | 575 | return; |
@@ -577,6 +579,10 @@ static void ieee80211_sta_wmm_params(struct ieee80211_local *local, | |||
577 | 579 | ||
578 | if (wmm_param_len < 8 || wmm_param[5] /* version */ != 1) | 580 | if (wmm_param_len < 8 || wmm_param[5] /* version */ != 1) |
579 | return; | 581 | return; |
582 | |||
583 | if (ifmgd->flags & IEEE80211_STA_UAPSD_ENABLED) | ||
584 | uapsd_queues = local->uapsd_queues; | ||
585 | |||
580 | count = wmm_param[6] & 0x0f; | 586 | count = wmm_param[6] & 0x0f; |
581 | if (count == ifmgd->wmm_last_param_set) | 587 | if (count == ifmgd->wmm_last_param_set) |
582 | return; | 588 | return; |
@@ -591,6 +597,7 @@ static void ieee80211_sta_wmm_params(struct ieee80211_local *local, | |||
591 | for (; left >= 4; left -= 4, pos += 4) { | 597 | for (; left >= 4; left -= 4, pos += 4) { |
592 | int aci = (pos[0] >> 5) & 0x03; | 598 | int aci = (pos[0] >> 5) & 0x03; |
593 | int acm = (pos[0] >> 4) & 0x01; | 599 | int acm = (pos[0] >> 4) & 0x01; |
600 | bool uapsd = false; | ||
594 | int queue; | 601 | int queue; |
595 | 602 | ||
596 | switch (aci) { | 603 | switch (aci) { |
@@ -598,22 +605,30 @@ static void ieee80211_sta_wmm_params(struct ieee80211_local *local, | |||
598 | queue = 3; | 605 | queue = 3; |
599 | if (acm) | 606 | if (acm) |
600 | local->wmm_acm |= BIT(1) | BIT(2); /* BK/- */ | 607 | local->wmm_acm |= BIT(1) | BIT(2); /* BK/- */ |
608 | if (uapsd_queues & IEEE80211_WMM_IE_STA_QOSINFO_AC_BK) | ||
609 | uapsd = true; | ||
601 | break; | 610 | break; |
602 | case 2: /* AC_VI */ | 611 | case 2: /* AC_VI */ |
603 | queue = 1; | 612 | queue = 1; |
604 | if (acm) | 613 | if (acm) |
605 | local->wmm_acm |= BIT(4) | BIT(5); /* CL/VI */ | 614 | local->wmm_acm |= BIT(4) | BIT(5); /* CL/VI */ |
615 | if (uapsd_queues & IEEE80211_WMM_IE_STA_QOSINFO_AC_VI) | ||
616 | uapsd = true; | ||
606 | break; | 617 | break; |
607 | case 3: /* AC_VO */ | 618 | case 3: /* AC_VO */ |
608 | queue = 0; | 619 | queue = 0; |
609 | if (acm) | 620 | if (acm) |
610 | local->wmm_acm |= BIT(6) | BIT(7); /* VO/NC */ | 621 | local->wmm_acm |= BIT(6) | BIT(7); /* VO/NC */ |
622 | if (uapsd_queues & IEEE80211_WMM_IE_STA_QOSINFO_AC_VO) | ||
623 | uapsd = true; | ||
611 | break; | 624 | break; |
612 | case 0: /* AC_BE */ | 625 | case 0: /* AC_BE */ |
613 | default: | 626 | default: |
614 | queue = 2; | 627 | queue = 2; |
615 | if (acm) | 628 | if (acm) |
616 | local->wmm_acm |= BIT(0) | BIT(3); /* BE/EE */ | 629 | local->wmm_acm |= BIT(0) | BIT(3); /* BE/EE */ |
630 | if (uapsd_queues & IEEE80211_WMM_IE_STA_QOSINFO_AC_BE) | ||
631 | uapsd = true; | ||
617 | break; | 632 | break; |
618 | } | 633 | } |
619 | 634 | ||
@@ -621,11 +636,14 @@ static void ieee80211_sta_wmm_params(struct ieee80211_local *local, | |||
621 | params.cw_max = ecw2cw((pos[1] & 0xf0) >> 4); | 636 | params.cw_max = ecw2cw((pos[1] & 0xf0) >> 4); |
622 | params.cw_min = ecw2cw(pos[1] & 0x0f); | 637 | params.cw_min = ecw2cw(pos[1] & 0x0f); |
623 | params.txop = get_unaligned_le16(pos + 2); | 638 | params.txop = get_unaligned_le16(pos + 2); |
639 | params.uapsd = uapsd; | ||
640 | |||
624 | #ifdef CONFIG_MAC80211_VERBOSE_DEBUG | 641 | #ifdef CONFIG_MAC80211_VERBOSE_DEBUG |
625 | printk(KERN_DEBUG "%s: WMM queue=%d aci=%d acm=%d aifs=%d " | 642 | printk(KERN_DEBUG "%s: WMM queue=%d aci=%d acm=%d aifs=%d " |
626 | "cWmin=%d cWmax=%d txop=%d\n", | 643 | "cWmin=%d cWmax=%d txop=%d uapsd=%d\n", |
627 | wiphy_name(local->hw.wiphy), queue, aci, acm, | 644 | wiphy_name(local->hw.wiphy), queue, aci, acm, |
628 | params.aifs, params.cw_min, params.cw_max, params.txop); | 645 | params.aifs, params.cw_min, params.cw_max, params.txop, |
646 | params.uapsd); | ||
629 | #endif | 647 | #endif |
630 | if (drv_conf_tx(local, queue, ¶ms) && local->ops->conf_tx) | 648 | if (drv_conf_tx(local, queue, ¶ms) && local->ops->conf_tx) |
631 | printk(KERN_DEBUG "%s: failed to set TX queue " | 649 | printk(KERN_DEBUG "%s: failed to set TX queue " |
@@ -652,6 +670,8 @@ static u32 ieee80211_handle_bss_capability(struct ieee80211_sub_if_data *sdata, | |||
652 | } | 670 | } |
653 | 671 | ||
654 | use_short_slot = !!(capab & WLAN_CAPABILITY_SHORT_SLOT_TIME); | 672 | use_short_slot = !!(capab & WLAN_CAPABILITY_SHORT_SLOT_TIME); |
673 | if (sdata->local->hw.conf.channel->band == IEEE80211_BAND_5GHZ) | ||
674 | use_short_slot = true; | ||
655 | 675 | ||
656 | if (use_protection != bss_conf->use_cts_prot) { | 676 | if (use_protection != bss_conf->use_cts_prot) { |
657 | bss_conf->use_cts_prot = use_protection; | 677 | bss_conf->use_cts_prot = use_protection; |
@@ -723,7 +743,7 @@ static void ieee80211_set_associated(struct ieee80211_sub_if_data *sdata, | |||
723 | ieee80211_recalc_smps(local, sdata); | 743 | ieee80211_recalc_smps(local, sdata); |
724 | mutex_unlock(&local->iflist_mtx); | 744 | mutex_unlock(&local->iflist_mtx); |
725 | 745 | ||
726 | netif_start_queue(sdata->dev); | 746 | netif_tx_start_all_queues(sdata->dev); |
727 | netif_carrier_on(sdata->dev); | 747 | netif_carrier_on(sdata->dev); |
728 | } | 748 | } |
729 | 749 | ||
@@ -759,7 +779,7 @@ static void ieee80211_set_disassoc(struct ieee80211_sub_if_data *sdata) | |||
759 | * time -- we don't want the scan code to enable queues. | 779 | * time -- we don't want the scan code to enable queues. |
760 | */ | 780 | */ |
761 | 781 | ||
762 | netif_stop_queue(sdata->dev); | 782 | netif_tx_stop_all_queues(sdata->dev); |
763 | netif_carrier_off(sdata->dev); | 783 | netif_carrier_off(sdata->dev); |
764 | 784 | ||
765 | rcu_read_lock(); | 785 | rcu_read_lock(); |
@@ -1096,7 +1116,7 @@ static bool ieee80211_assoc_success(struct ieee80211_work *wk, | |||
1096 | if (err) { | 1116 | if (err) { |
1097 | printk(KERN_DEBUG "%s: failed to insert STA entry for" | 1117 | printk(KERN_DEBUG "%s: failed to insert STA entry for" |
1098 | " the AP (error %d)\n", sdata->name, err); | 1118 | " the AP (error %d)\n", sdata->name, err); |
1099 | return RX_MGMT_CFG80211_ASSOC_ERROR; | 1119 | return false; |
1100 | } | 1120 | } |
1101 | 1121 | ||
1102 | if (elems.wmm_param) | 1122 | if (elems.wmm_param) |
@@ -1120,6 +1140,13 @@ static bool ieee80211_assoc_success(struct ieee80211_work *wk, | |||
1120 | ieee80211_set_associated(sdata, cbss, changed); | 1140 | ieee80211_set_associated(sdata, cbss, changed); |
1121 | 1141 | ||
1122 | /* | 1142 | /* |
1143 | * If we're using 4-addr mode, let the AP know that we're | ||
1144 | * doing so, so that it can create the STA VLAN on its side | ||
1145 | */ | ||
1146 | if (ifmgd->use_4addr) | ||
1147 | ieee80211_send_4addr_nullfunc(local, sdata); | ||
1148 | |||
1149 | /* | ||
1123 | * Start timer to probe the connection to the AP now. | 1150 | * Start timer to probe the connection to the AP now. |
1124 | * Also start the timer that will detect beacon loss. | 1151 | * Also start the timer that will detect beacon loss. |
1125 | */ | 1152 | */ |
@@ -1774,7 +1801,7 @@ int ieee80211_mgd_auth(struct ieee80211_sub_if_data *sdata, | |||
1774 | if (!wk) | 1801 | if (!wk) |
1775 | return -ENOMEM; | 1802 | return -ENOMEM; |
1776 | 1803 | ||
1777 | memcpy(wk->filter_ta, req->bss->bssid, ETH_ALEN);; | 1804 | memcpy(wk->filter_ta, req->bss->bssid, ETH_ALEN); |
1778 | 1805 | ||
1779 | if (req->ie && req->ie_len) { | 1806 | if (req->ie && req->ie_len) { |
1780 | memcpy(wk->ie, req->ie, req->ie_len); | 1807 | memcpy(wk->ie, req->ie, req->ie_len); |
@@ -1897,6 +1924,15 @@ int ieee80211_mgd_assoc(struct ieee80211_sub_if_data *sdata, | |||
1897 | wk->assoc.ht_information_ie = | 1924 | wk->assoc.ht_information_ie = |
1898 | ieee80211_bss_get_ie(req->bss, WLAN_EID_HT_INFORMATION); | 1925 | ieee80211_bss_get_ie(req->bss, WLAN_EID_HT_INFORMATION); |
1899 | 1926 | ||
1927 | if (bss->wmm_used && bss->uapsd_supported && | ||
1928 | (sdata->local->hw.flags & IEEE80211_HW_SUPPORTS_UAPSD)) { | ||
1929 | wk->assoc.uapsd_used = true; | ||
1930 | ifmgd->flags |= IEEE80211_STA_UAPSD_ENABLED; | ||
1931 | } else { | ||
1932 | wk->assoc.uapsd_used = false; | ||
1933 | ifmgd->flags &= ~IEEE80211_STA_UAPSD_ENABLED; | ||
1934 | } | ||
1935 | |||
1900 | ssid = ieee80211_bss_get_ie(req->bss, WLAN_EID_SSID); | 1936 | ssid = ieee80211_bss_get_ie(req->bss, WLAN_EID_SSID); |
1901 | memcpy(wk->assoc.ssid, ssid + 2, ssid[1]); | 1937 | memcpy(wk->assoc.ssid, ssid + 2, ssid[1]); |
1902 | wk->assoc.ssid_len = ssid[1]; | 1938 | wk->assoc.ssid_len = ssid[1]; |
diff --git a/net/mac80211/offchannel.c b/net/mac80211/offchannel.c index a7bbfc40a648..c36b1911987a 100644 --- a/net/mac80211/offchannel.c +++ b/net/mac80211/offchannel.c | |||
@@ -113,7 +113,7 @@ void ieee80211_offchannel_stop_beaconing(struct ieee80211_local *local) | |||
113 | */ | 113 | */ |
114 | if (sdata->vif.type != NL80211_IFTYPE_STATION && | 114 | if (sdata->vif.type != NL80211_IFTYPE_STATION && |
115 | sdata->vif.type != NL80211_IFTYPE_MONITOR) | 115 | sdata->vif.type != NL80211_IFTYPE_MONITOR) |
116 | netif_stop_queue(sdata->dev); | 116 | netif_tx_stop_all_queues(sdata->dev); |
117 | } | 117 | } |
118 | mutex_unlock(&local->iflist_mtx); | 118 | mutex_unlock(&local->iflist_mtx); |
119 | } | 119 | } |
@@ -131,7 +131,7 @@ void ieee80211_offchannel_stop_station(struct ieee80211_local *local) | |||
131 | continue; | 131 | continue; |
132 | 132 | ||
133 | if (sdata->vif.type == NL80211_IFTYPE_STATION) { | 133 | if (sdata->vif.type == NL80211_IFTYPE_STATION) { |
134 | netif_stop_queue(sdata->dev); | 134 | netif_tx_stop_all_queues(sdata->dev); |
135 | if (sdata->u.mgd.associated) | 135 | if (sdata->u.mgd.associated) |
136 | ieee80211_offchannel_ps_enable(sdata); | 136 | ieee80211_offchannel_ps_enable(sdata); |
137 | } | 137 | } |
@@ -153,9 +153,11 @@ void ieee80211_offchannel_return(struct ieee80211_local *local, | |||
153 | if (sdata->vif.type == NL80211_IFTYPE_STATION) { | 153 | if (sdata->vif.type == NL80211_IFTYPE_STATION) { |
154 | if (sdata->u.mgd.associated) | 154 | if (sdata->u.mgd.associated) |
155 | ieee80211_offchannel_ps_disable(sdata); | 155 | ieee80211_offchannel_ps_disable(sdata); |
156 | netif_wake_queue(sdata->dev); | ||
157 | } | 156 | } |
158 | 157 | ||
158 | if (sdata->vif.type != NL80211_IFTYPE_MONITOR) | ||
159 | netif_tx_wake_all_queues(sdata->dev); | ||
160 | |||
159 | /* re-enable beaconing */ | 161 | /* re-enable beaconing */ |
160 | if (enable_beaconing && | 162 | if (enable_beaconing && |
161 | (sdata->vif.type == NL80211_IFTYPE_AP || | 163 | (sdata->vif.type == NL80211_IFTYPE_AP || |
diff --git a/net/mac80211/rate.c b/net/mac80211/rate.c index b9007f80cb92..c74b7c85403c 100644 --- a/net/mac80211/rate.c +++ b/net/mac80211/rate.c | |||
@@ -207,6 +207,27 @@ static bool rc_no_data_or_no_ack(struct ieee80211_tx_rate_control *txrc) | |||
207 | return ((info->flags & IEEE80211_TX_CTL_NO_ACK) || !ieee80211_is_data(fc)); | 207 | return ((info->flags & IEEE80211_TX_CTL_NO_ACK) || !ieee80211_is_data(fc)); |
208 | } | 208 | } |
209 | 209 | ||
210 | static void rc_send_low_broadcast(s8 *idx, u32 basic_rates, u8 max_rate_idx) | ||
211 | { | ||
212 | u8 i; | ||
213 | |||
214 | if (basic_rates == 0) | ||
215 | return; /* assume basic rates unknown and accept rate */ | ||
216 | if (*idx < 0) | ||
217 | return; | ||
218 | if (basic_rates & (1 << *idx)) | ||
219 | return; /* selected rate is a basic rate */ | ||
220 | |||
221 | for (i = *idx + 1; i <= max_rate_idx; i++) { | ||
222 | if (basic_rates & (1 << i)) { | ||
223 | *idx = i; | ||
224 | return; | ||
225 | } | ||
226 | } | ||
227 | |||
228 | /* could not find a basic rate; use original selection */ | ||
229 | } | ||
230 | |||
210 | bool rate_control_send_low(struct ieee80211_sta *sta, | 231 | bool rate_control_send_low(struct ieee80211_sta *sta, |
211 | void *priv_sta, | 232 | void *priv_sta, |
212 | struct ieee80211_tx_rate_control *txrc) | 233 | struct ieee80211_tx_rate_control *txrc) |
@@ -218,12 +239,48 @@ bool rate_control_send_low(struct ieee80211_sta *sta, | |||
218 | info->control.rates[0].count = | 239 | info->control.rates[0].count = |
219 | (info->flags & IEEE80211_TX_CTL_NO_ACK) ? | 240 | (info->flags & IEEE80211_TX_CTL_NO_ACK) ? |
220 | 1 : txrc->hw->max_rate_tries; | 241 | 1 : txrc->hw->max_rate_tries; |
242 | if (!sta && txrc->ap) | ||
243 | rc_send_low_broadcast(&info->control.rates[0].idx, | ||
244 | txrc->bss_conf->basic_rates, | ||
245 | txrc->sband->n_bitrates); | ||
221 | return true; | 246 | return true; |
222 | } | 247 | } |
223 | return false; | 248 | return false; |
224 | } | 249 | } |
225 | EXPORT_SYMBOL(rate_control_send_low); | 250 | EXPORT_SYMBOL(rate_control_send_low); |
226 | 251 | ||
252 | static void rate_idx_match_mask(struct ieee80211_tx_rate *rate, | ||
253 | int n_bitrates, u32 mask) | ||
254 | { | ||
255 | int j; | ||
256 | |||
257 | /* See whether the selected rate or anything below it is allowed. */ | ||
258 | for (j = rate->idx; j >= 0; j--) { | ||
259 | if (mask & (1 << j)) { | ||
260 | /* Okay, found a suitable rate. Use it. */ | ||
261 | rate->idx = j; | ||
262 | return; | ||
263 | } | ||
264 | } | ||
265 | |||
266 | /* Try to find a higher rate that would be allowed */ | ||
267 | for (j = rate->idx + 1; j < n_bitrates; j++) { | ||
268 | if (mask & (1 << j)) { | ||
269 | /* Okay, found a suitable rate. Use it. */ | ||
270 | rate->idx = j; | ||
271 | return; | ||
272 | } | ||
273 | } | ||
274 | |||
275 | /* | ||
276 | * Uh.. No suitable rate exists. This should not really happen with | ||
277 | * sane TX rate mask configurations. However, should someone manage to | ||
278 | * configure supported rates and TX rate mask in incompatible way, | ||
279 | * allow the frame to be transmitted with whatever the rate control | ||
280 | * selected. | ||
281 | */ | ||
282 | } | ||
283 | |||
227 | void rate_control_get_rate(struct ieee80211_sub_if_data *sdata, | 284 | void rate_control_get_rate(struct ieee80211_sub_if_data *sdata, |
228 | struct sta_info *sta, | 285 | struct sta_info *sta, |
229 | struct ieee80211_tx_rate_control *txrc) | 286 | struct ieee80211_tx_rate_control *txrc) |
@@ -233,6 +290,7 @@ void rate_control_get_rate(struct ieee80211_sub_if_data *sdata, | |||
233 | struct ieee80211_sta *ista = NULL; | 290 | struct ieee80211_sta *ista = NULL; |
234 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(txrc->skb); | 291 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(txrc->skb); |
235 | int i; | 292 | int i; |
293 | u32 mask; | ||
236 | 294 | ||
237 | if (sta) { | 295 | if (sta) { |
238 | ista = &sta->sta; | 296 | ista = &sta->sta; |
@@ -245,23 +303,31 @@ void rate_control_get_rate(struct ieee80211_sub_if_data *sdata, | |||
245 | info->control.rates[i].count = 1; | 303 | info->control.rates[i].count = 1; |
246 | } | 304 | } |
247 | 305 | ||
248 | if (sta && sdata->force_unicast_rateidx > -1) { | 306 | ref->ops->get_rate(ref->priv, ista, priv_sta, txrc); |
249 | info->control.rates[0].idx = sdata->force_unicast_rateidx; | ||
250 | } else { | ||
251 | ref->ops->get_rate(ref->priv, ista, priv_sta, txrc); | ||
252 | info->flags |= IEEE80211_TX_INTFL_RCALGO; | ||
253 | } | ||
254 | 307 | ||
255 | /* | 308 | /* |
256 | * try to enforce the maximum rate the user wanted | 309 | * Try to enforce the rateidx mask the user wanted. skip this if the |
310 | * default mask (allow all rates) is used to save some processing for | ||
311 | * the common case. | ||
257 | */ | 312 | */ |
258 | if (sdata->max_ratectrl_rateidx > -1) | 313 | mask = sdata->rc_rateidx_mask[info->band]; |
314 | if (mask != (1 << txrc->sband->n_bitrates) - 1) { | ||
315 | if (sta) { | ||
316 | /* Filter out rates that the STA does not support */ | ||
317 | mask &= sta->sta.supp_rates[info->band]; | ||
318 | } | ||
319 | /* | ||
320 | * Make sure the rate index selected for each TX rate is | ||
321 | * included in the configured mask and change the rate indexes | ||
322 | * if needed. | ||
323 | */ | ||
259 | for (i = 0; i < IEEE80211_TX_MAX_RATES; i++) { | 324 | for (i = 0; i < IEEE80211_TX_MAX_RATES; i++) { |
325 | /* Rate masking supports only legacy rates for now */ | ||
260 | if (info->control.rates[i].flags & IEEE80211_TX_RC_MCS) | 326 | if (info->control.rates[i].flags & IEEE80211_TX_RC_MCS) |
261 | continue; | 327 | continue; |
262 | info->control.rates[i].idx = | 328 | rate_idx_match_mask(&info->control.rates[i], |
263 | min_t(s8, info->control.rates[i].idx, | 329 | txrc->sband->n_bitrates, mask); |
264 | sdata->max_ratectrl_rateidx); | 330 | } |
265 | } | 331 | } |
266 | 332 | ||
267 | BUG_ON(info->control.rates[0].idx < 0); | 333 | BUG_ON(info->control.rates[0].idx < 0); |
diff --git a/net/mac80211/rate.h b/net/mac80211/rate.h index cb9bd1f65e27..669dddd40521 100644 --- a/net/mac80211/rate.h +++ b/net/mac80211/rate.h | |||
@@ -44,10 +44,7 @@ static inline void rate_control_tx_status(struct ieee80211_local *local, | |||
44 | struct rate_control_ref *ref = local->rate_ctrl; | 44 | struct rate_control_ref *ref = local->rate_ctrl; |
45 | struct ieee80211_sta *ista = &sta->sta; | 45 | struct ieee80211_sta *ista = &sta->sta; |
46 | void *priv_sta = sta->rate_ctrl_priv; | 46 | void *priv_sta = sta->rate_ctrl_priv; |
47 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | 47 | ref->ops->tx_status(ref->priv, sband, ista, priv_sta, skb); |
48 | |||
49 | if (likely(info->flags & IEEE80211_TX_INTFL_RCALGO)) | ||
50 | ref->ops->tx_status(ref->priv, sband, ista, priv_sta, skb); | ||
51 | } | 48 | } |
52 | 49 | ||
53 | 50 | ||
diff --git a/net/mac80211/rx.c b/net/mac80211/rx.c index bfcf09eb64b4..efa6d3689c5e 100644 --- a/net/mac80211/rx.c +++ b/net/mac80211/rx.c | |||
@@ -1111,6 +1111,18 @@ ieee80211_rx_h_sta_process(struct ieee80211_rx_data *rx) | |||
1111 | if (ieee80211_is_nullfunc(hdr->frame_control) || | 1111 | if (ieee80211_is_nullfunc(hdr->frame_control) || |
1112 | ieee80211_is_qos_nullfunc(hdr->frame_control)) { | 1112 | ieee80211_is_qos_nullfunc(hdr->frame_control)) { |
1113 | I802_DEBUG_INC(rx->local->rx_handlers_drop_nullfunc); | 1113 | I802_DEBUG_INC(rx->local->rx_handlers_drop_nullfunc); |
1114 | |||
1115 | /* | ||
1116 | * If we receive a 4-addr nullfunc frame from a STA | ||
1117 | * that was not moved to a 4-addr STA vlan yet, drop | ||
1118 | * the frame to the monitor interface, to make sure | ||
1119 | * that hostapd sees it | ||
1120 | */ | ||
1121 | if (ieee80211_has_a4(hdr->frame_control) && | ||
1122 | (rx->sdata->vif.type == NL80211_IFTYPE_AP || | ||
1123 | (rx->sdata->vif.type == NL80211_IFTYPE_AP_VLAN && | ||
1124 | !rx->sdata->u.vlan.sta))) | ||
1125 | return RX_DROP_MONITOR; | ||
1114 | /* | 1126 | /* |
1115 | * Update counter and free packet here to avoid | 1127 | * Update counter and free packet here to avoid |
1116 | * counting this as a dropped packed. | 1128 | * counting this as a dropped packed. |
@@ -1665,7 +1677,9 @@ ieee80211_rx_h_mesh_fwding(struct ieee80211_rx_data *rx) | |||
1665 | memset(info, 0, sizeof(*info)); | 1677 | memset(info, 0, sizeof(*info)); |
1666 | info->flags |= IEEE80211_TX_INTFL_NEED_TXPROCESSING; | 1678 | info->flags |= IEEE80211_TX_INTFL_NEED_TXPROCESSING; |
1667 | info->control.vif = &rx->sdata->vif; | 1679 | info->control.vif = &rx->sdata->vif; |
1668 | ieee80211_select_queue(local, fwd_skb); | 1680 | skb_set_queue_mapping(skb, |
1681 | ieee80211_select_queue(rx->sdata, fwd_skb)); | ||
1682 | ieee80211_set_qos_hdr(local, skb); | ||
1669 | if (is_multicast_ether_addr(fwd_hdr->addr1)) | 1683 | if (is_multicast_ether_addr(fwd_hdr->addr1)) |
1670 | IEEE80211_IFSTA_MESH_CTR_INC(&sdata->u.mesh, | 1684 | IEEE80211_IFSTA_MESH_CTR_INC(&sdata->u.mesh, |
1671 | fwded_mcast); | 1685 | fwded_mcast); |
diff --git a/net/mac80211/scan.c b/net/mac80211/scan.c index 365f40975511..9afe2f9885dc 100644 --- a/net/mac80211/scan.c +++ b/net/mac80211/scan.c | |||
@@ -12,7 +12,6 @@ | |||
12 | * published by the Free Software Foundation. | 12 | * published by the Free Software Foundation. |
13 | */ | 13 | */ |
14 | 14 | ||
15 | #include <linux/wireless.h> | ||
16 | #include <linux/if_arp.h> | 15 | #include <linux/if_arp.h> |
17 | #include <linux/rtnetlink.h> | 16 | #include <linux/rtnetlink.h> |
18 | #include <net/mac80211.h> | 17 | #include <net/mac80211.h> |
@@ -55,6 +54,23 @@ void ieee80211_rx_bss_put(struct ieee80211_local *local, | |||
55 | cfg80211_put_bss(container_of((void *)bss, struct cfg80211_bss, priv)); | 54 | cfg80211_put_bss(container_of((void *)bss, struct cfg80211_bss, priv)); |
56 | } | 55 | } |
57 | 56 | ||
57 | static bool is_uapsd_supported(struct ieee802_11_elems *elems) | ||
58 | { | ||
59 | u8 qos_info; | ||
60 | |||
61 | if (elems->wmm_info && elems->wmm_info_len == 7 | ||
62 | && elems->wmm_info[5] == 1) | ||
63 | qos_info = elems->wmm_info[6]; | ||
64 | else if (elems->wmm_param && elems->wmm_param_len == 24 | ||
65 | && elems->wmm_param[5] == 1) | ||
66 | qos_info = elems->wmm_param[6]; | ||
67 | else | ||
68 | /* no valid wmm information or parameter element found */ | ||
69 | return false; | ||
70 | |||
71 | return qos_info & IEEE80211_WMM_IE_AP_QOSINFO_UAPSD; | ||
72 | } | ||
73 | |||
58 | struct ieee80211_bss * | 74 | struct ieee80211_bss * |
59 | ieee80211_bss_info_update(struct ieee80211_local *local, | 75 | ieee80211_bss_info_update(struct ieee80211_local *local, |
60 | struct ieee80211_rx_status *rx_status, | 76 | struct ieee80211_rx_status *rx_status, |
@@ -118,6 +134,7 @@ ieee80211_bss_info_update(struct ieee80211_local *local, | |||
118 | } | 134 | } |
119 | 135 | ||
120 | bss->wmm_used = elems->wmm_param || elems->wmm_info; | 136 | bss->wmm_used = elems->wmm_param || elems->wmm_info; |
137 | bss->uapsd_supported = is_uapsd_supported(elems); | ||
121 | 138 | ||
122 | if (!beacon) | 139 | if (!beacon) |
123 | bss->last_probe_resp = jiffies; | 140 | bss->last_probe_resp = jiffies; |
@@ -285,6 +302,7 @@ void ieee80211_scan_completed(struct ieee80211_hw *hw, bool aborted) | |||
285 | ieee80211_mlme_notify_scan_completed(local); | 302 | ieee80211_mlme_notify_scan_completed(local); |
286 | ieee80211_ibss_notify_scan_completed(local); | 303 | ieee80211_ibss_notify_scan_completed(local); |
287 | ieee80211_mesh_notify_scan_completed(local); | 304 | ieee80211_mesh_notify_scan_completed(local); |
305 | ieee80211_queue_work(&local->hw, &local->work_work); | ||
288 | } | 306 | } |
289 | EXPORT_SYMBOL(ieee80211_scan_completed); | 307 | EXPORT_SYMBOL(ieee80211_scan_completed); |
290 | 308 | ||
diff --git a/net/mac80211/sta_info.c b/net/mac80211/sta_info.c index 47da552ce8a6..f735826f055c 100644 --- a/net/mac80211/sta_info.c +++ b/net/mac80211/sta_info.c | |||
@@ -119,6 +119,27 @@ struct sta_info *sta_info_get(struct ieee80211_sub_if_data *sdata, | |||
119 | return sta; | 119 | return sta; |
120 | } | 120 | } |
121 | 121 | ||
122 | /* | ||
123 | * Get sta info either from the specified interface | ||
124 | * or from one of its vlans | ||
125 | */ | ||
126 | struct sta_info *sta_info_get_bss(struct ieee80211_sub_if_data *sdata, | ||
127 | const u8 *addr) | ||
128 | { | ||
129 | struct ieee80211_local *local = sdata->local; | ||
130 | struct sta_info *sta; | ||
131 | |||
132 | sta = rcu_dereference(local->sta_hash[STA_HASH(addr)]); | ||
133 | while (sta) { | ||
134 | if ((sta->sdata == sdata || | ||
135 | sta->sdata->bss == sdata->bss) && | ||
136 | memcmp(sta->sta.addr, addr, ETH_ALEN) == 0) | ||
137 | break; | ||
138 | sta = rcu_dereference(sta->hnext); | ||
139 | } | ||
140 | return sta; | ||
141 | } | ||
142 | |||
122 | struct sta_info *sta_info_get_by_idx(struct ieee80211_sub_if_data *sdata, | 143 | struct sta_info *sta_info_get_by_idx(struct ieee80211_sub_if_data *sdata, |
123 | int idx) | 144 | int idx) |
124 | { | 145 | { |
diff --git a/net/mac80211/sta_info.h b/net/mac80211/sta_info.h index c8208236e896..6f79bba5706e 100644 --- a/net/mac80211/sta_info.h +++ b/net/mac80211/sta_info.h | |||
@@ -408,6 +408,9 @@ static inline u32 get_sta_flags(struct sta_info *sta) | |||
408 | struct sta_info *sta_info_get(struct ieee80211_sub_if_data *sdata, | 408 | struct sta_info *sta_info_get(struct ieee80211_sub_if_data *sdata, |
409 | const u8 *addr); | 409 | const u8 *addr); |
410 | 410 | ||
411 | struct sta_info *sta_info_get_bss(struct ieee80211_sub_if_data *sdata, | ||
412 | const u8 *addr); | ||
413 | |||
411 | static inline | 414 | static inline |
412 | void for_each_sta_info_type_check(struct ieee80211_local *local, | 415 | void for_each_sta_info_type_check(struct ieee80211_local *local, |
413 | const u8 *addr, | 416 | const u8 *addr, |
diff --git a/net/mac80211/tx.c b/net/mac80211/tx.c index 7bba49d2b6ca..daf81048c1f7 100644 --- a/net/mac80211/tx.c +++ b/net/mac80211/tx.c | |||
@@ -180,6 +180,71 @@ static int inline is_ieee80211_device(struct ieee80211_local *local, | |||
180 | } | 180 | } |
181 | 181 | ||
182 | /* tx handlers */ | 182 | /* tx handlers */ |
183 | static ieee80211_tx_result debug_noinline | ||
184 | ieee80211_tx_h_dynamic_ps(struct ieee80211_tx_data *tx) | ||
185 | { | ||
186 | struct ieee80211_local *local = tx->local; | ||
187 | struct ieee80211_if_managed *ifmgd; | ||
188 | |||
189 | /* driver doesn't support power save */ | ||
190 | if (!(local->hw.flags & IEEE80211_HW_SUPPORTS_PS)) | ||
191 | return TX_CONTINUE; | ||
192 | |||
193 | /* hardware does dynamic power save */ | ||
194 | if (local->hw.flags & IEEE80211_HW_SUPPORTS_DYNAMIC_PS) | ||
195 | return TX_CONTINUE; | ||
196 | |||
197 | /* dynamic power save disabled */ | ||
198 | if (local->hw.conf.dynamic_ps_timeout <= 0) | ||
199 | return TX_CONTINUE; | ||
200 | |||
201 | /* we are scanning, don't enable power save */ | ||
202 | if (local->scanning) | ||
203 | return TX_CONTINUE; | ||
204 | |||
205 | if (!local->ps_sdata) | ||
206 | return TX_CONTINUE; | ||
207 | |||
208 | /* No point if we're going to suspend */ | ||
209 | if (local->quiescing) | ||
210 | return TX_CONTINUE; | ||
211 | |||
212 | /* dynamic ps is supported only in managed mode */ | ||
213 | if (tx->sdata->vif.type != NL80211_IFTYPE_STATION) | ||
214 | return TX_CONTINUE; | ||
215 | |||
216 | ifmgd = &tx->sdata->u.mgd; | ||
217 | |||
218 | /* | ||
219 | * Don't wakeup from power save if u-apsd is enabled, voip ac has | ||
220 | * u-apsd enabled and the frame is in voip class. This effectively | ||
221 | * means that even if all access categories have u-apsd enabled, in | ||
222 | * practise u-apsd is only used with the voip ac. This is a | ||
223 | * workaround for the case when received voip class packets do not | ||
224 | * have correct qos tag for some reason, due the network or the | ||
225 | * peer application. | ||
226 | * | ||
227 | * Note: local->uapsd_queues access is racy here. If the value is | ||
228 | * changed via debugfs, user needs to reassociate manually to have | ||
229 | * everything in sync. | ||
230 | */ | ||
231 | if ((ifmgd->flags & IEEE80211_STA_UAPSD_ENABLED) | ||
232 | && (local->uapsd_queues & IEEE80211_WMM_IE_STA_QOSINFO_AC_VO) | ||
233 | && skb_get_queue_mapping(tx->skb) == 0) | ||
234 | return TX_CONTINUE; | ||
235 | |||
236 | if (local->hw.conf.flags & IEEE80211_CONF_PS) { | ||
237 | ieee80211_stop_queues_by_reason(&local->hw, | ||
238 | IEEE80211_QUEUE_STOP_REASON_PS); | ||
239 | ieee80211_queue_work(&local->hw, | ||
240 | &local->dynamic_ps_disable_work); | ||
241 | } | ||
242 | |||
243 | mod_timer(&local->dynamic_ps_timer, jiffies + | ||
244 | msecs_to_jiffies(local->hw.conf.dynamic_ps_timeout)); | ||
245 | |||
246 | return TX_CONTINUE; | ||
247 | } | ||
183 | 248 | ||
184 | static ieee80211_tx_result debug_noinline | 249 | static ieee80211_tx_result debug_noinline |
185 | ieee80211_tx_h_check_assoc(struct ieee80211_tx_data *tx) | 250 | ieee80211_tx_h_check_assoc(struct ieee80211_tx_data *tx) |
@@ -519,7 +584,12 @@ ieee80211_tx_h_rate_ctrl(struct ieee80211_tx_data *tx) | |||
519 | txrc.bss_conf = &tx->sdata->vif.bss_conf; | 584 | txrc.bss_conf = &tx->sdata->vif.bss_conf; |
520 | txrc.skb = tx->skb; | 585 | txrc.skb = tx->skb; |
521 | txrc.reported_rate.idx = -1; | 586 | txrc.reported_rate.idx = -1; |
522 | txrc.max_rate_idx = tx->sdata->max_ratectrl_rateidx; | 587 | txrc.rate_idx_mask = tx->sdata->rc_rateidx_mask[tx->channel->band]; |
588 | if (txrc.rate_idx_mask == (1 << sband->n_bitrates) - 1) | ||
589 | txrc.max_rate_idx = -1; | ||
590 | else | ||
591 | txrc.max_rate_idx = fls(txrc.rate_idx_mask) - 1; | ||
592 | txrc.ap = tx->sdata->vif.type == NL80211_IFTYPE_AP; | ||
523 | 593 | ||
524 | /* set up RTS protection if desired */ | 594 | /* set up RTS protection if desired */ |
525 | if (len > tx->local->hw.wiphy->rts_threshold) { | 595 | if (len > tx->local->hw.wiphy->rts_threshold) { |
@@ -1051,8 +1121,11 @@ ieee80211_tx_prepare(struct ieee80211_sub_if_data *sdata, | |||
1051 | 1121 | ||
1052 | hdr = (struct ieee80211_hdr *) skb->data; | 1122 | hdr = (struct ieee80211_hdr *) skb->data; |
1053 | 1123 | ||
1054 | if (sdata->vif.type == NL80211_IFTYPE_AP_VLAN) | 1124 | if (sdata->vif.type == NL80211_IFTYPE_AP_VLAN) { |
1055 | tx->sta = rcu_dereference(sdata->u.vlan.sta); | 1125 | tx->sta = rcu_dereference(sdata->u.vlan.sta); |
1126 | if (!tx->sta && sdata->dev->ieee80211_ptr->use_4addr) | ||
1127 | return TX_DROP; | ||
1128 | } | ||
1056 | if (!tx->sta) | 1129 | if (!tx->sta) |
1057 | tx->sta = sta_info_get(sdata, hdr->addr1); | 1130 | tx->sta = sta_info_get(sdata, hdr->addr1); |
1058 | 1131 | ||
@@ -1215,6 +1288,7 @@ static int invoke_tx_handlers(struct ieee80211_tx_data *tx) | |||
1215 | goto txh_done; \ | 1288 | goto txh_done; \ |
1216 | } while (0) | 1289 | } while (0) |
1217 | 1290 | ||
1291 | CALL_TXH(ieee80211_tx_h_dynamic_ps); | ||
1218 | CALL_TXH(ieee80211_tx_h_check_assoc); | 1292 | CALL_TXH(ieee80211_tx_h_check_assoc); |
1219 | CALL_TXH(ieee80211_tx_h_ps_buf); | 1293 | CALL_TXH(ieee80211_tx_h_ps_buf); |
1220 | CALL_TXH(ieee80211_tx_h_select_key); | 1294 | CALL_TXH(ieee80211_tx_h_select_key); |
@@ -1397,34 +1471,6 @@ static int ieee80211_skb_resize(struct ieee80211_local *local, | |||
1397 | return 0; | 1471 | return 0; |
1398 | } | 1472 | } |
1399 | 1473 | ||
1400 | static bool need_dynamic_ps(struct ieee80211_local *local) | ||
1401 | { | ||
1402 | /* driver doesn't support power save */ | ||
1403 | if (!(local->hw.flags & IEEE80211_HW_SUPPORTS_PS)) | ||
1404 | return false; | ||
1405 | |||
1406 | /* hardware does dynamic power save */ | ||
1407 | if (local->hw.flags & IEEE80211_HW_SUPPORTS_DYNAMIC_PS) | ||
1408 | return false; | ||
1409 | |||
1410 | /* dynamic power save disabled */ | ||
1411 | if (local->hw.conf.dynamic_ps_timeout <= 0) | ||
1412 | return false; | ||
1413 | |||
1414 | /* we are scanning, don't enable power save */ | ||
1415 | if (local->scanning) | ||
1416 | return false; | ||
1417 | |||
1418 | if (!local->ps_sdata) | ||
1419 | return false; | ||
1420 | |||
1421 | /* No point if we're going to suspend */ | ||
1422 | if (local->quiescing) | ||
1423 | return false; | ||
1424 | |||
1425 | return true; | ||
1426 | } | ||
1427 | |||
1428 | static void ieee80211_xmit(struct ieee80211_sub_if_data *sdata, | 1474 | static void ieee80211_xmit(struct ieee80211_sub_if_data *sdata, |
1429 | struct sk_buff *skb) | 1475 | struct sk_buff *skb) |
1430 | { | 1476 | { |
@@ -1435,18 +1481,6 @@ static void ieee80211_xmit(struct ieee80211_sub_if_data *sdata, | |||
1435 | int headroom; | 1481 | int headroom; |
1436 | bool may_encrypt; | 1482 | bool may_encrypt; |
1437 | 1483 | ||
1438 | if (need_dynamic_ps(local)) { | ||
1439 | if (local->hw.conf.flags & IEEE80211_CONF_PS) { | ||
1440 | ieee80211_stop_queues_by_reason(&local->hw, | ||
1441 | IEEE80211_QUEUE_STOP_REASON_PS); | ||
1442 | ieee80211_queue_work(&local->hw, | ||
1443 | &local->dynamic_ps_disable_work); | ||
1444 | } | ||
1445 | |||
1446 | mod_timer(&local->dynamic_ps_timer, jiffies + | ||
1447 | msecs_to_jiffies(local->hw.conf.dynamic_ps_timeout)); | ||
1448 | } | ||
1449 | |||
1450 | rcu_read_lock(); | 1484 | rcu_read_lock(); |
1451 | 1485 | ||
1452 | if (unlikely(sdata->vif.type == NL80211_IFTYPE_MONITOR)) { | 1486 | if (unlikely(sdata->vif.type == NL80211_IFTYPE_MONITOR)) { |
@@ -1511,7 +1545,7 @@ static void ieee80211_xmit(struct ieee80211_sub_if_data *sdata, | |||
1511 | return; | 1545 | return; |
1512 | } | 1546 | } |
1513 | 1547 | ||
1514 | ieee80211_select_queue(local, skb); | 1548 | ieee80211_set_qos_hdr(local, skb); |
1515 | ieee80211_tx(sdata, skb, false); | 1549 | ieee80211_tx(sdata, skb, false); |
1516 | rcu_read_unlock(); | 1550 | rcu_read_unlock(); |
1517 | } | 1551 | } |
@@ -2060,6 +2094,7 @@ struct sk_buff *ieee80211_beacon_get_tim(struct ieee80211_hw *hw, | |||
2060 | struct beacon_data *beacon; | 2094 | struct beacon_data *beacon; |
2061 | struct ieee80211_supported_band *sband; | 2095 | struct ieee80211_supported_band *sband; |
2062 | enum ieee80211_band band = local->hw.conf.channel->band; | 2096 | enum ieee80211_band band = local->hw.conf.channel->band; |
2097 | struct ieee80211_tx_rate_control txrc; | ||
2063 | 2098 | ||
2064 | sband = local->hw.wiphy->bands[band]; | 2099 | sband = local->hw.wiphy->bands[band]; |
2065 | 2100 | ||
@@ -2167,21 +2202,25 @@ struct sk_buff *ieee80211_beacon_get_tim(struct ieee80211_hw *hw, | |||
2167 | info = IEEE80211_SKB_CB(skb); | 2202 | info = IEEE80211_SKB_CB(skb); |
2168 | 2203 | ||
2169 | info->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; | 2204 | info->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; |
2205 | info->flags |= IEEE80211_TX_CTL_NO_ACK; | ||
2170 | info->band = band; | 2206 | info->band = band; |
2171 | /* | 2207 | |
2172 | * XXX: For now, always use the lowest rate | 2208 | memset(&txrc, 0, sizeof(txrc)); |
2173 | */ | 2209 | txrc.hw = hw; |
2174 | info->control.rates[0].idx = 0; | 2210 | txrc.sband = sband; |
2175 | info->control.rates[0].count = 1; | 2211 | txrc.bss_conf = &sdata->vif.bss_conf; |
2176 | info->control.rates[1].idx = -1; | 2212 | txrc.skb = skb; |
2177 | info->control.rates[2].idx = -1; | 2213 | txrc.reported_rate.idx = -1; |
2178 | info->control.rates[3].idx = -1; | 2214 | txrc.rate_idx_mask = sdata->rc_rateidx_mask[band]; |
2179 | info->control.rates[4].idx = -1; | 2215 | if (txrc.rate_idx_mask == (1 << sband->n_bitrates) - 1) |
2180 | BUILD_BUG_ON(IEEE80211_TX_MAX_RATES != 5); | 2216 | txrc.max_rate_idx = -1; |
2217 | else | ||
2218 | txrc.max_rate_idx = fls(txrc.rate_idx_mask) - 1; | ||
2219 | txrc.ap = true; | ||
2220 | rate_control_get_rate(sdata, NULL, &txrc); | ||
2181 | 2221 | ||
2182 | info->control.vif = vif; | 2222 | info->control.vif = vif; |
2183 | 2223 | ||
2184 | info->flags |= IEEE80211_TX_CTL_NO_ACK; | ||
2185 | info->flags |= IEEE80211_TX_CTL_CLEAR_PS_FILT; | 2224 | info->flags |= IEEE80211_TX_CTL_CLEAR_PS_FILT; |
2186 | info->flags |= IEEE80211_TX_CTL_ASSIGN_SEQ; | 2225 | info->flags |= IEEE80211_TX_CTL_ASSIGN_SEQ; |
2187 | out: | 2226 | out: |
@@ -2190,6 +2229,134 @@ struct sk_buff *ieee80211_beacon_get_tim(struct ieee80211_hw *hw, | |||
2190 | } | 2229 | } |
2191 | EXPORT_SYMBOL(ieee80211_beacon_get_tim); | 2230 | EXPORT_SYMBOL(ieee80211_beacon_get_tim); |
2192 | 2231 | ||
2232 | struct sk_buff *ieee80211_pspoll_get(struct ieee80211_hw *hw, | ||
2233 | struct ieee80211_vif *vif) | ||
2234 | { | ||
2235 | struct ieee80211_sub_if_data *sdata; | ||
2236 | struct ieee80211_if_managed *ifmgd; | ||
2237 | struct ieee80211_pspoll *pspoll; | ||
2238 | struct ieee80211_local *local; | ||
2239 | struct sk_buff *skb; | ||
2240 | |||
2241 | if (WARN_ON(vif->type != NL80211_IFTYPE_STATION)) | ||
2242 | return NULL; | ||
2243 | |||
2244 | sdata = vif_to_sdata(vif); | ||
2245 | ifmgd = &sdata->u.mgd; | ||
2246 | local = sdata->local; | ||
2247 | |||
2248 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + sizeof(*pspoll)); | ||
2249 | if (!skb) { | ||
2250 | printk(KERN_DEBUG "%s: failed to allocate buffer for " | ||
2251 | "pspoll template\n", sdata->name); | ||
2252 | return NULL; | ||
2253 | } | ||
2254 | skb_reserve(skb, local->hw.extra_tx_headroom); | ||
2255 | |||
2256 | pspoll = (struct ieee80211_pspoll *) skb_put(skb, sizeof(*pspoll)); | ||
2257 | memset(pspoll, 0, sizeof(*pspoll)); | ||
2258 | pspoll->frame_control = cpu_to_le16(IEEE80211_FTYPE_CTL | | ||
2259 | IEEE80211_STYPE_PSPOLL); | ||
2260 | pspoll->aid = cpu_to_le16(ifmgd->aid); | ||
2261 | |||
2262 | /* aid in PS-Poll has its two MSBs each set to 1 */ | ||
2263 | pspoll->aid |= cpu_to_le16(1 << 15 | 1 << 14); | ||
2264 | |||
2265 | memcpy(pspoll->bssid, ifmgd->bssid, ETH_ALEN); | ||
2266 | memcpy(pspoll->ta, vif->addr, ETH_ALEN); | ||
2267 | |||
2268 | return skb; | ||
2269 | } | ||
2270 | EXPORT_SYMBOL(ieee80211_pspoll_get); | ||
2271 | |||
2272 | struct sk_buff *ieee80211_nullfunc_get(struct ieee80211_hw *hw, | ||
2273 | struct ieee80211_vif *vif) | ||
2274 | { | ||
2275 | struct ieee80211_hdr_3addr *nullfunc; | ||
2276 | struct ieee80211_sub_if_data *sdata; | ||
2277 | struct ieee80211_if_managed *ifmgd; | ||
2278 | struct ieee80211_local *local; | ||
2279 | struct sk_buff *skb; | ||
2280 | |||
2281 | if (WARN_ON(vif->type != NL80211_IFTYPE_STATION)) | ||
2282 | return NULL; | ||
2283 | |||
2284 | sdata = vif_to_sdata(vif); | ||
2285 | ifmgd = &sdata->u.mgd; | ||
2286 | local = sdata->local; | ||
2287 | |||
2288 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + sizeof(*nullfunc)); | ||
2289 | if (!skb) { | ||
2290 | printk(KERN_DEBUG "%s: failed to allocate buffer for nullfunc " | ||
2291 | "template\n", sdata->name); | ||
2292 | return NULL; | ||
2293 | } | ||
2294 | skb_reserve(skb, local->hw.extra_tx_headroom); | ||
2295 | |||
2296 | nullfunc = (struct ieee80211_hdr_3addr *) skb_put(skb, | ||
2297 | sizeof(*nullfunc)); | ||
2298 | memset(nullfunc, 0, sizeof(*nullfunc)); | ||
2299 | nullfunc->frame_control = cpu_to_le16(IEEE80211_FTYPE_DATA | | ||
2300 | IEEE80211_STYPE_NULLFUNC | | ||
2301 | IEEE80211_FCTL_TODS); | ||
2302 | memcpy(nullfunc->addr1, ifmgd->bssid, ETH_ALEN); | ||
2303 | memcpy(nullfunc->addr2, vif->addr, ETH_ALEN); | ||
2304 | memcpy(nullfunc->addr3, ifmgd->bssid, ETH_ALEN); | ||
2305 | |||
2306 | return skb; | ||
2307 | } | ||
2308 | EXPORT_SYMBOL(ieee80211_nullfunc_get); | ||
2309 | |||
2310 | struct sk_buff *ieee80211_probereq_get(struct ieee80211_hw *hw, | ||
2311 | struct ieee80211_vif *vif, | ||
2312 | const u8 *ssid, size_t ssid_len, | ||
2313 | const u8 *ie, size_t ie_len) | ||
2314 | { | ||
2315 | struct ieee80211_sub_if_data *sdata; | ||
2316 | struct ieee80211_local *local; | ||
2317 | struct ieee80211_hdr_3addr *hdr; | ||
2318 | struct sk_buff *skb; | ||
2319 | size_t ie_ssid_len; | ||
2320 | u8 *pos; | ||
2321 | |||
2322 | sdata = vif_to_sdata(vif); | ||
2323 | local = sdata->local; | ||
2324 | ie_ssid_len = 2 + ssid_len; | ||
2325 | |||
2326 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + sizeof(*hdr) + | ||
2327 | ie_ssid_len + ie_len); | ||
2328 | if (!skb) { | ||
2329 | printk(KERN_DEBUG "%s: failed to allocate buffer for probe " | ||
2330 | "request template\n", sdata->name); | ||
2331 | return NULL; | ||
2332 | } | ||
2333 | |||
2334 | skb_reserve(skb, local->hw.extra_tx_headroom); | ||
2335 | |||
2336 | hdr = (struct ieee80211_hdr_3addr *) skb_put(skb, sizeof(*hdr)); | ||
2337 | memset(hdr, 0, sizeof(*hdr)); | ||
2338 | hdr->frame_control = cpu_to_le16(IEEE80211_FTYPE_MGMT | | ||
2339 | IEEE80211_STYPE_PROBE_REQ); | ||
2340 | memset(hdr->addr1, 0xff, ETH_ALEN); | ||
2341 | memcpy(hdr->addr2, vif->addr, ETH_ALEN); | ||
2342 | memset(hdr->addr3, 0xff, ETH_ALEN); | ||
2343 | |||
2344 | pos = skb_put(skb, ie_ssid_len); | ||
2345 | *pos++ = WLAN_EID_SSID; | ||
2346 | *pos++ = ssid_len; | ||
2347 | if (ssid) | ||
2348 | memcpy(pos, ssid, ssid_len); | ||
2349 | pos += ssid_len; | ||
2350 | |||
2351 | if (ie) { | ||
2352 | pos = skb_put(skb, ie_len); | ||
2353 | memcpy(pos, ie, ie_len); | ||
2354 | } | ||
2355 | |||
2356 | return skb; | ||
2357 | } | ||
2358 | EXPORT_SYMBOL(ieee80211_probereq_get); | ||
2359 | |||
2193 | void ieee80211_rts_get(struct ieee80211_hw *hw, struct ieee80211_vif *vif, | 2360 | void ieee80211_rts_get(struct ieee80211_hw *hw, struct ieee80211_vif *vif, |
2194 | const void *frame, size_t frame_len, | 2361 | const void *frame, size_t frame_len, |
2195 | const struct ieee80211_tx_info *frame_txctl, | 2362 | const struct ieee80211_tx_info *frame_txctl, |
@@ -2289,6 +2456,9 @@ void ieee80211_tx_skb(struct ieee80211_sub_if_data *sdata, struct sk_buff *skb) | |||
2289 | skb_set_network_header(skb, 0); | 2456 | skb_set_network_header(skb, 0); |
2290 | skb_set_transport_header(skb, 0); | 2457 | skb_set_transport_header(skb, 0); |
2291 | 2458 | ||
2459 | /* send all internal mgmt frames on VO */ | ||
2460 | skb_set_queue_mapping(skb, 0); | ||
2461 | |||
2292 | /* | 2462 | /* |
2293 | * The other path calling ieee80211_xmit is from the tasklet, | 2463 | * The other path calling ieee80211_xmit is from the tasklet, |
2294 | * and while we can handle concurrent transmissions locking | 2464 | * and while we can handle concurrent transmissions locking |
diff --git a/net/mac80211/util.c b/net/mac80211/util.c index 7e38858a9280..ca170b417da6 100644 --- a/net/mac80211/util.c +++ b/net/mac80211/util.c | |||
@@ -18,7 +18,6 @@ | |||
18 | #include <linux/skbuff.h> | 18 | #include <linux/skbuff.h> |
19 | #include <linux/etherdevice.h> | 19 | #include <linux/etherdevice.h> |
20 | #include <linux/if_arp.h> | 20 | #include <linux/if_arp.h> |
21 | #include <linux/wireless.h> | ||
22 | #include <linux/bitmap.h> | 21 | #include <linux/bitmap.h> |
23 | #include <linux/crc32.h> | 22 | #include <linux/crc32.h> |
24 | #include <net/net_namespace.h> | 23 | #include <net/net_namespace.h> |
@@ -269,6 +268,7 @@ static void __ieee80211_wake_queue(struct ieee80211_hw *hw, int queue, | |||
269 | enum queue_stop_reason reason) | 268 | enum queue_stop_reason reason) |
270 | { | 269 | { |
271 | struct ieee80211_local *local = hw_to_local(hw); | 270 | struct ieee80211_local *local = hw_to_local(hw); |
271 | struct ieee80211_sub_if_data *sdata; | ||
272 | 272 | ||
273 | if (WARN_ON(queue >= hw->queues)) | 273 | if (WARN_ON(queue >= hw->queues)) |
274 | return; | 274 | return; |
@@ -281,6 +281,11 @@ static void __ieee80211_wake_queue(struct ieee80211_hw *hw, int queue, | |||
281 | 281 | ||
282 | if (!skb_queue_empty(&local->pending[queue])) | 282 | if (!skb_queue_empty(&local->pending[queue])) |
283 | tasklet_schedule(&local->tx_pending_tasklet); | 283 | tasklet_schedule(&local->tx_pending_tasklet); |
284 | |||
285 | rcu_read_lock(); | ||
286 | list_for_each_entry_rcu(sdata, &local->interfaces, list) | ||
287 | netif_tx_wake_queue(netdev_get_tx_queue(sdata->dev, queue)); | ||
288 | rcu_read_unlock(); | ||
284 | } | 289 | } |
285 | 290 | ||
286 | void ieee80211_wake_queue_by_reason(struct ieee80211_hw *hw, int queue, | 291 | void ieee80211_wake_queue_by_reason(struct ieee80211_hw *hw, int queue, |
@@ -305,11 +310,17 @@ static void __ieee80211_stop_queue(struct ieee80211_hw *hw, int queue, | |||
305 | enum queue_stop_reason reason) | 310 | enum queue_stop_reason reason) |
306 | { | 311 | { |
307 | struct ieee80211_local *local = hw_to_local(hw); | 312 | struct ieee80211_local *local = hw_to_local(hw); |
313 | struct ieee80211_sub_if_data *sdata; | ||
308 | 314 | ||
309 | if (WARN_ON(queue >= hw->queues)) | 315 | if (WARN_ON(queue >= hw->queues)) |
310 | return; | 316 | return; |
311 | 317 | ||
312 | __set_bit(reason, &local->queue_stop_reasons[queue]); | 318 | __set_bit(reason, &local->queue_stop_reasons[queue]); |
319 | |||
320 | rcu_read_lock(); | ||
321 | list_for_each_entry_rcu(sdata, &local->interfaces, list) | ||
322 | netif_tx_stop_queue(netdev_get_tx_queue(sdata->dev, queue)); | ||
323 | rcu_read_unlock(); | ||
313 | } | 324 | } |
314 | 325 | ||
315 | void ieee80211_stop_queue_by_reason(struct ieee80211_hw *hw, int queue, | 326 | void ieee80211_stop_queue_by_reason(struct ieee80211_hw *hw, int queue, |
@@ -781,6 +792,8 @@ void ieee80211_set_wmm_default(struct ieee80211_sub_if_data *sdata) | |||
781 | break; | 792 | break; |
782 | } | 793 | } |
783 | 794 | ||
795 | qparam.uapsd = false; | ||
796 | |||
784 | drv_conf_tx(local, queue, &qparam); | 797 | drv_conf_tx(local, queue, &qparam); |
785 | } | 798 | } |
786 | } | 799 | } |
@@ -989,40 +1002,33 @@ void ieee80211_send_probe_req(struct ieee80211_sub_if_data *sdata, u8 *dst, | |||
989 | struct ieee80211_local *local = sdata->local; | 1002 | struct ieee80211_local *local = sdata->local; |
990 | struct sk_buff *skb; | 1003 | struct sk_buff *skb; |
991 | struct ieee80211_mgmt *mgmt; | 1004 | struct ieee80211_mgmt *mgmt; |
992 | u8 *pos; | 1005 | size_t buf_len; |
993 | 1006 | u8 *buf; | |
994 | skb = dev_alloc_skb(local->hw.extra_tx_headroom + sizeof(*mgmt) + 200 + | 1007 | |
995 | ie_len); | 1008 | /* FIXME: come up with a proper value */ |
996 | if (!skb) { | 1009 | buf = kmalloc(200 + ie_len, GFP_KERNEL); |
997 | printk(KERN_DEBUG "%s: failed to allocate buffer for probe " | 1010 | if (!buf) { |
998 | "request\n", sdata->name); | 1011 | printk(KERN_DEBUG "%s: failed to allocate temporary IE " |
1012 | "buffer\n", sdata->name); | ||
999 | return; | 1013 | return; |
1000 | } | 1014 | } |
1001 | skb_reserve(skb, local->hw.extra_tx_headroom); | ||
1002 | 1015 | ||
1003 | mgmt = (struct ieee80211_mgmt *) skb_put(skb, 24); | 1016 | buf_len = ieee80211_build_preq_ies(local, buf, ie, ie_len, |
1004 | memset(mgmt, 0, 24); | 1017 | local->hw.conf.channel->band); |
1005 | mgmt->frame_control = cpu_to_le16(IEEE80211_FTYPE_MGMT | | 1018 | |
1006 | IEEE80211_STYPE_PROBE_REQ); | 1019 | skb = ieee80211_probereq_get(&local->hw, &sdata->vif, |
1007 | memcpy(mgmt->sa, sdata->vif.addr, ETH_ALEN); | 1020 | ssid, ssid_len, |
1021 | buf, buf_len); | ||
1022 | |||
1008 | if (dst) { | 1023 | if (dst) { |
1024 | mgmt = (struct ieee80211_mgmt *) skb->data; | ||
1009 | memcpy(mgmt->da, dst, ETH_ALEN); | 1025 | memcpy(mgmt->da, dst, ETH_ALEN); |
1010 | memcpy(mgmt->bssid, dst, ETH_ALEN); | 1026 | memcpy(mgmt->bssid, dst, ETH_ALEN); |
1011 | } else { | ||
1012 | memset(mgmt->da, 0xff, ETH_ALEN); | ||
1013 | memset(mgmt->bssid, 0xff, ETH_ALEN); | ||
1014 | } | 1027 | } |
1015 | pos = skb_put(skb, 2 + ssid_len); | ||
1016 | *pos++ = WLAN_EID_SSID; | ||
1017 | *pos++ = ssid_len; | ||
1018 | memcpy(pos, ssid, ssid_len); | ||
1019 | pos += ssid_len; | ||
1020 | |||
1021 | skb_put(skb, ieee80211_build_preq_ies(local, pos, ie, ie_len, | ||
1022 | local->hw.conf.channel->band)); | ||
1023 | 1028 | ||
1024 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; | 1029 | IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_INTFL_DONT_ENCRYPT; |
1025 | ieee80211_tx_skb(sdata, skb); | 1030 | ieee80211_tx_skb(sdata, skb); |
1031 | kfree(buf); | ||
1026 | } | 1032 | } |
1027 | 1033 | ||
1028 | u32 ieee80211_sta_get_rates(struct ieee80211_local *local, | 1034 | u32 ieee80211_sta_get_rates(struct ieee80211_local *local, |
@@ -1066,9 +1072,9 @@ void ieee80211_stop_device(struct ieee80211_local *local) | |||
1066 | ieee80211_led_radio(local, false); | 1072 | ieee80211_led_radio(local, false); |
1067 | 1073 | ||
1068 | cancel_work_sync(&local->reconfig_filter); | 1074 | cancel_work_sync(&local->reconfig_filter); |
1069 | drv_stop(local); | ||
1070 | 1075 | ||
1071 | flush_workqueue(local->workqueue); | 1076 | flush_workqueue(local->workqueue); |
1077 | drv_stop(local); | ||
1072 | } | 1078 | } |
1073 | 1079 | ||
1074 | int ieee80211_reconfig(struct ieee80211_local *local) | 1080 | int ieee80211_reconfig(struct ieee80211_local *local) |
@@ -1094,7 +1100,7 @@ int ieee80211_reconfig(struct ieee80211_local *local) | |||
1094 | if (res) { | 1100 | if (res) { |
1095 | WARN(local->suspended, "Harware became unavailable " | 1101 | WARN(local->suspended, "Harware became unavailable " |
1096 | "upon resume. This is could be a software issue" | 1102 | "upon resume. This is could be a software issue" |
1097 | "prior to suspend or a harware issue\n"); | 1103 | "prior to suspend or a hardware issue\n"); |
1098 | return res; | 1104 | return res; |
1099 | } | 1105 | } |
1100 | 1106 | ||
diff --git a/net/mac80211/wme.c b/net/mac80211/wme.c index b19b7696f3a2..34e6d02da779 100644 --- a/net/mac80211/wme.c +++ b/net/mac80211/wme.c | |||
@@ -44,22 +44,69 @@ static int wme_downgrade_ac(struct sk_buff *skb) | |||
44 | } | 44 | } |
45 | 45 | ||
46 | 46 | ||
47 | /* Indicate which queue to use. */ | 47 | /* Indicate which queue to use. */ |
48 | static u16 classify80211(struct ieee80211_local *local, struct sk_buff *skb) | 48 | u16 ieee80211_select_queue(struct ieee80211_sub_if_data *sdata, |
49 | struct sk_buff *skb) | ||
49 | { | 50 | { |
50 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) skb->data; | 51 | struct ieee80211_local *local = sdata->local; |
52 | struct sta_info *sta = NULL; | ||
53 | u32 sta_flags = 0; | ||
54 | const u8 *ra = NULL; | ||
55 | bool qos = false; | ||
51 | 56 | ||
52 | if (!ieee80211_is_data(hdr->frame_control)) { | 57 | if (local->hw.queues < 4 || skb->len < 6) { |
53 | /* management frames go on AC_VO queue, but are sent | 58 | skb->priority = 0; /* required for correct WPA/11i MIC */ |
54 | * without QoS control fields */ | 59 | return min_t(u16, local->hw.queues - 1, |
55 | return 0; | 60 | ieee802_1d_to_ac[skb->priority]); |
61 | } | ||
62 | |||
63 | rcu_read_lock(); | ||
64 | switch (sdata->vif.type) { | ||
65 | case NL80211_IFTYPE_AP_VLAN: | ||
66 | rcu_read_lock(); | ||
67 | sta = rcu_dereference(sdata->u.vlan.sta); | ||
68 | if (sta) | ||
69 | sta_flags = get_sta_flags(sta); | ||
70 | rcu_read_unlock(); | ||
71 | if (sta) | ||
72 | break; | ||
73 | case NL80211_IFTYPE_AP: | ||
74 | ra = skb->data; | ||
75 | break; | ||
76 | case NL80211_IFTYPE_WDS: | ||
77 | ra = sdata->u.wds.remote_addr; | ||
78 | break; | ||
79 | #ifdef CONFIG_MAC80211_MESH | ||
80 | case NL80211_IFTYPE_MESH_POINT: | ||
81 | /* | ||
82 | * XXX: This is clearly broken ... but already was before, | ||
83 | * because ieee80211_fill_mesh_addresses() would clear A1 | ||
84 | * except for multicast addresses. | ||
85 | */ | ||
86 | break; | ||
87 | #endif | ||
88 | case NL80211_IFTYPE_STATION: | ||
89 | ra = sdata->u.mgd.bssid; | ||
90 | break; | ||
91 | case NL80211_IFTYPE_ADHOC: | ||
92 | ra = skb->data; | ||
93 | break; | ||
94 | default: | ||
95 | break; | ||
56 | } | 96 | } |
57 | 97 | ||
58 | if (0 /* injected */) { | 98 | if (!sta && ra && !is_multicast_ether_addr(ra)) { |
59 | /* use AC from radiotap */ | 99 | sta = sta_info_get(sdata, ra); |
100 | if (sta) | ||
101 | sta_flags = get_sta_flags(sta); | ||
60 | } | 102 | } |
61 | 103 | ||
62 | if (!ieee80211_is_data_qos(hdr->frame_control)) { | 104 | if (sta_flags & WLAN_STA_WME) |
105 | qos = true; | ||
106 | |||
107 | rcu_read_unlock(); | ||
108 | |||
109 | if (!qos) { | ||
63 | skb->priority = 0; /* required for correct WPA/11i MIC */ | 110 | skb->priority = 0; /* required for correct WPA/11i MIC */ |
64 | return ieee802_1d_to_ac[skb->priority]; | 111 | return ieee802_1d_to_ac[skb->priority]; |
65 | } | 112 | } |
@@ -68,6 +115,12 @@ static u16 classify80211(struct ieee80211_local *local, struct sk_buff *skb) | |||
68 | * data frame has */ | 115 | * data frame has */ |
69 | skb->priority = cfg80211_classify8021d(skb); | 116 | skb->priority = cfg80211_classify8021d(skb); |
70 | 117 | ||
118 | return ieee80211_downgrade_queue(local, skb); | ||
119 | } | ||
120 | |||
121 | u16 ieee80211_downgrade_queue(struct ieee80211_local *local, | ||
122 | struct sk_buff *skb) | ||
123 | { | ||
71 | /* in case we are a client verify acm is not set for this ac */ | 124 | /* in case we are a client verify acm is not set for this ac */ |
72 | while (unlikely(local->wmm_acm & BIT(skb->priority))) { | 125 | while (unlikely(local->wmm_acm & BIT(skb->priority))) { |
73 | if (wme_downgrade_ac(skb)) { | 126 | if (wme_downgrade_ac(skb)) { |
@@ -85,24 +138,17 @@ static u16 classify80211(struct ieee80211_local *local, struct sk_buff *skb) | |||
85 | return ieee802_1d_to_ac[skb->priority]; | 138 | return ieee802_1d_to_ac[skb->priority]; |
86 | } | 139 | } |
87 | 140 | ||
88 | void ieee80211_select_queue(struct ieee80211_local *local, struct sk_buff *skb) | 141 | void ieee80211_set_qos_hdr(struct ieee80211_local *local, struct sk_buff *skb) |
89 | { | 142 | { |
90 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) skb->data; | 143 | struct ieee80211_hdr *hdr = (void *)skb->data; |
91 | u16 queue; | 144 | |
92 | u8 tid; | 145 | /* Fill in the QoS header if there is one. */ |
93 | |||
94 | queue = classify80211(local, skb); | ||
95 | if (unlikely(queue >= local->hw.queues)) | ||
96 | queue = local->hw.queues - 1; | ||
97 | |||
98 | /* | ||
99 | * Now we know the 1d priority, fill in the QoS header if | ||
100 | * there is one (and we haven't done this before). | ||
101 | */ | ||
102 | if (ieee80211_is_data_qos(hdr->frame_control)) { | 146 | if (ieee80211_is_data_qos(hdr->frame_control)) { |
103 | u8 *p = ieee80211_get_qos_ctl(hdr); | 147 | u8 *p = ieee80211_get_qos_ctl(hdr); |
104 | u8 ack_policy = 0; | 148 | u8 ack_policy = 0, tid; |
149 | |||
105 | tid = skb->priority & IEEE80211_QOS_CTL_TAG1D_MASK; | 150 | tid = skb->priority & IEEE80211_QOS_CTL_TAG1D_MASK; |
151 | |||
106 | if (unlikely(local->wifi_wme_noack_test)) | 152 | if (unlikely(local->wifi_wme_noack_test)) |
107 | ack_policy |= QOS_CONTROL_ACK_POLICY_NOACK << | 153 | ack_policy |= QOS_CONTROL_ACK_POLICY_NOACK << |
108 | QOS_CONTROL_ACK_POLICY_SHIFT; | 154 | QOS_CONTROL_ACK_POLICY_SHIFT; |
@@ -110,6 +156,4 @@ void ieee80211_select_queue(struct ieee80211_local *local, struct sk_buff *skb) | |||
110 | *p++ = ack_policy | tid; | 156 | *p++ = ack_policy | tid; |
111 | *p = 0; | 157 | *p = 0; |
112 | } | 158 | } |
113 | |||
114 | skb_set_queue_mapping(skb, queue); | ||
115 | } | 159 | } |
diff --git a/net/mac80211/wme.h b/net/mac80211/wme.h index d4fd87ca5118..6053b1c9feee 100644 --- a/net/mac80211/wme.h +++ b/net/mac80211/wme.h | |||
@@ -20,7 +20,11 @@ | |||
20 | 20 | ||
21 | extern const int ieee802_1d_to_ac[8]; | 21 | extern const int ieee802_1d_to_ac[8]; |
22 | 22 | ||
23 | void ieee80211_select_queue(struct ieee80211_local *local, | 23 | u16 ieee80211_select_queue(struct ieee80211_sub_if_data *sdata, |
24 | struct sk_buff *skb); | 24 | struct sk_buff *skb); |
25 | void ieee80211_set_qos_hdr(struct ieee80211_local *local, struct sk_buff *skb); | ||
26 | u16 ieee80211_downgrade_queue(struct ieee80211_local *local, | ||
27 | struct sk_buff *skb); | ||
28 | |||
25 | 29 | ||
26 | #endif /* _WME_H */ | 30 | #endif /* _WME_H */ |
diff --git a/net/mac80211/work.c b/net/mac80211/work.c index ea89ed70734d..81bd5d592bb4 100644 --- a/net/mac80211/work.c +++ b/net/mac80211/work.c | |||
@@ -202,7 +202,7 @@ static void ieee80211_send_assoc(struct ieee80211_sub_if_data *sdata, | |||
202 | struct ieee80211_local *local = sdata->local; | 202 | struct ieee80211_local *local = sdata->local; |
203 | struct sk_buff *skb; | 203 | struct sk_buff *skb; |
204 | struct ieee80211_mgmt *mgmt; | 204 | struct ieee80211_mgmt *mgmt; |
205 | u8 *pos; | 205 | u8 *pos, qos_info; |
206 | const u8 *ies; | 206 | const u8 *ies; |
207 | size_t offset = 0, noffset; | 207 | size_t offset = 0, noffset; |
208 | int i, len, count, rates_len, supp_rates_len; | 208 | int i, len, count, rates_len, supp_rates_len; |
@@ -375,6 +375,14 @@ static void ieee80211_send_assoc(struct ieee80211_sub_if_data *sdata, | |||
375 | } | 375 | } |
376 | 376 | ||
377 | if (wk->assoc.wmm_used && local->hw.queues >= 4) { | 377 | if (wk->assoc.wmm_used && local->hw.queues >= 4) { |
378 | if (wk->assoc.uapsd_used) { | ||
379 | qos_info = local->uapsd_queues; | ||
380 | qos_info |= (local->uapsd_max_sp_len << | ||
381 | IEEE80211_WMM_IE_STA_QOSINFO_SP_SHIFT); | ||
382 | } else { | ||
383 | qos_info = 0; | ||
384 | } | ||
385 | |||
378 | pos = skb_put(skb, 9); | 386 | pos = skb_put(skb, 9); |
379 | *pos++ = WLAN_EID_VENDOR_SPECIFIC; | 387 | *pos++ = WLAN_EID_VENDOR_SPECIFIC; |
380 | *pos++ = 7; /* len */ | 388 | *pos++ = 7; /* len */ |
@@ -384,7 +392,7 @@ static void ieee80211_send_assoc(struct ieee80211_sub_if_data *sdata, | |||
384 | *pos++ = 2; /* WME */ | 392 | *pos++ = 2; /* WME */ |
385 | *pos++ = 0; /* WME info */ | 393 | *pos++ = 0; /* WME info */ |
386 | *pos++ = 1; /* WME ver */ | 394 | *pos++ = 1; /* WME ver */ |
387 | *pos++ = 0; | 395 | *pos++ = qos_info; |
388 | } | 396 | } |
389 | 397 | ||
390 | /* add any remaining custom (i.e. vendor specific here) IEs */ | 398 | /* add any remaining custom (i.e. vendor specific here) IEs */ |
@@ -531,9 +539,9 @@ ieee80211_remain_on_channel_timeout(struct ieee80211_work *wk) | |||
531 | wk->remain.started = true; | 539 | wk->remain.started = true; |
532 | wk->timeout = jiffies + msecs_to_jiffies(wk->remain.duration); | 540 | wk->timeout = jiffies + msecs_to_jiffies(wk->remain.duration); |
533 | 541 | ||
534 | cfg80211_ready_on_channel(wk->sdata->dev, (u64)wk, wk->chan, | 542 | cfg80211_ready_on_channel(wk->sdata->dev, (unsigned long) wk, |
535 | wk->chan_type, wk->remain.duration, | 543 | wk->chan, wk->chan_type, |
536 | GFP_KERNEL); | 544 | wk->remain.duration, GFP_KERNEL); |
537 | 545 | ||
538 | return WORK_ACT_NONE; | 546 | return WORK_ACT_NONE; |
539 | } | 547 | } |
@@ -818,6 +826,7 @@ static void ieee80211_work_work(struct work_struct *work) | |||
818 | wk->chan == local->tmp_channel && | 826 | wk->chan == local->tmp_channel && |
819 | wk->chan_type == local->tmp_channel_type) { | 827 | wk->chan_type == local->tmp_channel_type) { |
820 | wk->started = true; | 828 | wk->started = true; |
829 | wk->timeout = jiffies; | ||
821 | } | 830 | } |
822 | 831 | ||
823 | if (!wk->started && !local->tmp_channel) { | 832 | if (!wk->started && !local->tmp_channel) { |
@@ -935,6 +944,9 @@ void ieee80211_add_work(struct ieee80211_work *wk) | |||
935 | if (WARN_ON(!wk->done)) | 944 | if (WARN_ON(!wk->done)) |
936 | return; | 945 | return; |
937 | 946 | ||
947 | if (WARN_ON(!ieee80211_sdata_running(wk->sdata))) | ||
948 | return; | ||
949 | |||
938 | wk->started = false; | 950 | wk->started = false; |
939 | 951 | ||
940 | local = wk->sdata->local; | 952 | local = wk->sdata->local; |
@@ -1027,7 +1039,7 @@ static enum work_done_result ieee80211_remain_done(struct ieee80211_work *wk, | |||
1027 | /* | 1039 | /* |
1028 | * We are done serving the remain-on-channel command. | 1040 | * We are done serving the remain-on-channel command. |
1029 | */ | 1041 | */ |
1030 | cfg80211_remain_on_channel_expired(wk->sdata->dev, (u64)wk, | 1042 | cfg80211_remain_on_channel_expired(wk->sdata->dev, (unsigned long) wk, |
1031 | wk->chan, wk->chan_type, | 1043 | wk->chan, wk->chan_type, |
1032 | GFP_KERNEL); | 1044 | GFP_KERNEL); |
1033 | 1045 | ||
@@ -1053,7 +1065,7 @@ int ieee80211_wk_remain_on_channel(struct ieee80211_sub_if_data *sdata, | |||
1053 | 1065 | ||
1054 | wk->remain.duration = duration; | 1066 | wk->remain.duration = duration; |
1055 | 1067 | ||
1056 | *cookie = (u64)wk; | 1068 | *cookie = (unsigned long) wk; |
1057 | 1069 | ||
1058 | ieee80211_add_work(wk); | 1070 | ieee80211_add_work(wk); |
1059 | 1071 | ||
@@ -1069,7 +1081,7 @@ int ieee80211_wk_cancel_remain_on_channel(struct ieee80211_sub_if_data *sdata, | |||
1069 | 1081 | ||
1070 | mutex_lock(&local->work_mtx); | 1082 | mutex_lock(&local->work_mtx); |
1071 | list_for_each_entry_safe(wk, tmp, &local->work_list, list) { | 1083 | list_for_each_entry_safe(wk, tmp, &local->work_list, list) { |
1072 | if ((u64)wk == cookie) { | 1084 | if ((unsigned long) wk == cookie) { |
1073 | wk->timeout = jiffies; | 1085 | wk->timeout = jiffies; |
1074 | found = true; | 1086 | found = true; |
1075 | break; | 1087 | break; |