ath10k: allow setting coverage class
authorBenjamin Berg <benjamin@sipsolutions.net>
Wed, 28 Sep 2016 12:11:58 +0000 (15:11 +0300)
committerKalle Valo <kvalo@qca.qualcomm.com>
Tue, 4 Oct 2016 15:03:22 +0000 (18:03 +0300)
Unfortunately ath10k does not generally allow modifying the coverage class
with the stock firmware and Qualcomm has so far refused to implement this
feature so that it can be properly supported in ath10k. If we however know
the registers that need to be modified for proper operation with a higher
coverage class, then we can do these modifications from the driver.

This is a hack and might cause subtle problems but as it's not enabled by
default (only when user space changes the coverage class explicitly) it should
not cause new problems for existing setups. But still this should be considered
as an experimental feature and used with caution.

This patch implements the support for first generation cards (QCA9880, QCA9887
and so on) which are based on a core that is similar to ath9k. The registers
are modified in place and need to be re-written every time the firmware sets
them. To achieve this the register status is verified after certain WMI events
from the firmware.

The coverage class may not be modified temporarily right after the card
re-initializes the registers. This is for example the case during scanning.

Thanks to Sebastian Gottschall <s.gottschall@dd-wrt.com> for initially
working on a userspace support for this. This patch wouldn't have been
possible without this documentation.

Signed-off-by: Benjamin Berg <benjamin@sipsolutions.net>
Signed-off-by: Simon Wunderlich <sw@simonwunderlich.de>
Signed-off-by: Mathias Kretschmer <mathias.kretschmer@fit.fraunhofer.de>
Signed-off-by: Kalle Valo <kvalo@qca.qualcomm.com>
drivers/net/wireless/ath/ath10k/core.c
drivers/net/wireless/ath/ath10k/core.h
drivers/net/wireless/ath/ath10k/debug.h
drivers/net/wireless/ath/ath10k/hw.c
drivers/net/wireless/ath/ath10k/hw.h
drivers/net/wireless/ath/ath10k/mac.c
drivers/net/wireless/ath/ath10k/wmi.c

index 867a8403b15b09ec5b1dffee01afc43b43829f1f..7005e2a9872676b1c77433ae489fb9bd64b763d8 100644 (file)
@@ -1562,6 +1562,15 @@ static void ath10k_core_restart(struct work_struct *work)
        mutex_unlock(&ar->conf_mutex);
 }
 
+static void ath10k_core_set_coverage_class_work(struct work_struct *work)
+{
+       struct ath10k *ar = container_of(work, struct ath10k,
+                                        set_coverage_class_work);
+
+       if (ar->hw_params.hw_ops->set_coverage_class)
+               ar->hw_params.hw_ops->set_coverage_class(ar, -1);
+}
+
 static int ath10k_core_init_firmware_features(struct ath10k *ar)
 {
        struct ath10k_fw_file *fw_file = &ar->normal_mode_fw.fw_file;
@@ -2344,6 +2353,8 @@ struct ath10k *ath10k_core_create(size_t priv_size, struct device *dev,
 
        INIT_WORK(&ar->register_work, ath10k_core_register_work);
        INIT_WORK(&ar->restart_work, ath10k_core_restart);
+       INIT_WORK(&ar->set_coverage_class_work,
+                 ath10k_core_set_coverage_class_work);
 
        init_dummy_netdev(&ar->napi_dev);
 
index c091306bca3326033b108862935d79bf1037a5da..6e5aa2d09699fbd2867a3987e309d85ffa7e9b11 100644 (file)
@@ -912,6 +912,19 @@ struct ath10k {
        struct net_device napi_dev;
        struct napi_struct napi;
 
+       struct work_struct set_coverage_class_work;
+       /* protected by conf_mutex */
+       struct {
+               /* writing also protected by data_lock */
+               s16 coverage_class;
+
+               u32 reg_phyclk;
+               u32 reg_slottime_conf;
+               u32 reg_slottime_orig;
+               u32 reg_ack_cts_timeout_conf;
+               u32 reg_ack_cts_timeout_orig;
+       } fw_coverage;
+
        /* must be last */
        u8 drv_priv[0] __aligned(sizeof(void *));
 };
index c458fa96a6d4205ffa79ab191cc26f50e536305f..335512b11ca2907ddc22dc7061a7730dd35c32af 100644 (file)
@@ -94,7 +94,19 @@ int ath10k_debug_get_et_sset_count(struct ieee80211_hw *hw,
 void ath10k_debug_get_et_stats(struct ieee80211_hw *hw,
                               struct ieee80211_vif *vif,
                               struct ethtool_stats *stats, u64 *data);
+
+static inline u64 ath10k_debug_get_fw_dbglog_mask(struct ath10k *ar)
+{
+       return ar->debug.fw_dbglog_mask;
+}
+
+static inline u32 ath10k_debug_get_fw_dbglog_level(struct ath10k *ar)
+{
+       return ar->debug.fw_dbglog_level;
+}
+
 #else
+
 static inline int ath10k_debug_start(struct ath10k *ar)
 {
        return 0;
@@ -144,6 +156,16 @@ ath10k_debug_get_new_fw_crash_data(struct ath10k *ar)
        return NULL;
 }
 
+static inline u64 ath10k_debug_get_fw_dbglog_mask(struct ath10k *ar)
+{
+       return 0;
+}
+
+static inline u32 ath10k_debug_get_fw_dbglog_level(struct ath10k *ar)
+{
+       return 0;
+}
+
 #define ATH10K_DFS_STAT_INC(ar, c) do { } while (0)
 
 #define ath10k_debug_get_et_strings NULL
index 675e75d66db224d886cf1470d1426c6aea6f9841..33fb26833cd0a6b259a131d66972bf7620606f5b 100644 (file)
 #include <linux/types.h>
 #include "core.h"
 #include "hw.h"
+#include "hif.h"
+#include "wmi-ops.h"
 
 const struct ath10k_hw_regs qca988x_regs = {
        .rtc_soc_base_address           = 0x00004000,
        .rtc_wmac_base_address          = 0x00005000,
        .soc_core_base_address          = 0x00009000,
+       .wlan_mac_base_address          = 0x00020000,
        .ce_wrapper_base_address        = 0x00057000,
        .ce0_base_address               = 0x00057400,
        .ce1_base_address               = 0x00057800,
@@ -48,6 +51,7 @@ const struct ath10k_hw_regs qca6174_regs = {
        .rtc_soc_base_address                   = 0x00000800,
        .rtc_wmac_base_address                  = 0x00001000,
        .soc_core_base_address                  = 0x0003a000,
+       .wlan_mac_base_address                  = 0x00020000,
        .ce_wrapper_base_address                = 0x00034000,
        .ce0_base_address                       = 0x00034400,
        .ce1_base_address                       = 0x00034800,
@@ -74,6 +78,7 @@ const struct ath10k_hw_regs qca99x0_regs = {
        .rtc_soc_base_address                   = 0x00080000,
        .rtc_wmac_base_address                  = 0x00000000,
        .soc_core_base_address                  = 0x00082000,
+       .wlan_mac_base_address                  = 0x00030000,
        .ce_wrapper_base_address                = 0x0004d000,
        .ce0_base_address                       = 0x0004a000,
        .ce1_base_address                       = 0x0004a400,
@@ -109,6 +114,7 @@ const struct ath10k_hw_regs qca99x0_regs = {
 const struct ath10k_hw_regs qca4019_regs = {
        .rtc_soc_base_address                   = 0x00080000,
        .soc_core_base_address                  = 0x00082000,
+       .wlan_mac_base_address                  = 0x00030000,
        .ce_wrapper_base_address                = 0x0004d000,
        .ce0_base_address                       = 0x0004a000,
        .ce1_base_address                       = 0x0004a400,
@@ -220,7 +226,143 @@ void ath10k_hw_fill_survey_time(struct ath10k *ar, struct survey_info *survey,
        survey->time_busy = CCNT_TO_MSEC(ar, rcc);
 }
 
+/* The firmware does not support setting the coverage class. Instead this
+ * function monitors and modifies the corresponding MAC registers.
+ */
+static void ath10k_hw_qca988x_set_coverage_class(struct ath10k *ar,
+                                                s16 value)
+{
+       u32 slottime_reg;
+       u32 slottime;
+       u32 timeout_reg;
+       u32 ack_timeout;
+       u32 cts_timeout;
+       u32 phyclk_reg;
+       u32 phyclk;
+       u64 fw_dbglog_mask;
+       u32 fw_dbglog_level;
+
+       mutex_lock(&ar->conf_mutex);
+
+       /* Only modify registers if the core is started. */
+       if ((ar->state != ATH10K_STATE_ON) &&
+           (ar->state != ATH10K_STATE_RESTARTED))
+               goto unlock;
+
+       /* Retrieve the current values of the two registers that need to be
+        * adjusted.
+        */
+       slottime_reg = ath10k_hif_read32(ar, WLAN_MAC_BASE_ADDRESS +
+                                            WAVE1_PCU_GBL_IFS_SLOT);
+       timeout_reg = ath10k_hif_read32(ar, WLAN_MAC_BASE_ADDRESS +
+                                           WAVE1_PCU_ACK_CTS_TIMEOUT);
+       phyclk_reg = ath10k_hif_read32(ar, WLAN_MAC_BASE_ADDRESS +
+                                          WAVE1_PHYCLK);
+       phyclk = MS(phyclk_reg, WAVE1_PHYCLK_USEC) + 1;
+
+       if (value < 0)
+               value = ar->fw_coverage.coverage_class;
+
+       /* Break out if the coverage class and registers have the expected
+        * value.
+        */
+       if (value == ar->fw_coverage.coverage_class &&
+           slottime_reg == ar->fw_coverage.reg_slottime_conf &&
+           timeout_reg == ar->fw_coverage.reg_ack_cts_timeout_conf &&
+           phyclk_reg == ar->fw_coverage.reg_phyclk)
+               goto unlock;
+
+       /* Store new initial register values from the firmware. */
+       if (slottime_reg != ar->fw_coverage.reg_slottime_conf)
+               ar->fw_coverage.reg_slottime_orig = slottime_reg;
+       if (timeout_reg != ar->fw_coverage.reg_ack_cts_timeout_conf)
+               ar->fw_coverage.reg_ack_cts_timeout_orig = timeout_reg;
+       ar->fw_coverage.reg_phyclk = phyclk_reg;
+
+       /* Calculat new value based on the (original) firmware calculation. */
+       slottime_reg = ar->fw_coverage.reg_slottime_orig;
+       timeout_reg = ar->fw_coverage.reg_ack_cts_timeout_orig;
+
+       /* Do some sanity checks on the slottime register. */
+       if (slottime_reg % phyclk) {
+               ath10k_warn(ar,
+                           "failed to set coverage class: expected integer microsecond value in register\n");
+
+               goto store_regs;
+       }
+
+       slottime = MS(slottime_reg, WAVE1_PCU_GBL_IFS_SLOT);
+       slottime = slottime / phyclk;
+       if (slottime != 9 && slottime != 20) {
+               ath10k_warn(ar,
+                           "failed to set coverage class: expected slot time of 9 or 20us in HW register. It is %uus.\n",
+                           slottime);
+
+               goto store_regs;
+       }
+
+       /* Recalculate the register values by adding the additional propagation
+        * delay (3us per coverage class).
+        */
+
+       slottime = MS(slottime_reg, WAVE1_PCU_GBL_IFS_SLOT);
+       slottime += value * 3 * phyclk;
+       slottime = min_t(u32, slottime, WAVE1_PCU_GBL_IFS_SLOT_MAX);
+       slottime = SM(slottime, WAVE1_PCU_GBL_IFS_SLOT);
+       slottime_reg = (slottime_reg & ~WAVE1_PCU_GBL_IFS_SLOT_MASK) | slottime;
+
+       /* Update ack timeout (lower halfword). */
+       ack_timeout = MS(timeout_reg, WAVE1_PCU_ACK_CTS_TIMEOUT_ACK);
+       ack_timeout += 3 * value * phyclk;
+       ack_timeout = min_t(u32, ack_timeout, WAVE1_PCU_ACK_CTS_TIMEOUT_MAX);
+       ack_timeout = SM(ack_timeout, WAVE1_PCU_ACK_CTS_TIMEOUT_ACK);
+
+       /* Update cts timeout (upper halfword). */
+       cts_timeout = MS(timeout_reg, WAVE1_PCU_ACK_CTS_TIMEOUT_CTS);
+       cts_timeout += 3 * value * phyclk;
+       cts_timeout = min_t(u32, cts_timeout, WAVE1_PCU_ACK_CTS_TIMEOUT_MAX);
+       cts_timeout = SM(cts_timeout, WAVE1_PCU_ACK_CTS_TIMEOUT_CTS);
+
+       timeout_reg = ack_timeout | cts_timeout;
+
+       ath10k_hif_write32(ar,
+                          WLAN_MAC_BASE_ADDRESS + WAVE1_PCU_GBL_IFS_SLOT,
+                          slottime_reg);
+       ath10k_hif_write32(ar,
+                          WLAN_MAC_BASE_ADDRESS + WAVE1_PCU_ACK_CTS_TIMEOUT,
+                          timeout_reg);
+
+       /* Ensure we have a debug level of WARN set for the case that the
+        * coverage class is larger than 0. This is important as we need to
+        * set the registers again if the firmware does an internal reset and
+        * this way we will be notified of the event.
+        */
+       fw_dbglog_mask = ath10k_debug_get_fw_dbglog_mask(ar);
+       fw_dbglog_level = ath10k_debug_get_fw_dbglog_level(ar);
+
+       if (value > 0) {
+               if (fw_dbglog_level > ATH10K_DBGLOG_LEVEL_WARN)
+                       fw_dbglog_level = ATH10K_DBGLOG_LEVEL_WARN;
+               fw_dbglog_mask = ~0;
+       }
+
+       ath10k_wmi_dbglog_cfg(ar, fw_dbglog_mask, fw_dbglog_level);
+
+store_regs:
+       /* After an error we will not retry setting the coverage class. */
+       spin_lock_bh(&ar->data_lock);
+       ar->fw_coverage.coverage_class = value;
+       spin_unlock_bh(&ar->data_lock);
+
+       ar->fw_coverage.reg_slottime_conf = slottime_reg;
+       ar->fw_coverage.reg_ack_cts_timeout_conf = timeout_reg;
+
+unlock:
+       mutex_unlock(&ar->conf_mutex);
+}
+
 const struct ath10k_hw_ops qca988x_ops = {
+       .set_coverage_class = ath10k_hw_qca988x_set_coverage_class,
 };
 
 static int ath10k_qca99x0_rx_desc_get_l3_pad_bytes(struct htt_rx_desc *rxd)
index 6038b7486f1dbfafe446593d027bf54b7e23dd36..883547f3347c2564e480c531e2534abd603d5ba0 100644 (file)
@@ -230,6 +230,7 @@ struct ath10k_hw_regs {
        u32 rtc_soc_base_address;
        u32 rtc_wmac_base_address;
        u32 soc_core_base_address;
+       u32 wlan_mac_base_address;
        u32 ce_wrapper_base_address;
        u32 ce0_base_address;
        u32 ce1_base_address;
@@ -418,6 +419,7 @@ struct htt_rx_desc;
 /* Defines needed for Rx descriptor abstraction */
 struct ath10k_hw_ops {
        int (*rx_desc_get_l3_pad_bytes)(struct htt_rx_desc *rxd);
+       void (*set_coverage_class)(struct ath10k *ar, s16 value);
 };
 
 extern const struct ath10k_hw_ops qca988x_ops;
@@ -614,7 +616,7 @@ ath10k_rx_desc_get_l3_pad_bytes(struct ath10k_hw_params *hw,
 #define WLAN_SI_BASE_ADDRESS                   0x00010000
 #define WLAN_GPIO_BASE_ADDRESS                 0x00014000
 #define WLAN_ANALOG_INTF_BASE_ADDRESS          0x0001c000
-#define WLAN_MAC_BASE_ADDRESS                  0x00020000
+#define WLAN_MAC_BASE_ADDRESS                  ar->regs->wlan_mac_base_address
 #define EFUSE_BASE_ADDRESS                     0x00030000
 #define FPGA_REG_BASE_ADDRESS                  0x00039000
 #define WLAN_UART2_BASE_ADDRESS                        0x00054c00
@@ -814,4 +816,28 @@ ath10k_rx_desc_get_l3_pad_bytes(struct ath10k_hw_params *hw,
 
 #define RTC_STATE_V_GET(x) (((x) & RTC_STATE_V_MASK) >> RTC_STATE_V_LSB)
 
+/* Register definitions for first generation ath10k cards. These cards include
+ * a mac thich has a register allocation similar to ath9k and at least some
+ * registers including the ones relevant for modifying the coverage class are
+ * identical to the ath9k definitions.
+ * These registers are usually managed by the ath10k firmware. However by
+ * overriding them it is possible to support coverage class modifications.
+ */
+#define WAVE1_PCU_ACK_CTS_TIMEOUT              0x8014
+#define WAVE1_PCU_ACK_CTS_TIMEOUT_MAX          0x00003FFF
+#define WAVE1_PCU_ACK_CTS_TIMEOUT_ACK_MASK     0x00003FFF
+#define WAVE1_PCU_ACK_CTS_TIMEOUT_ACK_LSB      0
+#define WAVE1_PCU_ACK_CTS_TIMEOUT_CTS_MASK     0x3FFF0000
+#define WAVE1_PCU_ACK_CTS_TIMEOUT_CTS_LSB      16
+
+#define WAVE1_PCU_GBL_IFS_SLOT                 0x1070
+#define WAVE1_PCU_GBL_IFS_SLOT_MASK            0x0000FFFF
+#define WAVE1_PCU_GBL_IFS_SLOT_MAX             0x0000FFFF
+#define WAVE1_PCU_GBL_IFS_SLOT_LSB             0
+#define WAVE1_PCU_GBL_IFS_SLOT_RESV0           0xFFFF0000
+
+#define WAVE1_PHYCLK                           0x801C
+#define WAVE1_PHYCLK_USEC_MASK                 0x0000007F
+#define WAVE1_PHYCLK_USEC_LSB                  0
+
 #endif /* _HW_H_ */
index ed71b95a661dcb123ee736434f55d2ea7840ffd3..2e5d2ca6d81615389c38c2903c911df667a57a9d 100644 (file)
@@ -5412,6 +5412,20 @@ static void ath10k_bss_info_changed(struct ieee80211_hw *hw,
        mutex_unlock(&ar->conf_mutex);
 }
 
+static void ath10k_mac_op_set_coverage_class(struct ieee80211_hw *hw, s16 value)
+{
+       struct ath10k *ar = hw->priv;
+
+       /* This function should never be called if setting the coverage class
+        * is not supported on this hardware.
+        */
+       if (!ar->hw_params.hw_ops->set_coverage_class) {
+               WARN_ON_ONCE(1);
+               return;
+       }
+       ar->hw_params.hw_ops->set_coverage_class(ar, value);
+}
+
 static int ath10k_hw_scan(struct ieee80211_hw *hw,
                          struct ieee80211_vif *vif,
                          struct ieee80211_scan_request *hw_req)
@@ -7437,6 +7451,7 @@ static const struct ieee80211_ops ath10k_ops = {
        .remove_interface               = ath10k_remove_interface,
        .configure_filter               = ath10k_configure_filter,
        .bss_info_changed               = ath10k_bss_info_changed,
+       .set_coverage_class             = ath10k_mac_op_set_coverage_class,
        .hw_scan                        = ath10k_hw_scan,
        .cancel_hw_scan                 = ath10k_cancel_hw_scan,
        .set_key                        = ath10k_set_key,
@@ -8123,6 +8138,10 @@ int ath10k_mac_register(struct ath10k *ar)
                goto err_dfs_detector_exit;
        }
 
+       /* Disable set_coverage_class for chipsets that do not support it. */
+       if (!ar->hw_params.hw_ops->set_coverage_class)
+               ar->ops->set_coverage_class = NULL;
+
        ret = ath_regd_init(&ar->ath_common.regulatory, ar->hw->wiphy,
                            ath10k_reg_notifier);
        if (ret) {
index b4cfaf9aa0ace4b6391be54cc5fad73f704ee63e..387c4eede388d56a8e60427495bb6640011bcb50 100644 (file)
@@ -4931,6 +4931,23 @@ exit:
        return 0;
 }
 
+static inline void ath10k_wmi_queue_set_coverage_class_work(struct ath10k *ar)
+{
+       if (ar->hw_params.hw_ops->set_coverage_class) {
+               spin_lock_bh(&ar->data_lock);
+
+               /* This call only ensures that the modified coverage class
+                * persists in case the firmware sets the registers back to
+                * their default value. So calling it is only necessary if the
+                * coverage class has a non-zero value.
+                */
+               if (ar->fw_coverage.coverage_class)
+                       queue_work(ar->workqueue, &ar->set_coverage_class_work);
+
+               spin_unlock_bh(&ar->data_lock);
+       }
+}
+
 static void ath10k_wmi_op_rx(struct ath10k *ar, struct sk_buff *skb)
 {
        struct wmi_cmd_hdr *cmd_hdr;
@@ -4951,6 +4968,7 @@ static void ath10k_wmi_op_rx(struct ath10k *ar, struct sk_buff *skb)
                return;
        case WMI_SCAN_EVENTID:
                ath10k_wmi_event_scan(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_CHAN_INFO_EVENTID:
                ath10k_wmi_event_chan_info(ar, skb);
@@ -4960,15 +4978,18 @@ static void ath10k_wmi_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_DEBUG_MESG_EVENTID:
                ath10k_wmi_event_debug_mesg(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_UPDATE_STATS_EVENTID:
                ath10k_wmi_event_update_stats(ar, skb);
                break;
        case WMI_VDEV_START_RESP_EVENTID:
                ath10k_wmi_event_vdev_start_resp(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_VDEV_STOPPED_EVENTID:
                ath10k_wmi_event_vdev_stopped(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_PEER_STA_KICKOUT_EVENTID:
                ath10k_wmi_event_peer_sta_kickout(ar, skb);
@@ -4984,12 +5005,14 @@ static void ath10k_wmi_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_ROAM_EVENTID:
                ath10k_wmi_event_roam(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_PROFILE_MATCH:
                ath10k_wmi_event_profile_match(ar, skb);
                break;
        case WMI_DEBUG_PRINT_EVENTID:
                ath10k_wmi_event_debug_print(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_PDEV_QVIT_EVENTID:
                ath10k_wmi_event_pdev_qvit(ar, skb);
@@ -5038,6 +5061,7 @@ static void ath10k_wmi_op_rx(struct ath10k *ar, struct sk_buff *skb)
                return;
        case WMI_READY_EVENTID:
                ath10k_wmi_event_ready(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        default:
                ath10k_warn(ar, "Unknown eventid: %d\n", id);
@@ -5081,6 +5105,7 @@ static void ath10k_wmi_10_1_op_rx(struct ath10k *ar, struct sk_buff *skb)
                return;
        case WMI_10X_SCAN_EVENTID:
                ath10k_wmi_event_scan(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_CHAN_INFO_EVENTID:
                ath10k_wmi_event_chan_info(ar, skb);
@@ -5090,15 +5115,18 @@ static void ath10k_wmi_10_1_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10X_DEBUG_MESG_EVENTID:
                ath10k_wmi_event_debug_mesg(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_UPDATE_STATS_EVENTID:
                ath10k_wmi_event_update_stats(ar, skb);
                break;
        case WMI_10X_VDEV_START_RESP_EVENTID:
                ath10k_wmi_event_vdev_start_resp(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_VDEV_STOPPED_EVENTID:
                ath10k_wmi_event_vdev_stopped(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_PEER_STA_KICKOUT_EVENTID:
                ath10k_wmi_event_peer_sta_kickout(ar, skb);
@@ -5114,12 +5142,14 @@ static void ath10k_wmi_10_1_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10X_ROAM_EVENTID:
                ath10k_wmi_event_roam(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_PROFILE_MATCH:
                ath10k_wmi_event_profile_match(ar, skb);
                break;
        case WMI_10X_DEBUG_PRINT_EVENTID:
                ath10k_wmi_event_debug_print(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_PDEV_QVIT_EVENTID:
                ath10k_wmi_event_pdev_qvit(ar, skb);
@@ -5159,6 +5189,7 @@ static void ath10k_wmi_10_1_op_rx(struct ath10k *ar, struct sk_buff *skb)
                return;
        case WMI_10X_READY_EVENTID:
                ath10k_wmi_event_ready(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10X_PDEV_UTF_EVENTID:
                /* ignore utf events */
@@ -5205,6 +5236,7 @@ static void ath10k_wmi_10_2_op_rx(struct ath10k *ar, struct sk_buff *skb)
                return;
        case WMI_10_2_SCAN_EVENTID:
                ath10k_wmi_event_scan(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_CHAN_INFO_EVENTID:
                ath10k_wmi_event_chan_info(ar, skb);
@@ -5214,15 +5246,18 @@ static void ath10k_wmi_10_2_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10_2_DEBUG_MESG_EVENTID:
                ath10k_wmi_event_debug_mesg(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_UPDATE_STATS_EVENTID:
                ath10k_wmi_event_update_stats(ar, skb);
                break;
        case WMI_10_2_VDEV_START_RESP_EVENTID:
                ath10k_wmi_event_vdev_start_resp(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_VDEV_STOPPED_EVENTID:
                ath10k_wmi_event_vdev_stopped(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_PEER_STA_KICKOUT_EVENTID:
                ath10k_wmi_event_peer_sta_kickout(ar, skb);
@@ -5238,12 +5273,14 @@ static void ath10k_wmi_10_2_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10_2_ROAM_EVENTID:
                ath10k_wmi_event_roam(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_PROFILE_MATCH:
                ath10k_wmi_event_profile_match(ar, skb);
                break;
        case WMI_10_2_DEBUG_PRINT_EVENTID:
                ath10k_wmi_event_debug_print(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_PDEV_QVIT_EVENTID:
                ath10k_wmi_event_pdev_qvit(ar, skb);
@@ -5274,15 +5311,18 @@ static void ath10k_wmi_10_2_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10_2_VDEV_STANDBY_REQ_EVENTID:
                ath10k_wmi_event_vdev_standby_req(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_VDEV_RESUME_REQ_EVENTID:
                ath10k_wmi_event_vdev_resume_req(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_SERVICE_READY_EVENTID:
                ath10k_wmi_event_service_ready(ar, skb);
                return;
        case WMI_10_2_READY_EVENTID:
                ath10k_wmi_event_ready(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_2_PDEV_TEMPERATURE_EVENTID:
                ath10k_wmi_event_temperature(ar, skb);
@@ -5345,12 +5385,14 @@ static void ath10k_wmi_10_4_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10_4_DEBUG_MESG_EVENTID:
                ath10k_wmi_event_debug_mesg(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_SERVICE_READY_EVENTID:
                ath10k_wmi_event_service_ready(ar, skb);
                return;
        case WMI_10_4_SCAN_EVENTID:
                ath10k_wmi_event_scan(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_CHAN_INFO_EVENTID:
                ath10k_wmi_event_chan_info(ar, skb);
@@ -5360,12 +5402,14 @@ static void ath10k_wmi_10_4_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10_4_READY_EVENTID:
                ath10k_wmi_event_ready(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_PEER_STA_KICKOUT_EVENTID:
                ath10k_wmi_event_peer_sta_kickout(ar, skb);
                break;
        case WMI_10_4_ROAM_EVENTID:
                ath10k_wmi_event_roam(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_HOST_SWBA_EVENTID:
                ath10k_wmi_event_host_swba(ar, skb);
@@ -5375,12 +5419,15 @@ static void ath10k_wmi_10_4_op_rx(struct ath10k *ar, struct sk_buff *skb)
                break;
        case WMI_10_4_DEBUG_PRINT_EVENTID:
                ath10k_wmi_event_debug_print(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_VDEV_START_RESP_EVENTID:
                ath10k_wmi_event_vdev_start_resp(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_VDEV_STOPPED_EVENTID:
                ath10k_wmi_event_vdev_stopped(ar, skb);
+               ath10k_wmi_queue_set_coverage_class_work(ar);
                break;
        case WMI_10_4_WOW_WAKEUP_HOST_EVENTID:
        case WMI_10_4_PEER_RATECODE_LIST_EVENTID:
@@ -6099,6 +6146,7 @@ void ath10k_wmi_start_scan_init(struct ath10k *ar,
                | WMI_SCAN_EVENT_COMPLETED
                | WMI_SCAN_EVENT_BSS_CHANNEL
                | WMI_SCAN_EVENT_FOREIGN_CHANNEL
+               | WMI_SCAN_EVENT_FOREIGN_CHANNEL_EXIT
                | WMI_SCAN_EVENT_DEQUEUED;
        arg->scan_ctrl_flags |= WMI_SCAN_CHAN_STAT_EVENT;
        arg->n_bssids = 1;