Commit 690390d8 authored by Chin-Yen Lee's avatar Chin-Yen Lee Committed by Kalle Valo

rtw88: sync the power state between driver and firmware

In current flow, driver issues power requests to firmware for entering or
leaving deep power mode. But driver does not wait for an ack from firmware
via reading CPWM register when driver requests to enter deep power mode.
The behavior will lead to unsynchronized state between firmware and driver.
Furthermore, consecutive same power requests may confuse firmware and leads
to unexpected value of CPWM register. This patch ensures that the power
request will wait for an ack from firmware and only send one power request
each time.
Signed-off-by: default avatarChin-Yen Lee <timlee@realtek.com>
Signed-off-by: default avatarTzu-En Huang <tehuang@realtek.com>
Signed-off-by: default avatarKalle Valo <kvalo@codeaurora.org>
Link: https://lore.kernel.org/r/20201030084826.9034-2-tehuang@realtek.com
parent 842280da
...@@ -68,48 +68,39 @@ int rtw_leave_ips(struct rtw_dev *rtwdev) ...@@ -68,48 +68,39 @@ int rtw_leave_ips(struct rtw_dev *rtwdev)
void rtw_power_mode_change(struct rtw_dev *rtwdev, bool enter) void rtw_power_mode_change(struct rtw_dev *rtwdev, bool enter)
{ {
u8 request, confirm, polling; u8 request, confirm, polling;
u8 polling_cnt; int ret;
u8 retry_cnt = 0;
for (retry_cnt = 0; retry_cnt < 3; retry_cnt++) {
request = rtw_read8(rtwdev, rtwdev->hci.rpwm_addr);
confirm = rtw_read8(rtwdev, rtwdev->hci.cpwm_addr);
/* toggle to request power mode, others remain 0 */
request ^= request | BIT_RPWM_TOGGLE;
if (!enter) {
request |= POWER_MODE_ACK;
} else {
request |= POWER_MODE_LCLK;
if (rtw_fw_lps_deep_mode == LPS_DEEP_MODE_PG)
request |= POWER_MODE_PG;
}
rtw_write8(rtwdev, rtwdev->hci.rpwm_addr, request);
if (enter)
return;
/* check confirm power mode has left power save state */ request = rtw_read8(rtwdev, rtwdev->hci.rpwm_addr);
for (polling_cnt = 0; polling_cnt < 50; polling_cnt++) { confirm = rtw_read8(rtwdev, rtwdev->hci.cpwm_addr);
polling = rtw_read8(rtwdev, rtwdev->hci.cpwm_addr);
if ((polling ^ confirm) & BIT_RPWM_TOGGLE) /* toggle to request power mode, others remain 0 */
return; request ^= request | BIT_RPWM_TOGGLE;
udelay(100); if (enter) {
} request |= POWER_MODE_LCLK;
if (rtw_fw_lps_deep_mode == LPS_DEEP_MODE_PG)
/* in case of fw/hw missed the request, retry */ request |= POWER_MODE_PG;
rtw_warn(rtwdev, "failed to leave deep PS, retry=%d\n",
retry_cnt);
} }
/* Each request require an ack from firmware */
request |= POWER_MODE_ACK;
/* Hit here means that driver failed to change hardware power mode to rtw_write8(rtwdev, rtwdev->hci.rpwm_addr, request);
* active state after retry 3 times. If the power state is locked at
* Deep sleep, most of the hardware circuits is not working, even /* Check firmware get the power requset and ack via cpwm register */
* register read/write. It should be treated as fatal error and ret = read_poll_timeout_atomic(rtw_read8, polling,
* requires an entire analysis about the firmware/hardware (polling ^ confirm) & BIT_RPWM_TOGGLE,
*/ 100, 15000, true, rtwdev,
WARN(1, "Hardware power state locked\n"); rtwdev->hci.cpwm_addr);
if (ret) {
/* Hit here means that driver failed to get an ack from firmware.
* The reason could be that hardware is locked at Deep sleep,
* so most of the hardware circuits are not working, even
* register read/write; or firmware is locked in some state and
* cannot get the request. It should be treated as fatal error
* and requires an entire analysis about the firmware/hardware.
*/
WARN(1, "firmware failed to ack driver for %s Deep Power mode\n",
enter ? "entering" : "leaving");
}
} }
EXPORT_SYMBOL(rtw_power_mode_change); EXPORT_SYMBOL(rtw_power_mode_change);
......
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