UPSTREAM: drm/rockchip: Optimization vop mode set
authorMark Yao <mark.yao@rock-chips.com>
Wed, 16 Dec 2015 10:08:17 +0000 (18:08 +0800)
committerMark Yao <mark.yao@rock-chips.com>
Thu, 17 Mar 2016 01:42:02 +0000 (09:42 +0800)
Rk3288 vop timing registers is immediately register, when configure
timing on display active time, will cause tearing. use dclk reset is
not a good idea to avoid this tearing. we can avoid tearing by using
standby register.

Vop standby register will take effect at end of current frame, and
go back to work immediately when exit standby.

So we can use standby register to protect this context.

Change-Id: Ib65433900c67f9fbd3957c4b0506d6172474e5c2
Signed-off-by: Mark Yao <mark.yao@rock-chips.com>
(cherry picked from commit ce3887ed0d996e6353d739e8139b8e5faeb726d5)

drivers/gpu/drm/rockchip/rockchip_drm_vop.c

index 44f61540a0d179f6e1a3eb923ec9e22ee5e3babf..9ace3ae1081981b703b79d54581611a534a6017d 100644 (file)
@@ -1097,10 +1097,40 @@ static void vop_crtc_enable(struct drm_crtc *crtc)
 
        vop_enable(crtc);
        /*
-        * disable dclk to stop frame scan, so that we can safe config mode and
-        * enable iommu.
+        * If dclk rate is zero, mean that scanout is stop,
+        * we don't need wait any more.
         */
-       clk_disable(vop->dclk);
+       if (clk_get_rate(vop->dclk)) {
+               /*
+                * Rk3288 vop timing register is immediately, when configure
+                * display timing on display time, may cause tearing.
+                *
+                * Vop standby will take effect at end of current frame,
+                * if dsp hold valid irq happen, it means standby complete.
+                *
+                * mode set:
+                *    standby and wait complete --> |----
+                *                                  | display time
+                *                                  |----
+                *                                  |---> dsp hold irq
+                *     configure display timing --> |
+                *         standby exit             |
+                *                                  | new frame start.
+                */
+
+               reinit_completion(&vop->dsp_hold_completion);
+               vop_dsp_hold_valid_irq_enable(vop);
+
+               spin_lock(&vop->reg_lock);
+
+               VOP_CTRL_SET(vop, standby, 1);
+
+               spin_unlock(&vop->reg_lock);
+
+               wait_for_completion(&vop->dsp_hold_completion);
+
+               vop_dsp_hold_valid_irq_disable(vop);
+       }
 
        switch (vop->connector_type) {
        case DRM_MODE_CONNECTOR_LVDS:
@@ -1115,7 +1145,6 @@ static void vop_crtc_enable(struct drm_crtc *crtc)
        default:
                DRM_ERROR("unsupport connector_type[%d]\n",
                          vop->connector_type);
-               goto out;
        };
        VOP_CTRL_SET(vop, out_mode, vop->connector_out_mode);
 
@@ -1136,19 +1165,9 @@ static void vop_crtc_enable(struct drm_crtc *crtc)
        VOP_CTRL_SET(vop, vact_st_end, val);
        VOP_CTRL_SET(vop, vpost_st_end, val);
 
-
-       /*
-        * reset dclk, take all mode config affect, so the clk would run in
-        * correct frame.
-        */
-       reset_control_assert(vop->dclk_rst);
-       usleep_range(10, 20);
-       reset_control_deassert(vop->dclk_rst);
-
        clk_set_rate(vop->dclk, adjusted_mode->clock * 1000);
-out:
-       if (clk_enable(vop->dclk) < 0)
-               dev_err(vop->dev, "failed to enable dclk\n");
+
+       VOP_CTRL_SET(vop, standby, 0);
 }
 
 static void vop_crtc_atomic_flush(struct drm_crtc *crtc,