Received: by 10.223.185.116 with SMTP id b49csp7701897wrg; Thu, 1 Mar 2018 09:37:39 -0800 (PST) X-Google-Smtp-Source: AG47ELueO1d27mWrbGFTp4VXw3uq34lwqoVEeO0PgSY9ikeElfugiEPiQJQ5xQJYPutnK0D27I9L X-Received: by 2002:a17:902:b690:: with SMTP id c16-v6mr2659843pls.264.1519925858939; Thu, 01 Mar 2018 09:37:38 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1519925858; cv=none; d=google.com; s=arc-20160816; b=WqvP6qgOQkBnLIxV2LeBUQ6uCYz89vPVWVfjrqs8OAfAkJfwaIjgGuovjUjELjII6j dDdhe/jMQSE5lJgXnQQCQVekTCKKpVhsyBEGjF4/T02Mop208YEms6Mds1/3N9/QmyIm BBXyZTvmBaq1gzkceyAxhAu2Wv57Rn4ZzxLBDAlx3MK0NhHkXLqhE1blax3fXGaBXsEs vYE6uMtgZmACSSpiD9bovc7fAy+e2GpWw7rw6kunvQC0uXQB4gS+yUcBh9WcehKivDHS dN2vmH4Tm0QD/9M6LGvHCvS3fhJ4nmk8YP4LuHhh5Pri8KND/JE5yRr04jyEK06v+2Fx 0cdQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:message-id:date:subject:cc:to:from :arc-authentication-results; bh=azGPk2Jz4kaAeeNNVGHtfL4ZeBz0OdNDETG6Yj+IkCE=; b=M6omM33pKTwbp6gt0lSA7YlgD+jDi6o8Td0eaCqeexA9LS79hOP47DRzNYRWxhRpA0 MSwE7ise7Ru104nfMA9u/zbfRHQP6mqkZhZjBrQqqn5Bi/t5/05OPeUP+Y0x/AczGg6G 2hSjVnY4yhG2WzlTyDgCjnqM5rh7Hn7/A6EQw5ZT9e/uwfPXlDVFIJY9ddJiGLY+fAsF GTLigLgE/TroVakNJKPOOs34zzUvik67XUaNIEkMzy1ISGvogQz7J+dCt2yJvE07/mty MmyOBefWXJbD3Rfu8qez7paMn4MijnZoj8at8CXwD8LaWg40K44wBxtcZGztZHk8/ePk RgPQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id f5-v6si3380754plf.223.2018.03.01.09.37.22; Thu, 01 Mar 2018 09:37:38 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1033420AbeCARgl (ORCPT + 99 others); Thu, 1 Mar 2018 12:36:41 -0500 Received: from fw-tnat.cambridge.arm.com ([217.140.96.140]:56170 "EHLO cam-smtp0.cambridge.arm.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1032900AbeCARgj (ORCPT ); Thu, 1 Mar 2018 12:36:39 -0500 Received: from e110455-lin.cambridge.arm.com (e110455-lin.cambridge.arm.com [10.2.131.15]) by cam-smtp0.cambridge.arm.com (8.13.8/8.13.8) with ESMTP id w21HaXTr004484; Thu, 1 Mar 2018 17:36:33 GMT From: Liviu Dudau To: Brian Starkey Cc: Mali DP Maintainers , Alexandru-Cosmin Gheorghe , DRI devel , LKML , Liviu Dudau Subject: [PATCH] drm/mali-dp: Fix malidp_atomic_commit_hw_done() for event sending. Date: Thu, 1 Mar 2018 17:36:33 +0000 Message-Id: <20180301173633.9506-1-Liviu.Dudau@arm.com> X-Mailer: git-send-email 2.16.2 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Mali DP hardware has a 'go' bit (config_valid) for making the new scene parameters active at the next page flip. The problem with the current code is that the driver first sets this bit and then proceeds to wait for confirmation from the hardware that the configuration has been updated before arming the vblank event. As config_valid is actually asserted by the hardware after the vblank event, during the prefetch phase, when we get to arming the vblank event we are going to send it at the next vblank, in effect halving the vblank rate from the userspace perspective. Fix it by sending the userspace event from the IRQ handler, when we handle the config_valid interrupt, which syncs with the time when the hardware is active with the new parameters. Reported-by: Alexandru-Cosmin Gheorghe Signed-off-by: Liviu Dudau --- drivers/gpu/drm/arm/malidp_drv.c | 27 ++++++++++++--------------- drivers/gpu/drm/arm/malidp_drv.h | 1 + drivers/gpu/drm/arm/malidp_hw.c | 12 +++++++++--- 3 files changed, 22 insertions(+), 18 deletions(-) diff --git a/drivers/gpu/drm/arm/malidp_drv.c b/drivers/gpu/drm/arm/malidp_drv.c index d88a3b9d59cc..7be5d48eefd0 100644 --- a/drivers/gpu/drm/arm/malidp_drv.c +++ b/drivers/gpu/drm/arm/malidp_drv.c @@ -185,27 +185,26 @@ static int malidp_set_and_wait_config_valid(struct drm_device *drm) static void malidp_atomic_commit_hw_done(struct drm_atomic_state *state) { - struct drm_pending_vblank_event *event; struct drm_device *drm = state->dev; struct malidp_drm *malidp = drm->dev_private; + malidp->event = malidp->crtc.state->event; + malidp->crtc.state->event = NULL; + + /* + * if we have an event to deliver to userspace, make sure + * the vblank is enabled as we are sending it from the IRQ + * handler. + */ + if (malidp->event) + drm_crtc_vblank_get(&malidp->crtc); + if (malidp->crtc.enabled) { /* only set config_valid if the CRTC is enabled */ - if (malidp_set_and_wait_config_valid(drm)) + if (malidp_set_and_wait_config_valid(drm) < 0) DRM_DEBUG_DRIVER("timed out waiting for updated configuration\n"); } - event = malidp->crtc.state->event; - if (event) { - malidp->crtc.state->event = NULL; - - spin_lock_irq(&drm->event_lock); - if (drm_crtc_vblank_get(&malidp->crtc) == 0) - drm_crtc_arm_vblank_event(&malidp->crtc, event); - else - drm_crtc_send_vblank_event(&malidp->crtc, event); - spin_unlock_irq(&drm->event_lock); - } drm_atomic_helper_commit_hw_done(state); } @@ -232,8 +231,6 @@ static void malidp_atomic_commit_tail(struct drm_atomic_state *state) malidp_atomic_commit_hw_done(state); - drm_atomic_helper_wait_for_vblanks(drm, state); - pm_runtime_put(drm->dev); drm_atomic_helper_cleanup_planes(drm, state); diff --git a/drivers/gpu/drm/arm/malidp_drv.h b/drivers/gpu/drm/arm/malidp_drv.h index e0d12c9fc6b8..c2375bb49619 100644 --- a/drivers/gpu/drm/arm/malidp_drv.h +++ b/drivers/gpu/drm/arm/malidp_drv.h @@ -22,6 +22,7 @@ struct malidp_drm { struct malidp_hw_device *dev; struct drm_crtc crtc; wait_queue_head_t wq; + struct drm_pending_vblank_event *event; atomic_t config_valid; u32 core_id; }; diff --git a/drivers/gpu/drm/arm/malidp_hw.c b/drivers/gpu/drm/arm/malidp_hw.c index 2bfb542135ac..8abd335ec313 100644 --- a/drivers/gpu/drm/arm/malidp_hw.c +++ b/drivers/gpu/drm/arm/malidp_hw.c @@ -782,9 +782,15 @@ static irqreturn_t malidp_de_irq(int irq, void *arg) /* first handle the config valid IRQ */ dc_status = malidp_hw_read(hwdev, hw->map.dc_base + MALIDP_REG_STATUS); if (dc_status & hw->map.dc_irq_map.vsync_irq) { - /* we have a page flip event */ - atomic_set(&malidp->config_valid, 1); malidp_hw_clear_irq(hwdev, MALIDP_DC_BLOCK, dc_status); + /* do we have a page flip event? */ + if (malidp->event != NULL) { + spin_lock(&drm->event_lock); + drm_crtc_send_vblank_event(&malidp->crtc, malidp->event); + malidp->event = NULL; + spin_unlock(&drm->event_lock); + } + atomic_set(&malidp->config_valid, 1); ret = IRQ_WAKE_THREAD; } @@ -794,7 +800,7 @@ static irqreturn_t malidp_de_irq(int irq, void *arg) mask = malidp_hw_read(hwdev, MALIDP_REG_MASKIRQ); status &= mask; - if (status & de->vsync_irq) + if ((status & de->vsync_irq) && malidp->crtc.enabled) drm_crtc_handle_vblank(&malidp->crtc); malidp_hw_clear_irq(hwdev, MALIDP_DE_BLOCK, status); -- 2.16.2