|
@@ -1,1288 +0,0 @@
|
|
|
-commit 6fb7eefaa4d8377e6b124435059656dd6f643e91
|
|
|
-Author: Karl Beldan <[email protected]>
|
|
|
-Date: Tue Oct 7 15:53:38 2014 +0200
|
|
|
-
|
|
|
- mac80211/trivial: fix typo in starting baserate for rts_cts_rate_idx
|
|
|
-
|
|
|
- Fixes: 5253ffb8 ("mac80211: always pick a basic rate to tx RTS/CTS for pre-HT rates")
|
|
|
- Signed-off-by: Karl Beldan <[email protected]>
|
|
|
-
|
|
|
-commit b18111d911980af52bead74ee45250cc96ad5108
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Tue Oct 7 10:14:37 2014 +0530
|
|
|
-
|
|
|
- ath9k: Fix crash in MCC mode
|
|
|
-
|
|
|
- When a channel context is removed, the hw_queue_base
|
|
|
- is set to -1, this will result in a panic because
|
|
|
- ath9k_chanctx_stop_queues() can be called on an interface
|
|
|
- that is not assigned to any context yet - for example,
|
|
|
- when trying to scan.
|
|
|
-
|
|
|
- Fix this issue by setting the hw_queue_base to zero
|
|
|
- when a channel context is removed.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit e2cba8d7590e76661e86f1f0987ef9f8c13c9a6d
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:20 2014 +0530
|
|
|
-
|
|
|
- ath9k: Fix flushing in MCC mode
|
|
|
-
|
|
|
- When we are attempting to switch to a new
|
|
|
- channel context, the TX queues are flushed, but
|
|
|
- the mac80211 queues are not stopped and traffic
|
|
|
- can still come down to the driver.
|
|
|
-
|
|
|
- This patch fixes it by stopping the queues
|
|
|
- assigned to the current context/vif before
|
|
|
- trying to flush.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit 5ba8d9d2f018f2c4e23f9e68b90ca5b9d5470457
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:19 2014 +0530
|
|
|
-
|
|
|
- ath9k: Fix queue handling for channel contexts
|
|
|
-
|
|
|
- When a full chip reset is done, all the queues
|
|
|
- across all VIFs are stopped, but if MCC is enabled,
|
|
|
- only the queues of the current context is awakened,
|
|
|
- when we complete the reset.
|
|
|
-
|
|
|
- This results in unfairness for the inactive context.
|
|
|
- Since frames are queued internally in the driver if
|
|
|
- there is a context mismatch, we can awaken all the
|
|
|
- queues when coming out of a reset.
|
|
|
-
|
|
|
- The VIF-specific queues are still used in flow control,
|
|
|
- to ensure fairness when traffic is high.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit a064eaa10ca4ec58d5a405c9a7f87efc6d2fa423
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:18 2014 +0530
|
|
|
-
|
|
|
- ath9k: Add ath9k_chanctx_stop_queues()
|
|
|
-
|
|
|
- This can be used when the queues of a context
|
|
|
- needs to be stopped.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit b39031536aab9cb1324328cf46fa4ef940bd975f
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:17 2014 +0530
|
|
|
-
|
|
|
- ath9k: Pass context to ath9k_chanctx_wake_queues()
|
|
|
-
|
|
|
- Change the ath9k_chanctx_wake_queues() API so
|
|
|
- that we can pass the channel context that needs its
|
|
|
- queues to be stopped.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit 4f82eecf73019c27537f65c160e90385e159afd8
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:16 2014 +0530
|
|
|
-
|
|
|
- ath9k: Fix queue handling in flush()
|
|
|
-
|
|
|
- When draining of the TX queues fails, a
|
|
|
- full HW reset is done. ath_reset() makes sure
|
|
|
- that the queues in mac80211 are restarted,
|
|
|
- so there is no need to wake them up again.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit 60913f4d2951f6410eed969aae4717c7ced37044
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:15 2014 +0530
|
|
|
-
|
|
|
- ath9k: Remove duplicate code
|
|
|
-
|
|
|
- ath9k_has_tx_pending() can be used to
|
|
|
- check if there are pending frames instead
|
|
|
- of having duplicate code.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit fc1314c75e0558c03cb434e2af2c257caa201e76
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:14 2014 +0530
|
|
|
-
|
|
|
- ath9k: Fix pending frame check
|
|
|
-
|
|
|
- Checking for the queue depth outside of
|
|
|
- the TX queue lock is incorrect and in this
|
|
|
- case, is not required since it is done inside
|
|
|
- ath9k_has_pending_frames().
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit b736728575af03488388e84fceac7bf0eac5dbb6
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:13 2014 +0530
|
|
|
-
|
|
|
- ath9k: Check pending frames properly
|
|
|
-
|
|
|
- There is no need to check if the current
|
|
|
- channel context has active ACs queued up
|
|
|
- if the TX queue is not empty.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit 4b60af4ab4363bd79eeba94bb6bed396cf2aaf62
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Thu Oct 2 06:33:12 2014 +0530
|
|
|
-
|
|
|
- ath9k: Print RoC expiration
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit 4d9f634b02e4240f86719f30e4c9e62f6a4c4d36
|
|
|
-Author: Sujith Manoharan <[email protected]>
|
|
|
-Date: Tue Sep 30 14:15:23 2014 +0530
|
|
|
-
|
|
|
- ath9k: Check early for HW reset
|
|
|
-
|
|
|
- chan_lock is not required for checking if
|
|
|
- we are in the middle of a HW reset, so do it
|
|
|
- early. This also removes the small window
|
|
|
- where the lock is dropped and reacquired.
|
|
|
-
|
|
|
- Signed-off-by: Sujith Manoharan <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit c393d179924685d5c8c72446c5b6401f25fdb2a0
|
|
|
-Author: Marek Puzyniak <[email protected]>
|
|
|
-Date: Tue Oct 7 17:04:30 2014 +0200
|
|
|
-
|
|
|
- ath9k_htc: avoid kernel panic in ath9k_hw_reset
|
|
|
-
|
|
|
- hw pointer of ath_hw is not assigned to proper value
|
|
|
- in function ath9k_hw_reset what finally causes kernel panic.
|
|
|
- This can be solved by proper initialization of ath_hw in
|
|
|
- ath9k_init_priv.
|
|
|
-
|
|
|
- Signed-off-by: Marek Puzyniak <[email protected]>
|
|
|
- Acked-by: Oleksij Rempel <[email protected]>
|
|
|
- Signed-off-by: John W. Linville <[email protected]>
|
|
|
-
|
|
|
-commit 065e0b64f71632f5ad7f00c102fde09c534cfbf0
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Tue Sep 30 11:00:33 2014 +0200
|
|
|
-
|
|
|
- ath9k: fix getting tx duration for dynack
|
|
|
-
|
|
|
- On AR9003, tx control and tx status are in separate descriptor rings.
|
|
|
- Tx duration is extracted from the tx control descriptor data, which
|
|
|
- ar9003_hw_proc_txdesc cannot access.
|
|
|
-
|
|
|
- Fix getting the duration by adding a separate callback for it.
|
|
|
-
|
|
|
- Acked-by: Lorenzo Bianconi <[email protected]>
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit fdf9a4517b60d847b9bc0a30249efd96559fa450
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Tue Sep 9 09:48:30 2014 +0200
|
|
|
-
|
|
|
- ath9k_hw: fix PLL clock initialization for newer SoC
|
|
|
-
|
|
|
- On AR934x and newer SoC devices, the layout of the AR_RTC_PLL_CONTROL
|
|
|
- register changed. This currently breaks at least 5/10 MHz operation.
|
|
|
- AR933x uses the old layout.
|
|
|
-
|
|
|
- It might also have been causing other stability issues because of the
|
|
|
- different location of the PLL_BYPASS bit which needs to be set during
|
|
|
- PLL clock initialization.
|
|
|
-
|
|
|
- This patch also removes more instances of hardcoded register values in
|
|
|
- favor of properly computed ones with the PLL_BYPASS bit added.
|
|
|
-
|
|
|
- Reported-by: Lorenzo Bianconi <[email protected]>
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit b6d1f51cd8bdc9d952147a960fbf1f261d8e4188
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Mon Sep 8 18:35:08 2014 +0200
|
|
|
-
|
|
|
- ath9k_hw: reduce ANI spur immunity setting on HT40 extension channel
|
|
|
-
|
|
|
- The cycpwr_thr1 value needs to be lower on the extension channel than on
|
|
|
- the control channel, similar to how the register settings are programmed
|
|
|
- in the initvals.
|
|
|
-
|
|
|
- Also drop the unnecessary check for HT40 - this register can always be
|
|
|
- written. This patch has been reported to improve HT40 stability and
|
|
|
- throughput in some environments.
|
|
|
-
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit 5ad2dfbaa19aa45d29184d30c8c5dae0e110074a
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Mon Sep 8 18:31:26 2014 +0200
|
|
|
-
|
|
|
- Revert "ath9k_hw: reduce ANI firstep range for older chips"
|
|
|
-
|
|
|
- This reverts commit 09efc56345be4146ab9fc87a55c837ed5d6ea1ab
|
|
|
-
|
|
|
- I've received reports that this change is decreasing throughput in some
|
|
|
- rare conditions on an AR9280 based device
|
|
|
-
|
|
|
- Cc: [email protected]
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit 4c82fc569cf2f29e6c66d98ef4a1b0f3b6a98e9d
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Sat Sep 27 22:39:27 2014 +0200
|
|
|
-
|
|
|
- ath9k_hw: disable hardware ad-hoc flag on ar934x rev 3
|
|
|
-
|
|
|
- On AR934x rev 3, settin the ad-hoc flag completely messes up hardware
|
|
|
- state - beacons get stuck, almost no packets make it out, hardware is
|
|
|
- constantly reset.
|
|
|
-
|
|
|
- When leaving out that flag and setting up the hw like in AP mode, TSF
|
|
|
- timers won't be automatically synced, but at least the rest works.
|
|
|
-
|
|
|
- AR934x rev 2 and older are not affected by this bug
|
|
|
-
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit ecfb4b3fff006372ac5c40871f9bb182fd00444f
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Sat Sep 27 22:15:43 2014 +0200
|
|
|
-
|
|
|
- ath9k: use ah->get_mac_revision for all SoC devices if available
|
|
|
-
|
|
|
- It is needed for AR934x as well
|
|
|
-
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit c11113bc25df22898fb995d3205bdc4f27c98073
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Sat Sep 27 18:04:58 2014 +0200
|
|
|
-
|
|
|
- ath5k: add missing include for debug code
|
|
|
-
|
|
|
- Needed for calling vmalloc()/vfree()
|
|
|
-
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit 83f76a9f9a42773c7eef90bb86b4b2c16b0b3755
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Sat Sep 27 15:58:51 2014 +0200
|
|
|
-
|
|
|
- ath5k: fix AHB kconfig dependency
|
|
|
-
|
|
|
- Signed-off-by: Felix Fietkau <[email protected]>
|
|
|
-
|
|
|
-commit ddd67f2a5cfd73fad4b78190025402d419b9f0a9
|
|
|
-Author: Felix Fietkau <[email protected]>
|
|
|
-Date: Sat Sep 27 15:57:09 2014 +0200
|
|
|
-
|
|
|
- Revert "ath5k: Remove AHB bus support"
|
|
|
-
|
|
|
- This reverts commit 093ec3c5337434f40d77c1af06c139da3e5ba6dc.
|
|
|
-
|
|
|
---- a/drivers/net/wireless/ath/ath5k/Kconfig
|
|
|
-+++ b/drivers/net/wireless/ath/ath5k/Kconfig
|
|
|
-@@ -2,12 +2,14 @@ config ATH5K
|
|
|
- tristate "Atheros 5xxx wireless cards support"
|
|
|
- depends on m
|
|
|
- depends on PCI && MAC80211
|
|
|
-+ depends on (PCI || ATHEROS_AR231X) && MAC80211
|
|
|
- select ATH_COMMON
|
|
|
- select MAC80211_LEDS
|
|
|
- select BACKPORT_LEDS_CLASS
|
|
|
- select BACKPORT_NEW_LEDS
|
|
|
- select BACKPORT_AVERAGE
|
|
|
-- select ATH5K_PCI
|
|
|
-+ select ATH5K_AHB if ATHEROS_AR231X
|
|
|
-+ select ATH5K_PCI if !ATHEROS_AR231X
|
|
|
- ---help---
|
|
|
- This module adds support for wireless adapters based on
|
|
|
- Atheros 5xxx chipset.
|
|
|
-@@ -52,9 +54,16 @@ config ATH5K_TRACER
|
|
|
-
|
|
|
- If unsure, say N.
|
|
|
-
|
|
|
-+config ATH5K_AHB
|
|
|
-+ bool "Atheros 5xxx AHB bus support"
|
|
|
-+ depends on ATHEROS_AR231X
|
|
|
-+ ---help---
|
|
|
-+ This adds support for WiSoC type chipsets of the 5xxx Atheros
|
|
|
-+ family.
|
|
|
-+
|
|
|
- config ATH5K_PCI
|
|
|
- bool "Atheros 5xxx PCI bus support"
|
|
|
-- depends on PCI
|
|
|
-+ depends on !ATHEROS_AR231X
|
|
|
- ---help---
|
|
|
- This adds support for PCI type chipsets of the 5xxx Atheros
|
|
|
- family.
|
|
|
---- /dev/null
|
|
|
-+++ b/drivers/net/wireless/ath/ath5k/ahb.c
|
|
|
-@@ -0,0 +1,234 @@
|
|
|
-+/*
|
|
|
-+ * Copyright (c) 2008-2009 Atheros Communications Inc.
|
|
|
-+ * Copyright (c) 2009 Gabor Juhos <[email protected]>
|
|
|
-+ * Copyright (c) 2009 Imre Kaloz <[email protected]>
|
|
|
-+ *
|
|
|
-+ * 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 <linux/nl80211.h>
|
|
|
-+#include <linux/platform_device.h>
|
|
|
-+#include <linux/etherdevice.h>
|
|
|
-+#include <linux/export.h>
|
|
|
-+#include <ar231x_platform.h>
|
|
|
-+#include "ath5k.h"
|
|
|
-+#include "debug.h"
|
|
|
-+#include "base.h"
|
|
|
-+#include "reg.h"
|
|
|
-+
|
|
|
-+/* return bus cachesize in 4B word units */
|
|
|
-+static void ath5k_ahb_read_cachesize(struct ath_common *common, int *csz)
|
|
|
-+{
|
|
|
-+ *csz = L1_CACHE_BYTES >> 2;
|
|
|
-+}
|
|
|
-+
|
|
|
-+static bool
|
|
|
-+ath5k_ahb_eeprom_read(struct ath_common *common, u32 off, u16 *data)
|
|
|
-+{
|
|
|
-+ struct ath5k_hw *ah = common->priv;
|
|
|
-+ struct platform_device *pdev = to_platform_device(ah->dev);
|
|
|
-+ struct ar231x_board_config *bcfg = dev_get_platdata(&pdev->dev);
|
|
|
-+ u16 *eeprom, *eeprom_end;
|
|
|
-+
|
|
|
-+ eeprom = (u16 *) bcfg->radio;
|
|
|
-+ eeprom_end = ((void *) bcfg->config) + BOARD_CONFIG_BUFSZ;
|
|
|
-+
|
|
|
-+ eeprom += off;
|
|
|
-+ if (eeprom > eeprom_end)
|
|
|
-+ return false;
|
|
|
-+
|
|
|
-+ *data = *eeprom;
|
|
|
-+ return true;
|
|
|
-+}
|
|
|
-+
|
|
|
-+int ath5k_hw_read_srev(struct ath5k_hw *ah)
|
|
|
-+{
|
|
|
-+ struct platform_device *pdev = to_platform_device(ah->dev);
|
|
|
-+ struct ar231x_board_config *bcfg = dev_get_platdata(&pdev->dev);
|
|
|
-+ ah->ah_mac_srev = bcfg->devid;
|
|
|
-+ return 0;
|
|
|
-+}
|
|
|
-+
|
|
|
-+static int ath5k_ahb_eeprom_read_mac(struct ath5k_hw *ah, u8 *mac)
|
|
|
-+{
|
|
|
-+ struct platform_device *pdev = to_platform_device(ah->dev);
|
|
|
-+ struct ar231x_board_config *bcfg = dev_get_platdata(&pdev->dev);
|
|
|
-+ u8 *cfg_mac;
|
|
|
-+
|
|
|
-+ if (to_platform_device(ah->dev)->id == 0)
|
|
|
-+ cfg_mac = bcfg->config->wlan0_mac;
|
|
|
-+ else
|
|
|
-+ cfg_mac = bcfg->config->wlan1_mac;
|
|
|
-+
|
|
|
-+ memcpy(mac, cfg_mac, ETH_ALEN);
|
|
|
-+ return 0;
|
|
|
-+}
|
|
|
-+
|
|
|
-+static const struct ath_bus_ops ath_ahb_bus_ops = {
|
|
|
-+ .ath_bus_type = ATH_AHB,
|
|
|
-+ .read_cachesize = ath5k_ahb_read_cachesize,
|
|
|
-+ .eeprom_read = ath5k_ahb_eeprom_read,
|
|
|
-+ .eeprom_read_mac = ath5k_ahb_eeprom_read_mac,
|
|
|
-+};
|
|
|
-+
|
|
|
-+/*Initialization*/
|
|
|
-+static int ath_ahb_probe(struct platform_device *pdev)
|
|
|
-+{
|
|
|
-+ struct ar231x_board_config *bcfg = dev_get_platdata(&pdev->dev);
|
|
|
-+ struct ath5k_hw *ah;
|
|
|
-+ struct ieee80211_hw *hw;
|
|
|
-+ struct resource *res;
|
|
|
-+ void __iomem *mem;
|
|
|
-+ int irq;
|
|
|
-+ int ret = 0;
|
|
|
-+ u32 reg;
|
|
|
-+
|
|
|
-+ if (!dev_get_platdata(&pdev->dev)) {
|
|
|
-+ dev_err(&pdev->dev, "no platform data specified\n");
|
|
|
-+ ret = -EINVAL;
|
|
|
-+ goto err_out;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
|
-+ if (res == NULL) {
|
|
|
-+ dev_err(&pdev->dev, "no memory resource found\n");
|
|
|
-+ ret = -ENXIO;
|
|
|
-+ goto err_out;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ mem = ioremap_nocache(res->start, resource_size(res));
|
|
|
-+ if (mem == NULL) {
|
|
|
-+ dev_err(&pdev->dev, "ioremap failed\n");
|
|
|
-+ ret = -ENOMEM;
|
|
|
-+ goto err_out;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
|
|
|
-+ if (res == NULL) {
|
|
|
-+ dev_err(&pdev->dev, "no IRQ resource found\n");
|
|
|
-+ ret = -ENXIO;
|
|
|
-+ goto err_iounmap;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ irq = res->start;
|
|
|
-+
|
|
|
-+ hw = ieee80211_alloc_hw(sizeof(struct ath5k_hw), &ath5k_hw_ops);
|
|
|
-+ if (hw == NULL) {
|
|
|
-+ dev_err(&pdev->dev, "no memory for ieee80211_hw\n");
|
|
|
-+ ret = -ENOMEM;
|
|
|
-+ goto err_iounmap;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ ah = hw->priv;
|
|
|
-+ ah->hw = hw;
|
|
|
-+ ah->dev = &pdev->dev;
|
|
|
-+ ah->iobase = mem;
|
|
|
-+ ah->irq = irq;
|
|
|
-+ ah->devid = bcfg->devid;
|
|
|
-+
|
|
|
-+ if (bcfg->devid >= AR5K_SREV_AR2315_R6) {
|
|
|
-+ /* Enable WMAC AHB arbitration */
|
|
|
-+ reg = ioread32((void __iomem *) AR5K_AR2315_AHB_ARB_CTL);
|
|
|
-+ reg |= AR5K_AR2315_AHB_ARB_CTL_WLAN;
|
|
|
-+ iowrite32(reg, (void __iomem *) AR5K_AR2315_AHB_ARB_CTL);
|
|
|
-+
|
|
|
-+ /* Enable global WMAC swapping */
|
|
|
-+ reg = ioread32((void __iomem *) AR5K_AR2315_BYTESWAP);
|
|
|
-+ reg |= AR5K_AR2315_BYTESWAP_WMAC;
|
|
|
-+ iowrite32(reg, (void __iomem *) AR5K_AR2315_BYTESWAP);
|
|
|
-+ } else {
|
|
|
-+ /* Enable WMAC DMA access (assuming 5312 or 231x*/
|
|
|
-+ /* TODO: check other platforms */
|
|
|
-+ reg = ioread32((void __iomem *) AR5K_AR5312_ENABLE);
|
|
|
-+ if (to_platform_device(ah->dev)->id == 0)
|
|
|
-+ reg |= AR5K_AR5312_ENABLE_WLAN0;
|
|
|
-+ else
|
|
|
-+ reg |= AR5K_AR5312_ENABLE_WLAN1;
|
|
|
-+ iowrite32(reg, (void __iomem *) AR5K_AR5312_ENABLE);
|
|
|
-+
|
|
|
-+ /*
|
|
|
-+ * On a dual-band AR5312, the multiband radio is only
|
|
|
-+ * used as pass-through. Disable 2 GHz support in the
|
|
|
-+ * driver for it
|
|
|
-+ */
|
|
|
-+ if (to_platform_device(ah->dev)->id == 0 &&
|
|
|
-+ (bcfg->config->flags & (BD_WLAN0 | BD_WLAN1)) ==
|
|
|
-+ (BD_WLAN1 | BD_WLAN0))
|
|
|
-+ ah->ah_capabilities.cap_needs_2GHz_ovr = true;
|
|
|
-+ else
|
|
|
-+ ah->ah_capabilities.cap_needs_2GHz_ovr = false;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ ret = ath5k_init_ah(ah, &ath_ahb_bus_ops);
|
|
|
-+ if (ret != 0) {
|
|
|
-+ dev_err(&pdev->dev, "failed to attach device, err=%d\n", ret);
|
|
|
-+ ret = -ENODEV;
|
|
|
-+ goto err_free_hw;
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ platform_set_drvdata(pdev, hw);
|
|
|
-+
|
|
|
-+ return 0;
|
|
|
-+
|
|
|
-+ err_free_hw:
|
|
|
-+ ieee80211_free_hw(hw);
|
|
|
-+ err_iounmap:
|
|
|
-+ iounmap(mem);
|
|
|
-+ err_out:
|
|
|
-+ return ret;
|
|
|
-+}
|
|
|
-+
|
|
|
-+static int ath_ahb_remove(struct platform_device *pdev)
|
|
|
-+{
|
|
|
-+ struct ar231x_board_config *bcfg = dev_get_platdata(&pdev->dev);
|
|
|
-+ struct ieee80211_hw *hw = platform_get_drvdata(pdev);
|
|
|
-+ struct ath5k_hw *ah;
|
|
|
-+ u32 reg;
|
|
|
-+
|
|
|
-+ if (!hw)
|
|
|
-+ return 0;
|
|
|
-+
|
|
|
-+ ah = hw->priv;
|
|
|
-+
|
|
|
-+ if (bcfg->devid >= AR5K_SREV_AR2315_R6) {
|
|
|
-+ /* Disable WMAC AHB arbitration */
|
|
|
-+ reg = ioread32((void __iomem *) AR5K_AR2315_AHB_ARB_CTL);
|
|
|
-+ reg &= ~AR5K_AR2315_AHB_ARB_CTL_WLAN;
|
|
|
-+ iowrite32(reg, (void __iomem *) AR5K_AR2315_AHB_ARB_CTL);
|
|
|
-+ } else {
|
|
|
-+ /*Stop DMA access */
|
|
|
-+ reg = ioread32((void __iomem *) AR5K_AR5312_ENABLE);
|
|
|
-+ if (to_platform_device(ah->dev)->id == 0)
|
|
|
-+ reg &= ~AR5K_AR5312_ENABLE_WLAN0;
|
|
|
-+ else
|
|
|
-+ reg &= ~AR5K_AR5312_ENABLE_WLAN1;
|
|
|
-+ iowrite32(reg, (void __iomem *) AR5K_AR5312_ENABLE);
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ ath5k_deinit_ah(ah);
|
|
|
-+ iounmap(ah->iobase);
|
|
|
-+ ieee80211_free_hw(hw);
|
|
|
-+
|
|
|
-+ return 0;
|
|
|
-+}
|
|
|
-+
|
|
|
-+static struct platform_driver ath_ahb_driver = {
|
|
|
-+ .probe = ath_ahb_probe,
|
|
|
-+ .remove = ath_ahb_remove,
|
|
|
-+ .driver = {
|
|
|
-+ .name = "ar231x-wmac",
|
|
|
-+ .owner = THIS_MODULE,
|
|
|
-+ },
|
|
|
-+};
|
|
|
-+
|
|
|
-+module_platform_driver(ath_ahb_driver);
|
|
|
---- a/drivers/net/wireless/ath/ath5k/ath5k.h
|
|
|
-+++ b/drivers/net/wireless/ath/ath5k/ath5k.h
|
|
|
-@@ -1647,6 +1647,32 @@ static inline struct ath_regulatory *ath
|
|
|
- return &(ath5k_hw_common(ah)->regulatory);
|
|
|
- }
|
|
|
-
|
|
|
-+#ifdef CONFIG_ATHEROS_AR231X
|
|
|
-+#define AR5K_AR2315_PCI_BASE ((void __iomem *)0xb0100000)
|
|
|
-+
|
|
|
-+static inline void __iomem *ath5k_ahb_reg(struct ath5k_hw *ah, u16 reg)
|
|
|
-+{
|
|
|
-+ /* On AR2315 and AR2317 the PCI clock domain registers
|
|
|
-+ * are outside of the WMAC register space */
|
|
|
-+ if (unlikely((reg >= 0x4000) && (reg < 0x5000) &&
|
|
|
-+ (ah->ah_mac_srev >= AR5K_SREV_AR2315_R6)))
|
|
|
-+ return AR5K_AR2315_PCI_BASE + reg;
|
|
|
-+
|
|
|
-+ return ah->iobase + reg;
|
|
|
-+}
|
|
|
-+
|
|
|
-+static inline u32 ath5k_hw_reg_read(struct ath5k_hw *ah, u16 reg)
|
|
|
-+{
|
|
|
-+ return ioread32(ath5k_ahb_reg(ah, reg));
|
|
|
-+}
|
|
|
-+
|
|
|
-+static inline void ath5k_hw_reg_write(struct ath5k_hw *ah, u32 val, u16 reg)
|
|
|
-+{
|
|
|
-+ iowrite32(val, ath5k_ahb_reg(ah, reg));
|
|
|
-+}
|
|
|
-+
|
|
|
-+#else
|
|
|
-+
|
|
|
- static inline u32 ath5k_hw_reg_read(struct ath5k_hw *ah, u16 reg)
|
|
|
- {
|
|
|
- return ioread32(ah->iobase + reg);
|
|
|
-@@ -1657,6 +1683,8 @@ static inline void ath5k_hw_reg_write(st
|
|
|
- iowrite32(val, ah->iobase + reg);
|
|
|
- }
|
|
|
-
|
|
|
-+#endif
|
|
|
-+
|
|
|
- static inline enum ath_bus_type ath5k_get_bus_type(struct ath5k_hw *ah)
|
|
|
- {
|
|
|
- return ath5k_hw_common(ah)->bus_ops->ath_bus_type;
|
|
|
---- a/drivers/net/wireless/ath/ath5k/base.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath5k/base.c
|
|
|
-@@ -99,6 +99,15 @@ static int ath5k_reset(struct ath5k_hw *
|
|
|
-
|
|
|
- /* Known SREVs */
|
|
|
- static const struct ath5k_srev_name srev_names[] = {
|
|
|
-+#ifdef CONFIG_ATHEROS_AR231X
|
|
|
-+ { "5312", AR5K_VERSION_MAC, AR5K_SREV_AR5312_R2 },
|
|
|
-+ { "5312", AR5K_VERSION_MAC, AR5K_SREV_AR5312_R7 },
|
|
|
-+ { "2313", AR5K_VERSION_MAC, AR5K_SREV_AR2313_R8 },
|
|
|
-+ { "2315", AR5K_VERSION_MAC, AR5K_SREV_AR2315_R6 },
|
|
|
-+ { "2315", AR5K_VERSION_MAC, AR5K_SREV_AR2315_R7 },
|
|
|
-+ { "2317", AR5K_VERSION_MAC, AR5K_SREV_AR2317_R1 },
|
|
|
-+ { "2317", AR5K_VERSION_MAC, AR5K_SREV_AR2317_R2 },
|
|
|
-+#else
|
|
|
- { "5210", AR5K_VERSION_MAC, AR5K_SREV_AR5210 },
|
|
|
- { "5311", AR5K_VERSION_MAC, AR5K_SREV_AR5311 },
|
|
|
- { "5311A", AR5K_VERSION_MAC, AR5K_SREV_AR5311A },
|
|
|
-@@ -117,6 +126,7 @@ static const struct ath5k_srev_name srev
|
|
|
- { "5418", AR5K_VERSION_MAC, AR5K_SREV_AR5418 },
|
|
|
- { "2425", AR5K_VERSION_MAC, AR5K_SREV_AR2425 },
|
|
|
- { "2417", AR5K_VERSION_MAC, AR5K_SREV_AR2417 },
|
|
|
-+#endif
|
|
|
- { "xxxxx", AR5K_VERSION_MAC, AR5K_SREV_UNKNOWN },
|
|
|
- { "5110", AR5K_VERSION_RAD, AR5K_SREV_RAD_5110 },
|
|
|
- { "5111", AR5K_VERSION_RAD, AR5K_SREV_RAD_5111 },
|
|
|
-@@ -132,6 +142,10 @@ static const struct ath5k_srev_name srev
|
|
|
- { "5413", AR5K_VERSION_RAD, AR5K_SREV_RAD_5413 },
|
|
|
- { "5424", AR5K_VERSION_RAD, AR5K_SREV_RAD_5424 },
|
|
|
- { "5133", AR5K_VERSION_RAD, AR5K_SREV_RAD_5133 },
|
|
|
-+#ifdef CONFIG_ATHEROS_AR231X
|
|
|
-+ { "2316", AR5K_VERSION_RAD, AR5K_SREV_RAD_2316 },
|
|
|
-+ { "2317", AR5K_VERSION_RAD, AR5K_SREV_RAD_2317 },
|
|
|
-+#endif
|
|
|
- { "xxxxx", AR5K_VERSION_RAD, AR5K_SREV_UNKNOWN },
|
|
|
- };
|
|
|
-
|
|
|
---- a/drivers/net/wireless/ath/ath5k/led.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath5k/led.c
|
|
|
-@@ -163,14 +163,20 @@ int ath5k_init_leds(struct ath5k_hw *ah)
|
|
|
- {
|
|
|
- int ret = 0;
|
|
|
- struct ieee80211_hw *hw = ah->hw;
|
|
|
-+#ifndef CONFIG_ATHEROS_AR231X
|
|
|
- struct pci_dev *pdev = ah->pdev;
|
|
|
-+#endif
|
|
|
- char name[ATH5K_LED_MAX_NAME_LEN + 1];
|
|
|
- const struct pci_device_id *match;
|
|
|
-
|
|
|
- if (!ah->pdev)
|
|
|
- return 0;
|
|
|
-
|
|
|
-+#ifdef CONFIG_ATHEROS_AR231X
|
|
|
-+ match = NULL;
|
|
|
-+#else
|
|
|
- match = pci_match_id(&ath5k_led_devices[0], pdev);
|
|
|
-+#endif
|
|
|
- if (match) {
|
|
|
- __set_bit(ATH_STAT_LEDSOFT, ah->status);
|
|
|
- ah->led_pin = ATH_PIN(match->driver_data);
|
|
|
---- a/drivers/net/wireless/ath/ath5k/debug.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath5k/debug.c
|
|
|
-@@ -65,6 +65,7 @@
|
|
|
-
|
|
|
- #include <linux/seq_file.h>
|
|
|
- #include <linux/list.h>
|
|
|
-+#include <linux/vmalloc.h>
|
|
|
- #include "debug.h"
|
|
|
- #include "ath5k.h"
|
|
|
- #include "reg.h"
|
|
|
---- a/drivers/net/wireless/ath/ath9k/hw.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/hw.c
|
|
|
-@@ -222,31 +222,28 @@ static void ath9k_hw_read_revisions(stru
|
|
|
- {
|
|
|
- u32 val;
|
|
|
-
|
|
|
-+ if (ah->get_mac_revision)
|
|
|
-+ ah->hw_version.macRev = ah->get_mac_revision();
|
|
|
-+
|
|
|
- switch (ah->hw_version.devid) {
|
|
|
- case AR5416_AR9100_DEVID:
|
|
|
- ah->hw_version.macVersion = AR_SREV_VERSION_9100;
|
|
|
- break;
|
|
|
- case AR9300_DEVID_AR9330:
|
|
|
- ah->hw_version.macVersion = AR_SREV_VERSION_9330;
|
|
|
-- if (ah->get_mac_revision) {
|
|
|
-- ah->hw_version.macRev = ah->get_mac_revision();
|
|
|
-- } else {
|
|
|
-+ if (!ah->get_mac_revision) {
|
|
|
- val = REG_READ(ah, AR_SREV);
|
|
|
- ah->hw_version.macRev = MS(val, AR_SREV_REVISION2);
|
|
|
- }
|
|
|
- return;
|
|
|
- case AR9300_DEVID_AR9340:
|
|
|
- ah->hw_version.macVersion = AR_SREV_VERSION_9340;
|
|
|
-- val = REG_READ(ah, AR_SREV);
|
|
|
-- ah->hw_version.macRev = MS(val, AR_SREV_REVISION2);
|
|
|
- return;
|
|
|
- case AR9300_DEVID_QCA955X:
|
|
|
- ah->hw_version.macVersion = AR_SREV_VERSION_9550;
|
|
|
- return;
|
|
|
- case AR9300_DEVID_AR953X:
|
|
|
- ah->hw_version.macVersion = AR_SREV_VERSION_9531;
|
|
|
-- if (ah->get_mac_revision)
|
|
|
-- ah->hw_version.macRev = ah->get_mac_revision();
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
-@@ -704,6 +701,8 @@ static void ath9k_hw_init_pll(struct ath
|
|
|
- {
|
|
|
- u32 pll;
|
|
|
-
|
|
|
-+ pll = ath9k_hw_compute_pll_control(ah, chan);
|
|
|
-+
|
|
|
- if (AR_SREV_9485(ah) || AR_SREV_9565(ah)) {
|
|
|
- /* program BB PLL ki and kd value, ki=0x4, kd=0x40 */
|
|
|
- REG_RMW_FIELD(ah, AR_CH0_BB_DPLL2,
|
|
|
-@@ -754,7 +753,8 @@ static void ath9k_hw_init_pll(struct ath
|
|
|
- REG_RMW_FIELD(ah, AR_CH0_DDR_DPLL3,
|
|
|
- AR_CH0_DPLL3_PHASE_SHIFT, 0x1);
|
|
|
-
|
|
|
-- REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x1142c);
|
|
|
-+ REG_WRITE(ah, AR_RTC_PLL_CONTROL,
|
|
|
-+ pll | AR_RTC_9300_PLL_BYPASS);
|
|
|
- udelay(1000);
|
|
|
-
|
|
|
- /* program refdiv, nint, frac to RTC register */
|
|
|
-@@ -770,7 +770,8 @@ static void ath9k_hw_init_pll(struct ath
|
|
|
- } else if (AR_SREV_9340(ah) || AR_SREV_9550(ah) || AR_SREV_9531(ah)) {
|
|
|
- u32 regval, pll2_divint, pll2_divfrac, refdiv;
|
|
|
-
|
|
|
-- REG_WRITE(ah, AR_RTC_PLL_CONTROL, 0x1142c);
|
|
|
-+ REG_WRITE(ah, AR_RTC_PLL_CONTROL,
|
|
|
-+ pll | AR_RTC_9300_SOC_PLL_BYPASS);
|
|
|
- udelay(1000);
|
|
|
-
|
|
|
- REG_SET_BIT(ah, AR_PHY_PLL_MODE, 0x1 << 16);
|
|
|
-@@ -843,7 +844,6 @@ static void ath9k_hw_init_pll(struct ath
|
|
|
- udelay(1000);
|
|
|
- }
|
|
|
-
|
|
|
-- pll = ath9k_hw_compute_pll_control(ah, chan);
|
|
|
- if (AR_SREV_9565(ah))
|
|
|
- pll |= 0x40000;
|
|
|
- REG_WRITE(ah, AR_RTC_PLL_CONTROL, pll);
|
|
|
-@@ -1192,9 +1192,12 @@ static void ath9k_hw_set_operating_mode(
|
|
|
-
|
|
|
- switch (opmode) {
|
|
|
- case NL80211_IFTYPE_ADHOC:
|
|
|
-- set |= AR_STA_ID1_ADHOC;
|
|
|
-- REG_SET_BIT(ah, AR_CFG, AR_CFG_AP_ADHOC_INDICATION);
|
|
|
-- break;
|
|
|
-+ if (!AR_SREV_9340_13(ah)) {
|
|
|
-+ set |= AR_STA_ID1_ADHOC;
|
|
|
-+ REG_SET_BIT(ah, AR_CFG, AR_CFG_AP_ADHOC_INDICATION);
|
|
|
-+ break;
|
|
|
-+ }
|
|
|
-+ /* fall through */
|
|
|
- case NL80211_IFTYPE_MESH_POINT:
|
|
|
- case NL80211_IFTYPE_AP:
|
|
|
- set |= AR_STA_ID1_STA_AP;
|
|
|
---- a/drivers/net/wireless/ath/ath9k/reg.h
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/reg.h
|
|
|
-@@ -903,6 +903,10 @@
|
|
|
- #define AR_SREV_9340(_ah) \
|
|
|
- (((_ah)->hw_version.macVersion == AR_SREV_VERSION_9340))
|
|
|
-
|
|
|
-+#define AR_SREV_9340_13(_ah) \
|
|
|
-+ (AR_SREV_9340((_ah)) && \
|
|
|
-+ ((_ah)->hw_version.macRev == AR_SREV_REVISION_9340_13))
|
|
|
-+
|
|
|
- #define AR_SREV_9340_13_OR_LATER(_ah) \
|
|
|
- (AR_SREV_9340((_ah)) && \
|
|
|
- ((_ah)->hw_version.macRev >= AR_SREV_REVISION_9340_13))
|
|
|
-@@ -1240,12 +1244,23 @@ enum {
|
|
|
- #define AR_CH0_DPLL3_PHASE_SHIFT_S 23
|
|
|
- #define AR_PHY_CCA_NOM_VAL_2GHZ -118
|
|
|
-
|
|
|
-+#define AR_RTC_9300_SOC_PLL_DIV_INT 0x0000003f
|
|
|
-+#define AR_RTC_9300_SOC_PLL_DIV_INT_S 0
|
|
|
-+#define AR_RTC_9300_SOC_PLL_DIV_FRAC 0x000fffc0
|
|
|
-+#define AR_RTC_9300_SOC_PLL_DIV_FRAC_S 6
|
|
|
-+#define AR_RTC_9300_SOC_PLL_REFDIV 0x01f00000
|
|
|
-+#define AR_RTC_9300_SOC_PLL_REFDIV_S 20
|
|
|
-+#define AR_RTC_9300_SOC_PLL_CLKSEL 0x06000000
|
|
|
-+#define AR_RTC_9300_SOC_PLL_CLKSEL_S 25
|
|
|
-+#define AR_RTC_9300_SOC_PLL_BYPASS 0x08000000
|
|
|
-+
|
|
|
- #define AR_RTC_9300_PLL_DIV 0x000003ff
|
|
|
- #define AR_RTC_9300_PLL_DIV_S 0
|
|
|
- #define AR_RTC_9300_PLL_REFDIV 0x00003C00
|
|
|
- #define AR_RTC_9300_PLL_REFDIV_S 10
|
|
|
- #define AR_RTC_9300_PLL_CLKSEL 0x0000C000
|
|
|
- #define AR_RTC_9300_PLL_CLKSEL_S 14
|
|
|
-+#define AR_RTC_9300_PLL_BYPASS 0x00010000
|
|
|
-
|
|
|
- #define AR_RTC_9160_PLL_DIV 0x000003ff
|
|
|
- #define AR_RTC_9160_PLL_DIV_S 0
|
|
|
---- a/drivers/net/wireless/ath/ath9k/ar5008_phy.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/ar5008_phy.c
|
|
|
-@@ -1004,9 +1004,11 @@ static bool ar5008_hw_ani_control_new(st
|
|
|
- case ATH9K_ANI_FIRSTEP_LEVEL:{
|
|
|
- u32 level = param;
|
|
|
-
|
|
|
-- value = level;
|
|
|
-+ value = level * 2;
|
|
|
- REG_RMW_FIELD(ah, AR_PHY_FIND_SIG,
|
|
|
- AR_PHY_FIND_SIG_FIRSTEP, value);
|
|
|
-+ REG_RMW_FIELD(ah, AR_PHY_FIND_SIG_LOW,
|
|
|
-+ AR_PHY_FIND_SIG_FIRSTEP_LOW, value);
|
|
|
-
|
|
|
- if (level != aniState->firstepLevel) {
|
|
|
- ath_dbg(common, ANI,
|
|
|
-@@ -1040,9 +1042,8 @@ static bool ar5008_hw_ani_control_new(st
|
|
|
- REG_RMW_FIELD(ah, AR_PHY_TIMING5,
|
|
|
- AR_PHY_TIMING5_CYCPWR_THR1, value);
|
|
|
-
|
|
|
-- if (IS_CHAN_HT40(ah->curchan))
|
|
|
-- REG_RMW_FIELD(ah, AR_PHY_EXT_CCA,
|
|
|
-- AR_PHY_EXT_TIMING5_CYCPWR_THR1, value);
|
|
|
-+ REG_RMW_FIELD(ah, AR_PHY_EXT_CCA,
|
|
|
-+ AR_PHY_EXT_TIMING5_CYCPWR_THR1, value - 1);
|
|
|
-
|
|
|
- if (level != aniState->spurImmunityLevel) {
|
|
|
- ath_dbg(common, ANI,
|
|
|
---- a/drivers/net/wireless/ath/ath9k/ar9003_phy.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/ar9003_phy.c
|
|
|
-@@ -517,6 +517,23 @@ static void ar9003_hw_spur_mitigate(stru
|
|
|
- ar9003_hw_spur_mitigate_ofdm(ah, chan);
|
|
|
- }
|
|
|
-
|
|
|
-+static u32 ar9003_hw_compute_pll_control_soc(struct ath_hw *ah,
|
|
|
-+ struct ath9k_channel *chan)
|
|
|
-+{
|
|
|
-+ u32 pll;
|
|
|
-+
|
|
|
-+ pll = SM(0x5, AR_RTC_9300_SOC_PLL_REFDIV);
|
|
|
-+
|
|
|
-+ if (chan && IS_CHAN_HALF_RATE(chan))
|
|
|
-+ pll |= SM(0x1, AR_RTC_9300_SOC_PLL_CLKSEL);
|
|
|
-+ else if (chan && IS_CHAN_QUARTER_RATE(chan))
|
|
|
-+ pll |= SM(0x2, AR_RTC_9300_SOC_PLL_CLKSEL);
|
|
|
-+
|
|
|
-+ pll |= SM(0x2c, AR_RTC_9300_SOC_PLL_DIV_INT);
|
|
|
-+
|
|
|
-+ return pll;
|
|
|
-+}
|
|
|
-+
|
|
|
- static u32 ar9003_hw_compute_pll_control(struct ath_hw *ah,
|
|
|
- struct ath9k_channel *chan)
|
|
|
- {
|
|
|
-@@ -1781,7 +1798,12 @@ void ar9003_hw_attach_phy_ops(struct ath
|
|
|
-
|
|
|
- priv_ops->rf_set_freq = ar9003_hw_set_channel;
|
|
|
- priv_ops->spur_mitigate_freq = ar9003_hw_spur_mitigate;
|
|
|
-- priv_ops->compute_pll_control = ar9003_hw_compute_pll_control;
|
|
|
-+
|
|
|
-+ if (AR_SREV_9340(ah) || AR_SREV_9550(ah) || AR_SREV_9531(ah))
|
|
|
-+ priv_ops->compute_pll_control = ar9003_hw_compute_pll_control_soc;
|
|
|
-+ else
|
|
|
-+ priv_ops->compute_pll_control = ar9003_hw_compute_pll_control;
|
|
|
-+
|
|
|
- priv_ops->set_channel_regs = ar9003_hw_set_channel_regs;
|
|
|
- priv_ops->init_bb = ar9003_hw_init_bb;
|
|
|
- priv_ops->process_ini = ar9003_hw_process_ini;
|
|
|
---- a/drivers/net/wireless/ath/ath9k/ar9002_mac.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/ar9002_mac.c
|
|
|
-@@ -381,16 +381,27 @@ static int ar9002_hw_proc_txdesc(struct
|
|
|
- ts->evm1 = ads->AR_TxEVM1;
|
|
|
- ts->evm2 = ads->AR_TxEVM2;
|
|
|
-
|
|
|
-- status = ACCESS_ONCE(ads->ds_ctl4);
|
|
|
-- ts->duration[0] = MS(status, AR_PacketDur0);
|
|
|
-- ts->duration[1] = MS(status, AR_PacketDur1);
|
|
|
-- status = ACCESS_ONCE(ads->ds_ctl5);
|
|
|
-- ts->duration[2] = MS(status, AR_PacketDur2);
|
|
|
-- ts->duration[3] = MS(status, AR_PacketDur3);
|
|
|
--
|
|
|
- return 0;
|
|
|
- }
|
|
|
-
|
|
|
-+static int ar9002_hw_get_duration(struct ath_hw *ah, const void *ds, int index)
|
|
|
-+{
|
|
|
-+ struct ar5416_desc *ads = AR5416DESC(ds);
|
|
|
-+
|
|
|
-+ switch (index) {
|
|
|
-+ case 0:
|
|
|
-+ return MS(ACCESS_ONCE(ads->ds_ctl4), AR_PacketDur0);
|
|
|
-+ case 1:
|
|
|
-+ return MS(ACCESS_ONCE(ads->ds_ctl4), AR_PacketDur1);
|
|
|
-+ case 2:
|
|
|
-+ return MS(ACCESS_ONCE(ads->ds_ctl5), AR_PacketDur2);
|
|
|
-+ case 3:
|
|
|
-+ return MS(ACCESS_ONCE(ads->ds_ctl5), AR_PacketDur3);
|
|
|
-+ default:
|
|
|
-+ return -1;
|
|
|
-+ }
|
|
|
-+}
|
|
|
-+
|
|
|
- void ath9k_hw_setuprxdesc(struct ath_hw *ah, struct ath_desc *ds,
|
|
|
- u32 size, u32 flags)
|
|
|
- {
|
|
|
-@@ -413,4 +424,5 @@ void ar9002_hw_attach_mac_ops(struct ath
|
|
|
- ops->get_isr = ar9002_hw_get_isr;
|
|
|
- ops->set_txdesc = ar9002_set_txdesc;
|
|
|
- ops->proc_txdesc = ar9002_hw_proc_txdesc;
|
|
|
-+ ops->get_duration = ar9002_hw_get_duration;
|
|
|
- }
|
|
|
---- a/drivers/net/wireless/ath/ath9k/ar9003_mac.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/ar9003_mac.c
|
|
|
-@@ -355,11 +355,9 @@ static int ar9003_hw_proc_txdesc(struct
|
|
|
- struct ath_tx_status *ts)
|
|
|
- {
|
|
|
- struct ar9003_txs *ads;
|
|
|
-- struct ar9003_txc *adc;
|
|
|
- u32 status;
|
|
|
-
|
|
|
- ads = &ah->ts_ring[ah->ts_tail];
|
|
|
-- adc = (struct ar9003_txc *)ads;
|
|
|
-
|
|
|
- status = ACCESS_ONCE(ads->status8);
|
|
|
- if ((status & AR_TxDone) == 0)
|
|
|
-@@ -428,18 +426,29 @@ static int ar9003_hw_proc_txdesc(struct
|
|
|
- ts->ts_rssi_ext1 = MS(status, AR_TxRSSIAnt11);
|
|
|
- ts->ts_rssi_ext2 = MS(status, AR_TxRSSIAnt12);
|
|
|
-
|
|
|
-- status = ACCESS_ONCE(adc->ctl15);
|
|
|
-- ts->duration[0] = MS(status, AR_PacketDur0);
|
|
|
-- ts->duration[1] = MS(status, AR_PacketDur1);
|
|
|
-- status = ACCESS_ONCE(adc->ctl16);
|
|
|
-- ts->duration[2] = MS(status, AR_PacketDur2);
|
|
|
-- ts->duration[3] = MS(status, AR_PacketDur3);
|
|
|
--
|
|
|
- memset(ads, 0, sizeof(*ads));
|
|
|
-
|
|
|
- return 0;
|
|
|
- }
|
|
|
-
|
|
|
-+static int ar9003_hw_get_duration(struct ath_hw *ah, const void *ds, int index)
|
|
|
-+{
|
|
|
-+ const struct ar9003_txc *adc = ds;
|
|
|
-+
|
|
|
-+ switch (index) {
|
|
|
-+ case 0:
|
|
|
-+ return MS(ACCESS_ONCE(adc->ctl15), AR_PacketDur0);
|
|
|
-+ case 1:
|
|
|
-+ return MS(ACCESS_ONCE(adc->ctl15), AR_PacketDur1);
|
|
|
-+ case 2:
|
|
|
-+ return MS(ACCESS_ONCE(adc->ctl16), AR_PacketDur2);
|
|
|
-+ case 3:
|
|
|
-+ return MS(ACCESS_ONCE(adc->ctl16), AR_PacketDur3);
|
|
|
-+ default:
|
|
|
-+ return 0;
|
|
|
-+ }
|
|
|
-+}
|
|
|
-+
|
|
|
- void ar9003_hw_attach_mac_ops(struct ath_hw *hw)
|
|
|
- {
|
|
|
- struct ath_hw_ops *ops = ath9k_hw_ops(hw);
|
|
|
-@@ -449,6 +458,7 @@ void ar9003_hw_attach_mac_ops(struct ath
|
|
|
- ops->get_isr = ar9003_hw_get_isr;
|
|
|
- ops->set_txdesc = ar9003_set_txdesc;
|
|
|
- ops->proc_txdesc = ar9003_hw_proc_txdesc;
|
|
|
-+ ops->get_duration = ar9003_hw_get_duration;
|
|
|
- }
|
|
|
-
|
|
|
- void ath9k_hw_set_rx_bufsize(struct ath_hw *ah, u16 buf_size)
|
|
|
---- a/drivers/net/wireless/ath/ath9k/dynack.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/dynack.c
|
|
|
-@@ -202,7 +202,7 @@ void ath_dynack_sample_tx_ts(struct ath_
|
|
|
- ridx = ts->ts_rateindex;
|
|
|
-
|
|
|
- da->st_rbf.ts[da->st_rbf.t_rb].tstamp = ts->ts_tstamp;
|
|
|
-- da->st_rbf.ts[da->st_rbf.t_rb].dur = ts->duration[ts->ts_rateindex];
|
|
|
-+ da->st_rbf.ts[da->st_rbf.t_rb].dur = ts->duration;
|
|
|
- ether_addr_copy(da->st_rbf.addr[da->st_rbf.t_rb].h_dest, hdr->addr1);
|
|
|
- ether_addr_copy(da->st_rbf.addr[da->st_rbf.t_rb].h_src, hdr->addr2);
|
|
|
-
|
|
|
---- a/drivers/net/wireless/ath/ath9k/hw-ops.h
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/hw-ops.h
|
|
|
-@@ -67,6 +67,12 @@ static inline int ath9k_hw_txprocdesc(st
|
|
|
- return ath9k_hw_ops(ah)->proc_txdesc(ah, ds, ts);
|
|
|
- }
|
|
|
-
|
|
|
-+static inline int ath9k_hw_get_duration(struct ath_hw *ah, const void *ds,
|
|
|
-+ int index)
|
|
|
-+{
|
|
|
-+ return ath9k_hw_ops(ah)->get_duration(ah, ds, index);
|
|
|
-+}
|
|
|
-+
|
|
|
- static inline void ath9k_hw_antdiv_comb_conf_get(struct ath_hw *ah,
|
|
|
- struct ath_hw_antcomb_conf *antconf)
|
|
|
- {
|
|
|
---- a/drivers/net/wireless/ath/ath9k/hw.h
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/hw.h
|
|
|
-@@ -691,6 +691,7 @@ struct ath_hw_ops {
|
|
|
- struct ath_tx_info *i);
|
|
|
- int (*proc_txdesc)(struct ath_hw *ah, void *ds,
|
|
|
- struct ath_tx_status *ts);
|
|
|
-+ int (*get_duration)(struct ath_hw *ah, const void *ds, int index);
|
|
|
- void (*antdiv_comb_conf_get)(struct ath_hw *ah,
|
|
|
- struct ath_hw_antcomb_conf *antconf);
|
|
|
- void (*antdiv_comb_conf_set)(struct ath_hw *ah,
|
|
|
---- a/drivers/net/wireless/ath/ath9k/mac.h
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/mac.h
|
|
|
-@@ -121,7 +121,7 @@ struct ath_tx_status {
|
|
|
- u32 evm0;
|
|
|
- u32 evm1;
|
|
|
- u32 evm2;
|
|
|
-- u32 duration[4];
|
|
|
-+ u32 duration;
|
|
|
- };
|
|
|
-
|
|
|
- struct ath_rx_status {
|
|
|
---- a/drivers/net/wireless/ath/ath9k/xmit.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/xmit.c
|
|
|
-@@ -683,6 +683,8 @@ static void ath_tx_process_buffer(struct
|
|
|
- if (bf_is_ampdu_not_probing(bf))
|
|
|
- txq->axq_ampdu_depth--;
|
|
|
-
|
|
|
-+ ts->duration = ath9k_hw_get_duration(sc->sc_ah, bf->bf_desc,
|
|
|
-+ ts->ts_rateindex);
|
|
|
- if (!bf_isampdu(bf)) {
|
|
|
- if (!flush) {
|
|
|
- info = IEEE80211_SKB_CB(bf->bf_mpdu);
|
|
|
---- a/drivers/net/wireless/ath/ath9k/ath9k.h
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/ath9k.h
|
|
|
-@@ -455,7 +455,8 @@ void ath9k_p2p_bss_info_changed(struct a
|
|
|
- void ath9k_beacon_add_noa(struct ath_softc *sc, struct ath_vif *avp,
|
|
|
- struct sk_buff *skb);
|
|
|
- void ath9k_p2p_ps_timer(void *priv);
|
|
|
--void ath9k_chanctx_wake_queues(struct ath_softc *sc);
|
|
|
-+void ath9k_chanctx_wake_queues(struct ath_softc *sc, struct ath_chanctx *ctx);
|
|
|
-+void ath9k_chanctx_stop_queues(struct ath_softc *sc, struct ath_chanctx *ctx);
|
|
|
- void ath_chanctx_check_active(struct ath_softc *sc, struct ath_chanctx *ctx);
|
|
|
-
|
|
|
- void ath_chanctx_beacon_recv_ev(struct ath_softc *sc,
|
|
|
-@@ -525,7 +526,12 @@ static inline void ath9k_beacon_add_noa(
|
|
|
- static inline void ath9k_p2p_ps_timer(struct ath_softc *sc)
|
|
|
- {
|
|
|
- }
|
|
|
--static inline void ath9k_chanctx_wake_queues(struct ath_softc *sc)
|
|
|
-+static inline void ath9k_chanctx_wake_queues(struct ath_softc *sc,
|
|
|
-+ struct ath_chanctx *ctx)
|
|
|
-+{
|
|
|
-+}
|
|
|
-+static inline void ath9k_chanctx_stop_queues(struct ath_softc *sc,
|
|
|
-+ struct ath_chanctx *ctx)
|
|
|
- {
|
|
|
- }
|
|
|
- static inline void ath_chanctx_check_active(struct ath_softc *sc,
|
|
|
---- a/drivers/net/wireless/ath/ath9k/channel.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/channel.c
|
|
|
-@@ -761,6 +761,13 @@ void ath_offchannel_next(struct ath_soft
|
|
|
-
|
|
|
- void ath_roc_complete(struct ath_softc *sc, bool abort)
|
|
|
- {
|
|
|
-+ struct ath_common *common = ath9k_hw_common(sc->sc_ah);
|
|
|
-+
|
|
|
-+ if (abort)
|
|
|
-+ ath_dbg(common, CHAN_CTX, "RoC aborted\n");
|
|
|
-+ else
|
|
|
-+ ath_dbg(common, CHAN_CTX, "RoC expired\n");
|
|
|
-+
|
|
|
- sc->offchannel.roc_vif = NULL;
|
|
|
- sc->offchannel.roc_chan = NULL;
|
|
|
- if (!abort)
|
|
|
-@@ -1037,9 +1044,11 @@ static void ath_offchannel_channel_chang
|
|
|
- void ath_chanctx_set_next(struct ath_softc *sc, bool force)
|
|
|
- {
|
|
|
- struct ath_common *common = ath9k_hw_common(sc->sc_ah);
|
|
|
-+ struct ath_chanctx *old_ctx;
|
|
|
- struct timespec ts;
|
|
|
- bool measure_time = false;
|
|
|
- bool send_ps = false;
|
|
|
-+ bool queues_stopped = false;
|
|
|
-
|
|
|
- spin_lock_bh(&sc->chan_lock);
|
|
|
- if (!sc->next_chan) {
|
|
|
-@@ -1069,6 +1078,10 @@ void ath_chanctx_set_next(struct ath_sof
|
|
|
- getrawmonotonic(&ts);
|
|
|
- measure_time = true;
|
|
|
- }
|
|
|
-+
|
|
|
-+ ath9k_chanctx_stop_queues(sc, sc->cur_chan);
|
|
|
-+ queues_stopped = true;
|
|
|
-+
|
|
|
- __ath9k_flush(sc->hw, ~0, true);
|
|
|
-
|
|
|
- if (ath_chanctx_send_ps_frame(sc, true))
|
|
|
-@@ -1082,6 +1095,7 @@ void ath_chanctx_set_next(struct ath_sof
|
|
|
- sc->cur_chan->tsf_val = ath9k_hw_gettsf64(sc->sc_ah);
|
|
|
- }
|
|
|
- }
|
|
|
-+ old_ctx = sc->cur_chan;
|
|
|
- sc->cur_chan = sc->next_chan;
|
|
|
- sc->cur_chan->stopped = false;
|
|
|
- sc->next_chan = NULL;
|
|
|
-@@ -1104,7 +1118,16 @@ void ath_chanctx_set_next(struct ath_sof
|
|
|
- if (measure_time)
|
|
|
- sc->sched.channel_switch_time =
|
|
|
- ath9k_hw_get_tsf_offset(&ts, NULL);
|
|
|
-+ /*
|
|
|
-+ * A reset will ensure that all queues are woken up,
|
|
|
-+ * so there is no need to awaken them again.
|
|
|
-+ */
|
|
|
-+ goto out;
|
|
|
- }
|
|
|
-+
|
|
|
-+ if (queues_stopped)
|
|
|
-+ ath9k_chanctx_wake_queues(sc, old_ctx);
|
|
|
-+out:
|
|
|
- if (send_ps)
|
|
|
- ath_chanctx_send_ps_frame(sc, false);
|
|
|
-
|
|
|
-@@ -1170,18 +1193,37 @@ bool ath9k_is_chanctx_enabled(void)
|
|
|
- /* Queue management */
|
|
|
- /********************/
|
|
|
-
|
|
|
--void ath9k_chanctx_wake_queues(struct ath_softc *sc)
|
|
|
-+void ath9k_chanctx_stop_queues(struct ath_softc *sc, struct ath_chanctx *ctx)
|
|
|
-+{
|
|
|
-+ struct ath_hw *ah = sc->sc_ah;
|
|
|
-+ int i;
|
|
|
-+
|
|
|
-+ if (ctx == &sc->offchannel.chan) {
|
|
|
-+ ieee80211_stop_queue(sc->hw,
|
|
|
-+ sc->hw->offchannel_tx_hw_queue);
|
|
|
-+ } else {
|
|
|
-+ for (i = 0; i < IEEE80211_NUM_ACS; i++)
|
|
|
-+ ieee80211_stop_queue(sc->hw,
|
|
|
-+ ctx->hw_queue_base + i);
|
|
|
-+ }
|
|
|
-+
|
|
|
-+ if (ah->opmode == NL80211_IFTYPE_AP)
|
|
|
-+ ieee80211_stop_queue(sc->hw, sc->hw->queues - 2);
|
|
|
-+}
|
|
|
-+
|
|
|
-+
|
|
|
-+void ath9k_chanctx_wake_queues(struct ath_softc *sc, struct ath_chanctx *ctx)
|
|
|
- {
|
|
|
- struct ath_hw *ah = sc->sc_ah;
|
|
|
- int i;
|
|
|
-
|
|
|
-- if (sc->cur_chan == &sc->offchannel.chan) {
|
|
|
-+ if (ctx == &sc->offchannel.chan) {
|
|
|
- ieee80211_wake_queue(sc->hw,
|
|
|
- sc->hw->offchannel_tx_hw_queue);
|
|
|
- } else {
|
|
|
- for (i = 0; i < IEEE80211_NUM_ACS; i++)
|
|
|
- ieee80211_wake_queue(sc->hw,
|
|
|
-- sc->cur_chan->hw_queue_base + i);
|
|
|
-+ ctx->hw_queue_base + i);
|
|
|
- }
|
|
|
-
|
|
|
- if (ah->opmode == NL80211_IFTYPE_AP)
|
|
|
---- a/drivers/net/wireless/ath/ath9k/htc_drv_init.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/htc_drv_init.c
|
|
|
-@@ -464,6 +464,7 @@ static int ath9k_init_priv(struct ath9k_
|
|
|
- return -ENOMEM;
|
|
|
-
|
|
|
- ah->dev = priv->dev;
|
|
|
-+ ah->hw = priv->hw;
|
|
|
- ah->hw_version.devid = devid;
|
|
|
- ah->hw_version.usbdev = drv_info;
|
|
|
- ah->ah_flags |= AH_USE_EEPROM;
|
|
|
---- a/drivers/net/wireless/ath/ath9k/main.c
|
|
|
-+++ b/drivers/net/wireless/ath/ath9k/main.c
|
|
|
-@@ -60,8 +60,10 @@ static bool ath9k_has_pending_frames(str
|
|
|
-
|
|
|
- spin_lock_bh(&txq->axq_lock);
|
|
|
-
|
|
|
-- if (txq->axq_depth)
|
|
|
-+ if (txq->axq_depth) {
|
|
|
- pending = true;
|
|
|
-+ goto out;
|
|
|
-+ }
|
|
|
-
|
|
|
- if (txq->mac80211_qnum >= 0) {
|
|
|
- struct list_head *list;
|
|
|
-@@ -70,6 +72,7 @@ static bool ath9k_has_pending_frames(str
|
|
|
- if (!list_empty(list))
|
|
|
- pending = true;
|
|
|
- }
|
|
|
-+out:
|
|
|
- spin_unlock_bh(&txq->axq_lock);
|
|
|
- return pending;
|
|
|
- }
|
|
|
-@@ -261,12 +264,7 @@ static bool ath_complete_reset(struct at
|
|
|
-
|
|
|
- ath9k_hw_set_interrupts(ah);
|
|
|
- ath9k_hw_enable_interrupts(ah);
|
|
|
--
|
|
|
-- if (!ath9k_is_chanctx_enabled())
|
|
|
-- ieee80211_wake_queues(sc->hw);
|
|
|
-- else
|
|
|
-- ath9k_chanctx_wake_queues(sc);
|
|
|
--
|
|
|
-+ ieee80211_wake_queues(sc->hw);
|
|
|
- ath9k_p2p_ps_timer(sc);
|
|
|
-
|
|
|
- return true;
|
|
|
-@@ -1971,9 +1969,6 @@ static bool ath9k_has_tx_pending(struct
|
|
|
- if (!ATH_TXQ_SETUP(sc, i))
|
|
|
- continue;
|
|
|
-
|
|
|
-- if (!sc->tx.txq[i].axq_depth)
|
|
|
-- continue;
|
|
|
--
|
|
|
- npend = ath9k_has_pending_frames(sc, &sc->tx.txq[i]);
|
|
|
- if (npend)
|
|
|
- break;
|
|
|
-@@ -1999,7 +1994,6 @@ void __ath9k_flush(struct ieee80211_hw *
|
|
|
- struct ath_common *common = ath9k_hw_common(ah);
|
|
|
- int timeout = HZ / 5; /* 200 ms */
|
|
|
- bool drain_txq;
|
|
|
-- int i;
|
|
|
-
|
|
|
- cancel_delayed_work_sync(&sc->tx_complete_work);
|
|
|
-
|
|
|
-@@ -2027,10 +2021,6 @@ void __ath9k_flush(struct ieee80211_hw *
|
|
|
- ath_reset(sc);
|
|
|
-
|
|
|
- ath9k_ps_restore(sc);
|
|
|
-- for (i = 0; i < IEEE80211_NUM_ACS; i++) {
|
|
|
-- ieee80211_wake_queue(sc->hw,
|
|
|
-- sc->cur_chan->hw_queue_base + i);
|
|
|
-- }
|
|
|
- }
|
|
|
-
|
|
|
- ieee80211_queue_delayed_work(hw, &sc->tx_complete_work, 0);
|
|
|
-@@ -2039,16 +2029,8 @@ void __ath9k_flush(struct ieee80211_hw *
|
|
|
- static bool ath9k_tx_frames_pending(struct ieee80211_hw *hw)
|
|
|
- {
|
|
|
- struct ath_softc *sc = hw->priv;
|
|
|
-- int i;
|
|
|
--
|
|
|
-- for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++) {
|
|
|
-- if (!ATH_TXQ_SETUP(sc, i))
|
|
|
-- continue;
|
|
|
-
|
|
|
-- if (ath9k_has_pending_frames(sc, &sc->tx.txq[i]))
|
|
|
-- return true;
|
|
|
-- }
|
|
|
-- return false;
|
|
|
-+ return ath9k_has_tx_pending(sc);
|
|
|
- }
|
|
|
-
|
|
|
- static int ath9k_tx_last_beacon(struct ieee80211_hw *hw)
|
|
|
-@@ -2350,7 +2332,7 @@ static void ath9k_remove_chanctx(struct
|
|
|
- conf->def.chan->center_freq);
|
|
|
-
|
|
|
- ctx->assigned = false;
|
|
|
-- ctx->hw_queue_base = -1;
|
|
|
-+ ctx->hw_queue_base = 0;
|
|
|
- ath_chanctx_event(sc, NULL, ATH_CHANCTX_EVENT_UNASSIGN);
|
|
|
-
|
|
|
- mutex_unlock(&sc->mutex);
|
|
|
---- a/net/mac80211/rate.c
|
|
|
-+++ b/net/mac80211/rate.c
|
|
|
-@@ -448,7 +448,7 @@ static void rate_fixup_ratelist(struct i
|
|
|
- */
|
|
|
- if (!(rates[0].flags & IEEE80211_TX_RC_MCS)) {
|
|
|
- u32 basic_rates = vif->bss_conf.basic_rates;
|
|
|
-- s8 baserate = basic_rates ? ffs(basic_rates - 1) : 0;
|
|
|
-+ s8 baserate = basic_rates ? ffs(basic_rates) - 1 : 0;
|
|
|
-
|
|
|
- rate = &sband->bitrates[rates[0].idx];
|
|
|
-
|