rtw88: add dynamic cck pd mechanism
authorTzu-En Huang <tehuang@realtek.com>
Mon, 9 Sep 2019 07:16:10 +0000 (15:16 +0800)
committerKalle Valo <kvalo@codeaurora.org>
Fri, 13 Sep 2019 15:03:21 +0000 (18:03 +0300)
This mechanism reduces the numbers of false alram in cck rate by
dynamically adjusting the value of power threshold and cs_ratio.
We determine the new value by three factors, which are rssi, false alarm
count and igi. Based on these factors, we define the current condition
into five levels. Compared to the previous level, if the level is changed,
we set the new values for power threshold and cs_ratio.

Signed-off-by: Tzu-En Huang <tehuang@realtek.com>
Signed-off-by: Yan-Hsuan Chuang <yhchuang@realtek.com>
Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
drivers/net/wireless/realtek/rtw88/main.h
drivers/net/wireless/realtek/rtw88/phy.c
drivers/net/wireless/realtek/rtw88/phy.h
drivers/net/wireless/realtek/rtw88/rtw8822c.c

index 76d925323819fe49ef00859e415777807d9fc5cd..bede3f38516e81cb3489e14f8c2f0f37084fa2b7 100644 (file)
@@ -642,6 +642,7 @@ struct rtw_chip_ops {
        void (*false_alarm_statistics)(struct rtw_dev *rtwdev);
        void (*phy_calibration)(struct rtw_dev *rtwdev);
        void (*dpk_track)(struct rtw_dev *rtwdev);
+       void (*cck_pd_set)(struct rtw_dev *rtwdev, u8 level);
 
        /* for coex */
        void (*coex_set_init)(struct rtw_dev *rtwdev);
@@ -1110,6 +1111,13 @@ struct rtw_dpk_info {
        u8 dpk_bw;
 };
 
+struct rtw_phy_cck_pd_reg {
+       u32 reg_pd;
+       u32 mask_pd;
+       u32 reg_cs;
+       u32 mask_cs;
+};
+
 #define DACK_MSBK_BACKUP_NUM   0xf
 #define DACK_DCK_BACKUP_NUM    0x2
 
@@ -1145,6 +1153,10 @@ struct rtw_dm_info {
        u8 dack_dck[RTW_RF_PATH_MAX][2][DACK_DCK_BACKUP_NUM];
 
        struct rtw_dpk_info dpk_info;
+
+       /* [bandwidth 0:20M/1:40M][number of path] */
+       u8 cck_pd_lv[2][RTW_RF_PATH_MAX];
+       u32 cck_fa_avg;
 };
 
 struct rtw_efuse {
@@ -1381,6 +1393,11 @@ static inline void rtw_flag_set(struct rtw_dev *rtwdev, enum rtw_flags flag)
        set_bit(flag, rtwdev->flags);
 }
 
+static inline bool rtw_is_assoc(struct rtw_dev *rtwdev)
+{
+       return !!rtwdev->sta_cnt;
+}
+
 void rtw_get_channel_params(struct cfg80211_chan_def *chandef,
                            struct rtw_channel_params *ch_param);
 bool check_hw_ready(struct rtw_dev *rtwdev, u32 addr, u32 mask, u32 target);
index 0293d73c82abd3c533f2f6b5cfc12537891cb76e..d3d3f40de75e03a580c447676c0bf0da5df5fca3 100644 (file)
@@ -111,6 +111,19 @@ enum rtw_phy_band_type {
        PHY_BAND_5G     = 1,
 };
 
+static void rtw_phy_cck_pd_init(struct rtw_dev *rtwdev)
+{
+       struct rtw_dm_info *dm_info = &rtwdev->dm_info;
+       u8 i, j;
+
+       for (i = 0; i <= RTW_CHANNEL_WIDTH_40; i++) {
+               for (j = 0; j < RTW_RF_PATH_MAX; j++)
+                       dm_info->cck_pd_lv[i][j] = 0;
+       }
+
+       dm_info->cck_fa_avg = CCK_FA_AVG_RESET;
+}
+
 void rtw_phy_init(struct rtw_dev *rtwdev)
 {
        struct rtw_chip_info *chip = rtwdev->chip;
@@ -129,6 +142,7 @@ void rtw_phy_init(struct rtw_dev *rtwdev)
        addr = chip->dig[0].addr;
        mask = chip->dig[0].mask;
        dm_info->igi_history[0] = rtw_read32_mask(rtwdev, addr, mask);
+       rtw_phy_cck_pd_init(rtwdev);
 }
 
 void rtw_phy_dig_write(struct rtw_dev *rtwdev, u8 igi)
@@ -447,11 +461,90 @@ static void rtw_phy_dpk_track(struct rtw_dev *rtwdev)
                chip->ops->dpk_track(rtwdev);
 }
 
+#define CCK_PD_LV_MAX          5
+#define CCK_PD_FA_LV1_MIN      1000
+#define CCK_PD_FA_LV0_MAX      500
+
+static u8 rtw_phy_cck_pd_lv_unlink(struct rtw_dev *rtwdev)
+{
+       struct rtw_dm_info *dm_info = &rtwdev->dm_info;
+       u32 cck_fa_avg = dm_info->cck_fa_avg;
+
+       if (cck_fa_avg > CCK_PD_FA_LV1_MIN)
+               return 1;
+
+       if (cck_fa_avg < CCK_PD_FA_LV0_MAX)
+               return 0;
+
+       return CCK_PD_LV_MAX;
+}
+
+#define CCK_PD_IGI_LV4_VAL 0x38
+#define CCK_PD_IGI_LV3_VAL 0x2a
+#define CCK_PD_IGI_LV2_VAL 0x24
+#define CCK_PD_RSSI_LV4_VAL 32
+#define CCK_PD_RSSI_LV3_VAL 32
+#define CCK_PD_RSSI_LV2_VAL 24
+
+static u8 rtw_phy_cck_pd_lv_link(struct rtw_dev *rtwdev)
+{
+       struct rtw_dm_info *dm_info = &rtwdev->dm_info;
+       u8 igi = dm_info->igi_history[0];
+       u8 rssi = dm_info->min_rssi;
+       u32 cck_fa_avg = dm_info->cck_fa_avg;
+
+       if (igi > CCK_PD_IGI_LV4_VAL && rssi > CCK_PD_RSSI_LV4_VAL)
+               return 4;
+       if (igi > CCK_PD_IGI_LV3_VAL && rssi > CCK_PD_RSSI_LV3_VAL)
+               return 3;
+       if (igi > CCK_PD_IGI_LV2_VAL || rssi > CCK_PD_RSSI_LV2_VAL)
+               return 2;
+       if (cck_fa_avg > CCK_PD_FA_LV1_MIN)
+               return 1;
+       if (cck_fa_avg < CCK_PD_FA_LV0_MAX)
+               return 0;
+
+       return CCK_PD_LV_MAX;
+}
+
+static u8 rtw_phy_cck_pd_lv(struct rtw_dev *rtwdev)
+{
+       if (!rtw_is_assoc(rtwdev))
+               return rtw_phy_cck_pd_lv_unlink(rtwdev);
+       else
+               return rtw_phy_cck_pd_lv_link(rtwdev);
+}
+
+static void rtw_phy_cck_pd(struct rtw_dev *rtwdev)
+{
+       struct rtw_dm_info *dm_info = &rtwdev->dm_info;
+       struct rtw_chip_info *chip = rtwdev->chip;
+       u32 cck_fa = dm_info->cck_fa_cnt;
+       u8 level;
+
+       if (rtwdev->hal.current_band_type != RTW_BAND_2G)
+               return;
+
+       if (dm_info->cck_fa_avg == CCK_FA_AVG_RESET)
+               dm_info->cck_fa_avg = cck_fa;
+       else
+               dm_info->cck_fa_avg = (dm_info->cck_fa_avg * 3 + cck_fa) >> 2;
+
+       level = rtw_phy_cck_pd_lv(rtwdev);
+
+       if (level >= CCK_PD_LV_MAX)
+               return;
+
+       if (chip->ops->cck_pd_set)
+               chip->ops->cck_pd_set(rtwdev, level);
+}
+
 void rtw_phy_dynamic_mechanism(struct rtw_dev *rtwdev)
 {
        /* for further calculation */
        rtw_phy_statistics(rtwdev);
        rtw_phy_dig(rtwdev);
+       rtw_phy_cck_pd(rtwdev);
        rtw_phy_ra_info_update(rtwdev);
        rtw_phy_dpk_track(rtwdev);
 }
index cc87b157f23e9713cf8f884c08c520181359c136..e79b084628e755f926f1746cc8ada1f4152cf283 100644 (file)
@@ -146,4 +146,6 @@ rtw_get_tx_power_params(struct rtw_dev *rtwdev, u8 path,
 #define MASKBYTE3LOWNIBBLE     0x0f000000
 #define        MASKL3BYTES             0x00ffffff
 
+#define CCK_FA_AVG_RESET 0xffffffff
+
 #endif
index 5a428e041bb8d98983f301701af97e9ad9c4c990..f28563d62fd569e36cb8c5935d824601cba0a1ce 100644 (file)
@@ -3116,6 +3116,81 @@ void rtw8822c_dpk_track(struct rtw_dev *rtwdev)
        }
 }
 
+static const struct rtw_phy_cck_pd_reg
+rtw8822c_cck_pd_reg[RTW_CHANNEL_WIDTH_40 + 1][RTW_RF_PATH_MAX] = {
+       {
+               {0x1ac8, 0x00ff, 0x1ad0, 0x01f},
+               {0x1ac8, 0xff00, 0x1ad0, 0x3e0}
+       },
+       {
+               {0x1acc, 0x00ff, 0x1ad0, 0x01F00000},
+               {0x1acc, 0xff00, 0x1ad0, 0x3E000000}
+       },
+};
+
+#define RTW_CCK_PD_MAX 255
+#define RTW_CCK_CS_MAX 31
+#define RTW_CCK_CS_ERR1 27
+#define RTW_CCK_CS_ERR2 29
+static void
+rtw8822c_phy_cck_pd_set_reg(struct rtw_dev *rtwdev,
+                           s8 pd_diff, s8 cs_diff, u8 bw, u8 nrx)
+{
+       u32 pd, cs;
+
+       if (WARN_ON(bw > RTW_CHANNEL_WIDTH_40 || nrx >= RTW_RF_PATH_MAX))
+               return;
+
+       pd = rtw_read32_mask(rtwdev,
+                            rtw8822c_cck_pd_reg[bw][nrx].reg_pd,
+                            rtw8822c_cck_pd_reg[bw][nrx].mask_pd);
+       cs = rtw_read32_mask(rtwdev,
+                            rtw8822c_cck_pd_reg[bw][nrx].reg_cs,
+                            rtw8822c_cck_pd_reg[bw][nrx].mask_cs);
+       pd += pd_diff;
+       cs += cs_diff;
+       if (pd > RTW_CCK_PD_MAX)
+               pd = RTW_CCK_PD_MAX;
+       if (cs == RTW_CCK_CS_ERR1 || cs == RTW_CCK_CS_ERR2)
+               cs++;
+       else if (cs > RTW_CCK_CS_MAX)
+               cs = RTW_CCK_CS_MAX;
+       rtw_write32_mask(rtwdev,
+                        rtw8822c_cck_pd_reg[bw][nrx].reg_pd,
+                        rtw8822c_cck_pd_reg[bw][nrx].mask_pd,
+                        pd);
+       rtw_write32_mask(rtwdev,
+                        rtw8822c_cck_pd_reg[bw][nrx].reg_cs,
+                        rtw8822c_cck_pd_reg[bw][nrx].mask_cs,
+                        cs);
+}
+
+static void rtw8822c_phy_cck_pd_set(struct rtw_dev *rtwdev, u8 new_lvl)
+{
+       struct rtw_dm_info *dm_info = &rtwdev->dm_info;
+       s8 pd_lvl[4] = {2, 4, 6, 8};
+       s8 cs_lvl[4] = {2, 2, 2, 4};
+       u8 cur_lvl;
+       u8 nrx, bw;
+
+       nrx = (u8)rtw_read32_mask(rtwdev, 0x1a2c, 0x60000);
+       bw = (u8)rtw_read32_mask(rtwdev, 0x9b0, 0xc);
+
+       if (dm_info->cck_pd_lv[bw][nrx] == new_lvl)
+               return;
+
+       cur_lvl = dm_info->cck_pd_lv[bw][nrx];
+
+       /* update cck pd info */
+       dm_info->cck_fa_avg = CCK_FA_AVG_RESET;
+
+       rtw8822c_phy_cck_pd_set_reg(rtwdev,
+                                   pd_lvl[new_lvl] - pd_lvl[cur_lvl],
+                                   cs_lvl[new_lvl] - cs_lvl[cur_lvl],
+                                   bw, nrx);
+       dm_info->cck_pd_lv[bw][nrx] = new_lvl;
+}
+
 static struct rtw_pwr_seq_cmd trans_carddis_to_cardemu_8822c[] = {
        {0x0086,
         RTW_PWR_CUT_ALL_MSK,
@@ -3495,6 +3570,7 @@ static struct rtw_chip_ops rtw8822c_ops = {
        .false_alarm_statistics = rtw8822c_false_alarm_statistics,
        .dpk_track              = rtw8822c_dpk_track,
        .phy_calibration        = rtw8822c_phy_calibration,
+       .cck_pd_set             = rtw8822c_phy_cck_pd_set,
 
        .coex_set_init          = rtw8822c_coex_cfg_init,
        .coex_set_ant_switch    = NULL,