Blame view

net/mac80211/offchannel.c 26.3 KB
b203ffc3a   Jouni Malinen   mac80211: General...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
  /*
   * Off-channel operation helpers
   *
   * Copyright 2003, Jouni Malinen <jkmaline@cc.hut.fi>
   * Copyright 2004, Instant802 Networks, Inc.
   * Copyright 2005, Devicescape Software, Inc.
   * Copyright 2006-2007	Jiri Benc <jbenc@suse.cz>
   * Copyright 2007, Michael Wu <flamingice@sourmilk.net>
   * Copyright 2009	Johannes Berg <johannes@sipsolutions.net>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
   */
bc3b2d7fb   Paul Gortmaker   net: Add export.h...
15
  #include <linux/export.h>
b203ffc3a   Jouni Malinen   mac80211: General...
16
17
  #include <net/mac80211.h>
  #include "ieee80211_i.h"
2eb278e08   Johannes Berg   mac80211: unify S...
18
  #include "driver-ops.h"
b203ffc3a   Jouni Malinen   mac80211: General...
19
20
  
  /*
b23b025fe   Ben Greear   mac80211: Optimiz...
21
22
23
24
25
   * Tell our hardware to disable PS.
   * Optionally inform AP that we will go to sleep so that it will buffer
   * the frames while we are doing off-channel work.  This is optional
   * because we *may* be doing work on-operating channel, and want our
   * hardware unconditionally awake, but still let the AP send us normal frames.
b203ffc3a   Jouni Malinen   mac80211: General...
26
   */
559cef996   Rajkumar Manoharan   mac80211: cleanup...
27
  static void ieee80211_offchannel_ps_enable(struct ieee80211_sub_if_data *sdata)
b203ffc3a   Jouni Malinen   mac80211: General...
28
29
  {
  	struct ieee80211_local *local = sdata->local;
4730d5977   Luis R. Rodriguez   mac80211: reset c...
30
  	struct ieee80211_if_managed *ifmgd = &sdata->u.mgd;
b203ffc3a   Jouni Malinen   mac80211: General...
31
32
33
34
35
36
  
  	local->offchannel_ps_enabled = false;
  
  	/* FIXME: what to do when local->pspolling is true? */
  
  	del_timer_sync(&local->dynamic_ps_timer);
3bc3c0d74   Luis R. Rodriguez   mac80211: disable...
37
  	del_timer_sync(&ifmgd->bcn_mon_timer);
4730d5977   Luis R. Rodriguez   mac80211: reset c...
38
  	del_timer_sync(&ifmgd->conn_mon_timer);
b203ffc3a   Jouni Malinen   mac80211: General...
39
40
41
42
43
44
45
  	cancel_work_sync(&local->dynamic_ps_enable_work);
  
  	if (local->hw.conf.flags & IEEE80211_CONF_PS) {
  		local->offchannel_ps_enabled = true;
  		local->hw.conf.flags &= ~IEEE80211_CONF_PS;
  		ieee80211_hw_config(local, IEEE80211_CONF_CHANGE_PS);
  	}
559cef996   Rajkumar Manoharan   mac80211: cleanup...
46
  	if (!local->offchannel_ps_enabled ||
30686bf7f   Johannes Berg   mac80211: convert...
47
  	    !ieee80211_hw_check(&local->hw, PS_NULLFUNC_STACK))
b203ffc3a   Jouni Malinen   mac80211: General...
48
49
50
51
52
53
54
55
56
57
  		/*
  		 * If power save was enabled, no need to send a nullfunc
  		 * frame because AP knows that we are sleeping. But if the
  		 * hardware is creating the nullfunc frame for power save
  		 * status (ie. IEEE80211_HW_PS_NULLFUNC_STACK is not
  		 * enabled) and power save was enabled, the firmware just
  		 * sent a null frame with power save disabled. So we need
  		 * to send a new nullfunc frame to inform the AP that we
  		 * are again sleeping.
  		 */
076cdcb12   Johannes Berg   mac80211: use boo...
58
  		ieee80211_send_nullfunc(local, sdata, true);
b203ffc3a   Jouni Malinen   mac80211: General...
59
60
61
62
63
64
65
66
  }
  
  /* inform AP that we are awake again, unless power save is enabled */
  static void ieee80211_offchannel_ps_disable(struct ieee80211_sub_if_data *sdata)
  {
  	struct ieee80211_local *local = sdata->local;
  
  	if (!local->ps_sdata)
076cdcb12   Johannes Berg   mac80211: use boo...
67
  		ieee80211_send_nullfunc(local, sdata, false);
b203ffc3a   Jouni Malinen   mac80211: General...
68
69
70
71
72
73
74
75
76
77
78
79
80
81
  	else if (local->offchannel_ps_enabled) {
  		/*
  		 * In !IEEE80211_HW_PS_NULLFUNC_STACK case the hardware
  		 * will send a nullfunc frame with the powersave bit set
  		 * even though the AP already knows that we are sleeping.
  		 * This could be avoided by sending a null frame with power
  		 * save bit disabled before enabling the power save, but
  		 * this doesn't gain anything.
  		 *
  		 * When IEEE80211_HW_PS_NULLFUNC_STACK is enabled, no need
  		 * to send a nullfunc frame because AP already knows that
  		 * we are sleeping, let's just enable power save mode in
  		 * hardware.
  		 */
b23b025fe   Ben Greear   mac80211: Optimiz...
82
83
84
  		/* TODO:  Only set hardware if CONF_PS changed?
  		 * TODO:  Should we set offchannel_ps_enabled to false?
  		 */
b203ffc3a   Jouni Malinen   mac80211: General...
85
86
87
88
89
90
91
92
93
  		local->hw.conf.flags |= IEEE80211_CONF_PS;
  		ieee80211_hw_config(local, IEEE80211_CONF_CHANGE_PS);
  	} else if (local->hw.conf.dynamic_ps_timeout > 0) {
  		/*
  		 * If IEEE80211_CONF_PS was not set and the dynamic_ps_timer
  		 * had been running before leaving the operating channel,
  		 * restart the timer now and send a nullfunc frame to inform
  		 * the AP that we are awake.
  		 */
076cdcb12   Johannes Berg   mac80211: use boo...
94
  		ieee80211_send_nullfunc(local, sdata, false);
b203ffc3a   Jouni Malinen   mac80211: General...
95
96
97
  		mod_timer(&local->dynamic_ps_timer, jiffies +
  			  msecs_to_jiffies(local->hw.conf.dynamic_ps_timeout));
  	}
4730d5977   Luis R. Rodriguez   mac80211: reset c...
98

3bc3c0d74   Luis R. Rodriguez   mac80211: disable...
99
  	ieee80211_sta_reset_beacon_monitor(sdata);
4730d5977   Luis R. Rodriguez   mac80211: reset c...
100
  	ieee80211_sta_reset_conn_monitor(sdata);
b203ffc3a   Jouni Malinen   mac80211: General...
101
  }
aacde9ee4   Stanislaw Gruszka   mac80211: synchro...
102
  void ieee80211_offchannel_stop_vifs(struct ieee80211_local *local)
b203ffc3a   Jouni Malinen   mac80211: General...
103
104
  {
  	struct ieee80211_sub_if_data *sdata;
fe57d9f5c   Johannes Berg   mac80211: track w...
105
106
  	if (WARN_ON(local->use_chanctx))
  		return;
b23b025fe   Ben Greear   mac80211: Optimiz...
107
108
109
110
  	/*
  	 * notify the AP about us leaving the channel and stop all
  	 * STA interfaces.
  	 */
6c17b77b6   Seth Forshee   mac80211: Fix tx ...
111

9c35d7d23   Seth Forshee   mac80211: Add flu...
112
113
114
115
  	/*
  	 * Stop queues and transmit all frames queued by the driver
  	 * before sending nullfunc to enable powersave at the AP.
  	 */
445ea4e83   Johannes Berg   mac80211: stop qu...
116
  	ieee80211_stop_queues_by_reason(&local->hw, IEEE80211_MAX_QUEUE_MAP,
cca07b00a   Luciano Coelho   mac80211: introdu...
117
118
  					IEEE80211_QUEUE_STOP_REASON_OFFCHANNEL,
  					false);
3b24f4c65   Emmanuel Grumbach   mac80211: let flu...
119
  	ieee80211_flush_queues(local, NULL, false);
6c17b77b6   Seth Forshee   mac80211: Fix tx ...
120

b203ffc3a   Jouni Malinen   mac80211: General...
121
122
123
124
  	mutex_lock(&local->iflist_mtx);
  	list_for_each_entry(sdata, &local->interfaces, list) {
  		if (!ieee80211_sdata_running(sdata))
  			continue;
708d50edb   Ayala Beker   mac80211: add boi...
125
126
  		if (sdata->vif.type == NL80211_IFTYPE_P2P_DEVICE ||
  		    sdata->vif.type == NL80211_IFTYPE_NAN)
f142c6b90   Johannes Berg   mac80211: support...
127
  			continue;
b23b025fe   Ben Greear   mac80211: Optimiz...
128
129
130
131
  		if (sdata->vif.type != NL80211_IFTYPE_MONITOR)
  			set_bit(SDATA_STATE_OFFCHANNEL, &sdata->state);
  
  		/* Check to see if we should disable beaconing. */
d6a832288   Johannes Berg   mac80211: track e...
132
133
134
135
  		if (sdata->vif.bss_conf.enable_beacon) {
  			set_bit(SDATA_STATE_OFFCHANNEL_BEACON_STOPPED,
  				&sdata->state);
  			sdata->vif.bss_conf.enable_beacon = false;
b203ffc3a   Jouni Malinen   mac80211: General...
136
137
  			ieee80211_bss_info_change_notify(
  				sdata, BSS_CHANGED_BEACON_ENABLED);
d6a832288   Johannes Berg   mac80211: track e...
138
  		}
b203ffc3a   Jouni Malinen   mac80211: General...
139

6c17b77b6   Seth Forshee   mac80211: Fix tx ...
140
141
142
  		if (sdata->vif.type == NL80211_IFTYPE_STATION &&
  		    sdata->u.mgd.associated)
  			ieee80211_offchannel_ps_enable(sdata);
b203ffc3a   Jouni Malinen   mac80211: General...
143
144
145
  	}
  	mutex_unlock(&local->iflist_mtx);
  }
aacde9ee4   Stanislaw Gruszka   mac80211: synchro...
146
  void ieee80211_offchannel_return(struct ieee80211_local *local)
b203ffc3a   Jouni Malinen   mac80211: General...
147
148
  {
  	struct ieee80211_sub_if_data *sdata;
fe57d9f5c   Johannes Berg   mac80211: track w...
149
150
  	if (WARN_ON(local->use_chanctx))
  		return;
b203ffc3a   Jouni Malinen   mac80211: General...
151
152
  	mutex_lock(&local->iflist_mtx);
  	list_for_each_entry(sdata, &local->interfaces, list) {
f142c6b90   Johannes Berg   mac80211: support...
153
154
  		if (sdata->vif.type == NL80211_IFTYPE_P2P_DEVICE)
  			continue;
f6e8cb72a   Eliad Peller   mac80211: always ...
155
156
  		if (sdata->vif.type != NL80211_IFTYPE_MONITOR)
  			clear_bit(SDATA_STATE_OFFCHANNEL, &sdata->state);
b203ffc3a   Jouni Malinen   mac80211: General...
157
158
159
160
  		if (!ieee80211_sdata_running(sdata))
  			continue;
  
  		/* Tell AP we're back */
aacde9ee4   Stanislaw Gruszka   mac80211: synchro...
161
162
163
  		if (sdata->vif.type == NL80211_IFTYPE_STATION &&
  		    sdata->u.mgd.associated)
  			ieee80211_offchannel_ps_disable(sdata);
b203ffc3a   Jouni Malinen   mac80211: General...
164

d6a832288   Johannes Berg   mac80211: track e...
165
166
167
  		if (test_and_clear_bit(SDATA_STATE_OFFCHANNEL_BEACON_STOPPED,
  				       &sdata->state)) {
  			sdata->vif.bss_conf.enable_beacon = true;
b203ffc3a   Jouni Malinen   mac80211: General...
168
169
  			ieee80211_bss_info_change_notify(
  				sdata, BSS_CHANGED_BEACON_ENABLED);
d6a832288   Johannes Berg   mac80211: track e...
170
  		}
b203ffc3a   Jouni Malinen   mac80211: General...
171
172
  	}
  	mutex_unlock(&local->iflist_mtx);
6c17b77b6   Seth Forshee   mac80211: Fix tx ...
173

445ea4e83   Johannes Berg   mac80211: stop qu...
174
  	ieee80211_wake_queues_by_reason(&local->hw, IEEE80211_MAX_QUEUE_MAP,
cca07b00a   Luciano Coelho   mac80211: introdu...
175
176
  					IEEE80211_QUEUE_STOP_REASON_OFFCHANNEL,
  					false);
b203ffc3a   Jouni Malinen   mac80211: General...
177
  }
21f835896   Johannes Berg   mac80211: impleme...
178

aaa016ccd   Johannes Berg   mac80211: rewrite...
179
  static void ieee80211_roc_notify_destroy(struct ieee80211_roc_work *roc)
2eb278e08   Johannes Berg   mac80211: unify S...
180
  {
aaa016ccd   Johannes Berg   mac80211: rewrite...
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
  	/* was never transmitted */
  	if (roc->frame) {
  		cfg80211_mgmt_tx_status(&roc->sdata->wdev, roc->mgmt_tx_cookie,
  					roc->frame->data, roc->frame->len,
  					false, GFP_KERNEL);
  		ieee80211_free_txskb(&roc->sdata->local->hw, roc->frame);
  	}
  
  	if (!roc->mgmt_tx_cookie)
  		cfg80211_remain_on_channel_expired(&roc->sdata->wdev,
  						   roc->cookie, roc->chan,
  						   GFP_KERNEL);
  
  	list_del(&roc->list);
  	kfree(roc);
  }
  
  static unsigned long ieee80211_end_finished_rocs(struct ieee80211_local *local,
  						 unsigned long now)
  {
  	struct ieee80211_roc_work *roc, *tmp;
  	long remaining_dur_min = LONG_MAX;
  
  	lockdep_assert_held(&local->mtx);
  
  	list_for_each_entry_safe(roc, tmp, &local->roc_list, list) {
  		long remaining;
  
  		if (!roc->started)
  			break;
  
  		remaining = roc->start_time +
  			    msecs_to_jiffies(roc->duration) -
  			    now;
1b894521e   Ilan Peer   mac80211: handle ...
215
216
217
218
219
  		/* In case of HW ROC, it is possible that the HW finished the
  		 * ROC session before the actual requested time. In such a case
  		 * end the ROC session (disregarding the remaining time).
  		 */
  		if (roc->abort || roc->hw_begun || remaining <= 0)
aaa016ccd   Johannes Berg   mac80211: rewrite...
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
  			ieee80211_roc_notify_destroy(roc);
  		else
  			remaining_dur_min = min(remaining_dur_min, remaining);
  	}
  
  	return remaining_dur_min;
  }
  
  static bool ieee80211_recalc_sw_work(struct ieee80211_local *local,
  				     unsigned long now)
  {
  	long dur = ieee80211_end_finished_rocs(local, now);
  
  	if (dur == LONG_MAX)
  		return false;
  
  	mod_delayed_work(local->workqueue, &local->roc_work, dur);
  	return true;
  }
  
  static void ieee80211_handle_roc_started(struct ieee80211_roc_work *roc,
  					 unsigned long start_time)
  {
aaa016ccd   Johannes Berg   mac80211: rewrite...
243
  	if (WARN_ON(roc->notified))
2eb278e08   Johannes Berg   mac80211: unify S...
244
  		return;
aaa016ccd   Johannes Berg   mac80211: rewrite...
245
246
  	roc->start_time = start_time;
  	roc->started = true;
aaa016ccd   Johannes Berg   mac80211: rewrite...
247

2eb278e08   Johannes Berg   mac80211: unify S...
248
249
  	if (roc->mgmt_tx_cookie) {
  		if (!WARN_ON(!roc->frame)) {
55de908ab   Johannes Berg   mac80211: use cha...
250
251
  			ieee80211_tx_skb_tid_band(roc->sdata, roc->frame, 7,
  						  roc->chan->band);
2eb278e08   Johannes Berg   mac80211: unify S...
252
253
254
  			roc->frame = NULL;
  		}
  	} else {
50febf6a1   Johannes Berg   mac80211: use a c...
255
  		cfg80211_ready_on_channel(&roc->sdata->wdev, roc->cookie,
42d97a599   Johannes Berg   cfg80211: remove ...
256
257
  					  roc->chan, roc->req_duration,
  					  GFP_KERNEL);
2eb278e08   Johannes Berg   mac80211: unify S...
258
259
260
261
  	}
  
  	roc->notified = true;
  }
21f835896   Johannes Berg   mac80211: impleme...
262
263
264
265
  static void ieee80211_hw_roc_start(struct work_struct *work)
  {
  	struct ieee80211_local *local =
  		container_of(work, struct ieee80211_local, hw_roc_start);
aaa016ccd   Johannes Berg   mac80211: rewrite...
266
  	struct ieee80211_roc_work *roc;
21f835896   Johannes Berg   mac80211: impleme...
267
268
  
  	mutex_lock(&local->mtx);
aaa016ccd   Johannes Berg   mac80211: rewrite...
269
270
271
  	list_for_each_entry(roc, &local->roc_list, list) {
  		if (!roc->started)
  			break;
2eb278e08   Johannes Berg   mac80211: unify S...
272

e6a8a3aaa   Johannes Berg   mac80211: fix rem...
273
  		roc->hw_begun = true;
aaa016ccd   Johannes Berg   mac80211: rewrite...
274
  		ieee80211_handle_roc_started(roc, local->hw_roc_start_time);
2eb278e08   Johannes Berg   mac80211: unify S...
275
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
276

21f835896   Johannes Berg   mac80211: impleme...
277
278
279
280
281
282
  	mutex_unlock(&local->mtx);
  }
  
  void ieee80211_ready_on_channel(struct ieee80211_hw *hw)
  {
  	struct ieee80211_local *local = hw_to_local(hw);
2eb278e08   Johannes Berg   mac80211: unify S...
283
  	local->hw_roc_start_time = jiffies;
21f835896   Johannes Berg   mac80211: impleme...
284
285
286
287
288
  	trace_api_ready_on_channel(local);
  
  	ieee80211_queue_work(hw, &local->hw_roc_start);
  }
  EXPORT_SYMBOL_GPL(ieee80211_ready_on_channel);
aaa016ccd   Johannes Berg   mac80211: rewrite...
289
  static void _ieee80211_start_next_roc(struct ieee80211_local *local)
21f835896   Johannes Berg   mac80211: impleme...
290
  {
aaa016ccd   Johannes Berg   mac80211: rewrite...
291
292
293
  	struct ieee80211_roc_work *roc, *tmp;
  	enum ieee80211_roc_type type;
  	u32 min_dur, max_dur;
21f835896   Johannes Berg   mac80211: impleme...
294

2eb278e08   Johannes Berg   mac80211: unify S...
295
  	lockdep_assert_held(&local->mtx);
21f835896   Johannes Berg   mac80211: impleme...
296

aaa016ccd   Johannes Berg   mac80211: rewrite...
297
  	if (WARN_ON(list_empty(&local->roc_list)))
21f835896   Johannes Berg   mac80211: impleme...
298
  		return;
21f835896   Johannes Berg   mac80211: impleme...
299

2eb278e08   Johannes Berg   mac80211: unify S...
300
301
  	roc = list_first_entry(&local->roc_list, struct ieee80211_roc_work,
  			       list);
71ecfa189   Johannes Berg   mac80211: clean u...
302

aaa016ccd   Johannes Berg   mac80211: rewrite...
303
  	if (WARN_ON(roc->started))
0f6b3f597   Johannes Berg   mac80211: fix dou...
304
  		return;
aaa016ccd   Johannes Berg   mac80211: rewrite...
305
306
307
  	min_dur = roc->duration;
  	max_dur = roc->duration;
  	type = roc->type;
71ecfa189   Johannes Berg   mac80211: clean u...
308

aaa016ccd   Johannes Berg   mac80211: rewrite...
309
310
311
312
313
314
315
316
317
  	list_for_each_entry(tmp, &local->roc_list, list) {
  		if (tmp == roc)
  			continue;
  		if (tmp->sdata != roc->sdata || tmp->chan != roc->chan)
  			break;
  		max_dur = max(tmp->duration, max_dur);
  		min_dur = min(tmp->duration, min_dur);
  		type = max(tmp->type, type);
  	}
2eb278e08   Johannes Berg   mac80211: unify S...
318

aaa016ccd   Johannes Berg   mac80211: rewrite...
319
320
321
  	if (local->ops->remain_on_channel) {
  		int ret = drv_remain_on_channel(local, roc->sdata, roc->chan,
  						max_dur, type);
2eb278e08   Johannes Berg   mac80211: unify S...
322
323
324
325
326
327
328
329
330
  
  		if (ret) {
  			wiphy_warn(local->hw.wiphy,
  				   "failed to start next HW ROC (%d)
  ", ret);
  			/*
  			 * queue the work struct again to avoid recursion
  			 * when multiple failures occur
  			 */
aaa016ccd   Johannes Berg   mac80211: rewrite...
331
332
333
334
335
336
337
338
339
  			list_for_each_entry(tmp, &local->roc_list, list) {
  				if (tmp->sdata != roc->sdata ||
  				    tmp->chan != roc->chan)
  					break;
  				tmp->started = true;
  				tmp->abort = true;
  			}
  			ieee80211_queue_work(&local->hw, &local->hw_roc_done);
  			return;
2eb278e08   Johannes Berg   mac80211: unify S...
340
  		}
b4b177a55   Johannes Berg   mac80211: fix on-...
341

aaa016ccd   Johannes Berg   mac80211: rewrite...
342
343
344
345
346
347
348
  		/* we'll notify about the start once the HW calls back */
  		list_for_each_entry(tmp, &local->roc_list, list) {
  			if (tmp->sdata != roc->sdata || tmp->chan != roc->chan)
  				break;
  			tmp->started = true;
  		}
  	} else {
b4b177a55   Johannes Berg   mac80211: fix on-...
349
350
351
352
353
354
355
356
  		/* If actually operating on the desired channel (with at least
  		 * 20 MHz channel width) don't stop all the operations but still
  		 * treat it as though the ROC operation started properly, so
  		 * other ROC operations won't interfere with this one.
  		 */
  		roc->on_channel = roc->chan == local->_oper_chandef.chan &&
  				  local->_oper_chandef.width != NL80211_CHAN_WIDTH_5 &&
  				  local->_oper_chandef.width != NL80211_CHAN_WIDTH_10;
21f835896   Johannes Berg   mac80211: impleme...
357

b4b177a55   Johannes Berg   mac80211: fix on-...
358
  		/* start this ROC */
2eb278e08   Johannes Berg   mac80211: unify S...
359
  		ieee80211_recalc_idle(local);
b4b177a55   Johannes Berg   mac80211: fix on-...
360
361
362
363
364
365
  		if (!roc->on_channel) {
  			ieee80211_offchannel_stop_vifs(local);
  
  			local->tmp_channel = roc->chan;
  			ieee80211_hw_config(local, 0);
  		}
2eb278e08   Johannes Berg   mac80211: unify S...
366

aaa016ccd   Johannes Berg   mac80211: rewrite...
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
  		ieee80211_queue_delayed_work(&local->hw, &local->roc_work,
  					     msecs_to_jiffies(min_dur));
  
  		/* tell userspace or send frame(s) */
  		list_for_each_entry(tmp, &local->roc_list, list) {
  			if (tmp->sdata != roc->sdata || tmp->chan != roc->chan)
  				break;
  
  			tmp->on_channel = roc->on_channel;
  			ieee80211_handle_roc_started(tmp, jiffies);
  		}
  	}
  }
  
  void ieee80211_start_next_roc(struct ieee80211_local *local)
  {
  	struct ieee80211_roc_work *roc;
  
  	lockdep_assert_held(&local->mtx);
  
  	if (list_empty(&local->roc_list)) {
  		ieee80211_run_deferred_scan(local);
  		return;
  	}
470f4d613   Eliad Peller   mac80211: avoid R...
391
392
393
  	/* defer roc if driver is not started (i.e. during reconfig) */
  	if (local->in_reconfig)
  		return;
aaa016ccd   Johannes Berg   mac80211: rewrite...
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
  	roc = list_first_entry(&local->roc_list, struct ieee80211_roc_work,
  			       list);
  
  	if (WARN_ON_ONCE(roc->started))
  		return;
  
  	if (local->ops->remain_on_channel) {
  		_ieee80211_start_next_roc(local);
  	} else {
  		/* delay it a bit */
  		ieee80211_queue_delayed_work(&local->hw, &local->roc_work,
  					     round_jiffies_relative(HZ/2));
  	}
  }
  
  static void __ieee80211_roc_work(struct ieee80211_local *local)
  {
  	struct ieee80211_roc_work *roc;
  	bool on_channel;
  
  	lockdep_assert_held(&local->mtx);
  
  	if (WARN_ON(local->ops->remain_on_channel))
  		return;
2eb278e08   Johannes Berg   mac80211: unify S...
418

aaa016ccd   Johannes Berg   mac80211: rewrite...
419
420
421
422
  	roc = list_first_entry_or_null(&local->roc_list,
  				       struct ieee80211_roc_work, list);
  	if (!roc)
  		return;
2eb278e08   Johannes Berg   mac80211: unify S...
423

aaa016ccd   Johannes Berg   mac80211: rewrite...
424
425
426
  	if (!roc->started) {
  		WARN_ON(local->use_chanctx);
  		_ieee80211_start_next_roc(local);
2eb278e08   Johannes Berg   mac80211: unify S...
427
  	} else {
b4b177a55   Johannes Berg   mac80211: fix on-...
428
  		on_channel = roc->on_channel;
aaa016ccd   Johannes Berg   mac80211: rewrite...
429
430
  		if (ieee80211_recalc_sw_work(local, jiffies))
  			return;
2eb278e08   Johannes Berg   mac80211: unify S...
431

aaa016ccd   Johannes Berg   mac80211: rewrite...
432
433
434
  		/* careful - roc pointer became invalid during recalc */
  
  		if (!on_channel) {
3b24f4c65   Emmanuel Grumbach   mac80211: let flu...
435
  			ieee80211_flush_queues(local, NULL, false);
2eb278e08   Johannes Berg   mac80211: unify S...
436
437
438
  
  			local->tmp_channel = NULL;
  			ieee80211_hw_config(local, 0);
aacde9ee4   Stanislaw Gruszka   mac80211: synchro...
439
  			ieee80211_offchannel_return(local);
2eb278e08   Johannes Berg   mac80211: unify S...
440
441
442
  		}
  
  		ieee80211_recalc_idle(local);
aaa016ccd   Johannes Berg   mac80211: rewrite...
443
  		ieee80211_start_next_roc(local);
2eb278e08   Johannes Berg   mac80211: unify S...
444
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
445
  }
2eb278e08   Johannes Berg   mac80211: unify S...
446

aaa016ccd   Johannes Berg   mac80211: rewrite...
447
448
449
450
451
452
453
  static void ieee80211_roc_work(struct work_struct *work)
  {
  	struct ieee80211_local *local =
  		container_of(work, struct ieee80211_local, roc_work.work);
  
  	mutex_lock(&local->mtx);
  	__ieee80211_roc_work(local);
2eb278e08   Johannes Berg   mac80211: unify S...
454
455
456
457
458
459
460
  	mutex_unlock(&local->mtx);
  }
  
  static void ieee80211_hw_roc_done(struct work_struct *work)
  {
  	struct ieee80211_local *local =
  		container_of(work, struct ieee80211_local, hw_roc_done);
2eb278e08   Johannes Berg   mac80211: unify S...
461
462
  
  	mutex_lock(&local->mtx);
aaa016ccd   Johannes Berg   mac80211: rewrite...
463
  	ieee80211_end_finished_rocs(local, jiffies);
2eb278e08   Johannes Berg   mac80211: unify S...
464
465
466
  
  	/* if there's another roc, start it now */
  	ieee80211_start_next_roc(local);
21f835896   Johannes Berg   mac80211: impleme...
467
468
469
470
471
472
473
474
475
476
477
478
  	mutex_unlock(&local->mtx);
  }
  
  void ieee80211_remain_on_channel_expired(struct ieee80211_hw *hw)
  {
  	struct ieee80211_local *local = hw_to_local(hw);
  
  	trace_api_remain_on_channel_expired(local);
  
  	ieee80211_queue_work(hw, &local->hw_roc_done);
  }
  EXPORT_SYMBOL_GPL(ieee80211_remain_on_channel_expired);
aaa016ccd   Johannes Berg   mac80211: rewrite...
479
480
481
482
  static bool
  ieee80211_coalesce_hw_started_roc(struct ieee80211_local *local,
  				  struct ieee80211_roc_work *new_roc,
  				  struct ieee80211_roc_work *cur_roc)
a2fcfccba   Johannes Berg   mac80211: move of...
483
484
  {
  	unsigned long now = jiffies;
aaa016ccd   Johannes Berg   mac80211: rewrite...
485
486
487
488
  	unsigned long remaining;
  
  	if (WARN_ON(!cur_roc->started))
  		return false;
a2fcfccba   Johannes Berg   mac80211: move of...
489

aaa016ccd   Johannes Berg   mac80211: rewrite...
490
491
492
493
  	/* if it was scheduled in the hardware, but not started yet,
  	 * we can only combine if the older one had a longer duration
  	 */
  	if (!cur_roc->hw_begun && new_roc->duration > cur_roc->duration)
a2fcfccba   Johannes Berg   mac80211: move of...
494
  		return false;
aaa016ccd   Johannes Berg   mac80211: rewrite...
495
496
497
  	remaining = cur_roc->start_time +
  		    msecs_to_jiffies(cur_roc->duration) -
  		    now;
a2fcfccba   Johannes Berg   mac80211: move of...
498
499
500
  	/* if it doesn't fit entirely, schedule a new one */
  	if (new_roc->duration > jiffies_to_msecs(remaining))
  		return false;
aaa016ccd   Johannes Berg   mac80211: rewrite...
501
502
503
504
505
506
507
  	/* add just after the current one so we combine their finish later */
  	list_add(&new_roc->list, &cur_roc->list);
  
  	/* if the existing one has already begun then let this one also
  	 * begin, otherwise they'll both be marked properly by the work
  	 * struct that runs once the driver notifies us of the beginning
  	 */
e6a8a3aaa   Johannes Berg   mac80211: fix rem...
508
509
  	if (cur_roc->hw_begun) {
  		new_roc->hw_begun = true;
aaa016ccd   Johannes Berg   mac80211: rewrite...
510
  		ieee80211_handle_roc_started(new_roc, now);
e6a8a3aaa   Johannes Berg   mac80211: fix rem...
511
  	}
a2fcfccba   Johannes Berg   mac80211: move of...
512

a2fcfccba   Johannes Berg   mac80211: move of...
513
514
515
516
517
518
519
520
521
522
523
  	return true;
  }
  
  static int ieee80211_start_roc_work(struct ieee80211_local *local,
  				    struct ieee80211_sub_if_data *sdata,
  				    struct ieee80211_channel *channel,
  				    unsigned int duration, u64 *cookie,
  				    struct sk_buff *txskb,
  				    enum ieee80211_roc_type type)
  {
  	struct ieee80211_roc_work *roc, *tmp;
aaa016ccd   Johannes Berg   mac80211: rewrite...
524
  	bool queued = false, combine_started = true;
a2fcfccba   Johannes Berg   mac80211: move of...
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
  	int ret;
  
  	lockdep_assert_held(&local->mtx);
  
  	if (local->use_chanctx && !local->ops->remain_on_channel)
  		return -EOPNOTSUPP;
  
  	roc = kzalloc(sizeof(*roc), GFP_KERNEL);
  	if (!roc)
  		return -ENOMEM;
  
  	/*
  	 * If the duration is zero, then the driver
  	 * wouldn't actually do anything. Set it to
  	 * 10 for now.
  	 *
  	 * TODO: cancel the off-channel operation
  	 *       when we get the SKB's TX status and
  	 *       the wait time was zero before.
  	 */
  	if (!duration)
  		duration = 10;
  
  	roc->chan = channel;
  	roc->duration = duration;
  	roc->req_duration = duration;
  	roc->frame = txskb;
  	roc->type = type;
  	roc->sdata = sdata;
a2fcfccba   Johannes Berg   mac80211: move of...
554
555
556
557
558
559
560
561
562
563
564
  
  	/*
  	 * cookie is either the roc cookie (for normal roc)
  	 * or the SKB (for mgmt TX)
  	 */
  	if (!txskb) {
  		roc->cookie = ieee80211_mgmt_tx_cookie(local);
  		*cookie = roc->cookie;
  	} else {
  		roc->mgmt_tx_cookie = *cookie;
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
  	/* if there's no need to queue, handle it immediately */
  	if (list_empty(&local->roc_list) &&
  	    !local->scanning && !ieee80211_is_radar_required(local)) {
  		/* if not HW assist, just queue & schedule work */
  		if (!local->ops->remain_on_channel) {
  			list_add_tail(&roc->list, &local->roc_list);
  			ieee80211_queue_delayed_work(&local->hw,
  						     &local->roc_work, 0);
  		} else {
  			/* otherwise actually kick it off here
  			 * (for error handling)
  			 */
  			ret = drv_remain_on_channel(local, sdata, channel,
  						    duration, type);
  			if (ret) {
  				kfree(roc);
  				return ret;
  			}
  			roc->started = true;
  			list_add_tail(&roc->list, &local->roc_list);
  		}
a2fcfccba   Johannes Berg   mac80211: move of...
586

aaa016ccd   Johannes Berg   mac80211: rewrite...
587
  		return 0;
a2fcfccba   Johannes Berg   mac80211: move of...
588
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
589
  	/* otherwise handle queueing */
a2fcfccba   Johannes Berg   mac80211: move of...
590

a2fcfccba   Johannes Berg   mac80211: move of...
591
592
593
594
595
  	list_for_each_entry(tmp, &local->roc_list, list) {
  		if (tmp->chan != channel || tmp->sdata != sdata)
  			continue;
  
  		/*
aaa016ccd   Johannes Berg   mac80211: rewrite...
596
597
  		 * Extend this ROC if possible: If it hasn't started, add
  		 * just after the new one to combine.
a2fcfccba   Johannes Berg   mac80211: move of...
598
599
  		 */
  		if (!tmp->started) {
aaa016ccd   Johannes Berg   mac80211: rewrite...
600
  			list_add(&roc->list, &tmp->list);
a2fcfccba   Johannes Berg   mac80211: move of...
601
602
603
  			queued = true;
  			break;
  		}
aaa016ccd   Johannes Berg   mac80211: rewrite...
604
605
  		if (!combine_started)
  			continue;
a2fcfccba   Johannes Berg   mac80211: move of...
606

aaa016ccd   Johannes Berg   mac80211: rewrite...
607
608
609
610
611
612
613
614
615
  		if (!local->ops->remain_on_channel) {
  			/* If there's no hardware remain-on-channel, and
  			 * doing so won't push us over the maximum r-o-c
  			 * we allow, then we can just add the new one to
  			 * the list and mark it as having started now.
  			 * If it would push over the limit, don't try to
  			 * combine with other started ones (that haven't
  			 * been running as long) but potentially sort it
  			 * with others that had the same fate.
a2fcfccba   Johannes Berg   mac80211: move of...
616
  			 */
aaa016ccd   Johannes Berg   mac80211: rewrite...
617
618
619
620
  			unsigned long now = jiffies;
  			u32 elapsed = jiffies_to_msecs(now - tmp->start_time);
  			struct wiphy *wiphy = local->hw.wiphy;
  			u32 max_roc = wiphy->max_remain_on_channel_duration;
a2fcfccba   Johannes Berg   mac80211: move of...
621

aaa016ccd   Johannes Berg   mac80211: rewrite...
622
623
624
625
  			if (elapsed + roc->duration > max_roc) {
  				combine_started = false;
  				continue;
  			}
a2fcfccba   Johannes Berg   mac80211: move of...
626

aaa016ccd   Johannes Berg   mac80211: rewrite...
627
628
629
630
  			list_add(&roc->list, &tmp->list);
  			queued = true;
  			roc->on_channel = tmp->on_channel;
  			ieee80211_handle_roc_started(roc, now);
e9db45578   Johannes Berg   mac80211: recalcu...
631
  			ieee80211_recalc_sw_work(local, now);
aaa016ccd   Johannes Berg   mac80211: rewrite...
632
  			break;
a2fcfccba   Johannes Berg   mac80211: move of...
633
  		}
aaa016ccd   Johannes Berg   mac80211: rewrite...
634
635
636
637
638
639
640
641
642
643
  
  		queued = ieee80211_coalesce_hw_started_roc(local, roc, tmp);
  		if (queued)
  			break;
  		/* if it wasn't queued, perhaps it can be combined with
  		 * another that also couldn't get combined previously,
  		 * but no need to check for already started ones, since
  		 * that can't work.
  		 */
  		combine_started = false;
a2fcfccba   Johannes Berg   mac80211: move of...
644
  	}
a2fcfccba   Johannes Berg   mac80211: move of...
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
  	if (!queued)
  		list_add_tail(&roc->list, &local->roc_list);
  
  	return 0;
  }
  
  int ieee80211_remain_on_channel(struct wiphy *wiphy, struct wireless_dev *wdev,
  				struct ieee80211_channel *chan,
  				unsigned int duration, u64 *cookie)
  {
  	struct ieee80211_sub_if_data *sdata = IEEE80211_WDEV_TO_SUB_IF(wdev);
  	struct ieee80211_local *local = sdata->local;
  	int ret;
  
  	mutex_lock(&local->mtx);
  	ret = ieee80211_start_roc_work(local, sdata, chan,
  				       duration, cookie, NULL,
  				       IEEE80211_ROC_TYPE_NORMAL);
  	mutex_unlock(&local->mtx);
  
  	return ret;
  }
  
  static int ieee80211_cancel_roc(struct ieee80211_local *local,
  				u64 cookie, bool mgmt_tx)
  {
  	struct ieee80211_roc_work *roc, *tmp, *found = NULL;
  	int ret;
7d37fcd40   Johannes Berg   mac80211: reject ...
673
674
  	if (!cookie)
  		return -ENOENT;
6e46d8ce8   Avraham Stern   mac80211: flush h...
675
  	flush_work(&local->hw_roc_start);
a2fcfccba   Johannes Berg   mac80211: move of...
676
677
  	mutex_lock(&local->mtx);
  	list_for_each_entry_safe(roc, tmp, &local->roc_list, list) {
a2fcfccba   Johannes Berg   mac80211: move of...
678
679
680
681
682
683
684
685
686
687
688
689
690
  		if (!mgmt_tx && roc->cookie != cookie)
  			continue;
  		else if (mgmt_tx && roc->mgmt_tx_cookie != cookie)
  			continue;
  
  		found = roc;
  		break;
  	}
  
  	if (!found) {
  		mutex_unlock(&local->mtx);
  		return -ENOENT;
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
691
692
693
694
  	if (!found->started) {
  		ieee80211_roc_notify_destroy(found);
  		goto out_unlock;
  	}
a2fcfccba   Johannes Berg   mac80211: move of...
695
696
  
  	if (local->ops->remain_on_channel) {
aaa016ccd   Johannes Berg   mac80211: rewrite...
697
698
699
700
  		ret = drv_cancel_remain_on_channel(local);
  		if (WARN_ON_ONCE(ret)) {
  			mutex_unlock(&local->mtx);
  			return ret;
a2fcfccba   Johannes Berg   mac80211: move of...
701
  		}
aaa016ccd   Johannes Berg   mac80211: rewrite...
702
703
704
705
706
707
708
709
710
711
712
713
  		/* TODO:
  		 * if multiple items were combined here then we really shouldn't
  		 * cancel them all - we should wait for as much time as needed
  		 * for the longest remaining one, and only then cancel ...
  		 */
  		list_for_each_entry_safe(roc, tmp, &local->roc_list, list) {
  			if (!roc->started)
  				break;
  			if (roc == found)
  				found = NULL;
  			ieee80211_roc_notify_destroy(roc);
  		}
a2fcfccba   Johannes Berg   mac80211: move of...
714

aaa016ccd   Johannes Berg   mac80211: rewrite...
715
716
  		/* that really must not happen - it was started */
  		WARN_ON(found);
a2fcfccba   Johannes Berg   mac80211: move of...
717

aaa016ccd   Johannes Berg   mac80211: rewrite...
718
  		ieee80211_start_next_roc(local);
a2fcfccba   Johannes Berg   mac80211: move of...
719
  	} else {
aaa016ccd   Johannes Berg   mac80211: rewrite...
720
  		/* go through work struct to return to the operating channel */
a2fcfccba   Johannes Berg   mac80211: move of...
721
  		found->abort = true;
aaa016ccd   Johannes Berg   mac80211: rewrite...
722
  		mod_delayed_work(local->workqueue, &local->roc_work, 0);
a2fcfccba   Johannes Berg   mac80211: move of...
723
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
724
725
   out_unlock:
  	mutex_unlock(&local->mtx);
a2fcfccba   Johannes Berg   mac80211: move of...
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
  	return 0;
  }
  
  int ieee80211_cancel_remain_on_channel(struct wiphy *wiphy,
  				       struct wireless_dev *wdev, u64 cookie)
  {
  	struct ieee80211_sub_if_data *sdata = IEEE80211_WDEV_TO_SUB_IF(wdev);
  	struct ieee80211_local *local = sdata->local;
  
  	return ieee80211_cancel_roc(local, cookie, false);
  }
  
  int ieee80211_mgmt_tx(struct wiphy *wiphy, struct wireless_dev *wdev,
  		      struct cfg80211_mgmt_tx_params *params, u64 *cookie)
  {
  	struct ieee80211_sub_if_data *sdata = IEEE80211_WDEV_TO_SUB_IF(wdev);
  	struct ieee80211_local *local = sdata->local;
5ee00dbd5   Johannes Berg   mac80211: simplif...
743
  	struct sk_buff *skb;
a2fcfccba   Johannes Berg   mac80211: move of...
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
  	struct sta_info *sta;
  	const struct ieee80211_mgmt *mgmt = (void *)params->buf;
  	bool need_offchan = false;
  	u32 flags;
  	int ret;
  	u8 *data;
  
  	if (params->dont_wait_for_ack)
  		flags = IEEE80211_TX_CTL_NO_ACK;
  	else
  		flags = IEEE80211_TX_INTFL_NL80211_FRAME_TX |
  			IEEE80211_TX_CTL_REQ_TX_STATUS;
  
  	if (params->no_cck)
  		flags |= IEEE80211_TX_CTL_NO_CCK_RATE;
  
  	switch (sdata->vif.type) {
  	case NL80211_IFTYPE_ADHOC:
  		if (!sdata->vif.bss_conf.ibss_joined)
  			need_offchan = true;
  		/* fall through */
  #ifdef CONFIG_MAC80211_MESH
  	case NL80211_IFTYPE_MESH_POINT:
  		if (ieee80211_vif_is_mesh(&sdata->vif) &&
  		    !sdata->u.mesh.mesh_id_len)
  			need_offchan = true;
  		/* fall through */
  #endif
  	case NL80211_IFTYPE_AP:
  	case NL80211_IFTYPE_AP_VLAN:
  	case NL80211_IFTYPE_P2P_GO:
  		if (sdata->vif.type != NL80211_IFTYPE_ADHOC &&
  		    !ieee80211_vif_is_mesh(&sdata->vif) &&
  		    !rcu_access_pointer(sdata->bss->beacon))
  			need_offchan = true;
  		if (!ieee80211_is_action(mgmt->frame_control) ||
  		    mgmt->u.action.category == WLAN_CATEGORY_PUBLIC ||
  		    mgmt->u.action.category == WLAN_CATEGORY_SELF_PROTECTED ||
  		    mgmt->u.action.category == WLAN_CATEGORY_SPECTRUM_MGMT)
  			break;
  		rcu_read_lock();
1d4de2e22   Michael Braun   mac80211: fix CMD...
785
  		sta = sta_info_get_bss(sdata, mgmt->da);
a2fcfccba   Johannes Berg   mac80211: move of...
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
  		rcu_read_unlock();
  		if (!sta)
  			return -ENOLINK;
  		break;
  	case NL80211_IFTYPE_STATION:
  	case NL80211_IFTYPE_P2P_CLIENT:
  		sdata_lock(sdata);
  		if (!sdata->u.mgd.associated ||
  		    (params->offchan && params->wait &&
  		     local->ops->remain_on_channel &&
  		     memcmp(sdata->u.mgd.associated->bssid,
  			    mgmt->bssid, ETH_ALEN)))
  			need_offchan = true;
  		sdata_unlock(sdata);
  		break;
  	case NL80211_IFTYPE_P2P_DEVICE:
  		need_offchan = true;
  		break;
cb3b7d876   Ayala Beker   cfg80211: add sta...
804
  	case NL80211_IFTYPE_NAN:
a2fcfccba   Johannes Berg   mac80211: move of...
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
  	default:
  		return -EOPNOTSUPP;
  	}
  
  	/* configurations requiring offchan cannot work if no channel has been
  	 * specified
  	 */
  	if (need_offchan && !params->chan)
  		return -EINVAL;
  
  	mutex_lock(&local->mtx);
  
  	/* Check if the operating channel is the requested channel */
  	if (!need_offchan) {
  		struct ieee80211_chanctx_conf *chanctx_conf;
  
  		rcu_read_lock();
  		chanctx_conf = rcu_dereference(sdata->vif.chanctx_conf);
  
  		if (chanctx_conf) {
  			need_offchan = params->chan &&
  				       (params->chan !=
  					chanctx_conf->def.chan);
  		} else if (!params->chan) {
  			ret = -EINVAL;
  			rcu_read_unlock();
  			goto out_unlock;
  		} else {
  			need_offchan = true;
  		}
  		rcu_read_unlock();
  	}
  
  	if (need_offchan && !params->offchan) {
  		ret = -EBUSY;
  		goto out_unlock;
  	}
  
  	skb = dev_alloc_skb(local->hw.extra_tx_headroom + params->len);
  	if (!skb) {
  		ret = -ENOMEM;
  		goto out_unlock;
  	}
  	skb_reserve(skb, local->hw.extra_tx_headroom);
59ae1d127   Johannes Berg   networking: intro...
849
  	data = skb_put_data(skb, params->buf, params->len);
a2fcfccba   Johannes Berg   mac80211: move of...
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
  
  	/* Update CSA counters */
  	if (sdata->vif.csa_active &&
  	    (sdata->vif.type == NL80211_IFTYPE_AP ||
  	     sdata->vif.type == NL80211_IFTYPE_MESH_POINT ||
  	     sdata->vif.type == NL80211_IFTYPE_ADHOC) &&
  	    params->n_csa_offsets) {
  		int i;
  		struct beacon_data *beacon = NULL;
  
  		rcu_read_lock();
  
  		if (sdata->vif.type == NL80211_IFTYPE_AP)
  			beacon = rcu_dereference(sdata->u.ap.beacon);
  		else if (sdata->vif.type == NL80211_IFTYPE_ADHOC)
  			beacon = rcu_dereference(sdata->u.ibss.presp);
  		else if (ieee80211_vif_is_mesh(&sdata->vif))
  			beacon = rcu_dereference(sdata->u.mesh.beacon);
  
  		if (beacon)
  			for (i = 0; i < params->n_csa_offsets; i++)
  				data[params->csa_offsets[i]] =
  					beacon->csa_current_counter;
  
  		rcu_read_unlock();
  	}
  
  	IEEE80211_SKB_CB(skb)->flags = flags;
  
  	skb->dev = sdata->dev;
  
  	if (!params->dont_wait_for_ack) {
  		/* make a copy to preserve the frame contents
  		 * in case of encryption.
  		 */
5ee00dbd5   Johannes Berg   mac80211: simplif...
885
886
  		ret = ieee80211_attach_ack_skb(local, skb, cookie, GFP_KERNEL);
  		if (ret) {
a2fcfccba   Johannes Berg   mac80211: move of...
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
  			kfree_skb(skb);
  			goto out_unlock;
  		}
  	} else {
  		/* Assign a dummy non-zero cookie, it's not sent to
  		 * userspace in this case but we rely on its value
  		 * internally in the need_offchan case to distinguish
  		 * mgmt-tx from remain-on-channel.
  		 */
  		*cookie = 0xffffffff;
  	}
  
  	if (!need_offchan) {
  		ieee80211_tx_skb(sdata, skb);
  		ret = 0;
  		goto out_unlock;
  	}
  
  	IEEE80211_SKB_CB(skb)->flags |= IEEE80211_TX_CTL_TX_OFFCHAN |
  					IEEE80211_TX_INTFL_OFFCHAN_TX_OK;
  	if (ieee80211_hw_check(&local->hw, QUEUE_CONTROL))
  		IEEE80211_SKB_CB(skb)->hw_queue =
  			local->hw.offchannel_tx_hw_queue;
  
  	/* This will handle all kinds of coalescing and immediate TX */
  	ret = ieee80211_start_roc_work(local, sdata, params->chan,
  				       params->wait, cookie, skb,
  				       IEEE80211_ROC_TYPE_MGMT_TX);
  	if (ret)
  		ieee80211_free_txskb(&local->hw, skb);
   out_unlock:
  	mutex_unlock(&local->mtx);
  	return ret;
  }
  
  int ieee80211_mgmt_tx_cancel_wait(struct wiphy *wiphy,
  				  struct wireless_dev *wdev, u64 cookie)
  {
  	struct ieee80211_local *local = wiphy_priv(wiphy);
  
  	return ieee80211_cancel_roc(local, cookie, true);
  }
2eb278e08   Johannes Berg   mac80211: unify S...
929
  void ieee80211_roc_setup(struct ieee80211_local *local)
21f835896   Johannes Berg   mac80211: impleme...
930
931
932
  {
  	INIT_WORK(&local->hw_roc_start, ieee80211_hw_roc_start);
  	INIT_WORK(&local->hw_roc_done, ieee80211_hw_roc_done);
aaa016ccd   Johannes Berg   mac80211: rewrite...
933
  	INIT_DELAYED_WORK(&local->roc_work, ieee80211_roc_work);
2eb278e08   Johannes Berg   mac80211: unify S...
934
935
  	INIT_LIST_HEAD(&local->roc_list);
  }
c8f994eec   Johannes Berg   mac80211: purge r...
936
937
  void ieee80211_roc_purge(struct ieee80211_local *local,
  			 struct ieee80211_sub_if_data *sdata)
2eb278e08   Johannes Berg   mac80211: unify S...
938
  {
2eb278e08   Johannes Berg   mac80211: unify S...
939
  	struct ieee80211_roc_work *roc, *tmp;
aaa016ccd   Johannes Berg   mac80211: rewrite...
940
  	bool work_to_do = false;
2eb278e08   Johannes Berg   mac80211: unify S...
941
942
943
  
  	mutex_lock(&local->mtx);
  	list_for_each_entry_safe(roc, tmp, &local->roc_list, list) {
c8f994eec   Johannes Berg   mac80211: purge r...
944
  		if (sdata && roc->sdata != sdata)
2eb278e08   Johannes Berg   mac80211: unify S...
945
  			continue;
aaa016ccd   Johannes Berg   mac80211: rewrite...
946
947
948
949
950
951
952
953
954
  		if (roc->started) {
  			if (local->ops->remain_on_channel) {
  				/* can race, so ignore return value */
  				drv_cancel_remain_on_channel(local);
  				ieee80211_roc_notify_destroy(roc);
  			} else {
  				roc->abort = true;
  				work_to_do = true;
  			}
2eb278e08   Johannes Berg   mac80211: unify S...
955
  		} else {
aaa016ccd   Johannes Berg   mac80211: rewrite...
956
  			ieee80211_roc_notify_destroy(roc);
2eb278e08   Johannes Berg   mac80211: unify S...
957
958
  		}
  	}
aaa016ccd   Johannes Berg   mac80211: rewrite...
959
960
961
  	if (work_to_do)
  		__ieee80211_roc_work(local);
  	mutex_unlock(&local->mtx);
21f835896   Johannes Berg   mac80211: impleme...
962
  }