htc_drv_main.c 45.2 KB
Newer Older
S
Sujith 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
/*
 * Copyright (c) 2010 Atheros Communications Inc.
 *
 * Permission to use, copy, modify, and/or distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */

#include "htc.h"

#ifdef CONFIG_ATH9K_HTC_DEBUGFS
static struct dentry *ath9k_debugfs_root;
#endif

/*************/
/* Utilities */
/*************/

/* HACK Alert: Use 11NG for 2.4, use 11NA for 5 */
static enum htc_phymode ath9k_htc_get_curmode(struct ath9k_htc_priv *priv,
					      struct ath9k_channel *ichan)
{
	enum htc_phymode mode;

	mode = HTC_MODE_AUTO;

	switch (ichan->chanmode) {
	case CHANNEL_G:
	case CHANNEL_G_HT20:
	case CHANNEL_G_HT40PLUS:
	case CHANNEL_G_HT40MINUS:
		mode = HTC_MODE_11NG;
		break;
	case CHANNEL_A:
	case CHANNEL_A_HT20:
	case CHANNEL_A_HT40PLUS:
	case CHANNEL_A_HT40MINUS:
		mode = HTC_MODE_11NA;
		break;
	default:
		break;
	}

	return mode;
}

S
Sujith Manoharan 已提交
55 56
bool ath9k_htc_setpower(struct ath9k_htc_priv *priv,
			enum ath9k_power_mode mode)
57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
{
	bool ret;

	mutex_lock(&priv->htc_pm_lock);
	ret = ath9k_hw_setpower(priv->ah, mode);
	mutex_unlock(&priv->htc_pm_lock);

	return ret;
}

void ath9k_htc_ps_wakeup(struct ath9k_htc_priv *priv)
{
	mutex_lock(&priv->htc_pm_lock);
	if (++priv->ps_usecount != 1)
		goto unlock;
	ath9k_hw_setpower(priv->ah, ATH9K_PM_AWAKE);

unlock:
	mutex_unlock(&priv->htc_pm_lock);
}

void ath9k_htc_ps_restore(struct ath9k_htc_priv *priv)
{
	mutex_lock(&priv->htc_pm_lock);
	if (--priv->ps_usecount != 0)
		goto unlock;

84 85 86
	if (priv->ps_idle)
		ath9k_hw_setpower(priv->ah, ATH9K_PM_FULL_SLEEP);
	else if (priv->ps_enabled)
87
		ath9k_hw_setpower(priv->ah, ATH9K_PM_NETWORK_SLEEP);
88

89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
unlock:
	mutex_unlock(&priv->htc_pm_lock);
}

void ath9k_ps_work(struct work_struct *work)
{
	struct ath9k_htc_priv *priv =
		container_of(work, struct ath9k_htc_priv,
			     ps_work);
	ath9k_htc_setpower(priv, ATH9K_PM_AWAKE);

	/* The chip wakes up after receiving the first beacon
	   while network sleep is enabled. For the driver to
	   be in sync with the hw, set the chip to awake and
	   only then set it to sleep.
	 */
	ath9k_htc_setpower(priv, ATH9K_PM_NETWORK_SLEEP);
}

108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
static void ath9k_htc_vif_iter(void *data, u8 *mac, struct ieee80211_vif *vif)
{
	struct ath9k_htc_priv *priv = data;
	struct ieee80211_bss_conf *bss_conf = &vif->bss_conf;

	if (bss_conf->assoc) {
		priv->rearm_ani = true;
		priv->reconfig_beacon = true;
	}
}

static void ath9k_htc_vif_reconfig(struct ath9k_htc_priv *priv)
{
	priv->rearm_ani = false;
	priv->reconfig_beacon = false;

	ieee80211_iterate_active_interfaces_atomic(priv->hw,
						   ath9k_htc_vif_iter, priv);
	if (priv->rearm_ani)
127
		ath9k_htc_start_ani(priv);
128 129 130 131 132 133 134 135

	if (priv->reconfig_beacon) {
		ath9k_htc_ps_wakeup(priv);
		ath9k_htc_beacon_reconfig(priv);
		ath9k_htc_ps_restore(priv);
	}
}

136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168
static void ath9k_htc_bssid_iter(void *data, u8 *mac, struct ieee80211_vif *vif)
{
	struct ath9k_vif_iter_data *iter_data = data;
	int i;

	for (i = 0; i < ETH_ALEN; i++)
		iter_data->mask[i] &= ~(iter_data->hw_macaddr[i] ^ mac[i]);
}

static void ath9k_htc_set_bssid_mask(struct ath9k_htc_priv *priv,
				     struct ieee80211_vif *vif)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_vif_iter_data iter_data;

	/*
	 * Use the hardware MAC address as reference, the hardware uses it
	 * together with the BSSID mask when matching addresses.
	 */
	iter_data.hw_macaddr = common->macaddr;
	memset(&iter_data.mask, 0xff, ETH_ALEN);

	if (vif)
		ath9k_htc_bssid_iter(&iter_data, vif->addr, vif);

	/* Get list of all active MAC addresses */
	ieee80211_iterate_active_interfaces_atomic(priv->hw, ath9k_htc_bssid_iter,
						   &iter_data);

	memcpy(common->bssidmask, iter_data.mask, ETH_ALEN);
	ath_hw_setbssidmask(common);
}

169 170 171 172 173 174 175 176 177 178 179 180
static void ath9k_htc_set_opmode(struct ath9k_htc_priv *priv)
{
	if (priv->num_ibss_vif)
		priv->ah->opmode = NL80211_IFTYPE_ADHOC;
	else if (priv->num_ap_vif)
		priv->ah->opmode = NL80211_IFTYPE_AP;
	else
		priv->ah->opmode = NL80211_IFTYPE_STATION;

	ath9k_hw_setopmode(priv->ah);
}

181 182 183 184 185
void ath9k_htc_reset(struct ath9k_htc_priv *priv)
{
	struct ath_hw *ah = priv->ah;
	struct ath_common *common = ath9k_hw_common(ah);
	struct ieee80211_channel *channel = priv->hw->conf.channel;
186
	struct ath9k_hw_cal_data *caldata = NULL;
187 188 189 190 191 192 193 194
	enum htc_phymode mode;
	__be16 htc_mode;
	u8 cmd_rsp;
	int ret;

	mutex_lock(&priv->mutex);
	ath9k_htc_ps_wakeup(priv);

195
	ath9k_htc_stop_ani(priv);
196 197 198 199 200 201
	ieee80211_stop_queues(priv->hw);
	htc_stop(priv->htc);
	WMI_CMD(WMI_DISABLE_INTR_CMDID);
	WMI_CMD(WMI_DRAIN_TXQ_ALL_CMDID);
	WMI_CMD(WMI_STOP_RECV_CMDID);

202
	caldata = &priv->caldata;
203 204 205 206 207 208 209
	ret = ath9k_hw_reset(ah, ah->curchan, caldata, false);
	if (ret) {
		ath_err(common,
			"Unable to reset device (%u Mhz) reset status %d\n",
			channel->center_freq, ret);
	}

210 211
	ath9k_cmn_update_txpow(ah, priv->curtxpow, priv->txpowlimit,
			       &priv->curtxpow);
212 213 214 215 216 217 218 219 220 221

	WMI_CMD(WMI_START_RECV_CMDID);
	ath9k_host_rx_init(priv);

	mode = ath9k_htc_get_curmode(priv, ah->curchan);
	htc_mode = cpu_to_be16(mode);
	WMI_CMD_BUF(WMI_SET_MODE_CMDID, &htc_mode);

	WMI_CMD(WMI_ENABLE_INTR_CMDID);
	htc_start(priv->htc);
222
	ath9k_htc_vif_reconfig(priv);
223 224 225 226 227 228
	ieee80211_wake_queues(priv->hw);

	ath9k_htc_ps_restore(priv);
	mutex_unlock(&priv->mutex);
}

S
Sujith 已提交
229 230 231 232 233 234 235
static int ath9k_htc_set_channel(struct ath9k_htc_priv *priv,
				 struct ieee80211_hw *hw,
				 struct ath9k_channel *hchan)
{
	struct ath_hw *ah = priv->ah;
	struct ath_common *common = ath9k_hw_common(ah);
	struct ieee80211_conf *conf = &common->hw->conf;
236
	bool fastcc;
S
Sujith 已提交
237
	struct ieee80211_channel *channel = hw->conf.channel;
238
	struct ath9k_hw_cal_data *caldata = NULL;
S
Sujith 已提交
239
	enum htc_phymode mode;
S
Sujith 已提交
240
	__be16 htc_mode;
S
Sujith 已提交
241 242 243 244 245 246
	u8 cmd_rsp;
	int ret;

	if (priv->op_flags & OP_INVALID)
		return -EIO;

247
	fastcc = !!(hw->conf.flags & IEEE80211_CONF_OFFCHANNEL);
S
Sujith 已提交
248

249
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
250 251 252 253 254
	htc_stop(priv->htc);
	WMI_CMD(WMI_DISABLE_INTR_CMDID);
	WMI_CMD(WMI_DRAIN_TXQ_ALL_CMDID);
	WMI_CMD(WMI_STOP_RECV_CMDID);

J
Joe Perches 已提交
255 256 257 258 259
	ath_dbg(common, ATH_DBG_CONFIG,
		"(%u MHz) -> (%u MHz), HT: %d, HT40: %d fastcc: %d\n",
		priv->ah->curchan->channel,
		channel->center_freq, conf_is_ht(conf), conf_is_ht40(conf),
		fastcc);
S
Sujith 已提交
260

261 262
	if (!fastcc)
		caldata = &priv->caldata;
263
	ret = ath9k_hw_reset(ah, hchan, caldata, fastcc);
S
Sujith 已提交
264
	if (ret) {
265 266 267
		ath_err(common,
			"Unable to reset channel (%u Mhz) reset status %d\n",
			channel->center_freq, ret);
S
Sujith 已提交
268 269 270
		goto err;
	}

271 272
	ath9k_cmn_update_txpow(ah, priv->curtxpow, priv->txpowlimit,
			       &priv->curtxpow);
S
Sujith 已提交
273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291

	WMI_CMD(WMI_START_RECV_CMDID);
	if (ret)
		goto err;

	ath9k_host_rx_init(priv);

	mode = ath9k_htc_get_curmode(priv, hchan);
	htc_mode = cpu_to_be16(mode);
	WMI_CMD_BUF(WMI_SET_MODE_CMDID, &htc_mode);
	if (ret)
		goto err;

	WMI_CMD(WMI_ENABLE_INTR_CMDID);
	if (ret)
		goto err;

	htc_start(priv->htc);
err:
292
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
293 294 295
	return ret;
}

296 297 298 299 300 301 302
/*
 * Monitor mode handling is a tad complicated because the firmware requires
 * an interface to be created exclusively, while mac80211 doesn't associate
 * an interface with the mode.
 *
 * So, for now, only one monitor interface can be configured.
 */
303 304 305 306 307 308 309 310 311
static void __ath9k_htc_remove_monitor_interface(struct ath9k_htc_priv *priv)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_target_vif hvif;
	int ret = 0;
	u8 cmd_rsp;

	memset(&hvif, 0, sizeof(struct ath9k_htc_target_vif));
	memcpy(&hvif.myaddr, common->macaddr, ETH_ALEN);
312
	hvif.index = priv->mon_vif_idx;
313 314
	WMI_CMD_BUF(WMI_VAP_REMOVE_CMDID, &hvif);
	priv->nvifs--;
315
	priv->vif_slot &= ~(1 << priv->mon_vif_idx);
316 317
}

318 319 320 321
static int ath9k_htc_add_monitor_interface(struct ath9k_htc_priv *priv)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_target_vif hvif;
322
	struct ath9k_htc_target_sta tsta;
323
	int ret = 0, sta_idx;
324 325
	u8 cmd_rsp;

326 327 328 329 330
	if ((priv->nvifs >= ATH9K_HTC_MAX_VIF) ||
	    (priv->nstations >= ATH9K_HTC_MAX_STA)) {
		ret = -ENOBUFS;
		goto err_vif;
	}
331

332 333 334 335 336
	sta_idx = ffz(priv->sta_slot);
	if ((sta_idx < 0) || (sta_idx > ATH9K_HTC_MAX_STA)) {
		ret = -ENOBUFS;
		goto err_vif;
	}
337 338 339 340

	/*
	 * Add an interface.
	 */
341 342 343 344
	memset(&hvif, 0, sizeof(struct ath9k_htc_target_vif));
	memcpy(&hvif.myaddr, common->macaddr, ETH_ALEN);

	hvif.opmode = cpu_to_be32(HTC_M_MONITOR);
345
	hvif.index = ffz(priv->vif_slot);
346 347 348

	WMI_CMD_BUF(WMI_VAP_CREATE_CMDID, &hvif);
	if (ret)
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363
		goto err_vif;

	/*
	 * Assign the monitor interface index as a special case here.
	 * This is needed when the interface is brought down.
	 */
	priv->mon_vif_idx = hvif.index;
	priv->vif_slot |= (1 << hvif.index);

	/*
	 * Set the hardware mode to monitor only if there are no
	 * other interfaces.
	 */
	if (!priv->nvifs)
		priv->ah->opmode = NL80211_IFTYPE_MONITOR;
364 365

	priv->nvifs++;
366 367 368 369 370 371 372 373 374

	/*
	 * Associate a station with the interface for packet injection.
	 */
	memset(&tsta, 0, sizeof(struct ath9k_htc_target_sta));

	memcpy(&tsta.macaddr, common->macaddr, ETH_ALEN);

	tsta.is_vif_sta = 1;
375
	tsta.sta_index = sta_idx;
376 377 378 379 380 381
	tsta.vif_index = hvif.index;
	tsta.maxampdu = 0xffff;

	WMI_CMD_BUF(WMI_NODE_CREATE_CMDID, &tsta);
	if (ret) {
		ath_err(common, "Unable to add station entry for monitor mode\n");
382
		goto err_sta;
383 384
	}

385
	priv->sta_slot |= (1 << sta_idx);
386
	priv->nstations++;
387
	priv->vif_sta_pos[priv->mon_vif_idx] = sta_idx;
388 389
	priv->ah->is_monitoring = true;

390 391 392 393
	ath_dbg(common, ATH_DBG_CONFIG,
		"Attached a monitor interface at idx: %d, sta idx: %d\n",
		priv->mon_vif_idx, sta_idx);

394
	return 0;
395

396
err_sta:
397 398 399 400
	/*
	 * Remove the interface from the target.
	 */
	__ath9k_htc_remove_monitor_interface(priv);
401 402 403
err_vif:
	ath_dbg(common, ATH_DBG_FATAL, "Unable to attach a monitor interface\n");

404
	return ret;
405 406 407 408 409 410
}

static int ath9k_htc_remove_monitor_interface(struct ath9k_htc_priv *priv)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	int ret = 0;
411
	u8 cmd_rsp, sta_idx;
412

413
	__ath9k_htc_remove_monitor_interface(priv);
414

415
	sta_idx = priv->vif_sta_pos[priv->mon_vif_idx];
416 417 418 419 420 421 422

	WMI_CMD_BUF(WMI_NODE_REMOVE_CMDID, &sta_idx);
	if (ret) {
		ath_err(common, "Unable to remove station entry for monitor mode\n");
		return ret;
	}

423
	priv->sta_slot &= ~(1 << sta_idx);
424
	priv->nstations--;
425
	priv->ah->is_monitoring = false;
426

427 428 429 430
	ath_dbg(common, ATH_DBG_CONFIG,
		"Removed a monitor interface at idx: %d, sta idx: %d\n",
		priv->mon_vif_idx, sta_idx);

431
	return 0;
432 433
}

S
Sujith 已提交
434 435 436 437 438 439 440 441
static int ath9k_htc_add_station(struct ath9k_htc_priv *priv,
				 struct ieee80211_vif *vif,
				 struct ieee80211_sta *sta)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_target_sta tsta;
	struct ath9k_htc_vif *avp = (struct ath9k_htc_vif *) vif->drv_priv;
	struct ath9k_htc_sta *ista;
442
	int ret, sta_idx;
S
Sujith 已提交
443 444 445 446 447
	u8 cmd_rsp;

	if (priv->nstations >= ATH9K_HTC_MAX_STA)
		return -ENOBUFS;

448 449 450 451
	sta_idx = ffz(priv->sta_slot);
	if ((sta_idx < 0) || (sta_idx > ATH9K_HTC_MAX_STA))
		return -ENOBUFS;

S
Sujith 已提交
452 453 454 455 456 457 458 459 460
	memset(&tsta, 0, sizeof(struct ath9k_htc_target_sta));

	if (sta) {
		ista = (struct ath9k_htc_sta *) sta->drv_priv;
		memcpy(&tsta.macaddr, sta->addr, ETH_ALEN);
		memcpy(&tsta.bssid, common->curbssid, ETH_ALEN);
		tsta.associd = common->curaid;
		tsta.is_vif_sta = 0;
		tsta.valid = true;
461
		ista->index = sta_idx;
S
Sujith 已提交
462 463 464 465 466
	} else {
		memcpy(&tsta.macaddr, vif->addr, ETH_ALEN);
		tsta.is_vif_sta = 1;
	}

467
	tsta.sta_index = sta_idx;
S
Sujith 已提交
468 469 470 471 472 473 474 475
	tsta.vif_index = avp->index;
	tsta.maxampdu = 0xffff;
	if (sta && sta->ht_cap.ht_supported)
		tsta.flags = cpu_to_be16(ATH_HTC_STA_HT);

	WMI_CMD_BUF(WMI_NODE_CREATE_CMDID, &tsta);
	if (ret) {
		if (sta)
476 477 478
			ath_err(common,
				"Unable to add station entry for: %pM\n",
				sta->addr);
S
Sujith 已提交
479 480 481
		return ret;
	}

482
	if (sta) {
J
Joe Perches 已提交
483 484 485
		ath_dbg(common, ATH_DBG_CONFIG,
			"Added a station entry for: %pM (idx: %d)\n",
			sta->addr, tsta.sta_index);
486 487 488 489 490
	} else {
		ath_dbg(common, ATH_DBG_CONFIG,
			"Added a station entry for VIF %d (idx: %d)\n",
			avp->index, tsta.sta_index);
	}
S
Sujith 已提交
491

492
	priv->sta_slot |= (1 << sta_idx);
S
Sujith 已提交
493
	priv->nstations++;
494 495 496
	if (!sta)
		priv->vif_sta_pos[avp->index] = sta_idx;

S
Sujith 已提交
497 498 499 500 501 502 503 504
	return 0;
}

static int ath9k_htc_remove_station(struct ath9k_htc_priv *priv,
				    struct ieee80211_vif *vif,
				    struct ieee80211_sta *sta)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
505
	struct ath9k_htc_vif *avp = (struct ath9k_htc_vif *) vif->drv_priv;
S
Sujith 已提交
506 507 508 509 510 511 512 513
	struct ath9k_htc_sta *ista;
	int ret;
	u8 cmd_rsp, sta_idx;

	if (sta) {
		ista = (struct ath9k_htc_sta *) sta->drv_priv;
		sta_idx = ista->index;
	} else {
514
		sta_idx = priv->vif_sta_pos[avp->index];
S
Sujith 已提交
515 516 517 518 519
	}

	WMI_CMD_BUF(WMI_NODE_REMOVE_CMDID, &sta_idx);
	if (ret) {
		if (sta)
520 521 522
			ath_err(common,
				"Unable to remove station entry for: %pM\n",
				sta->addr);
S
Sujith 已提交
523 524 525
		return ret;
	}

526
	if (sta) {
J
Joe Perches 已提交
527 528 529
		ath_dbg(common, ATH_DBG_CONFIG,
			"Removed a station entry for: %pM (idx: %d)\n",
			sta->addr, sta_idx);
530 531 532 533 534
	} else {
		ath_dbg(common, ATH_DBG_CONFIG,
			"Removed a station entry for VIF %d (idx: %d)\n",
			avp->index, sta_idx);
	}
S
Sujith 已提交
535

536
	priv->sta_slot &= ~(1 << sta_idx);
S
Sujith 已提交
537
	priv->nstations--;
538

S
Sujith 已提交
539 540 541
	return 0;
}

542
int ath9k_htc_update_cap_target(struct ath9k_htc_priv *priv)
S
Sujith 已提交
543 544 545 546 547 548 549 550 551 552 553 554
{
	struct ath9k_htc_cap_target tcap;
	int ret;
	u8 cmd_rsp;

	memset(&tcap, 0, sizeof(struct ath9k_htc_cap_target));

	/* FIXME: Values are hardcoded */
	tcap.flags = 0x240c40;
	tcap.flags_ext = 0x80601000;
	tcap.ampdu_limit = 0xffff0000;
	tcap.ampdu_subframes = 20;
S
Sujith 已提交
555
	tcap.tx_chainmask_legacy = priv->ah->caps.tx_chainmask;
S
Sujith 已提交
556
	tcap.protmode = 1;
S
Sujith 已提交
557
	tcap.tx_chainmask = priv->ah->caps.tx_chainmask;
S
Sujith 已提交
558 559 560 561 562 563

	WMI_CMD_BUF(WMI_TARGET_IC_UPDATE_CMDID, &tcap);

	return ret;
}

S
Sujith 已提交
564 565 566
static void ath9k_htc_setup_rate(struct ath9k_htc_priv *priv,
				 struct ieee80211_sta *sta,
				 struct ath9k_htc_target_rate *trate)
S
Sujith 已提交
567 568 569 570
{
	struct ath9k_htc_sta *ista = (struct ath9k_htc_sta *) sta->drv_priv;
	struct ieee80211_supported_band *sband;
	u32 caps = 0;
S
Sujith 已提交
571
	int i, j;
S
Sujith 已提交
572

S
Sujith 已提交
573
	sband = priv->hw->wiphy->bands[priv->hw->conf.channel->band];
S
Sujith 已提交
574 575 576

	for (i = 0, j = 0; i < sband->n_bitrates; i++) {
		if (sta->supp_rates[sband->band] & BIT(i)) {
S
Sujith 已提交
577
			trate->rates.legacy_rates.rs_rates[j]
S
Sujith 已提交
578 579 580 581
				= (sband->bitrates[i].bitrate * 2) / 10;
			j++;
		}
	}
S
Sujith 已提交
582
	trate->rates.legacy_rates.rs_nrates = j;
S
Sujith 已提交
583 584 585 586

	if (sta->ht_cap.ht_supported) {
		for (i = 0, j = 0; i < 77; i++) {
			if (sta->ht_cap.mcs.rx_mask[i/8] & (1<<(i%8)))
S
Sujith 已提交
587
				trate->rates.ht_rates.rs_rates[j++] = i;
S
Sujith 已提交
588 589 590
			if (j == ATH_HTC_RATE_MAX)
				break;
		}
S
Sujith 已提交
591
		trate->rates.ht_rates.rs_nrates = j;
S
Sujith 已提交
592 593

		caps = WLAN_RC_HT_FLAG;
594 595
		if (sta->ht_cap.mcs.rx_mask[1])
			caps |= WLAN_RC_DS_FLAG;
596 597
		if ((sta->ht_cap.cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40) &&
		     (conf_is_ht40(&priv->hw->conf)))
S
Sujith 已提交
598
			caps |= WLAN_RC_40_FLAG;
599 600 601 602 603
		if (conf_is_ht40(&priv->hw->conf) &&
		    (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40))
			caps |= WLAN_RC_SGI_FLAG;
		else if (conf_is_ht20(&priv->hw->conf) &&
			 (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20))
S
Sujith 已提交
604 605 606
			caps |= WLAN_RC_SGI_FLAG;
	}

S
Sujith 已提交
607 608 609 610 611 612 613 614 615 616 617
	trate->sta_index = ista->index;
	trate->isnew = 1;
	trate->capflags = cpu_to_be32(caps);
}

static int ath9k_htc_send_rate_cmd(struct ath9k_htc_priv *priv,
				    struct ath9k_htc_target_rate *trate)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	int ret;
	u8 cmd_rsp;
S
Sujith 已提交
618

S
Sujith 已提交
619
	WMI_CMD_BUF(WMI_RC_RATE_UPDATE_CMDID, trate);
S
Sujith 已提交
620
	if (ret) {
621 622
		ath_err(common,
			"Unable to initialize Rate information on target\n");
S
Sujith 已提交
623 624
	}

S
Sujith 已提交
625
	return ret;
S
Sujith 已提交
626 627
}

S
Sujith 已提交
628 629
static void ath9k_htc_init_rate(struct ath9k_htc_priv *priv,
				struct ieee80211_sta *sta)
S
Sujith 已提交
630 631
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
S
Sujith 已提交
632
	struct ath9k_htc_target_rate trate;
S
Sujith 已提交
633 634
	int ret;

S
Sujith 已提交
635 636 637 638
	memset(&trate, 0, sizeof(struct ath9k_htc_target_rate));
	ath9k_htc_setup_rate(priv, sta, &trate);
	ret = ath9k_htc_send_rate_cmd(priv, &trate);
	if (!ret)
J
Joe Perches 已提交
639 640 641
		ath_dbg(common, ATH_DBG_CONFIG,
			"Updated target sta: %pM, rate caps: 0x%X\n",
			sta->addr, be32_to_cpu(trate.capflags));
S
Sujith 已提交
642 643
}

644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665
static void ath9k_htc_update_rate(struct ath9k_htc_priv *priv,
				  struct ieee80211_vif *vif,
				  struct ieee80211_bss_conf *bss_conf)
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_target_rate trate;
	struct ieee80211_sta *sta;
	int ret;

	memset(&trate, 0, sizeof(struct ath9k_htc_target_rate));

	rcu_read_lock();
	sta = ieee80211_find_sta(vif, bss_conf->bssid);
	if (!sta) {
		rcu_read_unlock();
		return;
	}
	ath9k_htc_setup_rate(priv, sta, &trate);
	rcu_read_unlock();

	ret = ath9k_htc_send_rate_cmd(priv, &trate);
	if (!ret)
J
Joe Perches 已提交
666 667 668
		ath_dbg(common, ATH_DBG_CONFIG,
			"Updated target sta: %pM, rate caps: 0x%X\n",
			bss_conf->bssid, be32_to_cpu(trate.capflags));
669 670
}

671 672 673 674 675
static int ath9k_htc_tx_aggr_oper(struct ath9k_htc_priv *priv,
				  struct ieee80211_vif *vif,
				  struct ieee80211_sta *sta,
				  enum ieee80211_ampdu_mlme_action action,
				  u16 tid)
S
Sujith 已提交
676 677 678
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_target_aggr aggr;
679
	struct ath9k_htc_sta *ista;
S
Sujith 已提交
680 681 682
	int ret = 0;
	u8 cmd_rsp;

683
	if (tid >= ATH9K_HTC_MAX_TID)
S
Sujith 已提交
684 685
		return -EINVAL;

686 687
	memset(&aggr, 0, sizeof(struct ath9k_htc_target_aggr));
	ista = (struct ath9k_htc_sta *) sta->drv_priv;
S
Sujith 已提交
688 689

	aggr.sta_index = ista->index;
S
Sujith 已提交
690 691
	aggr.tidno = tid & 0xf;
	aggr.aggr_enable = (action == IEEE80211_AMPDU_TX_START) ? true : false;
S
Sujith 已提交
692 693 694

	WMI_CMD_BUF(WMI_TX_AGGR_ENABLE_CMDID, &aggr);
	if (ret)
J
Joe Perches 已提交
695 696 697
		ath_dbg(common, ATH_DBG_CONFIG,
			"Unable to %s TX aggregation for (%pM, %d)\n",
			(aggr.aggr_enable) ? "start" : "stop", sta->addr, tid);
S
Sujith 已提交
698
	else
J
Joe Perches 已提交
699 700 701 702
		ath_dbg(common, ATH_DBG_CONFIG,
			"%s TX aggregation for (%pM, %d)\n",
			(aggr.aggr_enable) ? "Starting" : "Stopping",
			sta->addr, tid);
S
Sujith 已提交
703

S
Sujith 已提交
704 705 706
	spin_lock_bh(&priv->tx_lock);
	ista->tid_state[tid] = (aggr.aggr_enable && !ret) ? AGGR_START : AGGR_STOP;
	spin_unlock_bh(&priv->tx_lock);
S
Sujith 已提交
707

S
Sujith 已提交
708
	return ret;
S
Sujith 已提交
709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725
}

/*********/
/* DEBUG */
/*********/

#ifdef CONFIG_ATH9K_HTC_DEBUGFS

static int ath9k_debugfs_open(struct inode *inode, struct file *file)
{
	file->private_data = inode->i_private;
	return 0;
}

static ssize_t read_file_tgt_stats(struct file *file, char __user *user_buf,
				   size_t count, loff_t *ppos)
{
726
	struct ath9k_htc_priv *priv = file->private_data;
S
Sujith 已提交
727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756
	struct ath9k_htc_target_stats cmd_rsp;
	char buf[512];
	unsigned int len = 0;
	int ret = 0;

	memset(&cmd_rsp, 0, sizeof(cmd_rsp));

	WMI_CMD(WMI_TGT_STATS_CMDID);
	if (ret)
		return -EINVAL;


	len += snprintf(buf + len, sizeof(buf) - len,
			"%19s : %10u\n", "TX Short Retries",
			be32_to_cpu(cmd_rsp.tx_shortretry));
	len += snprintf(buf + len, sizeof(buf) - len,
			"%19s : %10u\n", "TX Long Retries",
			be32_to_cpu(cmd_rsp.tx_longretry));
	len += snprintf(buf + len, sizeof(buf) - len,
			"%19s : %10u\n", "TX Xretries",
			be32_to_cpu(cmd_rsp.tx_xretries));
	len += snprintf(buf + len, sizeof(buf) - len,
			"%19s : %10u\n", "TX Unaggr. Xretries",
			be32_to_cpu(cmd_rsp.ht_txunaggr_xretry));
	len += snprintf(buf + len, sizeof(buf) - len,
			"%19s : %10u\n", "TX Xretries (HT)",
			be32_to_cpu(cmd_rsp.ht_tx_xretries));
	len += snprintf(buf + len, sizeof(buf) - len,
			"%19s : %10u\n", "TX Rate", priv->debug.txrate);

757 758 759
	if (len > sizeof(buf))
		len = sizeof(buf);

S
Sujith 已提交
760 761 762 763 764 765
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static const struct file_operations fops_tgt_stats = {
	.read = read_file_tgt_stats,
	.open = ath9k_debugfs_open,
766 767
	.owner = THIS_MODULE,
	.llseek = default_llseek,
S
Sujith 已提交
768 769 770 771 772
};

static ssize_t read_file_xmit(struct file *file, char __user *user_buf,
			      size_t count, loff_t *ppos)
{
773
	struct ath9k_htc_priv *priv = file->private_data;
S
Sujith 已提交
774 775 776 777 778 779 780 781 782 783 784 785 786 787 788
	char buf[512];
	unsigned int len = 0;

	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "Buffers queued",
			priv->debug.tx_stats.buf_queued);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "Buffers completed",
			priv->debug.tx_stats.buf_completed);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "SKBs queued",
			priv->debug.tx_stats.skb_queued);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "SKBs completed",
			priv->debug.tx_stats.skb_completed);
789 790 791
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "SKBs dropped",
			priv->debug.tx_stats.skb_dropped);
S
Sujith 已提交
792

793 794 795 796 797 798 799 800 801 802 803 804 805
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "BE queued",
			priv->debug.tx_stats.queue_stats[WME_AC_BE]);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "BK queued",
			priv->debug.tx_stats.queue_stats[WME_AC_BK]);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "VI queued",
			priv->debug.tx_stats.queue_stats[WME_AC_VI]);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "VO queued",
			priv->debug.tx_stats.queue_stats[WME_AC_VO]);

806 807 808
	if (len > sizeof(buf))
		len = sizeof(buf);

S
Sujith 已提交
809 810 811 812 813 814
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static const struct file_operations fops_xmit = {
	.read = read_file_xmit,
	.open = ath9k_debugfs_open,
815 816
	.owner = THIS_MODULE,
	.llseek = default_llseek,
S
Sujith 已提交
817 818 819 820 821
};

static ssize_t read_file_recv(struct file *file, char __user *user_buf,
			      size_t count, loff_t *ppos)
{
822
	struct ath9k_htc_priv *priv = file->private_data;
S
Sujith 已提交
823 824 825 826 827 828 829 830 831 832 833 834 835
	char buf[512];
	unsigned int len = 0;

	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "SKBs allocated",
			priv->debug.rx_stats.skb_allocated);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "SKBs completed",
			priv->debug.rx_stats.skb_completed);
	len += snprintf(buf + len, sizeof(buf) - len,
			"%20s : %10u\n", "SKBs Dropped",
			priv->debug.rx_stats.skb_dropped);

836 837 838
	if (len > sizeof(buf))
		len = sizeof(buf);

S
Sujith 已提交
839 840 841 842 843 844
	return simple_read_from_buffer(user_buf, count, ppos, buf, len);
}

static const struct file_operations fops_recv = {
	.read = read_file_recv,
	.open = ath9k_debugfs_open,
845 846
	.owner = THIS_MODULE,
	.llseek = default_llseek,
S
Sujith 已提交
847 848
};

849
int ath9k_htc_init_debug(struct ath_hw *ah)
S
Sujith 已提交
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
{
	struct ath_common *common = ath9k_hw_common(ah);
	struct ath9k_htc_priv *priv = (struct ath9k_htc_priv *) common->priv;

	if (!ath9k_debugfs_root)
		return -ENOENT;

	priv->debug.debugfs_phy = debugfs_create_dir(wiphy_name(priv->hw->wiphy),
						     ath9k_debugfs_root);
	if (!priv->debug.debugfs_phy)
		goto err;

	priv->debug.debugfs_tgt_stats = debugfs_create_file("tgt_stats", S_IRUSR,
						    priv->debug.debugfs_phy,
						    priv, &fops_tgt_stats);
	if (!priv->debug.debugfs_tgt_stats)
		goto err;


	priv->debug.debugfs_xmit = debugfs_create_file("xmit", S_IRUSR,
						       priv->debug.debugfs_phy,
						       priv, &fops_xmit);
	if (!priv->debug.debugfs_xmit)
		goto err;

	priv->debug.debugfs_recv = debugfs_create_file("recv", S_IRUSR,
						       priv->debug.debugfs_phy,
						       priv, &fops_recv);
	if (!priv->debug.debugfs_recv)
		goto err;

	return 0;

err:
884
	ath9k_htc_exit_debug(ah);
S
Sujith 已提交
885 886 887
	return -ENOMEM;
}

888
void ath9k_htc_exit_debug(struct ath_hw *ah)
S
Sujith 已提交
889 890 891 892 893 894 895 896 897 898
{
	struct ath_common *common = ath9k_hw_common(ah);
	struct ath9k_htc_priv *priv = (struct ath9k_htc_priv *) common->priv;

	debugfs_remove(priv->debug.debugfs_recv);
	debugfs_remove(priv->debug.debugfs_xmit);
	debugfs_remove(priv->debug.debugfs_tgt_stats);
	debugfs_remove(priv->debug.debugfs_phy);
}

899
int ath9k_htc_debug_create_root(void)
S
Sujith 已提交
900 901 902 903 904 905 906 907
{
	ath9k_debugfs_root = debugfs_create_dir(KBUILD_MODNAME, NULL);
	if (!ath9k_debugfs_root)
		return -ENOENT;

	return 0;
}

908
void ath9k_htc_debug_remove_root(void)
S
Sujith 已提交
909 910 911 912 913 914 915 916 917 918 919
{
	debugfs_remove(ath9k_debugfs_root);
	ath9k_debugfs_root = NULL;
}

#endif /* CONFIG_ATH9K_HTC_DEBUGFS */

/*******/
/* ANI */
/*******/

920
void ath9k_htc_start_ani(struct ath9k_htc_priv *priv)
S
Sujith 已提交
921 922 923 924 925 926 927 928
{
	struct ath_common *common = ath9k_hw_common(priv->ah);
	unsigned long timestamp = jiffies_to_msecs(jiffies);

	common->ani.longcal_timer = timestamp;
	common->ani.shortcal_timer = timestamp;
	common->ani.checkani_timer = timestamp;

929 930 931
	priv->op_flags |= OP_ANI_RUNNING;

	ieee80211_queue_delayed_work(common->hw, &priv->ani_work,
S
Sujith 已提交
932 933 934
				     msecs_to_jiffies(ATH_ANI_POLLINTERVAL));
}

935 936 937 938 939 940 941
void ath9k_htc_stop_ani(struct ath9k_htc_priv *priv)
{
	cancel_delayed_work_sync(&priv->ani_work);
	priv->op_flags &= ~OP_ANI_RUNNING;
}

void ath9k_htc_ani_work(struct work_struct *work)
S
Sujith 已提交
942 943
{
	struct ath9k_htc_priv *priv =
944
		container_of(work, struct ath9k_htc_priv, ani_work.work);
S
Sujith 已提交
945 946 947 948 949 950 951 952
	struct ath_hw *ah = priv->ah;
	struct ath_common *common = ath9k_hw_common(ah);
	bool longcal = false;
	bool shortcal = false;
	bool aniflag = false;
	unsigned int timestamp = jiffies_to_msecs(jiffies);
	u32 cal_interval, short_cal_interval;

953 954
	short_cal_interval = (ah->opmode == NL80211_IFTYPE_AP) ?
		ATH_AP_SHORT_CALINTERVAL : ATH_STA_SHORT_CALINTERVAL;
S
Sujith 已提交
955

956 957 958 959
	/* Only calibrate if awake */
	if (ah->power_mode != ATH9K_PM_AWAKE)
		goto set_timer;

S
Sujith 已提交
960 961 962
	/* Long calibration runs independently of short calibration. */
	if ((timestamp - common->ani.longcal_timer) >= ATH_LONG_CALINTERVAL) {
		longcal = true;
J
Joe Perches 已提交
963
		ath_dbg(common, ATH_DBG_ANI, "longcal @%lu\n", jiffies);
S
Sujith 已提交
964 965 966 967 968 969 970 971
		common->ani.longcal_timer = timestamp;
	}

	/* Short calibration applies only while caldone is false */
	if (!common->ani.caldone) {
		if ((timestamp - common->ani.shortcal_timer) >=
		    short_cal_interval) {
			shortcal = true;
J
Joe Perches 已提交
972 973
			ath_dbg(common, ATH_DBG_ANI,
				"shortcal @%lu\n", jiffies);
S
Sujith 已提交
974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993
			common->ani.shortcal_timer = timestamp;
			common->ani.resetcal_timer = timestamp;
		}
	} else {
		if ((timestamp - common->ani.resetcal_timer) >=
		    ATH_RESTART_CALINTERVAL) {
			common->ani.caldone = ath9k_hw_reset_calvalid(ah);
			if (common->ani.caldone)
				common->ani.resetcal_timer = timestamp;
		}
	}

	/* Verify whether we must check ANI */
	if ((timestamp - common->ani.checkani_timer) >= ATH_ANI_POLLINTERVAL) {
		aniflag = true;
		common->ani.checkani_timer = timestamp;
	}

	/* Skip all processing if there's nothing to do. */
	if (longcal || shortcal || aniflag) {
994 995 996

		ath9k_htc_ps_wakeup(priv);

S
Sujith 已提交
997 998 999 1000 1001
		/* Call ANI routine if necessary */
		if (aniflag)
			ath9k_hw_ani_monitor(ah, ah->curchan);

		/* Perform calibration if necessary */
1002
		if (longcal || shortcal)
S
Sujith 已提交
1003 1004 1005 1006 1007
			common->ani.caldone =
				ath9k_hw_calibrate(ah, ah->curchan,
						   common->rx_chainmask,
						   longcal);

1008
		ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1009 1010
	}

1011
set_timer:
S
Sujith 已提交
1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022
	/*
	* Set timer interval based on previous results.
	* The interval must be the shortest necessary to satisfy ANI,
	* short calibration and long calibration.
	*/
	cal_interval = ATH_LONG_CALINTERVAL;
	if (priv->ah->config.enable_ani)
		cal_interval = min(cal_interval, (u32)ATH_ANI_POLLINTERVAL);
	if (!common->ani.caldone)
		cal_interval = min(cal_interval, (u32)short_cal_interval);

1023
	ieee80211_queue_delayed_work(common->hw, &priv->ani_work,
S
Sujith 已提交
1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034
				     msecs_to_jiffies(cal_interval));
}

/**********************/
/* mac80211 Callbacks */
/**********************/

static int ath9k_htc_tx(struct ieee80211_hw *hw, struct sk_buff *skb)
{
	struct ieee80211_hdr *hdr;
	struct ath9k_htc_priv *priv = hw->priv;
S
Sujith 已提交
1035
	int padpos, padsize, ret;
S
Sujith 已提交
1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048

	hdr = (struct ieee80211_hdr *) skb->data;

	/* Add the padding after the header if this is not already done */
	padpos = ath9k_cmn_padpos(hdr->frame_control);
	padsize = padpos & 3;
	if (padsize && skb->len > padpos) {
		if (skb_headroom(skb) < padsize)
			return -1;
		skb_push(skb, padsize);
		memmove(skb->data, skb->data + padsize, padpos);
	}

S
Sujith 已提交
1049 1050 1051
	ret = ath9k_htc_tx_start(priv, skb);
	if (ret != 0) {
		if (ret == -ENOMEM) {
J
Joe Perches 已提交
1052 1053
			ath_dbg(ath9k_hw_common(priv->ah), ATH_DBG_XMIT,
				"Stopping TX queues\n");
S
Sujith 已提交
1054 1055 1056 1057 1058
			ieee80211_stop_queues(hw);
			spin_lock_bh(&priv->tx_lock);
			priv->tx_queues_stop = true;
			spin_unlock_bh(&priv->tx_lock);
		} else {
J
Joe Perches 已提交
1059 1060
			ath_dbg(ath9k_hw_common(priv->ah), ATH_DBG_XMIT,
				"Tx failed\n");
S
Sujith 已提交
1061
		}
S
Sujith 已提交
1062 1063 1064 1065 1066 1067 1068 1069 1070 1071
		goto fail_tx;
	}

	return 0;

fail_tx:
	dev_kfree_skb_any(skb);
	return 0;
}

S
Sujith 已提交
1072
static int ath9k_htc_start(struct ieee80211_hw *hw)
S
Sujith 已提交
1073 1074 1075 1076 1077 1078 1079 1080
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_hw *ah = priv->ah;
	struct ath_common *common = ath9k_hw_common(ah);
	struct ieee80211_channel *curchan = hw->conf.channel;
	struct ath9k_channel *init_channel;
	int ret = 0;
	enum htc_phymode mode;
S
Sujith 已提交
1081
	__be16 htc_mode;
S
Sujith 已提交
1082 1083
	u8 cmd_rsp;

S
Sujith 已提交
1084 1085
	mutex_lock(&priv->mutex);

J
Joe Perches 已提交
1086 1087 1088
	ath_dbg(common, ATH_DBG_CONFIG,
		"Starting driver with initial channel: %d MHz\n",
		curchan->center_freq);
S
Sujith 已提交
1089

S
Sujith 已提交
1090 1091 1092 1093
	/* Ensure that HW is awake before flushing RX */
	ath9k_htc_setpower(priv, ATH9K_PM_AWAKE);
	WMI_CMD(WMI_FLUSH_RECV_CMDID);

S
Sujith 已提交
1094 1095 1096 1097
	/* setup initial channel */
	init_channel = ath9k_cmn_get_curchannel(hw, ah);

	ath9k_hw_htc_resetinit(ah);
1098
	ret = ath9k_hw_reset(ah, init_channel, ah->caldata, false);
S
Sujith 已提交
1099
	if (ret) {
1100 1101 1102
		ath_err(common,
			"Unable to reset hardware; reset status %d (freq %u MHz)\n",
			ret, curchan->center_freq);
S
Sujith 已提交
1103
		mutex_unlock(&priv->mutex);
1104
		return ret;
S
Sujith 已提交
1105 1106
	}

1107 1108
	ath9k_cmn_update_txpow(ah, priv->curtxpow, priv->txpowlimit,
			       &priv->curtxpow);
S
Sujith 已提交
1109 1110 1111 1112 1113 1114 1115 1116 1117

	mode = ath9k_htc_get_curmode(priv, init_channel);
	htc_mode = cpu_to_be16(mode);
	WMI_CMD_BUF(WMI_SET_MODE_CMDID, &htc_mode);
	WMI_CMD(WMI_ATH_INIT_CMDID);
	WMI_CMD(WMI_START_RECV_CMDID);

	ath9k_host_rx_init(priv);

1118 1119 1120 1121 1122
	ret = ath9k_htc_update_cap_target(priv);
	if (ret)
		ath_dbg(common, ATH_DBG_CONFIG,
			"Failed to update capability in target\n");

S
Sujith 已提交
1123 1124 1125
	priv->op_flags &= ~OP_INVALID;
	htc_start(priv->htc);

S
Sujith 已提交
1126 1127 1128 1129 1130 1131
	spin_lock_bh(&priv->tx_lock);
	priv->tx_queues_stop = false;
	spin_unlock_bh(&priv->tx_lock);

	ieee80211_wake_queues(hw);

1132 1133 1134 1135 1136 1137
	if (ah->btcoex_hw.scheme == ATH_BTCOEX_CFG_3WIRE) {
		ath9k_hw_btcoex_set_weight(ah, AR_BT_COEX_WGHT,
					   AR_STOMP_LOW_WLAN_WGHT);
		ath9k_hw_btcoex_enable(ah);
		ath_htc_resume_btcoex_work(priv);
	}
S
Sujith 已提交
1138
	mutex_unlock(&priv->mutex);
1139

S
Sujith 已提交
1140 1141 1142
	return ret;
}

S
Sujith 已提交
1143
static void ath9k_htc_stop(struct ieee80211_hw *hw)
S
Sujith 已提交
1144 1145 1146 1147 1148 1149 1150
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_hw *ah = priv->ah;
	struct ath_common *common = ath9k_hw_common(ah);
	int ret = 0;
	u8 cmd_rsp;

S
Sujith 已提交
1151 1152
	mutex_lock(&priv->mutex);

S
Sujith 已提交
1153
	if (priv->op_flags & OP_INVALID) {
J
Joe Perches 已提交
1154
		ath_dbg(common, ATH_DBG_ANY, "Device not present\n");
S
Sujith 已提交
1155
		mutex_unlock(&priv->mutex);
S
Sujith 已提交
1156 1157 1158
		return;
	}

1159
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1160 1161 1162 1163
	htc_stop(priv->htc);
	WMI_CMD(WMI_DISABLE_INTR_CMDID);
	WMI_CMD(WMI_DRAIN_TXQ_ALL_CMDID);
	WMI_CMD(WMI_STOP_RECV_CMDID);
1164 1165 1166 1167 1168

	tasklet_kill(&priv->swba_tasklet);
	tasklet_kill(&priv->rx_tasklet);
	tasklet_kill(&priv->tx_tasklet);

S
Sujith 已提交
1169 1170
	skb_queue_purge(&priv->tx_queue);

1171 1172 1173 1174 1175 1176
	mutex_unlock(&priv->mutex);

	/* Cancel all the running timers/work .. */
	cancel_work_sync(&priv->fatal_work);
	cancel_work_sync(&priv->ps_work);
	cancel_delayed_work_sync(&priv->ath9k_led_blink_work);
1177
	ath9k_htc_stop_ani(priv);
1178 1179 1180 1181
	ath9k_led_stop_brightness(priv);

	mutex_lock(&priv->mutex);

1182 1183 1184 1185 1186 1187
	if (ah->btcoex_hw.enabled) {
		ath9k_hw_btcoex_disable(ah);
		if (ah->btcoex_hw.scheme == ATH_BTCOEX_CFG_3WIRE)
			ath_htc_cancel_btcoex_work(priv);
	}

1188 1189 1190 1191
	/* Remove a monitor interface if it's present. */
	if (priv->ah->is_monitoring)
		ath9k_htc_remove_monitor_interface(priv);

1192 1193 1194 1195 1196
	ath9k_hw_phy_disable(ah);
	ath9k_hw_disable(ah);
	ath9k_htc_ps_restore(priv);
	ath9k_htc_setpower(priv, ATH9K_PM_FULL_SLEEP);

S
Sujith 已提交
1197 1198
	priv->op_flags |= OP_INVALID;

J
Joe Perches 已提交
1199
	ath_dbg(common, ATH_DBG_CONFIG, "Driver halt\n");
1200 1201 1202
	mutex_unlock(&priv->mutex);
}

S
Sujith 已提交
1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214
static int ath9k_htc_add_interface(struct ieee80211_hw *hw,
				   struct ieee80211_vif *vif)
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath9k_htc_vif *avp = (void *)vif->drv_priv;
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_target_vif hvif;
	int ret = 0;
	u8 cmd_rsp;

	mutex_lock(&priv->mutex);

1215
	if (priv->nvifs >= ATH9K_HTC_MAX_VIF) {
1216
		mutex_unlock(&priv->mutex);
1217 1218 1219 1220 1221 1222 1223 1224
		return -ENOBUFS;
	}

	if (priv->num_ibss_vif ||
	    (priv->nvifs && vif->type == NL80211_IFTYPE_ADHOC)) {
		ath_err(common, "IBSS coexistence with other modes is not allowed\n");
		mutex_unlock(&priv->mutex);
		return -ENOBUFS;
S
Sujith 已提交
1225 1226
	}

1227 1228 1229 1230 1231 1232 1233 1234
	if (((vif->type == NL80211_IFTYPE_AP) ||
	     (vif->type == NL80211_IFTYPE_ADHOC)) &&
	    ((priv->num_ap_vif + priv->num_ibss_vif) >= ATH9K_HTC_MAX_BCN_VIF)) {
		ath_err(common, "Max. number of beaconing interfaces reached\n");
		mutex_unlock(&priv->mutex);
		return -ENOBUFS;
	}

1235
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1236 1237 1238 1239 1240 1241 1242 1243 1244 1245
	memset(&hvif, 0, sizeof(struct ath9k_htc_target_vif));
	memcpy(&hvif.myaddr, vif->addr, ETH_ALEN);

	switch (vif->type) {
	case NL80211_IFTYPE_STATION:
		hvif.opmode = cpu_to_be32(HTC_M_STA);
		break;
	case NL80211_IFTYPE_ADHOC:
		hvif.opmode = cpu_to_be32(HTC_M_IBSS);
		break;
1246 1247 1248
	case NL80211_IFTYPE_AP:
		hvif.opmode = cpu_to_be32(HTC_M_HOSTAP);
		break;
S
Sujith 已提交
1249
	default:
1250
		ath_err(common,
S
Sujith 已提交
1251 1252 1253 1254 1255 1256
			"Interface type %d not yet supported\n", vif->type);
		ret = -EOPNOTSUPP;
		goto out;
	}

	/* Index starts from zero on the target */
1257
	avp->index = hvif.index = ffz(priv->vif_slot);
S
Sujith 已提交
1258 1259 1260 1261 1262 1263 1264 1265 1266 1267
	hvif.rtsthreshold = cpu_to_be16(2304);
	WMI_CMD_BUF(WMI_VAP_CREATE_CMDID, &hvif);
	if (ret)
		goto out;

	/*
	 * We need a node in target to tx mgmt frames
	 * before association.
	 */
	ret = ath9k_htc_add_station(priv, vif, NULL);
1268 1269
	if (ret) {
		WMI_CMD_BUF(WMI_VAP_REMOVE_CMDID, &hvif);
S
Sujith 已提交
1270
		goto out;
1271
	}
S
Sujith 已提交
1272

1273 1274
	ath9k_htc_set_bssid_mask(priv, vif);

1275
	priv->vif_slot |= (1 << avp->index);
1276
	priv->nvifs++;
S
Sujith 已提交
1277
	priv->vif = vif;
1278

1279
	INC_VIF(priv, vif->type);
1280
	ath9k_htc_set_opmode(priv);
1281

1282 1283 1284 1285
	if ((priv->ah->opmode == NL80211_IFTYPE_AP) &&
	    !(priv->op_flags & OP_ANI_RUNNING))
		ath9k_htc_start_ani(priv);

1286 1287 1288
	ath_dbg(common, ATH_DBG_CONFIG,
		"Attach a VIF of type: %d at idx: %d\n", vif->type, avp->index);

S
Sujith 已提交
1289
out:
1290
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1291
	mutex_unlock(&priv->mutex);
S
Sujith 已提交
1292

S
Sujith 已提交
1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306
	return ret;
}

static void ath9k_htc_remove_interface(struct ieee80211_hw *hw,
				       struct ieee80211_vif *vif)
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_htc_vif *avp = (void *)vif->drv_priv;
	struct ath9k_htc_target_vif hvif;
	int ret = 0;
	u8 cmd_rsp;

	mutex_lock(&priv->mutex);
S
Sujith 已提交
1307
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1308 1309 1310 1311 1312 1313

	memset(&hvif, 0, sizeof(struct ath9k_htc_target_vif));
	memcpy(&hvif.myaddr, vif->addr, ETH_ALEN);
	hvif.index = avp->index;
	WMI_CMD_BUF(WMI_VAP_REMOVE_CMDID, &hvif);
	priv->nvifs--;
1314
	priv->vif_slot &= ~(1 << avp->index);
S
Sujith 已提交
1315 1316 1317 1318

	ath9k_htc_remove_station(priv, vif, NULL);
	priv->vif = NULL;

1319
	DEC_VIF(priv, vif->type);
1320
	ath9k_htc_set_opmode(priv);
1321

1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332
	/*
	 * Stop ANI only if there are no associated station interfaces.
	 */
	if ((vif->type == NL80211_IFTYPE_AP) && (priv->num_ap_vif == 0)) {
		priv->rearm_ani = false;
		ieee80211_iterate_active_interfaces_atomic(priv->hw,
						   ath9k_htc_vif_iter, priv);
		if (!priv->rearm_ani)
			ath9k_htc_stop_ani(priv);
	}

1333 1334
	ath_dbg(common, ATH_DBG_CONFIG, "Detach Interface at idx: %d\n", avp->index);

S
Sujith 已提交
1335
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346
	mutex_unlock(&priv->mutex);
}

static int ath9k_htc_config(struct ieee80211_hw *hw, u32 changed)
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ieee80211_conf *conf = &hw->conf;

	mutex_lock(&priv->mutex);

1347 1348 1349 1350
	if (changed & IEEE80211_CONF_CHANGE_IDLE) {
		bool enable_radio = false;
		bool idle = !!(conf->flags & IEEE80211_CONF_IDLE);

S
Sujith 已提交
1351
		mutex_lock(&priv->htc_pm_lock);
1352 1353 1354
		if (!idle && priv->ps_idle)
			enable_radio = true;
		priv->ps_idle = idle;
S
Sujith 已提交
1355
		mutex_unlock(&priv->htc_pm_lock);
1356 1357

		if (enable_radio) {
J
Joe Perches 已提交
1358 1359
			ath_dbg(common, ATH_DBG_CONFIG,
				"not-idle: enabling radio\n");
S
Sujith 已提交
1360 1361
			ath9k_htc_setpower(priv, ATH9K_PM_AWAKE);
			ath9k_htc_radio_enable(hw);
1362 1363 1364
		}
	}

1365 1366 1367 1368 1369
	/*
	 * Monitor interface should be added before
	 * IEEE80211_CONF_CHANGE_CHANNEL is handled.
	 */
	if (changed & IEEE80211_CONF_CHANGE_MONITOR) {
1370 1371 1372 1373 1374
		if ((conf->flags & IEEE80211_CONF_MONITOR) &&
		    !priv->ah->is_monitoring)
			ath9k_htc_add_monitor_interface(priv);
		else if (priv->ah->is_monitoring)
			ath9k_htc_remove_monitor_interface(priv);
1375 1376
	}

S
Sujith 已提交
1377 1378 1379 1380
	if (changed & IEEE80211_CONF_CHANGE_CHANNEL) {
		struct ieee80211_channel *curchan = hw->conf.channel;
		int pos = curchan->hw_value;

J
Joe Perches 已提交
1381 1382
		ath_dbg(common, ATH_DBG_CONFIG, "Set channel: %d MHz\n",
			curchan->center_freq);
S
Sujith 已提交
1383

1384 1385 1386
		ath9k_cmn_update_ichannel(&priv->ah->channels[pos],
					  hw->conf.channel,
					  hw->conf.channel_type);
S
Sujith 已提交
1387 1388

		if (ath9k_htc_set_channel(priv, hw, &priv->ah->channels[pos]) < 0) {
1389
			ath_err(common, "Unable to set channel\n");
S
Sujith 已提交
1390 1391 1392 1393 1394
			mutex_unlock(&priv->mutex);
			return -EINVAL;
		}

	}
1395

1396 1397 1398 1399 1400 1401 1402 1403 1404 1405
	if (changed & IEEE80211_CONF_CHANGE_PS) {
		if (conf->flags & IEEE80211_CONF_PS) {
			ath9k_htc_setpower(priv, ATH9K_PM_NETWORK_SLEEP);
			priv->ps_enabled = true;
		} else {
			priv->ps_enabled = false;
			cancel_work_sync(&priv->ps_work);
			ath9k_htc_setpower(priv, ATH9K_PM_AWAKE);
		}
	}
S
Sujith 已提交
1406

1407 1408
	if (changed & IEEE80211_CONF_CHANGE_POWER) {
		priv->txpowlimit = 2 * conf->power_level;
1409 1410
		ath9k_cmn_update_txpow(priv->ah, priv->curtxpow,
				       priv->txpowlimit, &priv->curtxpow);
1411 1412
	}

S
Sujith 已提交
1413 1414 1415 1416 1417 1418 1419 1420
	if (changed & IEEE80211_CONF_CHANGE_IDLE) {
		mutex_lock(&priv->htc_pm_lock);
		if (!priv->ps_idle) {
			mutex_unlock(&priv->htc_pm_lock);
			goto out;
		}
		mutex_unlock(&priv->htc_pm_lock);

J
Joe Perches 已提交
1421 1422
		ath_dbg(common, ATH_DBG_CONFIG,
			"idle: disabling radio\n");
S
Sujith 已提交
1423
		ath9k_htc_radio_disable(hw);
1424 1425
	}

S
Sujith 已提交
1426
out:
S
Sujith 已提交
1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437
	mutex_unlock(&priv->mutex);
	return 0;
}

#define SUPPORTED_FILTERS			\
	(FIF_PROMISC_IN_BSS |			\
	FIF_ALLMULTI |				\
	FIF_CONTROL |				\
	FIF_PSPOLL |				\
	FIF_OTHER_BSS |				\
	FIF_BCN_PRBRESP_PROMISC |		\
1438
	FIF_PROBE_REQ |				\
S
Sujith 已提交
1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449
	FIF_FCSFAIL)

static void ath9k_htc_configure_filter(struct ieee80211_hw *hw,
				       unsigned int changed_flags,
				       unsigned int *total_flags,
				       u64 multicast)
{
	struct ath9k_htc_priv *priv = hw->priv;
	u32 rfilt;

	mutex_lock(&priv->mutex);
1450
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1451

S
Sujith 已提交
1452 1453 1454 1455
	changed_flags &= SUPPORTED_FILTERS;
	*total_flags &= SUPPORTED_FILTERS;

	priv->rxfilter = *total_flags;
1456
	rfilt = ath9k_htc_calcrxfilter(priv);
S
Sujith 已提交
1457 1458
	ath9k_hw_setrxfilter(priv->ah, rfilt);

J
Joe Perches 已提交
1459 1460
	ath_dbg(ath9k_hw_common(priv->ah), ATH_DBG_CONFIG,
		"Set HW RX filter: 0x%x\n", rfilt);
S
Sujith 已提交
1461

1462
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1463 1464 1465
	mutex_unlock(&priv->mutex);
}

1466 1467 1468
static int ath9k_htc_sta_add(struct ieee80211_hw *hw,
			     struct ieee80211_vif *vif,
			     struct ieee80211_sta *sta)
S
Sujith 已提交
1469 1470 1471 1472
{
	struct ath9k_htc_priv *priv = hw->priv;
	int ret;

1473
	mutex_lock(&priv->mutex);
S
Sujith 已提交
1474
	ath9k_htc_ps_wakeup(priv);
1475 1476 1477 1478 1479
	ret = ath9k_htc_add_station(priv, vif, sta);
	if (!ret)
		ath9k_htc_init_rate(priv, sta);
	ath9k_htc_ps_restore(priv);
	mutex_unlock(&priv->mutex);
1480

1481 1482 1483 1484 1485 1486 1487 1488 1489
	return ret;
}

static int ath9k_htc_sta_remove(struct ieee80211_hw *hw,
				struct ieee80211_vif *vif,
				struct ieee80211_sta *sta)
{
	struct ath9k_htc_priv *priv = hw->priv;
	int ret;
1490

1491 1492 1493
	mutex_lock(&priv->mutex);
	ath9k_htc_ps_wakeup(priv);
	ret = ath9k_htc_remove_station(priv, vif, sta);
S
Sujith 已提交
1494
	ath9k_htc_ps_restore(priv);
1495
	mutex_unlock(&priv->mutex);
1496 1497

	return ret;
S
Sujith 已提交
1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511
}

static int ath9k_htc_conf_tx(struct ieee80211_hw *hw, u16 queue,
			     const struct ieee80211_tx_queue_params *params)
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_common *common = ath9k_hw_common(priv->ah);
	struct ath9k_tx_queue_info qi;
	int ret = 0, qnum;

	if (queue >= WME_NUM_AC)
		return 0;

	mutex_lock(&priv->mutex);
S
Sujith 已提交
1512
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1513 1514 1515 1516 1517 1518 1519 1520 1521 1522

	memset(&qi, 0, sizeof(struct ath9k_tx_queue_info));

	qi.tqi_aifs = params->aifs;
	qi.tqi_cwmin = params->cw_min;
	qi.tqi_cwmax = params->cw_max;
	qi.tqi_burstTime = params->txop;

	qnum = get_hw_qnum(queue, priv->hwq_map);

J
Joe Perches 已提交
1523 1524 1525 1526
	ath_dbg(common, ATH_DBG_CONFIG,
		"Configure tx [queue/hwq] [%d/%d],  aifs: %d, cw_min: %d, cw_max: %d, txop: %d\n",
		queue, qnum, params->aifs, params->cw_min,
		params->cw_max, params->txop);
S
Sujith 已提交
1527

1528
	ret = ath_htc_txq_update(priv, qnum, &qi);
S
Sujith 已提交
1529
	if (ret) {
1530
		ath_err(common, "TXQ Update failed\n");
S
Sujith 已提交
1531 1532
		goto out;
	}
S
Sujith 已提交
1533

S
Sujith 已提交
1534
	if ((priv->ah->opmode == NL80211_IFTYPE_ADHOC) &&
1535
	    (qnum == priv->hwq_map[WME_AC_BE]))
S
Sujith 已提交
1536 1537
		    ath9k_htc_beaconq_config(priv);
out:
S
Sujith 已提交
1538
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1539 1540 1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552 1553
	mutex_unlock(&priv->mutex);

	return ret;
}

static int ath9k_htc_set_key(struct ieee80211_hw *hw,
			     enum set_key_cmd cmd,
			     struct ieee80211_vif *vif,
			     struct ieee80211_sta *sta,
			     struct ieee80211_key_conf *key)
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_common *common = ath9k_hw_common(priv->ah);
	int ret = 0;

1554
	if (htc_modparam_nohwcrypt)
S
Sujith 已提交
1555 1556 1557
		return -ENOSPC;

	mutex_lock(&priv->mutex);
J
Joe Perches 已提交
1558
	ath_dbg(common, ATH_DBG_CONFIG, "Set HW Key\n");
1559
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1560 1561 1562

	switch (cmd) {
	case SET_KEY:
1563
		ret = ath_key_config(common, vif, sta, key);
S
Sujith 已提交
1564 1565 1566 1567
		if (ret >= 0) {
			key->hw_key_idx = ret;
			/* push IV and Michael MIC generation to stack */
			key->flags |= IEEE80211_KEY_FLAG_GENERATE_IV;
1568
			if (key->cipher == WLAN_CIPHER_SUITE_TKIP)
S
Sujith 已提交
1569
				key->flags |= IEEE80211_KEY_FLAG_GENERATE_MMIC;
1570 1571
			if (priv->ah->sw_mgmt_crypto &&
			    key->cipher == WLAN_CIPHER_SUITE_CCMP)
S
Sujith 已提交
1572 1573 1574 1575 1576
				key->flags |= IEEE80211_KEY_FLAG_SW_MGMT;
			ret = 0;
		}
		break;
	case DISABLE_KEY:
1577
		ath_key_delete(common, key);
S
Sujith 已提交
1578 1579 1580 1581 1582
		break;
	default:
		ret = -EINVAL;
	}

1583
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598
	mutex_unlock(&priv->mutex);

	return ret;
}

static void ath9k_htc_bss_info_changed(struct ieee80211_hw *hw,
				       struct ieee80211_vif *vif,
				       struct ieee80211_bss_conf *bss_conf,
				       u32 changed)
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath_hw *ah = priv->ah;
	struct ath_common *common = ath9k_hw_common(ah);

	mutex_lock(&priv->mutex);
1599
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1600 1601 1602 1603

	if (changed & BSS_CHANGED_ASSOC) {
		common->curaid = bss_conf->assoc ?
				 bss_conf->aid : 0;
J
Joe Perches 已提交
1604
		ath_dbg(common, ATH_DBG_CONFIG, "BSS Changed ASSOC %d\n",
S
Sujith 已提交
1605 1606
			bss_conf->assoc);

1607
		if (bss_conf->assoc)
1608
			ath9k_htc_start_ani(priv);
1609
		else
1610
			ath9k_htc_stop_ani(priv);
S
Sujith 已提交
1611 1612 1613 1614 1615 1616 1617
	}

	if (changed & BSS_CHANGED_BSSID) {
		/* Set BSSID */
		memcpy(common->curbssid, bss_conf->bssid, ETH_ALEN);
		ath9k_hw_write_associd(ah);

J
Joe Perches 已提交
1618 1619 1620
		ath_dbg(common, ATH_DBG_CONFIG,
			"BSSID: %pM aid: 0x%x\n",
			common->curbssid, common->curaid);
S
Sujith 已提交
1621 1622 1623 1624 1625 1626 1627
	}

	if ((changed & BSS_CHANGED_BEACON_INT) ||
	    (changed & BSS_CHANGED_BEACON) ||
	    ((changed & BSS_CHANGED_BEACON_ENABLED) &&
	    bss_conf->enable_beacon)) {
		priv->op_flags |= OP_ENABLE_BEACON;
1628
		ath9k_htc_beacon_config(priv, vif);
S
Sujith 已提交
1629 1630 1631 1632 1633
	}

	if ((changed & BSS_CHANGED_BEACON_ENABLED) &&
	    !bss_conf->enable_beacon) {
		priv->op_flags &= ~OP_ENABLE_BEACON;
1634
		ath9k_htc_beacon_config(priv, vif);
S
Sujith 已提交
1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645
	}

	if (changed & BSS_CHANGED_ERP_SLOT) {
		if (bss_conf->use_short_slot)
			ah->slottime = 9;
		else
			ah->slottime = 20;

		ath9k_hw_init_global_settings(ah);
	}

1646 1647 1648
	if (changed & BSS_CHANGED_HT)
		ath9k_htc_update_rate(priv, vif, bss_conf);

1649
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1650 1651 1652 1653 1654 1655 1656 1657 1658
	mutex_unlock(&priv->mutex);
}

static u64 ath9k_htc_get_tsf(struct ieee80211_hw *hw)
{
	struct ath9k_htc_priv *priv = hw->priv;
	u64 tsf;

	mutex_lock(&priv->mutex);
S
Sujith 已提交
1659
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1660
	tsf = ath9k_hw_gettsf64(priv->ah);
S
Sujith 已提交
1661
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1662 1663 1664 1665 1666 1667 1668 1669 1670 1671
	mutex_unlock(&priv->mutex);

	return tsf;
}

static void ath9k_htc_set_tsf(struct ieee80211_hw *hw, u64 tsf)
{
	struct ath9k_htc_priv *priv = hw->priv;

	mutex_lock(&priv->mutex);
S
Sujith 已提交
1672
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1673
	ath9k_hw_settsf64(priv->ah, tsf);
S
Sujith 已提交
1674
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1675 1676 1677 1678 1679 1680 1681 1682
	mutex_unlock(&priv->mutex);
}

static void ath9k_htc_reset_tsf(struct ieee80211_hw *hw)
{
	struct ath9k_htc_priv *priv = hw->priv;

	mutex_lock(&priv->mutex);
S
Sujith 已提交
1683
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1684
	ath9k_hw_reset_tsf(priv->ah);
1685
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1686
	mutex_unlock(&priv->mutex);
S
Sujith 已提交
1687 1688 1689 1690 1691 1692
}

static int ath9k_htc_ampdu_action(struct ieee80211_hw *hw,
				  struct ieee80211_vif *vif,
				  enum ieee80211_ampdu_mlme_action action,
				  struct ieee80211_sta *sta,
1693
				  u16 tid, u16 *ssn, u8 buf_size)
S
Sujith 已提交
1694 1695 1696
{
	struct ath9k_htc_priv *priv = hw->priv;
	struct ath9k_htc_sta *ista;
S
Sujith 已提交
1697
	int ret = 0;
S
Sujith 已提交
1698

1699 1700
	mutex_lock(&priv->mutex);

S
Sujith 已提交
1701 1702 1703 1704 1705 1706
	switch (action) {
	case IEEE80211_AMPDU_RX_START:
		break;
	case IEEE80211_AMPDU_RX_STOP:
		break;
	case IEEE80211_AMPDU_TX_START:
S
Sujith 已提交
1707 1708 1709 1710
		ret = ath9k_htc_tx_aggr_oper(priv, vif, sta, action, tid);
		if (!ret)
			ieee80211_start_tx_ba_cb_irqsafe(vif, sta->addr, tid);
		break;
S
Sujith 已提交
1711
	case IEEE80211_AMPDU_TX_STOP:
S
Sujith 已提交
1712 1713
		ath9k_htc_tx_aggr_oper(priv, vif, sta, action, tid);
		ieee80211_stop_tx_ba_cb_irqsafe(vif, sta->addr, tid);
S
Sujith 已提交
1714 1715 1716
		break;
	case IEEE80211_AMPDU_TX_OPERATIONAL:
		ista = (struct ath9k_htc_sta *) sta->drv_priv;
S
Sujith 已提交
1717
		spin_lock_bh(&priv->tx_lock);
S
Sujith 已提交
1718
		ista->tid_state[tid] = AGGR_OPERATIONAL;
S
Sujith 已提交
1719
		spin_unlock_bh(&priv->tx_lock);
S
Sujith 已提交
1720 1721
		break;
	default:
1722
		ath_err(ath9k_hw_common(priv->ah), "Unknown AMPDU action\n");
S
Sujith 已提交
1723 1724
	}

1725 1726
	mutex_unlock(&priv->mutex);

S
Sujith 已提交
1727
	return ret;
S
Sujith 已提交
1728 1729 1730 1731 1732 1733 1734 1735 1736 1737
}

static void ath9k_htc_sw_scan_start(struct ieee80211_hw *hw)
{
	struct ath9k_htc_priv *priv = hw->priv;

	mutex_lock(&priv->mutex);
	spin_lock_bh(&priv->beacon_lock);
	priv->op_flags |= OP_SCANNING;
	spin_unlock_bh(&priv->beacon_lock);
1738
	cancel_work_sync(&priv->ps_work);
1739
	ath9k_htc_stop_ani(priv);
S
Sujith 已提交
1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750
	mutex_unlock(&priv->mutex);
}

static void ath9k_htc_sw_scan_complete(struct ieee80211_hw *hw)
{
	struct ath9k_htc_priv *priv = hw->priv;

	mutex_lock(&priv->mutex);
	spin_lock_bh(&priv->beacon_lock);
	priv->op_flags &= ~OP_SCANNING;
	spin_unlock_bh(&priv->beacon_lock);
1751 1752
	ath9k_htc_ps_wakeup(priv);
	ath9k_htc_vif_reconfig(priv);
1753
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1754
	mutex_unlock(&priv->mutex);
S
Sujith 已提交
1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767
}

static int ath9k_htc_set_rts_threshold(struct ieee80211_hw *hw, u32 value)
{
	return 0;
}

static void ath9k_htc_set_coverage_class(struct ieee80211_hw *hw,
					 u8 coverage_class)
{
	struct ath9k_htc_priv *priv = hw->priv;

	mutex_lock(&priv->mutex);
S
Sujith 已提交
1768
	ath9k_htc_ps_wakeup(priv);
S
Sujith 已提交
1769 1770
	priv->ah->coverage_class = coverage_class;
	ath9k_hw_init_global_settings(priv->ah);
S
Sujith 已提交
1771
	ath9k_htc_ps_restore(priv);
S
Sujith 已提交
1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782
	mutex_unlock(&priv->mutex);
}

struct ieee80211_ops ath9k_htc_ops = {
	.tx                 = ath9k_htc_tx,
	.start              = ath9k_htc_start,
	.stop               = ath9k_htc_stop,
	.add_interface      = ath9k_htc_add_interface,
	.remove_interface   = ath9k_htc_remove_interface,
	.config             = ath9k_htc_config,
	.configure_filter   = ath9k_htc_configure_filter,
1783 1784
	.sta_add            = ath9k_htc_sta_add,
	.sta_remove         = ath9k_htc_sta_remove,
S
Sujith 已提交
1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797
	.conf_tx            = ath9k_htc_conf_tx,
	.bss_info_changed   = ath9k_htc_bss_info_changed,
	.set_key            = ath9k_htc_set_key,
	.get_tsf            = ath9k_htc_get_tsf,
	.set_tsf            = ath9k_htc_set_tsf,
	.reset_tsf          = ath9k_htc_reset_tsf,
	.ampdu_action       = ath9k_htc_ampdu_action,
	.sw_scan_start      = ath9k_htc_sw_scan_start,
	.sw_scan_complete   = ath9k_htc_sw_scan_complete,
	.set_rts_threshold  = ath9k_htc_set_rts_threshold,
	.rfkill_poll        = ath9k_htc_rfkill_poll_state,
	.set_coverage_class = ath9k_htc_set_coverage_class,
};