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

Madhav Chauhan madhav.chauhan at intel.com
Mon Oct 22 11:15:05 UTC 2018


On 10/15/2018 7:58 PM, Jani Nikula wrote:
> From: Madhav Chauhan <madhav.chauhan at intel.com>
>
> 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 <madhav.chauhan at intel.com>
> Signed-off-by: Jani Nikula <jani.nikula at intel.com>
> ---
>   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 */
> +	for_each_dsi_port(port, intel_dsi->ports) {
> +		dsi_trans = dsi_port_to_transcoder(port);
> +		I915_WRITE(VSYNC(dsi_trans),
> +			   (vsync_start - 1) | ((vsync_end - 1) << 16));
> +	}
> +
> +	/*
> +	 * FIXME: It has to be programmed only for interlaced
> +	 * modes. Put the check condition here once interlaced
> +	 * info available as described above.
> +	 * program TRANS_VSYNCSHIFT register
> +	 */
> +	for_each_dsi_port(port, intel_dsi->ports) {
> +		dsi_trans = dsi_port_to_transcoder(port);
> +		I915_WRITE(VSYNCSHIFT(dsi_trans), vsync_shift);
> +	}
> +}
> +
> +static void
>   gen11_dsi_enable_port_and_phy(struct intel_encoder *encoder,
>   			      const struct intel_crtc_state *pipe_config)
>   {
> @@ -512,4 +627,7 @@ gen11_dsi_pre_enable(struct intel_encoder *encoder,
>   
>   	/* step4: enable DSI port and DPHY */
>   	gen11_dsi_enable_port_and_phy(encoder, pipe_config);
> +
> +	/* step6c: configure transcoder timings */
> +	gen11_dsi_set_transcoder_timings(encoder, pipe_config);
>   }



More information about the Intel-gfx mailing list