Re: [Intel-gfx] [PATCH v7 15/23] drm/i915/icl: Configure DSI transcoder timings

2018-10-22 Thread Madhav Chauhan

On 10/15/2018 7:58 PM, Jani Nikula wrote:

From: Madhav Chauhan 

As part of DSI enable sequence, transcoder timings
(horizontal & vertical) need to be set so that transcoder
will generate the stream output as per those timings.
This patch set required transcoder timings as per BSPEC.

v2: Remove TRANS_TIMING_SHIFT usage

v3 by Jani:
  - Rebase
  - Reduce temp variable use
  - Checkpatch fix


v3 changes are fine.

Regards,
Madhav



Signed-off-by: Madhav Chauhan 
Signed-off-by: Jani Nikula 
---
  drivers/gpu/drm/i915/icl_dsi.c | 118 +
  1 file changed, 118 insertions(+)

diff --git a/drivers/gpu/drm/i915/icl_dsi.c b/drivers/gpu/drm/i915/icl_dsi.c
index 87d5e6435791..f6ed57b28676 100644
--- a/drivers/gpu/drm/i915/icl_dsi.c
+++ b/drivers/gpu/drm/i915/icl_dsi.c
@@ -477,6 +477,121 @@ gen11_dsi_configure_transcoder(struct intel_encoder 
*encoder,
  }
  
  static void

+gen11_dsi_set_transcoder_timings(struct intel_encoder *encoder,
+const struct intel_crtc_state *pipe_config)
+{
+   struct drm_i915_private *dev_priv = to_i915(encoder->base.dev);
+   struct intel_dsi *intel_dsi = enc_to_intel_dsi(&encoder->base);
+   const struct drm_display_mode *adjusted_mode =
+   &pipe_config->base.adjusted_mode;
+   enum port port;
+   enum transcoder dsi_trans;
+   /* horizontal timings */
+   u16 htotal, hactive, hsync_start, hsync_end, hsync_size;
+   u16 hfront_porch, hback_porch;
+   /* vertical timings */
+   u16 vtotal, vactive, vsync_start, vsync_end, vsync_shift;
+
+   hactive = adjusted_mode->crtc_hdisplay;
+   htotal = adjusted_mode->crtc_htotal;
+   hsync_start = adjusted_mode->crtc_hsync_start;
+   hsync_end = adjusted_mode->crtc_hsync_end;
+   hsync_size  = hsync_end - hsync_start;
+   hfront_porch = (adjusted_mode->crtc_hsync_start -
+   adjusted_mode->crtc_hdisplay);
+   hback_porch = (adjusted_mode->crtc_htotal -
+  adjusted_mode->crtc_hsync_end);
+   vactive = adjusted_mode->crtc_vdisplay;
+   vtotal = adjusted_mode->crtc_vtotal;
+   vsync_start = adjusted_mode->crtc_vsync_start;
+   vsync_end = adjusted_mode->crtc_vsync_end;
+   vsync_shift = hsync_start - htotal / 2;
+
+   if (intel_dsi->dual_link) {
+   hactive /= 2;
+   if (intel_dsi->dual_link == DSI_DUAL_LINK_FRONT_BACK)
+   hactive += intel_dsi->pixel_overlap;
+   htotal /= 2;
+   }
+
+   /* minimum hactive as per bspec: 256 pixels */
+   if (adjusted_mode->crtc_hdisplay < 256)
+   DRM_ERROR("hactive is less then 256 pixels\n");
+
+   /* if RGB666 format, then hactive must be multiple of 4 pixels */
+   if (intel_dsi->pixel_format == MIPI_DSI_FMT_RGB666 && hactive % 4 != 0)
+   DRM_ERROR("hactive pixels are not multiple of 4\n");
+
+   /* program TRANS_HTOTAL register */
+   for_each_dsi_port(port, intel_dsi->ports) {
+   dsi_trans = dsi_port_to_transcoder(port);
+   I915_WRITE(HTOTAL(dsi_trans),
+  (hactive - 1) | ((htotal - 1) << 16));
+   }
+
+   /* TRANS_HSYNC register to be programmed only for video mode */
+   if (intel_dsi->operation_mode == INTEL_DSI_VIDEO_MODE) {
+   if (intel_dsi->video_mode_format ==
+   VIDEO_MODE_NON_BURST_WITH_SYNC_PULSE) {
+   /* BSPEC: hsync size should be atleast 16 pixels */
+   if (hsync_size < 16)
+   DRM_ERROR("hsync size < 16 pixels\n");
+   }
+
+   if (hback_porch < 16)
+   DRM_ERROR("hback porch < 16 pixels\n");
+
+   if (intel_dsi->dual_link) {
+   hsync_start /= 2;
+   hsync_end /= 2;
+   }
+
+   for_each_dsi_port(port, intel_dsi->ports) {
+   dsi_trans = dsi_port_to_transcoder(port);
+   I915_WRITE(HSYNC(dsi_trans),
+  (hsync_start - 1) | ((hsync_end - 1) << 16));
+   }
+   }
+
+   /* program TRANS_VTOTAL register */
+   for_each_dsi_port(port, intel_dsi->ports) {
+   dsi_trans = dsi_port_to_transcoder(port);
+   /*
+* FIXME: Programing this by assuming progressive mode, since
+* non-interlaced info from VBT is not saved inside
+* struct drm_display_mode.
+* For interlace mode: program required pixel minus 2
+*/
+   I915_WRITE(VTOTAL(dsi_trans),
+  (vactive - 1) | ((vtotal - 1) << 16));
+   }
+
+   if (vsync_end < vsync_start || vsync_end > vtotal)
+   DRM_ERROR("Invalid vsync_end value\n");
+
+   if (vsync_start < vactive)
+

[Intel-gfx] [PATCH v7 15/23] drm/i915/icl: Configure DSI transcoder timings

2018-10-15 Thread Jani Nikula
From: Madhav Chauhan 

As part of DSI enable sequence, transcoder timings
(horizontal & vertical) need to be set so that transcoder
will generate the stream output as per those timings.
This patch set required transcoder timings as per BSPEC.

v2: Remove TRANS_TIMING_SHIFT usage

v3 by Jani:
 - Rebase
 - Reduce temp variable use
 - Checkpatch fix

Signed-off-by: Madhav Chauhan 
Signed-off-by: Jani Nikula 
---
 drivers/gpu/drm/i915/icl_dsi.c | 118 +
 1 file changed, 118 insertions(+)

diff --git a/drivers/gpu/drm/i915/icl_dsi.c b/drivers/gpu/drm/i915/icl_dsi.c
index 87d5e6435791..f6ed57b28676 100644
--- a/drivers/gpu/drm/i915/icl_dsi.c
+++ b/drivers/gpu/drm/i915/icl_dsi.c
@@ -477,6 +477,121 @@ gen11_dsi_configure_transcoder(struct intel_encoder 
*encoder,
 }
 
 static void
+gen11_dsi_set_transcoder_timings(struct intel_encoder *encoder,
+const struct intel_crtc_state *pipe_config)
+{
+   struct drm_i915_private *dev_priv = to_i915(encoder->base.dev);
+   struct intel_dsi *intel_dsi = enc_to_intel_dsi(&encoder->base);
+   const struct drm_display_mode *adjusted_mode =
+   &pipe_config->base.adjusted_mode;
+   enum port port;
+   enum transcoder dsi_trans;
+   /* horizontal timings */
+   u16 htotal, hactive, hsync_start, hsync_end, hsync_size;
+   u16 hfront_porch, hback_porch;
+   /* vertical timings */
+   u16 vtotal, vactive, vsync_start, vsync_end, vsync_shift;
+
+   hactive = adjusted_mode->crtc_hdisplay;
+   htotal = adjusted_mode->crtc_htotal;
+   hsync_start = adjusted_mode->crtc_hsync_start;
+   hsync_end = adjusted_mode->crtc_hsync_end;
+   hsync_size  = hsync_end - hsync_start;
+   hfront_porch = (adjusted_mode->crtc_hsync_start -
+   adjusted_mode->crtc_hdisplay);
+   hback_porch = (adjusted_mode->crtc_htotal -
+  adjusted_mode->crtc_hsync_end);
+   vactive = adjusted_mode->crtc_vdisplay;
+   vtotal = adjusted_mode->crtc_vtotal;
+   vsync_start = adjusted_mode->crtc_vsync_start;
+   vsync_end = adjusted_mode->crtc_vsync_end;
+   vsync_shift = hsync_start - htotal / 2;
+
+   if (intel_dsi->dual_link) {
+   hactive /= 2;
+   if (intel_dsi->dual_link == DSI_DUAL_LINK_FRONT_BACK)
+   hactive += intel_dsi->pixel_overlap;
+   htotal /= 2;
+   }
+
+   /* minimum hactive as per bspec: 256 pixels */
+   if (adjusted_mode->crtc_hdisplay < 256)
+   DRM_ERROR("hactive is less then 256 pixels\n");
+
+   /* if RGB666 format, then hactive must be multiple of 4 pixels */
+   if (intel_dsi->pixel_format == MIPI_DSI_FMT_RGB666 && hactive % 4 != 0)
+   DRM_ERROR("hactive pixels are not multiple of 4\n");
+
+   /* program TRANS_HTOTAL register */
+   for_each_dsi_port(port, intel_dsi->ports) {
+   dsi_trans = dsi_port_to_transcoder(port);
+   I915_WRITE(HTOTAL(dsi_trans),
+  (hactive - 1) | ((htotal - 1) << 16));
+   }
+
+   /* TRANS_HSYNC register to be programmed only for video mode */
+   if (intel_dsi->operation_mode == INTEL_DSI_VIDEO_MODE) {
+   if (intel_dsi->video_mode_format ==
+   VIDEO_MODE_NON_BURST_WITH_SYNC_PULSE) {
+   /* BSPEC: hsync size should be atleast 16 pixels */
+   if (hsync_size < 16)
+   DRM_ERROR("hsync size < 16 pixels\n");
+   }
+
+   if (hback_porch < 16)
+   DRM_ERROR("hback porch < 16 pixels\n");
+
+   if (intel_dsi->dual_link) {
+   hsync_start /= 2;
+   hsync_end /= 2;
+   }
+
+   for_each_dsi_port(port, intel_dsi->ports) {
+   dsi_trans = dsi_port_to_transcoder(port);
+   I915_WRITE(HSYNC(dsi_trans),
+  (hsync_start - 1) | ((hsync_end - 1) << 16));
+   }
+   }
+
+   /* program TRANS_VTOTAL register */
+   for_each_dsi_port(port, intel_dsi->ports) {
+   dsi_trans = dsi_port_to_transcoder(port);
+   /*
+* FIXME: Programing this by assuming progressive mode, since
+* non-interlaced info from VBT is not saved inside
+* struct drm_display_mode.
+* For interlace mode: program required pixel minus 2
+*/
+   I915_WRITE(VTOTAL(dsi_trans),
+  (vactive - 1) | ((vtotal - 1) << 16));
+   }
+
+   if (vsync_end < vsync_start || vsync_end > vtotal)
+   DRM_ERROR("Invalid vsync_end value\n");
+
+   if (vsync_start < vactive)
+   DRM_ERROR("vsync_start less than vactive\n");
+
+   /* program TRANS_VSYNC register */
+