[PATCH 3/4] drm/exynos/decon5433: fix vblank event handling
Inki Dae
inki.dae at samsung.com
Tue Mar 7 09:12:27 UTC 2017
Thanks for fixing it.
Andrzej,
DECON_CRFMID register is new to me. Where did you refer this register description from? I couldn't find this register in datasheet I have for Exynos5433.
Below are a little bit trivial comments.
2017년 02월 23일 01:05에 Andrzej Hajda 이(가) 쓴 글:
> Current implementation of event handling assumes that vblank interrupt is
> always called at the right time. It is not true, it can be delayed due to
> various reasons. As a result different races can happen. The patch fixes
> the issue by using hardware frame counter present in DECON to serialize
> vblank and commit completion events.
>
> Signed-off-by: Andrzej Hajda <a.hajda at samsung.com>
> ---
> drivers/gpu/drm/exynos/exynos5433_drm_decon.c | 62 +++++++++++++++++++++++++--
> include/video/exynos5433_decon.h | 9 ++++
> 2 files changed, 67 insertions(+), 4 deletions(-)
>
> diff --git a/drivers/gpu/drm/exynos/exynos5433_drm_decon.c b/drivers/gpu/drm/exynos/exynos5433_drm_decon.c
> index 147911e..bfa9396 100644
> --- a/drivers/gpu/drm/exynos/exynos5433_drm_decon.c
> +++ b/drivers/gpu/drm/exynos/exynos5433_drm_decon.c
> @@ -68,6 +68,8 @@ struct decon_context {
> unsigned long flags;
> unsigned long out_type;
> int first_win;
> + spinlock_t vblank_lock;
> + u32 frame_id;
> };
>
> static const uint32_t decon_formats[] = {
> @@ -365,25 +367,32 @@ static void decon_disable_plane(struct exynos_drm_crtc *crtc,
> static void decon_atomic_flush(struct exynos_drm_crtc *crtc)
> {
> struct decon_context *ctx = crtc->ctx;
> + unsigned long flags;
> int i;
>
> if (test_bit(BIT_SUSPENDED, &ctx->flags))
> return;
>
> + spin_lock_irqsave(&ctx->vblank_lock, flags);
> +
> for (i = ctx->first_win; i < WINDOWS_NR; i++)
> decon_shadow_protect_win(ctx, i, false);
>
> + if (ctx->out_type & IFTYPE_I80)
> + set_bit(BIT_WIN_UPDATED, &ctx->flags);
> +
> if (test_and_clear_bit(BIT_REQUEST_UPDATE, &ctx->flags))
> decon_set_bits(ctx, DECON_UPDATE, STANDALONE_UPDATE_F, ~0);
>
> - if (ctx->out_type & IFTYPE_I80)
> - set_bit(BIT_WIN_UPDATED, &ctx->flags);
> - exynos_crtc_handle_event(crtc);
> + exynos_crtc_handle_event(ctx->crtc);
You don't have to change 'crtc' to 'ctx->crtc'. Keep 'crtc'.
> +
> + spin_unlock_irqrestore(&ctx->vblank_lock, flags);
> }
>
> static void decon_swreset(struct decon_context *ctx)
> {
> unsigned int tries;
> + unsigned long flags;
>
> writel(0, ctx->addr + DECON_VIDCON0);
> for (tries = 2000; tries; --tries) {
> @@ -401,6 +410,10 @@ static void decon_swreset(struct decon_context *ctx)
>
> WARN(tries == 0, "failed to software reset DECON\n");
>
> + spin_lock_irqsave(&ctx->vblank_lock, flags);
> + ctx->frame_id = 0;
> + spin_unlock_irqrestore(&ctx->vblank_lock, flags);
> +
> if (!(ctx->out_type & IFTYPE_HDMI))
> return;
>
> @@ -579,6 +592,46 @@ static const struct component_ops decon_component_ops = {
> .unbind = decon_unbind,
> };
>
> +static void decon_handle_vblank(struct decon_context *ctx)
> +{
> + u32 frm, pfrm, status, cnt;
> +
> + spin_lock(&ctx->vblank_lock);
> +
> + /* To get consistent result repeat read until frame id is stable. */
> + frm = readl(ctx->addr + DECON_CRFMID);
> + cnt = 3;
Is there some guide that initial value of cnt should be 3?
> + do {
> + status = readl(ctx->addr + DECON_VIDCON1);
> + pfrm = frm;
> + frm = readl(ctx->addr + DECON_CRFMID);
> + } while (frm != pfrm && --cnt);
> +
> + status &= VIDCON1_VSTATUS_MASK | VIDCON1_I80_ACTIVE;
I couldn't find I80_ACTIVE field on DECON_VIDCON1 register descrption. Do you have other datasheet, new one?
> +
> + /* In case of delayed vblank CRFMID could be already incremented,
> + * it should be taken into account.
> + */
> + if (frm > 0)
> + switch (status) {
> + case VIDCON1_VSTATUS_VS:
> + if (ctx->out_type & IFTYPE_I80)
> + break;
> + case VIDCON1_I80_ACTIVE:
> + case VIDCON1_VSTATUS_BP:
> + case VIDCON1_VSTATUS_AC:
> + --frm;
Let's add 'break;' and 'default: break;' explicitly.
> + }
> +
> + if (frm != ctx->frame_id) {
> + if (frm > ctx->frame_id)
> + drm_crtc_handle_vblank(&ctx->crtc->base);
> + ctx->frame_id = frm;
> + }
> +
> + spin_unlock(&ctx->vblank_lock);
> +}
> +
> static irqreturn_t decon_irq_handler(int irq, void *dev_id)
> {
> struct decon_context *ctx = dev_id;
> @@ -599,7 +652,7 @@ static irqreturn_t decon_irq_handler(int irq, void *dev_id)
> (VIDOUT_INTERLACE_EN_F | VIDOUT_INTERLACE_FIELD_F))
> return IRQ_HANDLED;
> }
> - drm_crtc_handle_vblank(&ctx->crtc->base);
> + decon_handle_vblank(ctx);
> }
>
> out:
> @@ -672,6 +725,7 @@ static int exynos5433_decon_probe(struct platform_device *pdev)
> __set_bit(BIT_SUSPENDED, &ctx->flags);
> ctx->dev = dev;
> ctx->out_type = (unsigned long)of_device_get_match_data(dev);
> + spin_lock_init(&ctx->vblank_lock);
>
> if (ctx->out_type & IFTYPE_HDMI) {
> ctx->first_win = 1;
> diff --git a/include/video/exynos5433_decon.h b/include/video/exynos5433_decon.h
> index ef8e2a8..beefc62 100644
> --- a/include/video/exynos5433_decon.h
> +++ b/include/video/exynos5433_decon.h
> @@ -46,6 +46,8 @@
> #define DECON_FRAMEFIFO_STATUS 0x0524
> #define DECON_CMU 0x1404
> #define DECON_UPDATE 0x1410
> +#define DECON_CRFMID 0x1414
> +#define DECON_RRFRMID 0x1418
Above definition is not used in this patch.
Thanks.
> #define DECON_UPDATE_SCHEME 0x1438
> #define DECON_VIDCON1 0x2000
> #define DECON_VIDCON2 0x2004
> @@ -142,6 +144,13 @@
> #define STANDALONE_UPDATE_F (1 << 0)
>
> /* DECON_VIDCON1 */
> +#define VIDCON1_LINECNT_MASK (0x0fff << 16)
> +#define VIDCON1_I80_ACTIVE (1 << 15)
> +#define VIDCON1_VSTATUS_MASK (0x3 << 13)
> +#define VIDCON1_VSTATUS_VS (0 << 13)
> +#define VIDCON1_VSTATUS_BP (1 << 13)
> +#define VIDCON1_VSTATUS_AC (2 << 13)
> +#define VIDCON1_VSTATUS_FP (3 << 13)
> #define VIDCON1_VCLK_MASK (0x3 << 9)
> #define VIDCON1_VCLK_RUN_VDEN_DISABLE (0x3 << 9)
> #define VIDCON1_VCLK_HOLD (0x0 << 9)
>
More information about the dri-devel
mailing list