driver-ops.h 8.9 KB
Newer Older
1 2 3 4 5
#ifndef __MAC80211_DRIVER_OPS
#define __MAC80211_DRIVER_OPS

#include <net/mac80211.h>
#include "ieee80211_i.h"
6
#include "driver-trace.h"
7 8 9 10 11 12 13 14

static inline int drv_tx(struct ieee80211_local *local, struct sk_buff *skb)
{
	return local->ops->tx(&local->hw, skb);
}

static inline int drv_start(struct ieee80211_local *local)
{
15 16
	int ret;

17 18
	might_sleep();

19 20 21
	local->started = true;
	smp_mb();
	ret = local->ops->start(&local->hw);
22 23
	trace_drv_start(local, ret);
	return ret;
24 25 26 27
}

static inline void drv_stop(struct ieee80211_local *local)
{
28 29
	might_sleep();

30
	local->ops->stop(&local->hw);
31
	trace_drv_stop(local);
32 33 34 35 36 37 38 39

	/* sync away all work on the tasklet before clearing started */
	tasklet_disable(&local->tasklet);
	tasklet_enable(&local->tasklet);

	barrier();

	local->started = false;
40 41 42
}

static inline int drv_add_interface(struct ieee80211_local *local,
43
				    struct ieee80211_vif *vif)
44
{
45 46 47 48 49
	int ret;

	might_sleep();

	ret = local->ops->add_interface(&local->hw, vif);
50
	trace_drv_add_interface(local, vif_to_sdata(vif), ret);
51
	return ret;
52 53 54
}

static inline void drv_remove_interface(struct ieee80211_local *local,
55
					struct ieee80211_vif *vif)
56
{
57 58
	might_sleep();

59 60
	local->ops->remove_interface(&local->hw, vif);
	trace_drv_remove_interface(local, vif_to_sdata(vif));
61 62 63 64
}

static inline int drv_config(struct ieee80211_local *local, u32 changed)
{
65 66 67 68 69
	int ret;

	might_sleep();

	ret = local->ops->config(&local->hw, changed);
70 71
	trace_drv_config(local, changed, ret);
	return ret;
72 73 74
}

static inline void drv_bss_info_changed(struct ieee80211_local *local,
J
Johannes Berg 已提交
75
					struct ieee80211_sub_if_data *sdata,
76 77 78
					struct ieee80211_bss_conf *info,
					u32 changed)
{
79 80
	might_sleep();

81
	if (local->ops->bss_info_changed)
J
Johannes Berg 已提交
82 83
		local->ops->bss_info_changed(&local->hw, &sdata->vif, info, changed);
	trace_drv_bss_info_changed(local, sdata, info, changed);
84 85
}

86
static inline u64 drv_prepare_multicast(struct ieee80211_local *local,
87
					struct netdev_hw_addr_list *mc_list)
88 89 90 91
{
	u64 ret = 0;

	if (local->ops->prepare_multicast)
92
		ret = local->ops->prepare_multicast(&local->hw, mc_list);
93

94
	trace_drv_prepare_multicast(local, mc_list->count, ret);
95 96 97 98

	return ret;
}

99 100 101
static inline void drv_configure_filter(struct ieee80211_local *local,
					unsigned int changed_flags,
					unsigned int *total_flags,
102
					u64 multicast)
103
{
104 105
	might_sleep();

106
	local->ops->configure_filter(&local->hw, changed_flags, total_flags,
107
				     multicast);
108
	trace_drv_configure_filter(local, changed_flags, total_flags,
109
				   multicast);
110 111 112 113 114
}

static inline int drv_set_tim(struct ieee80211_local *local,
			      struct ieee80211_sta *sta, bool set)
{
115
	int ret = 0;
116
	if (local->ops->set_tim)
117 118 119
		ret = local->ops->set_tim(&local->hw, sta, set);
	trace_drv_set_tim(local, sta, set, ret);
	return ret;
120 121 122
}

static inline int drv_set_key(struct ieee80211_local *local,
J
Johannes Berg 已提交
123 124
			      enum set_key_cmd cmd,
			      struct ieee80211_sub_if_data *sdata,
125 126 127
			      struct ieee80211_sta *sta,
			      struct ieee80211_key_conf *key)
{
128 129 130 131 132
	int ret;

	might_sleep();

	ret = local->ops->set_key(&local->hw, cmd, &sdata->vif, sta, key);
J
Johannes Berg 已提交
133
	trace_drv_set_key(local, cmd, sdata, sta, key, ret);
134
	return ret;
135 136 137
}

static inline void drv_update_tkip_key(struct ieee80211_local *local,
138
				       struct ieee80211_sub_if_data *sdata,
139
				       struct ieee80211_key_conf *conf,
140
				       struct sta_info *sta, u32 iv32,
141 142
				       u16 *phase1key)
{
143 144 145 146 147
	struct ieee80211_sta *ista = NULL;

	if (sta)
		ista = &sta->sta;

148
	if (local->ops->update_tkip_key)
149 150 151
		local->ops->update_tkip_key(&local->hw, &sdata->vif, conf,
					    ista, iv32, phase1key);
	trace_drv_update_tkip_key(local, sdata, conf, ista, iv32);
152 153 154
}

static inline int drv_hw_scan(struct ieee80211_local *local,
155
			      struct ieee80211_sub_if_data *sdata,
156 157
			      struct cfg80211_scan_request *req)
{
158 159 160 161
	int ret;

	might_sleep();

162 163
	ret = local->ops->hw_scan(&local->hw, &sdata->vif, req);
	trace_drv_hw_scan(local, sdata, req, ret);
164
	return ret;
165 166 167 168
}

static inline void drv_sw_scan_start(struct ieee80211_local *local)
{
169 170
	might_sleep();

171 172
	if (local->ops->sw_scan_start)
		local->ops->sw_scan_start(&local->hw);
173
	trace_drv_sw_scan_start(local);
174 175 176 177
}

static inline void drv_sw_scan_complete(struct ieee80211_local *local)
{
178 179
	might_sleep();

180 181
	if (local->ops->sw_scan_complete)
		local->ops->sw_scan_complete(&local->hw);
182
	trace_drv_sw_scan_complete(local);
183 184 185 186 187
}

static inline int drv_get_stats(struct ieee80211_local *local,
				struct ieee80211_low_level_stats *stats)
{
188 189
	int ret = -EOPNOTSUPP;

190 191
	might_sleep();

192 193 194 195 196
	if (local->ops->get_stats)
		ret = local->ops->get_stats(&local->hw, stats);
	trace_drv_get_stats(local, stats, ret);

	return ret;
197 198 199 200 201 202 203
}

static inline void drv_get_tkip_seq(struct ieee80211_local *local,
				    u8 hw_key_idx, u32 *iv32, u16 *iv16)
{
	if (local->ops->get_tkip_seq)
		local->ops->get_tkip_seq(&local->hw, hw_key_idx, iv32, iv16);
204
	trace_drv_get_tkip_seq(local, hw_key_idx, iv32, iv16);
205 206 207 208 209
}

static inline int drv_set_rts_threshold(struct ieee80211_local *local,
					u32 value)
{
210
	int ret = 0;
211 212 213

	might_sleep();

214
	if (local->ops->set_rts_threshold)
215 216 217
		ret = local->ops->set_rts_threshold(&local->hw, value);
	trace_drv_set_rts_threshold(local, value, ret);
	return ret;
218 219
}

220 221 222 223 224 225 226 227 228 229 230 231 232 233 234
static inline int drv_set_coverage_class(struct ieee80211_local *local,
					 u8 value)
{
	int ret = 0;
	might_sleep();

	if (local->ops->set_coverage_class)
		local->ops->set_coverage_class(&local->hw, value);
	else
		ret = -EOPNOTSUPP;

	trace_drv_set_coverage_class(local, value, ret);
	return ret;
}

235
static inline void drv_sta_notify(struct ieee80211_local *local,
J
Johannes Berg 已提交
236
				  struct ieee80211_sub_if_data *sdata,
237 238 239 240
				  enum sta_notify_cmd cmd,
				  struct ieee80211_sta *sta)
{
	if (local->ops->sta_notify)
J
Johannes Berg 已提交
241 242
		local->ops->sta_notify(&local->hw, &sdata->vif, cmd, sta);
	trace_drv_sta_notify(local, sdata, cmd, sta);
243 244
}

245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
static inline int drv_sta_add(struct ieee80211_local *local,
			      struct ieee80211_sub_if_data *sdata,
			      struct ieee80211_sta *sta)
{
	int ret = 0;

	might_sleep();

	if (local->ops->sta_add)
		ret = local->ops->sta_add(&local->hw, &sdata->vif, sta);
	else if (local->ops->sta_notify)
		local->ops->sta_notify(&local->hw, &sdata->vif,
					STA_NOTIFY_ADD, sta);

	trace_drv_sta_add(local, sdata, sta, ret);

	return ret;
}

static inline void drv_sta_remove(struct ieee80211_local *local,
				  struct ieee80211_sub_if_data *sdata,
				  struct ieee80211_sta *sta)
{
	might_sleep();

	if (local->ops->sta_remove)
		local->ops->sta_remove(&local->hw, &sdata->vif, sta);
	else if (local->ops->sta_notify)
		local->ops->sta_notify(&local->hw, &sdata->vif,
					STA_NOTIFY_REMOVE, sta);

	trace_drv_sta_remove(local, sdata, sta);
}

279 280 281
static inline int drv_conf_tx(struct ieee80211_local *local, u16 queue,
			      const struct ieee80211_tx_queue_params *params)
{
282
	int ret = -EOPNOTSUPP;
283 284 285

	might_sleep();

286
	if (local->ops->conf_tx)
287 288 289
		ret = local->ops->conf_tx(&local->hw, queue, params);
	trace_drv_conf_tx(local, queue, params, ret);
	return ret;
290 291 292 293
}

static inline u64 drv_get_tsf(struct ieee80211_local *local)
{
294
	u64 ret = -1ULL;
295 296 297

	might_sleep();

298
	if (local->ops->get_tsf)
299 300 301
		ret = local->ops->get_tsf(&local->hw);
	trace_drv_get_tsf(local, ret);
	return ret;
302 303 304 305
}

static inline void drv_set_tsf(struct ieee80211_local *local, u64 tsf)
{
306 307
	might_sleep();

308 309
	if (local->ops->set_tsf)
		local->ops->set_tsf(&local->hw, tsf);
310
	trace_drv_set_tsf(local, tsf);
311 312 313 314
}

static inline void drv_reset_tsf(struct ieee80211_local *local)
{
315 316
	might_sleep();

317 318
	if (local->ops->reset_tsf)
		local->ops->reset_tsf(&local->hw);
319
	trace_drv_reset_tsf(local);
320 321 322 323
}

static inline int drv_tx_last_beacon(struct ieee80211_local *local)
{
324
	int ret = 1;
325 326 327

	might_sleep();

328
	if (local->ops->tx_last_beacon)
329 330 331
		ret = local->ops->tx_last_beacon(&local->hw);
	trace_drv_tx_last_beacon(local, ret);
	return ret;
332 333 334
}

static inline int drv_ampdu_action(struct ieee80211_local *local,
J
Johannes Berg 已提交
335
				   struct ieee80211_sub_if_data *sdata,
336 337 338 339
				   enum ieee80211_ampdu_mlme_action action,
				   struct ieee80211_sta *sta, u16 tid,
				   u16 *ssn)
{
340
	int ret = -EOPNOTSUPP;
341
	if (local->ops->ampdu_action)
J
Johannes Berg 已提交
342
		ret = local->ops->ampdu_action(&local->hw, &sdata->vif, action,
343
					       sta, tid, ssn);
J
Johannes Berg 已提交
344
	trace_drv_ampdu_action(local, sdata, action, sta, tid, ssn, ret);
345
	return ret;
346
}
J
Johannes Berg 已提交
347

348 349 350 351 352 353 354 355 356
static inline int drv_get_survey(struct ieee80211_local *local, int idx,
				struct survey_info *survey)
{
	int ret = -EOPNOTSUPP;
	if (local->ops->conf_tx)
		ret = local->ops->get_survey(&local->hw, idx, survey);
	/* trace_drv_get_survey(local, idx, survey, ret); */
	return ret;
}
J
Johannes Berg 已提交
357 358 359

static inline void drv_rfkill_poll(struct ieee80211_local *local)
{
360 361
	might_sleep();

J
Johannes Berg 已提交
362 363 364
	if (local->ops->rfkill_poll)
		local->ops->rfkill_poll(&local->hw);
}
365 366 367

static inline void drv_flush(struct ieee80211_local *local, bool drop)
{
368 369
	might_sleep();

370 371 372 373
	trace_drv_flush(local, drop);
	if (local->ops->flush)
		local->ops->flush(&local->hw, drop);
}
374 375 376 377 378 379 380 381 382 383 384

static inline void drv_channel_switch(struct ieee80211_local *local,
				     struct ieee80211_channel_switch *ch_switch)
{
	might_sleep();

	local->ops->channel_switch(&local->hw, ch_switch);

	trace_drv_channel_switch(local, ch_switch);
}

385
#endif /* __MAC80211_DRIVER_OPS */