Commit 3a2dd6b7 authored by Chin-Yen Lee's avatar Chin-Yen Lee Committed by Kalle Valo

rtw88: check firmware leave lps successfully

Driver needs to wait for firmware to restore hardware setting
to active mode after leaving lps.

After getting H2C from driver for leaving lps, firmware will
issue null packet without PS bit to inform AP driver is active,
and then restore REG_TCR Register if AP has receiced null packet.

But the transmission of null packet may cost much more time
in noisy environment. If driver does not wait for firmware,
null packet with PS bit could be sent due to incorrect REG_TCR setting.
And AP will be confused.

In our test, 100ms is enough for firmware to send null packet
to AP. If REG_TCR Register is still wrong after 100ms, we will
modify it directly, force the PS bit to be cleared
Signed-off-by: default avatarChin-Yen Lee <timlee@realtek.com>
Signed-off-by: default avatarYan-Hsuan Chuang <yhchuang@realtek.com>
Signed-off-by: default avatarKalle Valo <kvalo@codeaurora.org>
parent 97ef1226
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
#include "mac.h" #include "mac.h"
#include "coex.h" #include "coex.h"
#include "debug.h" #include "debug.h"
#include "reg.h"
static int rtw_ips_pwr_up(struct rtw_dev *rtwdev) static int rtw_ips_pwr_up(struct rtw_dev *rtwdev)
{ {
...@@ -118,6 +119,32 @@ static void __rtw_leave_lps_deep(struct rtw_dev *rtwdev) ...@@ -118,6 +119,32 @@ static void __rtw_leave_lps_deep(struct rtw_dev *rtwdev)
rtw_hci_deep_ps(rtwdev, false); rtw_hci_deep_ps(rtwdev, false);
} }
static void rtw_fw_leave_lps_state_check(struct rtw_dev *rtwdev)
{
int i;
/* Driver needs to wait for firmware to leave LPS state
* successfully. Firmware will send null packet to inform AP,
* and see if AP sends an ACK back, then firmware will restore
* the REG_TCR register.
*
* If driver does not wait for firmware, null packet with
* PS bit could be sent due to incorrect REG_TCR setting.
*
* In our test, 100ms should be enough for firmware to finish
* the flow. If REG_TCR Register is still incorrect after 100ms,
* just modify it directly, and throw a warn message.
*/
for (i = 0 ; i < LEAVE_LPS_TRY_CNT; i++) {
if (rtw_read32_mask(rtwdev, REG_TCR, BIT_PWRMGT_HWDATA_EN) == 0)
return;
msleep(20);
}
rtw_write32_mask(rtwdev, REG_TCR, BIT_PWRMGT_HWDATA_EN, 0);
rtw_warn(rtwdev, "firmware failed to restore hardware setting\n");
}
static void rtw_leave_lps_core(struct rtw_dev *rtwdev) static void rtw_leave_lps_core(struct rtw_dev *rtwdev)
{ {
struct rtw_lps_conf *conf = &rtwdev->lps_conf; struct rtw_lps_conf *conf = &rtwdev->lps_conf;
...@@ -128,6 +155,8 @@ static void rtw_leave_lps_core(struct rtw_dev *rtwdev) ...@@ -128,6 +155,8 @@ static void rtw_leave_lps_core(struct rtw_dev *rtwdev)
conf->smart_ps = 0; conf->smart_ps = 0;
rtw_fw_set_pwr_mode(rtwdev); rtw_fw_set_pwr_mode(rtwdev);
rtw_fw_leave_lps_state_check(rtwdev);
clear_bit(RTW_FLAG_LEISURE_PS, rtwdev->flags); clear_bit(RTW_FLAG_LEISURE_PS, rtwdev->flags);
rtw_coex_lps_notify(rtwdev, COEX_LPS_DISABLE); rtw_coex_lps_notify(rtwdev, COEX_LPS_DISABLE);
......
...@@ -11,6 +11,8 @@ ...@@ -11,6 +11,8 @@
#define POWER_MODE_PG BIT(4) #define POWER_MODE_PG BIT(4)
#define POWER_MODE_LCLK BIT(0) #define POWER_MODE_LCLK BIT(0)
#define LEAVE_LPS_TRY_CNT 5
int rtw_enter_ips(struct rtw_dev *rtwdev); int rtw_enter_ips(struct rtw_dev *rtwdev);
int rtw_leave_ips(struct rtw_dev *rtwdev); int rtw_leave_ips(struct rtw_dev *rtwdev);
......
...@@ -271,6 +271,7 @@ ...@@ -271,6 +271,7 @@
#define BIT_TSFT_SEL_TIMER0 (BIT(4) | BIT(5) | BIT(6)) #define BIT_TSFT_SEL_TIMER0 (BIT(4) | BIT(5) | BIT(6))
#define REG_TCR 0x0604 #define REG_TCR 0x0604
#define BIT_PWRMGT_HWDATA_EN BIT(7)
#define REG_RCR 0x0608 #define REG_RCR 0x0608
#define BIT_APP_FCS BIT(31) #define BIT_APP_FCS BIT(31)
#define BIT_APP_MIC BIT(30) #define BIT_APP_MIC BIT(30)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment