Commit ff57c651 authored by Da Lv's avatar Da Lv Committed by Sam Ravnborg

drm: kirin: Fix for hikey620 display offset problem

The original HiKey (620) board has had a long running issue
where when using a 1080p montior, the display would occasionally
blink and come come back with a horizontal offset (usually also
shifting the colors, depending on the value of the offset%4).

After lots of analysis by HiSi developers, they found the issue
was due to when running at 1080p, it was possible to hit the
device memory bandwidth limits, which could cause the DSI signal
to get out of sync.

Unfortunately the DSI logic doesn't have the ability to
automatically recover from this situation, but we can get a an
LDI underflow interrupt when it happens.

To then correct the issue, when we get an LDI underflow irq, we
we can simply suspend and resume the display, which resets the
hardware.

Thus, this patch enables the ldi underflow interrupt, and
initializes a workqueue that is used to suspend/resume the
display to recover. Then when the irq occurs we clear it and
schedule the workqueue to reset display engine.

Cc: Rongrong Zou <zourongrong@gmail.com>
Cc: Xinliang Liu <z.liuxinliang@hisilicon.com>
Cc: David Airlie <airlied@linux.ie>
Cc: Daniel Vetter <daniel@ffwll.ch>
Cc: dri-devel <dri-devel@lists.freedesktop.org>
Cc: Sam Ravnborg <sam@ravnborg.org>
Acked-by: default avatarXinliang Liu <z.liuxinliang@hisilicon.com>
Reviewed-by: default avatarSam Ravnborg <sam@ravnborg.org>
Signed-off-by: default avatarDa Lv <lvda3@hisilicon.com>
Signed-off-by: default avatarYidong Lin <linyidong@huawei.com>
[jstultz: Reworded the commit message, checkpatch cleanups]
Signed-off-by: default avatarJohn Stultz <john.stultz@linaro.org>
Signed-off-by: default avatarSam Ravnborg <sam@ravnborg.org>
Link: https://patchwork.freedesktop.org/patch/msgid/20190820230626.23253-2-john.stultz@linaro.org
parent 019cbd4a
...@@ -83,6 +83,7 @@ ...@@ -83,6 +83,7 @@
#define VSIZE_OFST 20 #define VSIZE_OFST 20
#define LDI_INT_EN 0x741C #define LDI_INT_EN 0x741C
#define FRAME_END_INT_EN_OFST 1 #define FRAME_END_INT_EN_OFST 1
#define UNDERFLOW_INT_EN_OFST 2
#define LDI_CTRL 0x7420 #define LDI_CTRL 0x7420
#define BPP_OFST 3 #define BPP_OFST 3
#define DATA_GATE_EN BIT(2) #define DATA_GATE_EN BIT(2)
......
...@@ -58,6 +58,7 @@ struct ade_hw_ctx { ...@@ -58,6 +58,7 @@ struct ade_hw_ctx {
struct ade_crtc { struct ade_crtc {
struct drm_crtc base; struct drm_crtc base;
struct ade_hw_ctx *ctx; struct ade_hw_ctx *ctx;
struct work_struct display_reset_wq;
bool enable; bool enable;
u32 out_format; u32 out_format;
}; };
...@@ -176,6 +177,7 @@ static void ade_init(struct ade_hw_ctx *ctx) ...@@ -176,6 +177,7 @@ static void ade_init(struct ade_hw_ctx *ctx)
*/ */
ade_update_bits(base + ADE_CTRL, FRM_END_START_OFST, ade_update_bits(base + ADE_CTRL, FRM_END_START_OFST,
FRM_END_START_MASK, REG_EFFECTIVE_IN_ADEEN_FRMEND); FRM_END_START_MASK, REG_EFFECTIVE_IN_ADEEN_FRMEND);
ade_update_bits(base + LDI_INT_EN, UNDERFLOW_INT_EN_OFST, MASK(1), 1);
} }
static bool ade_crtc_mode_fixup(struct drm_crtc *crtc, static bool ade_crtc_mode_fixup(struct drm_crtc *crtc,
...@@ -345,6 +347,17 @@ static void ade_crtc_disable_vblank(struct drm_crtc *crtc) ...@@ -345,6 +347,17 @@ static void ade_crtc_disable_vblank(struct drm_crtc *crtc)
MASK(1), 0); MASK(1), 0);
} }
static void drm_underflow_wq(struct work_struct *work)
{
struct ade_crtc *acrtc = container_of(work, struct ade_crtc,
display_reset_wq);
struct drm_device *drm_dev = (&acrtc->base)->dev;
struct drm_atomic_state *state;
state = drm_atomic_helper_suspend(drm_dev);
drm_atomic_helper_resume(drm_dev, state);
}
static irqreturn_t ade_irq_handler(int irq, void *data) static irqreturn_t ade_irq_handler(int irq, void *data)
{ {
struct ade_crtc *acrtc = data; struct ade_crtc *acrtc = data;
...@@ -362,6 +375,12 @@ static irqreturn_t ade_irq_handler(int irq, void *data) ...@@ -362,6 +375,12 @@ static irqreturn_t ade_irq_handler(int irq, void *data)
MASK(1), 1); MASK(1), 1);
drm_crtc_handle_vblank(crtc); drm_crtc_handle_vblank(crtc);
} }
if (status & BIT(UNDERFLOW_INT_EN_OFST)) {
ade_update_bits(base + LDI_INT_CLR, UNDERFLOW_INT_EN_OFST,
MASK(1), 1);
DRM_ERROR("LDI underflow!");
schedule_work(&acrtc->display_reset_wq);
}
return IRQ_HANDLED; return IRQ_HANDLED;
} }
...@@ -1038,6 +1057,9 @@ static int ade_drm_init(struct platform_device *pdev) ...@@ -1038,6 +1057,9 @@ static int ade_drm_init(struct platform_device *pdev)
/* vblank irq init */ /* vblank irq init */
ret = devm_request_irq(dev->dev, ctx->irq, ade_irq_handler, ret = devm_request_irq(dev->dev, ctx->irq, ade_irq_handler,
IRQF_SHARED, dev->driver->name, acrtc); IRQF_SHARED, dev->driver->name, acrtc);
INIT_WORK(&acrtc->display_reset_wq, drm_underflow_wq);
if (ret) if (ret)
return ret; return ret;
......
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