]> git.baikalelectronics.ru Git - kernel.git/commitdiff
ath11k: fix read fail for htt_stats and htt_peer_stats for single pdev
authorWen Gong <quic_wgong@quicinc.com>
Mon, 22 Nov 2021 11:13:58 +0000 (13:13 +0200)
committerKalle Valo <kvalo@codeaurora.org>
Mon, 22 Nov 2021 14:33:48 +0000 (16:33 +0200)
The pdev id is set to 0 for single pdev configured hardware, the real
pdev id is not 0 in firmware, for example, its pdev id is 1 for 5G/6G
phy and 2 for 2G band phy. For HTT_H2T_MSG_TYPE_EXT_STATS_CFG message,
firmware parse the pdev_mask to its pdev id, ath11k set it to 0 for
single pdev, it is not correct, need set it with the real pdev id of
firmware.

Save the real pdev id report by firmware and set it correctly.

Below commands run success with this patch:
cat /sys/kernel/debug/ieee80211/phy0/ath11k/htt_stats
cat /sys/kernel/debug/ieee80211/phy0/netdev\:wls1/stations/00\:03\:7f\:75\:59\:85/htt_peer_stats

Tested-on: QCA6390 hw2.0 PCI WLAN.HST.1.0.1-01740-QCAHSTSWPLZ_V2_TO_X86-1

Signed-off-by: Wen Gong <quic_wgong@quicinc.com>
Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
Link: https://lore.kernel.org/r/20211118095700.8149-1-quic_wgong@quicinc.com
drivers/net/wireless/ath/ath11k/core.h
drivers/net/wireless/ath/ath11k/dp_tx.c
drivers/net/wireless/ath/ath11k/mac.c
drivers/net/wireless/ath/ath11k/mac.h
drivers/net/wireless/ath/ath11k/wmi.c
drivers/net/wireless/ath/ath11k/wmi.h

index 74dce3518e9b8447bebf19b3f7909bdb859decd0..0103cfd0508d5df6a0103786f81f407c2cc2cd0a 100644 (file)
@@ -713,6 +713,11 @@ struct ath11k_base {
        /* Protects data like peers */
        spinlock_t base_lock;
        struct ath11k_pdev pdevs[MAX_RADIOS];
+       struct {
+               enum WMI_HOST_WLAN_BAND supported_bands;
+               u32 pdev_id;
+       } target_pdev_ids[MAX_RADIOS];
+       u8 target_pdev_count;
        struct ath11k_pdev __rcu *pdevs_active[MAX_RADIOS];
        struct ath11k_hal_reg_capabilities_ext hal_reg_cap[MAX_RADIOS];
        unsigned long long free_vdev_map;
index 88abd64e904701b0581bd259af0750b5fb8518d8..7587e1679ec3fc6621b5f60b0e78daa3c94b237f 100644 (file)
@@ -9,6 +9,7 @@
 #include "debugfs_sta.h"
 #include "hw.h"
 #include "peer.h"
+#include "mac.h"
 
 static enum hal_tcl_encap_type
 ath11k_dp_tx_get_encap_type(struct ath11k_vif *arvif, struct sk_buff *skb)
@@ -985,6 +986,7 @@ ath11k_dp_tx_htt_h2t_ext_stats_req(struct ath11k *ar, u8 type,
        struct ath11k_dp *dp = &ab->dp;
        struct sk_buff *skb;
        struct htt_ext_stats_cfg_cmd *cmd;
+       u32 pdev_id;
        int len = sizeof(*cmd);
        int ret;
 
@@ -998,7 +1000,12 @@ ath11k_dp_tx_htt_h2t_ext_stats_req(struct ath11k *ar, u8 type,
        memset(cmd, 0, sizeof(*cmd));
        cmd->hdr.msg_type = HTT_H2T_MSG_TYPE_EXT_STATS_CFG;
 
-       cmd->hdr.pdev_mask = 1 << ar->pdev->pdev_id;
+       if (ab->hw_params.single_pdev_only)
+               pdev_id = ath11k_mac_get_target_pdev_id(ar);
+       else
+               pdev_id = ar->pdev->pdev_id;
+
+       cmd->hdr.pdev_mask = 1 << pdev_id;
 
        cmd->hdr.stats_type = type;
        cmd->cfg_param0 = cfg_params->cfg0;
index b85e9cfc5dcb341854ea0cc002e996162464e434..6913d655354fd10235d39e19b1abbae4b7157d2e 100644 (file)
@@ -553,6 +553,67 @@ struct ath11k *ath11k_mac_get_ar_by_pdev_id(struct ath11k_base *ab, u32 pdev_id)
        return NULL;
 }
 
+struct ath11k_vif *ath11k_mac_get_vif_up(struct ath11k_base *ab)
+{
+       struct ath11k *ar;
+       struct ath11k_pdev *pdev;
+       struct ath11k_vif *arvif;
+       int i;
+
+       for (i = 0; i < ab->num_radios; i++) {
+               pdev = &ab->pdevs[i];
+               ar = pdev->ar;
+               list_for_each_entry(arvif, &ar->arvifs, list) {
+                       if (arvif->is_up)
+                               return arvif;
+               }
+       }
+
+       return NULL;
+}
+
+static bool ath11k_mac_band_match(enum nl80211_band band1, enum WMI_HOST_WLAN_BAND band2)
+{
+       return (((band1 == NL80211_BAND_2GHZ) && (band2 & WMI_HOST_WLAN_2G_CAP)) ||
+               (((band1 == NL80211_BAND_5GHZ) || (band1 == NL80211_BAND_6GHZ)) &&
+                  (band2 & WMI_HOST_WLAN_5G_CAP)));
+}
+
+u8 ath11k_mac_get_target_pdev_id_from_vif(struct ath11k_vif *arvif)
+{
+       struct ath11k *ar = arvif->ar;
+       struct ath11k_base *ab = ar->ab;
+       struct ieee80211_vif *vif = arvif->vif;
+       struct cfg80211_chan_def def;
+       enum nl80211_band band;
+       u8 pdev_id = ab->target_pdev_ids[0].pdev_id;
+       int i;
+
+       if (WARN_ON(ath11k_mac_vif_chan(vif, &def)))
+               return pdev_id;
+
+       band = def.chan->band;
+
+       for (i = 0; i < ab->target_pdev_count; i++) {
+               if (ath11k_mac_band_match(band, ab->target_pdev_ids[i].supported_bands))
+                       return ab->target_pdev_ids[i].pdev_id;
+       }
+
+       return pdev_id;
+}
+
+u8 ath11k_mac_get_target_pdev_id(struct ath11k *ar)
+{
+       struct ath11k_vif *arvif;
+
+       arvif = ath11k_mac_get_vif_up(ar->ab);
+
+       if (arvif)
+               return ath11k_mac_get_target_pdev_id_from_vif(arvif);
+       else
+               return ar->ab->target_pdev_ids[0].pdev_id;
+}
+
 static void ath11k_pdev_caps_update(struct ath11k *ar)
 {
        struct ath11k_base *ab = ar->ab;
index f6f37e8c8c6adb093ce8b5fc4df515060f6a25e0..359e5e9ca904d2f4bd61133c235a67615c6030d5 100644 (file)
@@ -144,6 +144,10 @@ void ath11k_mac_scan_finish(struct ath11k *ar);
 struct ath11k_vif *ath11k_mac_get_arvif(struct ath11k *ar, u32 vdev_id);
 struct ath11k_vif *ath11k_mac_get_arvif_by_vdev_id(struct ath11k_base *ab,
                                                   u32 vdev_id);
+u8 ath11k_mac_get_target_pdev_id(struct ath11k *ar);
+u8 ath11k_mac_get_target_pdev_id_from_vif(struct ath11k_vif *arvif);
+struct ath11k_vif *ath11k_mac_get_vif_up(struct ath11k_base *ab);
+
 struct ath11k *ath11k_mac_get_ar_by_vdev_id(struct ath11k_base *ab, u32 vdev_id);
 struct ath11k *ath11k_mac_get_ar_by_pdev_id(struct ath11k_base *ab, u32 pdev_id);
 
index 2e9fb87d8bf6c711184b6c4402c93d054b38c3cc..87351e0a269dcf00f8f0abdc354cad6650e80bcb 100644 (file)
@@ -337,6 +337,7 @@ ath11k_pull_mac_phy_cap_svc_ready_ext(struct ath11k_pdev_wmi *wmi_handle,
                                      struct ath11k_pdev *pdev)
 {
        struct wmi_mac_phy_capabilities *mac_phy_caps;
+       struct ath11k_base *ab = wmi_handle->wmi_ab->ab;
        struct ath11k_band_cap *cap_band;
        struct ath11k_pdev_cap *pdev_cap = &pdev->cap;
        u32 phy_map;
@@ -368,6 +369,10 @@ ath11k_pull_mac_phy_cap_svc_ready_ext(struct ath11k_pdev_wmi *wmi_handle,
        pdev->pdev_id = mac_phy_caps->pdev_id;
        pdev_cap->supported_bands |= mac_phy_caps->supported_bands;
        pdev_cap->ampdu_density = mac_phy_caps->ampdu_density;
+       ab->target_pdev_ids[ab->target_pdev_count].supported_bands =
+               mac_phy_caps->supported_bands;
+       ab->target_pdev_ids[ab->target_pdev_count].pdev_id = mac_phy_caps->pdev_id;
+       ab->target_pdev_count++;
 
        /* Take non-zero tx/rx chainmask. If tx/rx chainmask differs from
         * band to band for a single radio, need to see how this should be
@@ -4230,6 +4235,7 @@ static int ath11k_wmi_tlv_ext_soc_hal_reg_caps_parse(struct ath11k_base *soc,
        svc_rdy_ext->param.num_phy = svc_rdy_ext->soc_hal_reg_caps->num_phy;
 
        soc->num_radios = 0;
+       soc->target_pdev_count = 0;
        phy_id_map = svc_rdy_ext->pref_hw_mode_caps.phy_id_map;
 
        while (phy_id_map && soc->num_radios < MAX_RADIOS) {
index 4eb06cb7f88306454519a7477c0f9b2681167e42..8e8bac1b737089a014c7a83ef9cd3bdcb568fdd5 100644 (file)
@@ -113,10 +113,10 @@ enum wmi_host_hw_mode_priority {
        WMI_HOST_HW_MODE_MAX_PRI
 };
 
-enum {
+enum WMI_HOST_WLAN_BAND {
        WMI_HOST_WLAN_2G_CAP    = 0x1,
        WMI_HOST_WLAN_5G_CAP    = 0x2,
-       WMI_HOST_WLAN_2G_5G_CAP = 0x3,
+       WMI_HOST_WLAN_2G_5G_CAP = WMI_HOST_WLAN_2G_CAP | WMI_HOST_WLAN_5G_CAP,
 };
 
 /* Parameters used for WMI_VDEV_PARAM_AUTORATE_MISC_CFG command.