Commit 2a7b44c5 authored by zain wang's avatar zain wang Committed by Andrzej Hajda

drm/rockchip: Restore psr->state when enable/disable psr failed

If we failed disable psr, it would hang the display until next psr
cycle coming. So we should restore psr->state when it failed.

Cc: Tomasz Figa <tfiga@chromium.org>
Signed-off-by: default avatarzain wang <wzz@rock-chips.com>
Signed-off-by: default avatarDouglas Anderson <dianders@chromium.org>
Signed-off-by: default avatarSean Paul <seanpaul@chromium.org>
Signed-off-by: default avatarThierry Escande <thierry.escande@collabora.com>
Signed-off-by: default avatarEnric Balletbo i Serra <enric.balletbo@collabora.com>
Tested-by: default avatarMarek Szyprowski <m.szyprowski@samsung.com>
Reviewed-by: default avatarHeiko Stuebner <heiko@sntech.de>
Reviewed-by: default avatarArchit Taneja <architt@codeaurora.org>
Signed-off-by: default avatarAndrzej Hajda <a.hajda@samsung.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20180423105003.9004-14-enric.balletbo@collabora.com
parent d44ba844
...@@ -153,8 +153,10 @@ int analogix_dp_disable_psr(struct analogix_dp_device *dp) ...@@ -153,8 +153,10 @@ int analogix_dp_disable_psr(struct analogix_dp_device *dp)
psr_vsc.DB1 = 0; psr_vsc.DB1 = 0;
ret = drm_dp_dpcd_writeb(&dp->aux, DP_SET_POWER, DP_SET_POWER_D0); ret = drm_dp_dpcd_writeb(&dp->aux, DP_SET_POWER, DP_SET_POWER_D0);
if (ret != 1) if (ret != 1) {
dev_err(dp->dev, "Failed to set DP Power0 %d\n", ret); dev_err(dp->dev, "Failed to set DP Power0 %d\n", ret);
return ret;
}
return analogix_dp_send_psr_spd(dp, &psr_vsc, false); return analogix_dp_send_psr_spd(dp, &psr_vsc, false);
} }
......
...@@ -77,13 +77,13 @@ struct rockchip_dp_device { ...@@ -77,13 +77,13 @@ struct rockchip_dp_device {
struct analogix_dp_plat_data plat_data; struct analogix_dp_plat_data plat_data;
}; };
static void analogix_dp_psr_set(struct drm_encoder *encoder, bool enabled) static int analogix_dp_psr_set(struct drm_encoder *encoder, bool enabled)
{ {
struct rockchip_dp_device *dp = to_dp(encoder); struct rockchip_dp_device *dp = to_dp(encoder);
int ret; int ret;
if (!analogix_dp_psr_enabled(dp->adp)) if (!analogix_dp_psr_enabled(dp->adp))
return; return 0;
DRM_DEV_DEBUG(dp->dev, "%s PSR...\n", enabled ? "Entry" : "Exit"); DRM_DEV_DEBUG(dp->dev, "%s PSR...\n", enabled ? "Entry" : "Exit");
...@@ -91,13 +91,13 @@ static void analogix_dp_psr_set(struct drm_encoder *encoder, bool enabled) ...@@ -91,13 +91,13 @@ static void analogix_dp_psr_set(struct drm_encoder *encoder, bool enabled)
PSR_WAIT_LINE_FLAG_TIMEOUT_MS); PSR_WAIT_LINE_FLAG_TIMEOUT_MS);
if (ret) { if (ret) {
DRM_DEV_ERROR(dp->dev, "line flag interrupt did not arrive\n"); DRM_DEV_ERROR(dp->dev, "line flag interrupt did not arrive\n");
return; return -ETIMEDOUT;
} }
if (enabled) if (enabled)
analogix_dp_enable_psr(dp->adp); return analogix_dp_enable_psr(dp->adp);
else else
analogix_dp_disable_psr(dp->adp); return analogix_dp_disable_psr(dp->adp);
} }
static int rockchip_dp_pre_init(struct rockchip_dp_device *dp) static int rockchip_dp_pre_init(struct rockchip_dp_device *dp)
......
...@@ -36,7 +36,7 @@ struct psr_drv { ...@@ -36,7 +36,7 @@ struct psr_drv {
struct delayed_work flush_work; struct delayed_work flush_work;
void (*set)(struct drm_encoder *encoder, bool enable); int (*set)(struct drm_encoder *encoder, bool enable);
}; };
static struct psr_drv *find_psr_by_crtc(struct drm_crtc *crtc) static struct psr_drv *find_psr_by_crtc(struct drm_crtc *crtc)
...@@ -93,19 +93,25 @@ static void psr_set_state_locked(struct psr_drv *psr, enum psr_state state) ...@@ -93,19 +93,25 @@ static void psr_set_state_locked(struct psr_drv *psr, enum psr_state state)
return; return;
} }
psr->state = state;
/* Actually commit the state change to hardware */ /* Actually commit the state change to hardware */
switch (psr->state) { switch (state) {
case PSR_ENABLE: case PSR_ENABLE:
psr->set(psr->encoder, true); if (psr->set(psr->encoder, true))
return;
break; break;
case PSR_DISABLE: case PSR_DISABLE:
case PSR_FLUSH: case PSR_FLUSH:
psr->set(psr->encoder, false); if (psr->set(psr->encoder, false))
return;
break; break;
default:
pr_err("%s: Unknown state %d\n", __func__, state);
return;
} }
psr->state = state;
} }
static void psr_set_state(struct psr_drv *psr, enum psr_state state) static void psr_set_state(struct psr_drv *psr, enum psr_state state)
...@@ -229,7 +235,7 @@ EXPORT_SYMBOL(rockchip_drm_psr_flush_all); ...@@ -229,7 +235,7 @@ EXPORT_SYMBOL(rockchip_drm_psr_flush_all);
* Zero on success, negative errno on failure. * Zero on success, negative errno on failure.
*/ */
int rockchip_drm_psr_register(struct drm_encoder *encoder, int rockchip_drm_psr_register(struct drm_encoder *encoder,
void (*psr_set)(struct drm_encoder *, bool enable)) int (*psr_set)(struct drm_encoder *, bool enable))
{ {
struct rockchip_drm_private *drm_drv = encoder->dev->dev_private; struct rockchip_drm_private *drm_drv = encoder->dev->dev_private;
struct psr_drv *psr; struct psr_drv *psr;
......
...@@ -22,7 +22,7 @@ int rockchip_drm_psr_activate(struct drm_encoder *encoder); ...@@ -22,7 +22,7 @@ int rockchip_drm_psr_activate(struct drm_encoder *encoder);
int rockchip_drm_psr_deactivate(struct drm_encoder *encoder); int rockchip_drm_psr_deactivate(struct drm_encoder *encoder);
int rockchip_drm_psr_register(struct drm_encoder *encoder, int rockchip_drm_psr_register(struct drm_encoder *encoder,
void (*psr_set)(struct drm_encoder *, bool enable)); int (*psr_set)(struct drm_encoder *, bool enable));
void rockchip_drm_psr_unregister(struct drm_encoder *encoder); void rockchip_drm_psr_unregister(struct drm_encoder *encoder);
#endif /* __ROCKCHIP_DRM_PSR__ */ #endif /* __ROCKCHIP_DRM_PSR__ */
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