lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <pwd4hocrxrnfymby6szzp7irlveoa36er7yn5ivlht5mwxrpdz@r237bd3epols>
Date: Mon, 15 Sep 2025 14:06:58 +0300
From: Dmitry Baryshkov <dmitry.baryshkov@....qualcomm.com>
To: Harikrishna Shenoy <h-shenoy@...com>
Cc: andrzej.hajda@...el.com, neil.armstrong@...aro.org, rfoss@...nel.org,
        Laurent.pinchart@...asonboard.com, jonas@...boo.se,
        jernej.skrabec@...il.com, airlied@...il.com, simona@...ll.ch,
        maarten.lankhorst@...ux.intel.com, mripard@...nel.org,
        tzimmermann@...e.de, robh@...nel.org, krzk+dt@...nel.org,
        conor+dt@...nel.org, sjakhade@...ence.com, yamonkar@...ence.com,
        lumag@...nel.org, dianders@...omium.org, jani.nikula@...el.com,
        luca.ceresoli@...tlin.com, andy.yan@...k-chips.com,
        dri-devel@...ts.freedesktop.org, devicetree@...r.kernel.org,
        linux-kernel@...r.kernel.org, devarsht@...com, u-kumar1@...com,
        s-jain1@...com, tomi.valkeinen@...asonboard.com
Subject: Re: [PATCH v5 2/2] drm: bridge: cdns-mhdp8546: Add support for DSC
 and FEC

On Mon, Sep 15, 2025 at 04:00:41PM +0530, Harikrishna Shenoy wrote:
> From: Swapnil Jakhade <sjakhade@...ence.com>
> 
> Enable support for Display Stream Compression (DSC) in independent
> mode with a single stream, along with Forward Error Correction (FEC)
> in the Cadence MHDP8546 DisplayPort controller driver.
> 
> FEC is required when DSC is enabled to ensure reliable transmission
> of the compressed stream.
> 
> Signed-off-by: Swapnil Jakhade <sjakhade@...ence.com>
> Signed-off-by: Harikrishna Shenoy <h-shenoy@...com>
> ---
>  drivers/gpu/drm/bridge/cadence/Makefile       |   2 +-
>  .../drm/bridge/cadence/cdns-mhdp8546-core.c   | 367 ++++++++-
>  .../drm/bridge/cadence/cdns-mhdp8546-core.h   |  68 ++
>  .../drm/bridge/cadence/cdns-mhdp8546-dsc.c    | 695 ++++++++++++++++++
>  .../drm/bridge/cadence/cdns-mhdp8546-dsc.h    | 285 +++++++
>  5 files changed, 1392 insertions(+), 25 deletions(-)
>  create mode 100644 drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.c
>  create mode 100644 drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.h
> 
> diff --git a/drivers/gpu/drm/bridge/cadence/Makefile b/drivers/gpu/drm/bridge/cadence/Makefile
> index c95fd5b81d13..7957fbdc1b78 100644
> --- a/drivers/gpu/drm/bridge/cadence/Makefile
> +++ b/drivers/gpu/drm/bridge/cadence/Makefile
> @@ -3,5 +3,5 @@ obj-$(CONFIG_DRM_CDNS_DSI) += cdns-dsi.o
>  cdns-dsi-y := cdns-dsi-core.o
>  cdns-dsi-$(CONFIG_DRM_CDNS_DSI_J721E) += cdns-dsi-j721e.o
>  obj-$(CONFIG_DRM_CDNS_MHDP8546) += cdns-mhdp8546.o
> -cdns-mhdp8546-y := cdns-mhdp8546-core.o cdns-mhdp8546-hdcp.o
> +cdns-mhdp8546-y := cdns-mhdp8546-core.o cdns-mhdp8546-hdcp.o cdns-mhdp8546-dsc.o
>  cdns-mhdp8546-$(CONFIG_DRM_CDNS_MHDP8546_J721E) += cdns-mhdp8546-j721e.o
> diff --git a/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.c b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.c
> index a614d1384f71..df8bbf8919b2 100644
> --- a/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.c
> +++ b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.c
> @@ -50,6 +50,7 @@
>  #include <linux/unaligned.h>
>  
>  #include "cdns-mhdp8546-core.h"
> +#include "cdns-mhdp8546-dsc.h"
>  #include "cdns-mhdp8546-hdcp.h"
>  #include "cdns-mhdp8546-j721e.h"
>  
> @@ -545,6 +546,116 @@ int cdns_mhdp_adjust_lt(struct cdns_mhdp_device *mhdp, unsigned int nlanes,
>  	return ret;
>  }
>  
> +static int cdns_mhdp_wait_for_fec(struct cdns_mhdp_device *mhdp,
> +				  bool expected_status)
> +{
> +	u32 fec_status;
> +	unsigned long timeout = jiffies + msecs_to_jiffies(1000);
> +
> +	cdns_mhdp_reg_read(mhdp, CDNS_DP_FEC_STATUS, &fec_status);
> +	while (((fec_status & CDNS_DP_FEC_BUSY) != expected_status) &&
> +	       time_before(jiffies, timeout)) {
> +		cdns_mhdp_reg_read(mhdp, CDNS_DP_FEC_STATUS, &fec_status);
> +		cpu_relax();
> +	}
> +
> +	if (time_after_eq(jiffies, timeout)) {
> +		DRM_DEV_ERROR(mhdp->dev, "Timeout while waiting for FEC\n");
> +		return -ETIMEDOUT;
> +	}
> +
> +	return 0;
> +}
> +
> +static int cdns_mhdp_fec_sink_support(struct cdns_mhdp_device *mhdp)
> +{
> +	int ret;
> +	u16 dpcd_buffer;
> +
> +	ret = drm_dp_dpcd_read(&mhdp->aux, DP_FEC_CAPABILITY, &dpcd_buffer, 1);
> +	if (ret < 0)
> +		goto err;

Only since DP 1.4 if I'm not mistaken.

> +
> +	if (!(dpcd_buffer & DP_FEC_CAPABLE)) {
> +		ret = -EOPNOTSUPP;
> +		DRM_DEV_ERROR(mhdp->dev, "sink does not support FEC: %d\n", ret);

This is too loud.

> +		goto err;
> +	}
> +
> +	if (ret > 0)
> +		return 0;
> +err:
> +	return ret;
> +}

Consider extracting a common helper and using it here and in the Intel
DP driver. Also please use new DPCD helpers which return 0 instead of
size.

> +
> +static int cdns_mhdp_fec_sink_set_ready(struct cdns_mhdp_device *mhdp,
> +					bool enable)
> +{
> +	int ret;
> +	u8 dpcd_buffer;
> +
> +	ret = drm_dp_dpcd_read(&mhdp->aux, DP_FEC_CONFIGURATION, &dpcd_buffer, 1);
> +	if (ret < 0)
> +		goto err;

No need to read it back, it's controlled by DPTX and you don't set
anything else.

> +
> +	if (enable)
> +		dpcd_buffer |= DP_FEC_READY;
> +	else
> +		dpcd_buffer &= ~DP_FEC_READY;
> +
> +	ret = drm_dp_dpcd_write(&mhdp->aux, DP_FEC_CONFIGURATION, &dpcd_buffer, 1);
> +
> +	if (ret > 0)
> +		return 0;
> +err:
> +	DRM_DEV_ERROR(mhdp->dev, "cannot set sink FEC ready: %d\n", ret);
> +	return  ret;
> +}
> +
> +static int cdns_mhdp_fec_set_ready(struct cdns_mhdp_device *mhdp, bool enable)
> +{
> +	int ret;
> +
> +	ret = cdns_mhdp_fec_sink_support(mhdp);
> +	if (ret)
> +		goto err;
> +
> +	ret = cdns_mhdp_fec_sink_set_ready(mhdp, enable);
> +	if (ret)
> +		goto err;
> +
> +	ret = cdns_mhdp_reg_write_bit(mhdp, CDNS_DP_FEC_CTRL, 1, 1, enable);
> +	if (ret)
> +		goto err;
> +
> +err:
> +	return ret;
> +}
> +
> +static int cdns_mhdp_fec_enable(struct cdns_mhdp_device *mhdp, bool enable)
> +{
> +	int ret;
> +	u32 resp;
> +
> +	ret = cdns_mhdp_reg_read(mhdp, CDNS_DP_FRAMER_GLOBAL_CONFIG, &resp);
> +	if (ret < 0)
> +		goto err;
> +
> +	if (!(resp & CDNS_DP_NO_VIDEO_MODE)) {
> +		ret = -EIO;
> +		goto err;
> +	}
> +
> +	ret = cdns_mhdp_reg_write_bit(mhdp, CDNS_DP_FEC_CTRL, 0, 1, enable);
> +	if (ret)
> +		goto err;
> +
> +	return cdns_mhdp_wait_for_fec(mhdp, enable);
> +err:
> +	DRM_DEV_ERROR(mhdp->dev, "set fec enable failed: %d\n", ret);
> +	return ret;
> +}
> +
>  /**
>   * cdns_mhdp_link_configure() - configure a DisplayPort link
>   * @aux: DisplayPort AUX channel
> @@ -1407,6 +1518,13 @@ static int cdns_mhdp_link_up(struct cdns_mhdp_device *mhdp)
>  	amp[1] = DP_SET_ANSI_8B10B;
>  	drm_dp_dpcd_write(&mhdp->aux, DP_DOWNSPREAD_CTRL, amp, 2);
>  
> +	if (cdns_mhdp_fec_set_ready(mhdp, true)) {
> +		mhdp->fec_enabled = false;
> +		dev_info(mhdp->dev, "Cannot set FEC ready.\n");

If DPRX doesn't support FEC, we've just got several messages for no
particular reason. It's an optional feature.

> +	} else {
> +		mhdp->fec_enabled = true;
> +	}
> +
>  	if (mhdp->host.fast_link & mhdp->sink.fast_link) {
>  		dev_err(mhdp->dev, "fastlink not supported\n");
>  		return -EOPNOTSUPP;
> @@ -1494,10 +1612,14 @@ static int cdns_mhdp_connector_detect(struct drm_connector *conn,
>  	return cdns_mhdp_detect(mhdp);
>  }
>  
> -static u32 cdns_mhdp_get_bpp(struct cdns_mhdp_display_fmt *fmt)
> +static u32 cdns_mhdp_get_bpp(struct cdns_mhdp_device *mhdp)
>  {
> +	struct cdns_mhdp_display_fmt *fmt = &mhdp->display_fmt;
>  	u32 bpp;
>  
> +	if (mhdp->dsc_params.compression_enable)
> +		return mhdp->dsc_params.compressed_bpp;
> +
>  	if (fmt->y_only)
>  		return fmt->bpc;
>  
> @@ -1533,7 +1655,7 @@ bool cdns_mhdp_bandwidth_ok(struct cdns_mhdp_device *mhdp,
>  	 * units of the rate parameter.
>  	 */
>  
> -	bpp = cdns_mhdp_get_bpp(&mhdp->display_fmt);
> +	bpp = cdns_mhdp_get_bpp(mhdp);
>  	req_bw = mode->clock * bpp / 8;
>  	max_bw = lanes * rate;
>  	if (req_bw > max_bw) {
> @@ -1693,6 +1815,74 @@ static int cdns_mhdp_attach(struct drm_bridge *bridge,
>  	return ret;
>  }
>  
> +static int cdns_mhdp_dsc_set_stream_compressed(struct cdns_mhdp_device *mhdp,
> +					       int stream_id, bool compressed)
> +{
> +	u32 reg_val;
> +	int ret;
> +
> +	ret = cdns_mhdp_reg_read(mhdp, CDNS_DP_VB_ID(stream_id), &reg_val);
> +	if (ret < 0)
> +		return ret;
> +
> +	if (compressed)
> +		reg_val |= CDNS_DP_VB_ID_COMPRESSED;
> +	else
> +		reg_val &= ~CDNS_DP_VB_ID_COMPRESSED;
> +
> +	return cdns_mhdp_reg_write(mhdp, CDNS_DP_VB_ID(stream_id), reg_val);
> +}
> +
> +static int cdns_mhdp_dsc_wait_for_completion(struct cdns_mhdp_device *mhdp,
> +					     u8 event_bit, int stream_id)
> +{
> +	u32 dsc_ctrl;
> +	int ret;
> +	unsigned long timeout;
> +
> +	ret = cdns_mhdp_reg_write_bit(mhdp, CDNS_DP_DSC_CTRL(stream_id),
> +				      event_bit, 1, true);
> +	if (ret)
> +		return ret;
> +
> +	timeout = jiffies + msecs_to_jiffies(1000);
> +
> +	do {
> +		cdns_mhdp_reg_read(mhdp, CDNS_DP_DSC_CTRL(stream_id), &dsc_ctrl);
> +		cpu_relax();
> +	} while (((dsc_ctrl & (1 << event_bit)) != 0) && time_before(jiffies, timeout));
> +
> +	if (time_after_eq(jiffies, timeout)) {
> +		DRM_DEV_ERROR(mhdp->dev, "Timeout while waiting for event %d\n", event_bit);
> +		return -ETIMEDOUT;
> +	}
> +
> +	return 0;
> +}
> +
> +static int cdns_mhdp_dsc_reset(struct cdns_mhdp_device *mhdp)
> +{
> +	/* Setting reset bit in any stream resets entire DSC. Stream 0 may always be used for it. */
> +	return cdns_mhdp_dsc_wait_for_completion(mhdp, CDNS_DP_DSC_CTRL_SW_RST_BIT, 0);
> +}
> +
> +static int cdns_mhdp_dsc_update(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	return cdns_mhdp_dsc_wait_for_completion(mhdp, CDNS_DP_DSC_CTRL_REG_UPDATE_BIT,
> +						 stream_id);
> +}
> +
> +static int cdns_mhdp_dsc_enable(struct cdns_mhdp_device *mhdp, int stream_id, bool enable)
> +{
> +	return cdns_mhdp_reg_write_bit(mhdp, CDNS_DP_DSC_CTRL(stream_id),
> +				       CDNS_DP_DSC_CTRL_EN_BIT, 1, enable ? 1 : 0);
> +}
> +
> +static int cdns_mhdp_dsc_sink_enable(struct cdns_mhdp_device *mhdp, bool enable)
> +{
> +	return drm_dp_dpcd_writeb(&mhdp->aux, DP_DSC_ENABLE, enable) != 1;
> +}
> +
>  static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  				      const struct drm_display_mode *mode)
>  {
> @@ -1702,7 +1892,7 @@ static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  		front_porch, back_porch, msa_h0, msa_v0, hsync, vsync,
>  		dp_vertical_1;
>  	u8 stream_id = mhdp->stream_id;
> -	u32 bpp, bpc, pxlfmt, framer;
> +	u32 bpp, bpc, pxlfmt, framer, dp_byte_count;
>  	int ret;
>  
>  	pxlfmt = mhdp->display_fmt.color_format;
> @@ -1716,7 +1906,7 @@ static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  	     pxlfmt == DRM_COLOR_FORMAT_YCBCR422) && mode->crtc_vdisplay >= 720)
>  		misc0 = DP_YCBCR_COEFFICIENTS_ITU709;
>  
> -	bpp = cdns_mhdp_get_bpp(&mhdp->display_fmt);
> +	bpp = cdns_mhdp_get_bpp(mhdp);
>  
>  	switch (pxlfmt) {
>  	case DRM_COLOR_FORMAT_RGB444:
> @@ -1765,6 +1955,9 @@ static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  	if (mode->flags & DRM_MODE_FLAG_INTERLACE)
>  		bnd_hsync2vsync |= CDNS_IP_DET_INTERLACE_FORMAT;
>  
> +	if (mhdp->dsc_params.compression_enable)
> +		bnd_hsync2vsync |= CDNS_IP_VIF_ALIGNMENT_LSB;
> +
>  	cdns_mhdp_reg_write(mhdp, CDNS_BND_HSYNC2VSYNC(stream_id),
>  			    bnd_hsync2vsync);
>  
> @@ -1792,8 +1985,14 @@ static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  			    CDNS_DP_FRONT_PORCH(front_porch) |
>  			    CDNS_DP_BACK_PORCH(back_porch));
>  
> -	cdns_mhdp_reg_write(mhdp, CDNS_DP_BYTE_COUNT(stream_id),
> -			    mode->crtc_hdisplay * bpp / 8);
> +	if (mhdp->dsc_params.compression_enable) {
> +		dp_byte_count = (((mhdp->dsc_config.slice_chunk_size / mhdp->link.num_lanes) + 1) <<
> +				CDNS_DP_BYTE_COUNT_BYTES_IN_CHUNK_SHIFT) |
> +				((mhdp->dsc_config.slice_chunk_size / mhdp->link.num_lanes) + 1);
> +	} else {
> +		dp_byte_count = (mode->crtc_hdisplay * bpp / 8);
> +	}
> +	cdns_mhdp_reg_write(mhdp, CDNS_DP_BYTE_COUNT(stream_id), dp_byte_count);
>  
>  	msa_h0 = mode->crtc_htotal - mode->crtc_hsync_start;
>  	cdns_mhdp_reg_write(mhdp, CDNS_DP_MSA_HORIZONTAL_0(stream_id),
> @@ -1852,6 +2051,9 @@ static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  				(mode->flags & DRM_MODE_FLAG_INTERLACE) ?
>  				CDNS_DP_VB_ID_INTERLACED : 0);
>  
> +	if (mhdp->dsc_params.compression_enable)
> +		cdns_mhdp_dsc_set_stream_compressed(mhdp, stream_id, true);
> +
>  	ret = cdns_mhdp_reg_read(mhdp, CDNS_DP_FRAMER_GLOBAL_CONFIG, &framer);
>  	if (ret < 0) {
>  		dev_err(mhdp->dev,
> @@ -1860,7 +2062,6 @@ static void cdns_mhdp_configure_video(struct cdns_mhdp_device *mhdp,
>  		return;
>  	}
>  	framer |= CDNS_DP_FRAMER_EN;
> -	framer &= ~CDNS_DP_NO_VIDEO_MODE;
>  	cdns_mhdp_reg_write(mhdp, CDNS_DP_FRAMER_GLOBAL_CONFIG, framer);
>  }
>  
> @@ -1869,20 +2070,67 @@ static void cdns_mhdp_sst_enable(struct cdns_mhdp_device *mhdp,
>  {
>  	u32 rate, vs, required_bandwidth, available_bandwidth;
>  	s32 line_thresh1, line_thresh2, line_thresh = 0;
> -	int pxlclock = mode->crtc_clock;
> +	int ret, pxlclock = mode->crtc_clock;
> +	u32 vs_f, framer, pxl_repr;
>  	u32 tu_size = 64;
>  	u32 bpp;
>  
>  	/* Get rate in MSymbols per second per lane */
>  	rate = mhdp->link.rate / 1000;
>  
> -	bpp = cdns_mhdp_get_bpp(&mhdp->display_fmt);
> +	if (mhdp->fec_enabled) {
> +		if (cdns_mhdp_fec_enable(mhdp, true))
> +			mhdp->fec_enabled = false;
> +	} else {
> +		cdns_mhdp_fec_enable(mhdp, false);
> +	}
> +
> +	if (mhdp->dsc_supported && !cdns_mhdp_dsc_sink_support(mhdp)) {

Should be happening only if FEC is supported and enabled.

> +		ret = cdns_mhdp_dsc_reset(mhdp);
> +		if (ret)
> +			dev_err(mhdp->dev, "DSC reset failed. ret = %d\n", ret);
> +
> +		mhdp->dsc_params.compressed_bpp = 8;
> +		mhdp->dsc_params.slice_count = 10;

Why is it 10?

> +
> +		ret = cdns_mhdp_compute_dsc_params(mhdp);
> +		if (ret < 0) {
> +			mhdp->dsc_params.compression_enable = false;
> +			dev_err(mhdp->dev, "DSC params computation failed. ret = %d\n", ret);
> +		} else {
> +			mhdp->dsc_params.compression_enable = true;
> +			/* Write config for stream 0 */
> +			cdns_mhdp_dsc_write_enc_config(mhdp, 0, mode);
> +			cdns_mhdp_dsc_update(mhdp, 0);
> +			cdns_mhdp_dsc_write_config(mhdp);
> +		}
> +	} else {
> +		if (mhdp->dsc_params.compression_enable) {
> +			cdns_mhdp_dsc_sink_enable(mhdp, false);
> +			cdns_mhdp_dsc_set_stream_compressed(mhdp, 0, false);
> +		}
> +		mhdp->dsc_params.compression_enable = false;
> +	}
> +
> +	/* Enable DSC for stream 0 */
> +	if (mhdp->dsc_params.compression_enable) {
> +		cdns_mhdp_dsc_enable(mhdp, 0, true);
> +
> +		if (cdns_mhdp_dsc_sink_enable(mhdp, true))
> +			dev_err(mhdp->dev, "Cannot enable DSC in sink.\n");
> +		cdns_mhdp_dsc_send_pps_sdp(mhdp, 0);
> +	}
> +
> +	bpp = cdns_mhdp_get_bpp(mhdp);
>  
>  	required_bandwidth = pxlclock * bpp / 8;
>  	available_bandwidth = mhdp->link.num_lanes * rate;
>  
> -	vs = tu_size * required_bandwidth / available_bandwidth;
> -	vs /= 1000;
> +	vs_f = tu_size * required_bandwidth / available_bandwidth;
> +	if (mhdp->fec_enabled)
> +		vs_f = (vs_f * 1024) / 1000;	//2.4%
> +	vs = vs_f / 1000;
> +	vs_f = vs_f % 1000;
>  
>  	if (vs == tu_size)
>  		vs = tu_size - 1;
> @@ -1907,6 +2155,70 @@ static void cdns_mhdp_sst_enable(struct cdns_mhdp_device *mhdp,
>  						   0 : tu_size - vs));
>  
>  	cdns_mhdp_configure_video(mhdp, mode);
> +
> +	if (mhdp->dsc_params.compression_enable) {
> +		cdns_mhdp_reg_read(mhdp, CDNS_DP_FRAMER_PXL_REPR(0), &pxl_repr);
> +		pxl_repr &= ~CDNS_DP_FRAMER_PXL_REPR_M;
> +		pxl_repr &= ~CDNS_DP_FRAMER_PXL_REPR_DIFF;
> +		pxl_repr |= (((vs_f / 10) << CDNS_DP_FRAMER_PXL_REPR_M_SHIFT)
> +				& CDNS_DP_FRAMER_PXL_REPR_M) |
> +				(((100 - (vs_f / 10)) << CDNS_DP_FRAMER_PXL_REPR_DIFF_SHIFT)
> +				& CDNS_DP_FRAMER_PXL_REPR_DIFF);
> +		cdns_mhdp_reg_write(mhdp, CDNS_DP_FRAMER_PXL_REPR(0), pxl_repr);
> +	}
> +
> +	/* Enable video mode */
> +	cdns_mhdp_reg_read(mhdp, CDNS_DP_FRAMER_GLOBAL_CONFIG, &framer);
> +	framer &= ~CDNS_DP_NO_VIDEO_MODE;
> +	cdns_mhdp_reg_write(mhdp, CDNS_DP_FRAMER_GLOBAL_CONFIG, framer);
> +}
> +
> +static int cdns_mhdp_configure_car(struct cdns_mhdp_device *mhdp, bool enable)
> +{
> +	u32 dptx_car, source_pkt_car;
> +	int ret;
> +
> +	ret = cdns_mhdp_reg_read(mhdp, CDNS_DPTX_CAR, &dptx_car);
> +	if (ret < 0) {
> +		dev_err(mhdp->dev, "Failed to read CDNS_DPTX_CAR %d\n", ret);
> +		goto out;
> +	}
> +
> +	ret = cdns_mhdp_reg_read(mhdp, CDNS_SOURCE_PKT_CAR, &source_pkt_car);
> +	if (ret < 0) {
> +		dev_err(mhdp->dev, "Failed to read CDNS_SOURCE_PKT_CAR %d\n", ret);
> +		goto out;
> +	}
> +
> +	if (enable) {
> +		dev_dbg(mhdp->dev, "%s: Enable clocks for VIF and PIF\n", __func__);
> +
> +		/* Enable VIF clock for stream 0 */
> +		cdns_mhdp_reg_write(mhdp, CDNS_DPTX_CAR,
> +				    dptx_car | CDNS_VIF_CLK_EN | CDNS_VIF_CLK_RSTN);
> +
> +		/* Enable PKT clock */
> +		cdns_mhdp_reg_write(mhdp, CDNS_SOURCE_PKT_CAR,
> +				    source_pkt_car | CDNS_PKT_DATA_CLK_EN
> +						   | CDNS_PKT_DATA_RSTN
> +						   | CDNS_PKT_SYS_CLK_EN
> +						   | CDNS_PKT_SYS_RSTN);
> +	} else {
> +		dev_dbg(mhdp->dev, "%s: Disable clocks for VIF and PIF\n", __func__);
> +
> +		/* Disable VIF clock for stream 0 */
> +		cdns_mhdp_reg_write(mhdp, CDNS_DPTX_CAR,
> +				    dptx_car & ~(CDNS_VIF_CLK_EN | CDNS_VIF_CLK_RSTN));
> +
> +		/* Disable PKT clock */
> +		cdns_mhdp_reg_write(mhdp, CDNS_SOURCE_PKT_CAR,
> +				    source_pkt_car & ~(CDNS_PKT_DATA_CLK_EN
> +						   | CDNS_PKT_DATA_RSTN
> +						   | CDNS_PKT_SYS_CLK_EN
> +						   | CDNS_PKT_SYS_RSTN));
> +	}
> +out:
> +	return ret;
>  }
>  
>  static void cdns_mhdp_atomic_enable(struct drm_bridge *bridge,
> @@ -1919,8 +2231,7 @@ static void cdns_mhdp_atomic_enable(struct drm_bridge *bridge,
>  	struct drm_connector_state *conn_state;
>  	struct drm_bridge_state *new_state;
>  	const struct drm_display_mode *mode;
> -	u32 resp;
> -	int ret;
> +	int ret = 0;
>  
>  	dev_dbg(mhdp->dev, "bridge enable\n");
>  
> @@ -1935,16 +2246,13 @@ static void cdns_mhdp_atomic_enable(struct drm_bridge *bridge,
>  	if (mhdp->info && mhdp->info->ops && mhdp->info->ops->enable)
>  		mhdp->info->ops->enable(mhdp);
>  
> -	/* Enable VIF clock for stream 0 */
> -	ret = cdns_mhdp_reg_read(mhdp, CDNS_DPTX_CAR, &resp);
> +	/* Enable clocks for video and packet interfaces */
> +	ret = cdns_mhdp_configure_car(mhdp, true);
>  	if (ret < 0) {
> -		dev_err(mhdp->dev, "Failed to read CDNS_DPTX_CAR %d\n", ret);
> +		dev_err(mhdp->dev, "Failed to enable clocks %d\n", ret);
>  		goto out;
>  	}
>  
> -	cdns_mhdp_reg_write(mhdp, CDNS_DPTX_CAR,
> -			    resp | CDNS_VIF_CLK_EN | CDNS_VIF_CLK_RSTN);
> -
>  	connector = drm_atomic_get_new_connector_for_encoder(state,
>  							     bridge->encoder);
>  	if (WARN_ON(!connector))
> @@ -2020,10 +2328,8 @@ static void cdns_mhdp_atomic_disable(struct drm_bridge *bridge,
>  
>  	cdns_mhdp_link_down(mhdp);
>  
> -	/* Disable VIF clock for stream 0 */
> -	cdns_mhdp_reg_read(mhdp, CDNS_DPTX_CAR, &resp);
> -	cdns_mhdp_reg_write(mhdp, CDNS_DPTX_CAR,
> -			    resp & ~(CDNS_VIF_CLK_EN | CDNS_VIF_CLK_RSTN));
> +	/* Disable clocks for video and packet interfaces */
> +	cdns_mhdp_configure_car(mhdp, false);
>  
>  	if (mhdp->info && mhdp->info->ops && mhdp->info->ops->disable)
>  		mhdp->info->ops->disable(mhdp);
> @@ -2119,7 +2425,7 @@ static int cdns_mhdp_atomic_check(struct drm_bridge *bridge,
>  				  struct drm_connector_state *conn_state)
>  {
>  	struct cdns_mhdp_device *mhdp = bridge_to_mhdp(bridge);
> -	const struct drm_display_mode *mode = &crtc_state->adjusted_mode;
> +	struct drm_display_mode *mode = &crtc_state->adjusted_mode;
>  
>  	mutex_lock(&mhdp->link_mutex);
>  
> @@ -2139,6 +2445,10 @@ static int cdns_mhdp_atomic_check(struct drm_bridge *bridge,
>  	if (mhdp->info)
>  		bridge_state->input_bus_cfg.flags = *mhdp->info->input_bus_flags;
>  
> +	//TODO For DSC. Might need proper handling. 1920x1080 mode doesn't work without this.
> +	mode->flags |= (DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC);
> +	mode->flags &= ~(DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC);
> +
>  	mutex_unlock(&mhdp->link_mutex);
>  	return 0;
>  }
> @@ -2426,6 +2736,15 @@ static int cdns_mhdp_probe(struct platform_device *pdev)
>  		mhdp->hdcp_supported = true;
>  	}
>  
> +	mhdp->dsc_regs = devm_platform_ioremap_resource_byname(pdev, "dsc");
> +	if (IS_ERR(mhdp->dsc_regs)) {
> +		mhdp->dsc_supported = false;
> +		dev_info(dev,
> +			 "Failed to get DSC memory resource, DSC not supported\n");
> +	} else {
> +		mhdp->dsc_supported = true;
> +	}
> +
>  	mhdp->phy = devm_of_phy_get_by_index(dev, pdev->dev.of_node, 0);
>  	if (IS_ERR(mhdp->phy)) {
>  		dev_err(dev, "no PHY configured\n");
> diff --git a/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.h b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.h
> index bad2fc0c7306..c8f8071776d7 100644
> --- a/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.h
> +++ b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-core.h
> @@ -16,6 +16,7 @@
>  #include <linux/spinlock.h>
>  
>  #include <drm/display/drm_dp_helper.h>
> +#include <drm/display/drm_dsc.h>
>  #include <drm/drm_bridge.h>
>  #include <drm/drm_connector.h>
>  
> @@ -66,18 +67,30 @@ struct phy;
>  #define CDNS_VIF_CLK_EN				BIT(0)
>  #define CDNS_VIF_CLK_RSTN			BIT(1)
>  
> +#define CDNS_SOURCE_PKT_CAR			0x00918
> +#define CDNS_PKT_DATA_CLK_EN			BIT(0)
> +#define CDNS_PKT_DATA_RSTN			BIT(1)
> +#define CDNS_PKT_SYS_CLK_EN			BIT(2)
> +#define CDNS_PKT_SYS_RSTN			BIT(3)
> +
>  #define CDNS_SOURCE_VIDEO_IF(s)			(0x00b00 + ((s) * 0x20))
>  #define CDNS_BND_HSYNC2VSYNC(s)			(CDNS_SOURCE_VIDEO_IF(s) + \
>  						 0x00)
>  #define CDNS_IP_DTCT_WIN			GENMASK(11, 0)
>  #define CDNS_IP_DET_INTERLACE_FORMAT		BIT(12)
>  #define CDNS_IP_BYPASS_V_INTERFACE		BIT(13)
> +#define CDNS_IP_VIF_ALIGNMENT_LSB		BIT(14)
>  
>  #define CDNS_HSYNC2VSYNC_POL_CTRL(s)		(CDNS_SOURCE_VIDEO_IF(s) + \
>  						 0x10)
>  #define CDNS_H2V_HSYNC_POL_ACTIVE_LOW		BIT(1)
>  #define CDNS_H2V_VSYNC_POL_ACTIVE_LOW		BIT(2)
>  
> +#define CDNS_DP_DSC_CTRL(s)		        (CDNS_SOURCE_VIDEO_IF(s) + 0x14)
> +#define CDNS_DP_DSC_CTRL_EN_BIT			0
> +#define CDNS_DP_DSC_CTRL_SW_RST_BIT		1
> +#define CDNS_DP_DSC_CTRL_REG_UPDATE_BIT		2
> +
>  #define CDNS_DPTX_PHY_CONFIG			0x02000
>  #define CDNS_PHY_TRAINING_EN			BIT(0)
>  #define CDNS_PHY_TRAINING_TYPE(x)		(((x) & GENMASK(3, 0)) << 1)
> @@ -121,6 +134,10 @@ struct phy;
>  
>  #define CDNS_DP_ENHNCD				0x02304
>  
> +#define CDNS_DP_FEC_CTRL			0x02310
> +#define CDNS_DP_FEC_STATUS			0x02314
> +#define CDNS_DP_FEC_BUSY			BIT(0)
> +
>  #define CDNS_DPTX_STREAM(s)			(0x03000 + (s) * 0x80)
>  #define CDNS_DP_MSA_HORIZONTAL_0(s)		(CDNS_DPTX_STREAM(s) + 0x00)
>  #define CDNS_DP_MSAH0_H_TOTAL(x)		(x)
> @@ -178,6 +195,10 @@ struct phy;
>  #define CDNS_DP_FRAMER_YCBCR422			BIT(2)
>  #define CDNS_DP_FRAMER_YCBCR420			BIT(3)
>  #define CDNS_DP_FRAMER_Y_ONLY			BIT(4)
> +#define CDNS_DP_FRAMER_PXL_REPR_M		GENMASK(22, 16)
> +#define CDNS_DP_FRAMER_PXL_REPR_DIFF		GENMASK(30, 24)
> +#define CDNS_DP_FRAMER_PXL_REPR_M_SHIFT		16
> +#define CDNS_DP_FRAMER_PXL_REPR_DIFF_SHIFT	24
>  
>  #define CDNS_DP_FRAMER_SP(s)			(CDNS_DPTX_STREAM(s) + 0x50)
>  #define CDNS_DP_FRAMER_VSYNC_POL_LOW		BIT(0)
> @@ -198,6 +219,38 @@ struct phy;
>  #define CDNS_DP_BYTE_COUNT(s)			(CDNS_DPTX_STREAM(s) + 0x7c)
>  #define CDNS_DP_BYTE_COUNT_BYTES_IN_CHUNK_SHIFT	16
>  
> +#define CDNS_SOURCE_PACKET_IF(s)		(0x30800 + ((s) * 0x40))
> +#define CDNS_SOURCE_PIF_WR_ADDR(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x00)
> +
> +#define CDNS_SOURCE_PIF_WR_REQ(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x04)
> +#define SOURCE_PIF_WR_REQ_HOST_WR		BIT(0)
> +
> +#define CDNS_SOURCE_PIF_RD_ADDR(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x08)
> +#define CDNS_SOURCE_PIF_RD_REQ(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x0c)
> +#define CDNS_SOURCE_PIF_DATA_WR(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x10)
> +#define CDNS_SOURCE_PIF_DATA_RD(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x14)
> +#define CDNS_SOURCE_PIF_FIFO1_FLUSH(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x18)
> +#define CDNS_SOURCE_PIF_FIFO2_FLUSH(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x1c)
> +#define CDNS_SOURCE_PIF_STATUS(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x20)
> +#define CDNS_SOURCE_PIF_INT_SOURCE(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x24)
> +#define CDNS_SOURCE_PIF_INT_MASK(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x28)
> +
> +#define CDNS_SOURCE_PIF_PKT_ALLOC_REG(s)	(CDNS_SOURCE_PACKET_IF(s) + 0x2c)
> +#define SOURCE_PIF_PKT_ALLOC_REG_ACTIVE_IDLE_TYPE	BIT(17)
> +#define SOURCE_PIF_PKT_ALLOC_REG_TYPE_VALID		BIT(16)
> +#define SOURCE_PIF_PKT_ALLOC_REG_PACKET_TYPE		GENMASK(15, 8)
> +#define SOURCE_PIF_PKT_ALLOC_REG_PACKET_TYPE_SHIFT	8
> +#define SOURCE_PIF_PKT_ALLOC_REG_PKT_ALLOC_ADDR		GENMASK(3, 0)
> +
> +#define CDNS_SOURCE_PIF_PKT_ALLOC_WR_EN(s)	(CDNS_SOURCE_PACKET_IF(s) + 0x30)
> +#define SOURCE_PIF_PKT_ALLOC_WR_EN_EN		BIT(0)
> +
> +#define CDNS_SOURCE_PIF_SW_RST(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x34)
> +#define CDNS_SOURCE_PIF_PPS_HEADER(s)		(CDNS_SOURCE_PACKET_IF(s) + 0x38)
> +
> +#define CDNS_SOURCE_PIF_PPS(s)			(CDNS_SOURCE_PACKET_IF(s) + 0x3c)
> +#define SOURCE_PIF_PPS_PPS			BIT(0)
> +
>  /* mailbox */
>  #define MAILBOX_RETRY_US			1000
>  #define MAILBOX_TIMEOUT_US			2000000
> @@ -354,6 +407,7 @@ struct cdns_mhdp_hdcp {
>  struct cdns_mhdp_device {
>  	void __iomem *regs;
>  	void __iomem *sapb_regs;
> +	void __iomem *dsc_regs;
>  	void __iomem *j721e_regs;
>  
>  	struct device *dev;
> @@ -412,6 +466,20 @@ struct cdns_mhdp_device {
>  
>  	struct cdns_mhdp_hdcp hdcp;
>  	bool hdcp_supported;
> +
> +	struct drm_dsc_config dsc_config;
> +
> +	/* Display Stream Compression state */
> +	bool dsc_supported;
> +	struct {
> +		bool compression_enable;
> +		bool dsc_split;
> +		u16 compressed_bpp;
> +		u8 slice_count;
> +		u8 dsc_cap[DP_DSC_RECEIVER_CAP_SIZE];
> +	} dsc_params;
> +
> +	bool fec_enabled;
>  };
>  
>  #define connector_to_mhdp(x) container_of(x, struct cdns_mhdp_device, connector)
> diff --git a/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.c b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.c
> new file mode 100644
> index 000000000000..e540e8f2dd4a
> --- /dev/null
> +++ b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.c
> @@ -0,0 +1,695 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * Copyright (C) Cadence Design Systems, Inc.
> + *
> + * Author: Swapnil Jakhade <sjakhade@...ence.com>
> + */
> +
> +#include "cdns-mhdp8546-core.h"
> +#include "cdns-mhdp8546-dsc.h"
> +
> +void cdns_mhdp_dsc_write_config(struct cdns_mhdp_device *mhdp)
> +{
> +	u32 main_conf = 0;
> +
> +	main_conf = CDNS_DP_COM_MAIN_CONF_INPUT_MODE |
> +				CDNS_DP_COM_MAIN_CONF_AUTO_DB_UPDATE;
> +
> +	if (mhdp->dsc_params.dsc_split) {
> +		main_conf |= CDNS_DP_COM_MAIN_CONF_MUX_MODE |
> +					 CDNS_DP_COM_MAIN_CONF_SPLIT_PANEL;
> +	}
> +
> +	writel(main_conf, mhdp->dsc_regs + CDNS_DP_COM_MAIN_CONF);
> +}
> +
> +static u8 cdns_mhdp_dsc_compute_initial_lines(struct cdns_mhdp_device *mhdp)
> +{
> +	struct drm_dsc_config *dsc_cfg = &mhdp->dsc_config;
> +	unsigned long k1, real_bpp;
> +	u8 initial_lines;
> +
> +	real_bpp = (unsigned long)dsc_cfg->bits_per_pixel / 16;
> +
> +	if (dsc_cfg->bits_per_component == 8)
> +		k1 = 296;
> +	else
> +		k1 = 320;
> +
> +	if (mhdp->dsc_params.dsc_split) {
> +		initial_lines = (u8)DIV_ROUND_UP
> +				((k1 + (unsigned long)dsc_cfg->initial_xmit_delay +
> +				((((unsigned long)dsc_cfg->slice_chunk_size * 8) + 144) /
> +				real_bpp)), (unsigned long)dsc_cfg->slice_width);
> +	} else {
> +		initial_lines = (u8)DIV_ROUND_UP
> +				((k1 + (unsigned long)dsc_cfg->initial_xmit_delay +
> +				((DIV_ROUND_UP(((1 - (real_bpp / 48)) *
> +				((unsigned long)dsc_cfg->slice_chunk_size * 8)), 1) + 144) /
> +				real_bpp)), (unsigned long)dsc_cfg->slice_width);
> +	}

Is there a way to make it possible to comprehend?

> +
> +	return initial_lines;
> +}
> +
> +static void write_enc_main_conf(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	if (mhdp->dsc_config.convert_rgb)
> +		reg_val |= CDNS_DP_ENC_MAIN_CONF_CONVERT_RGB;
> +
> +	reg_val |= (mhdp->dsc_config.line_buf_depth <<
> +		   CDNS_DP_ENC_MAIN_CONF_LINEBUF_DEPTH_SHIFT) &
> +		   CDNS_DP_ENC_MAIN_CONF_LINEBUF_DEPTH;
> +
> +	reg_val |= (mhdp->dsc_config.bits_per_pixel <<
> +		   CDNS_DP_ENC_MAIN_CONF_BITS_PER_PIXEL_SHIFT) &
> +		   CDNS_DP_ENC_MAIN_CONF_BITS_PER_PIXEL;
> +
> +	if (mhdp->dsc_config.block_pred_enable)
> +		reg_val |= CDNS_DP_ENC_MAIN_CONF_BLOCK_PRED_EN;
> +
> +	reg_val |= CDNS_DP_ENC_MAIN_CONF_VIDEO_MODE;
> +
> +	reg_val |= (cdns_mhdp_dsc_compute_initial_lines(mhdp) <<
> +		   CDNS_DP_ENC_MAIN_CONF_INITIAL_LINES_SHIFT) &
> +		   CDNS_DP_ENC_MAIN_CONF_INITIAL_LINES;
> +
> +	if (mhdp->dsc_config.bits_per_component == 10)
> +		reg_val |= 0x1;
> +	else if (mhdp->dsc_config.bits_per_component != 8)
> +		dev_err(mhdp->dev, "Unsupported bits_per_component = %d\n",
> +			mhdp->dsc_config.bits_per_component);
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_MAIN_CONF(stream_id));
> +}
> +
> +static void write_enc_picture_size(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = (mhdp->dsc_config.pic_width &
> +		   CDNS_DP_ENC_PICTURE_SIZE_WIDTH);
> +
> +	reg_val |= (mhdp->dsc_config.pic_height <<
> +		   CDNS_DP_ENC_PICTURE_SIZE_HEIGHT_SHIFT) &
> +		   CDNS_DP_ENC_PICTURE_SIZE_HEIGHT;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_PICTURE_SIZE(stream_id));
> +}
> +
> +static void write_enc_slice_size(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = (mhdp->dsc_config.slice_width &
> +		   CDNS_DP_ENC_SLICE_SIZE_WIDTH);
> +
> +	reg_val |= (mhdp->dsc_config.slice_height <<
> +		   CDNS_DP_ENC_SLICE_SIZE_HEIGHT_SHIFT) &
> +		   CDNS_DP_ENC_SLICE_SIZE_HEIGHT;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_SLICE_SIZE(stream_id));
> +}
> +
> +static void write_enc_misc_size(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = ((mhdp->dsc_config.slice_width + 2) % 3) &
> +		   CDNS_DP_ENC_MISC_SIZE_LAST_GRP_SIZE;
> +
> +	reg_val |= (DSC_OUTPUT_BUFFER_MAX_ADDRESS <<
> +		   CDNS_DP_ENC_MISC_SIZE_OB_MAX_ADDR_SHIFT) &
> +		   CDNS_DP_ENC_MISC_SIZE_OB_MAX_ADDR;
> +
> +	reg_val |= (mhdp->dsc_config.slice_chunk_size <<
> +		   CDNS_DP_ENC_MISC_SIZE_CHUNK_SIZE_SHIFT) &
> +		   CDNS_DP_ENC_MISC_SIZE_CHUNK_SIZE;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_MISC_SIZE(stream_id));
> +}
> +
> +static void write_enc_hrd_delays(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = (mhdp->dsc_config.initial_xmit_delay &
> +		   CDNS_DP_ENC_HRD_DELAYS_INIT_XMIT_DELAY);
> +
> +	reg_val |= (mhdp->dsc_config.initial_dec_delay <<
> +		   CDNS_DP_ENC_HRD_DELAYS_INIT_DEC_DELAY_SHIFT) &
> +		   CDNS_DP_ENC_HRD_DELAYS_INIT_DEC_DELAY;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_HRD_DELAYS(stream_id));
> +}
> +
> +static void write_enc_rc_scale(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = (mhdp->dsc_config.initial_scale_value &
> +		   CDNS_DP_ENC_RC_SCALE_INIT_SCALE_VALUE);
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_SCALE(stream_id));
> +}
> +
> +static void write_enc_rc_scale_inc_dec(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = (mhdp->dsc_config.scale_increment_interval &
> +		   CDNS_DP_ENC_RC_SCALE_INC_INTERVAL);
> +
> +	reg_val |= (mhdp->dsc_config.scale_decrement_interval <<
> +		   CDNS_DP_ENC_RC_SCALE_DEC_INTERVAL_SHIFT) &
> +		   CDNS_DP_ENC_RC_SCALE_DEC_INTERVAL;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_SCALE_INC_DEC(stream_id));
> +}
> +
> +static void write_enc_rc_offsets(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +
> +	reg_val = (mhdp->dsc_config.first_line_bpg_offset &
> +		     CDNS_DP_ENC_RC_OFFSETS_1_FL_BPG_OFFSET);
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_OFFSETS_1(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.nfl_bpg_offset &
> +		   CDNS_DP_ENC_RC_OFFSETS_2_NFL_BPG_OFFSET);
> +
> +	reg_val |= (mhdp->dsc_config.slice_bpg_offset <<
> +		   CDNS_DP_ENC_RC_OFFSETS_2_SL_BPG_OFFSET_SHIFT) &
> +		   CDNS_DP_ENC_RC_OFFSETS_2_SL_BPG_OFFSET;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_OFFSETS_2(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.initial_offset &
> +		   CDNS_DP_ENC_RC_OFFSETS_3_INIT_OFFSET);
> +
> +	reg_val |= (mhdp->dsc_config.final_offset <<
> +		   CDNS_DP_ENC_RC_OFFSETS_3_FINAL_OFFSET_SHIFT) &
> +		   CDNS_DP_ENC_RC_OFFSETS_3_FINAL_OFFSET;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_OFFSETS_3(stream_id));
> +}
> +
> +static void write_enc_flatness_detection(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +
> +	reg_val = (mhdp->dsc_config.flatness_min_qp &
> +		   CDNS_DP_ENC_FLATNESS_DETECTION_MIN_QP);
> +
> +	reg_val |= (mhdp->dsc_config.flatness_max_qp <<
> +		   CDNS_DP_ENC_FLATNESS_DETECTION_MAX_QP_SHIFT) &
> +		   CDNS_DP_ENC_FLATNESS_DETECTION_MAX_QP;
> +
> +	reg_val |= (drm_dsc_flatness_det_thresh(&mhdp->dsc_config) <<
> +		   CDNS_DP_ENC_FLATNESS_DETECTION_THRESH_SHIFT) &
> +		   CDNS_DP_ENC_FLATNESS_DETECTION_THRESH;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_FLATNESS_DETECTION(stream_id));
> +}
> +
> +static void write_enc_rc_model_size(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +
> +	reg_val = (mhdp->dsc_config.rc_model_size &
> +		   CDNS_DP_ENC_RC_MODEL_SIZE_RC_MODEL_SIZE);
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MODEL_SIZE(stream_id));
> +}
> +
> +static void write_enc_rc_config(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +
> +	reg_val = (mhdp->dsc_config.rc_edge_factor &
> +		   CDNS_DP_ENC_RC_CONFIG_EDGE_FACTOR);
> +
> +	reg_val |= (mhdp->dsc_config.rc_quant_incr_limit0 <<
> +		   CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_0_SHIFT) &
> +		   CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_0;
> +
> +	reg_val |= (mhdp->dsc_config.rc_quant_incr_limit1 <<
> +		   CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_1;
> +
> +	reg_val |= (mhdp->dsc_config.rc_tgt_offset_high <<
> +		   CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_HI_SHIFT) &
> +		   CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_HI;
> +
> +	reg_val |= (mhdp->dsc_config.rc_tgt_offset_low <<
> +		   CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_LO_SHIFT) &
> +		   CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_LO;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_CONFIG(stream_id));
> +}
> +
> +static void write_enc_rc_buf_thresh(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +	int index = 0;
> +
> +	reg_val = (mhdp->dsc_config.rc_buf_thresh[index++] &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_0);
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_1;
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_2;
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_3;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_BUF_THRESH_0(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_buf_thresh[index++] &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_4);
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_5;
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_6;
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_7;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_BUF_THRESH_1(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_buf_thresh[index++] &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_8);
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_9;
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_10;
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_11;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_BUF_THRESH_2(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_buf_thresh[index++] &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_3_THRESH_12);
> +
> +	reg_val |= (mhdp->dsc_config.rc_buf_thresh[index++] <<
> +		   CDNS_DP_ENC_RC_BUF_THRESH_THRESH_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_BUF_THRESH_3_THRESH_13;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_BUF_THRESH_3(stream_id));
> +}
> +
> +static void write_enc_rc_min_qp(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +	int index = 0;
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_min_qp &
> +		   CDNS_DP_ENC_RC_MIN_QP_0_RANGE_0);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_0_RANGE_1;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_0_RANGE_2;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_0_RANGE_3;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_0_RANGE_4;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MIN_QP_0(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_min_qp &
> +		   CDNS_DP_ENC_RC_MIN_QP_1_RANGE_5);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_1_RANGE_6;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_1_RANGE_7;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_1_RANGE_8;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_1_RANGE_9;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MIN_QP_1(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_min_qp &
> +		   CDNS_DP_ENC_RC_MIN_QP_2_RANGE_10);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_2_RANGE_11;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_2_RANGE_12;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_2_RANGE_13;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_min_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_MIN_QP_2_RANGE_14;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MIN_QP_2(stream_id));
> +}
> +
> +static void write_enc_rc_max_qp(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +	int index = 0;
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_max_qp &
> +		   CDNS_DP_ENC_RC_MAX_QP_0_RANGE_0);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_0_RANGE_1;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_0_RANGE_2;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_0_RANGE_3;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_0_RANGE_4;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MAX_QP_0(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_max_qp &
> +		   CDNS_DP_ENC_RC_MAX_QP_1_RANGE_5);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_1_RANGE_6;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_1_RANGE_7;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_1_RANGE_8;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_1_RANGE_9;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MAX_QP_1(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_max_qp &
> +		   CDNS_DP_ENC_RC_MAX_QP_2_RANGE_10);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_2_RANGE_11;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_2_RANGE_12;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_2_RANGE_13;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_max_qp <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_MAX_QP_2_RANGE_14;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_MAX_QP_2(stream_id));
> +}
> +
> +static void write_enc_rc_range_bpg(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	u32 reg_val;
> +	int index = 0;
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_0);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_1;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_2;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_3;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_4;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_5);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_6;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_7;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_8;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_9;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_1(stream_id));
> +
> +	reg_val = (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_10);
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_11;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_12;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_13;
> +
> +	reg_val |= (mhdp->dsc_config.rc_range_params[index++].range_bpg_offset <<
> +		   CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT) &
> +		   CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_14;
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_2(stream_id));
> +}
> +
> +static void write_enc_dpi_ctrl(struct cdns_mhdp_device *mhdp, int stream_id,
> +			       const struct drm_display_mode *mode)
> +{
> +	u32 reg_val = 0;
> +
> +	reg_val = ((mode->crtc_htotal * cdns_mhdp_dsc_compute_initial_lines(mhdp)) &
> +		  CDNS_DP_ENC_DPI_CTRL_OUT_DELAY_CYCLES);
> +
> +	writel(reg_val, mhdp->dsc_regs + CDNS_DP_ENC_DPI_CTRL_OUT_DELAY(stream_id));
> +}
> +
> +void cdns_mhdp_dsc_write_enc_config(struct cdns_mhdp_device *mhdp, int stream_id,
> +				    const struct drm_display_mode *mode)
> +{
> +	write_enc_main_conf(mhdp, stream_id);
> +	write_enc_picture_size(mhdp, stream_id);
> +	write_enc_slice_size(mhdp, stream_id);
> +	write_enc_misc_size(mhdp, stream_id);
> +	write_enc_hrd_delays(mhdp, stream_id);
> +	write_enc_rc_scale(mhdp, stream_id);
> +	write_enc_rc_scale_inc_dec(mhdp, stream_id);
> +	write_enc_rc_offsets(mhdp, stream_id);
> +	write_enc_flatness_detection(mhdp, stream_id);
> +	write_enc_rc_model_size(mhdp, stream_id);
> +	write_enc_rc_config(mhdp, stream_id);
> +	write_enc_rc_buf_thresh(mhdp, stream_id);
> +	write_enc_rc_min_qp(mhdp, stream_id);
> +	write_enc_rc_max_qp(mhdp, stream_id);
> +	write_enc_rc_range_bpg(mhdp, stream_id);
> +	write_enc_dpi_ctrl(mhdp, stream_id, mode);
> +}
> +
> +int cdns_mhdp_dsc_sink_support(struct cdns_mhdp_device *mhdp)
> +{
> +	int ret;
> +
> +	ret = drm_dp_dpcd_read(&mhdp->aux, DP_DSC_SUPPORT, &mhdp->dsc_params.dsc_cap,
> +			       DP_DSC_RECEIVER_CAP_SIZE);
> +	if (ret != DP_DSC_RECEIVER_CAP_SIZE) {
> +		DRM_DEV_ERROR(mhdp->dev, "cannot read sink DSC DPCD: %d\n", ret);
> +		goto err;
> +	}
> +	if (!(mhdp->dsc_params.dsc_cap[0] & DP_DSC_DECOMPRESSION_IS_SUPPORTED)) {
> +		ret = -EOPNOTSUPP;
> +		DRM_DEV_ERROR(mhdp->dev, "sink does not support DSC: %d\n", ret);
> +		goto err;
> +	}
> +
> +	ret = 0;
> +err:
> +	return ret;
> +}
> +
> +int cdns_mhdp_compute_dsc_params(struct cdns_mhdp_device *mhdp)
> +{
> +	struct drm_bridge *bridge = &mhdp->bridge;
> +	struct drm_dsc_config *dsc_cfg = &mhdp->dsc_config;
> +	struct drm_crtc_state *crtc_state = bridge->encoder->crtc->state;
> +	u8 *dsc_dpcd = (u8 *)&mhdp->dsc_params.dsc_cap;
> +	int ret;
> +
> +	dsc_cfg->pic_width = crtc_state->adjusted_mode.crtc_hdisplay;
> +	dsc_cfg->pic_height = crtc_state->adjusted_mode.crtc_vdisplay;
> +
> +	dsc_cfg->slice_width = dsc_cfg->pic_width;
> +	dsc_cfg->slice_height = DIV_ROUND_UP(dsc_cfg->pic_height,
> +					     mhdp->dsc_params.slice_count);
> +
> +	dsc_cfg->dsc_version_major = (dsc_dpcd[DP_DSC_REV - DP_DSC_SUPPORT] &
> +				      DP_DSC_MAJOR_MASK) >> DP_DSC_MAJOR_SHIFT;
> +	dsc_cfg->dsc_version_minor =
> +		min(DSC_SUPPORTED_VERSION_MIN,
> +		    (dsc_dpcd[DP_DSC_REV - DP_DSC_SUPPORT] &
> +		     DP_DSC_MINOR_MASK) >> DP_DSC_MINOR_SHIFT);
> +
> +	dsc_cfg->convert_rgb =
> +		dsc_dpcd[DP_DSC_DEC_COLOR_FORMAT_CAP - DP_DSC_SUPPORT] & DP_DSC_RGB;
> +
> +	dsc_cfg->vbr_enable = false;
> +
> +	dsc_cfg->block_pred_enable =
> +		dsc_dpcd[DP_DSC_BLK_PREDICTION_SUPPORT - DP_DSC_SUPPORT] &
> +		DP_DSC_BLK_PREDICTION_IS_SUPPORTED;
> +
> +	dsc_cfg->bits_per_pixel = mhdp->dsc_params.compressed_bpp << 4;
> +
> +	dsc_cfg->bits_per_component = mhdp->display_fmt.bpc;
> +
> +	if (mhdp->dsc_config.bits_per_component == 8)
> +		dsc_cfg->line_buf_depth = min(9, drm_dp_dsc_sink_line_buf_depth(dsc_dpcd));
> +	else if (mhdp->dsc_config.bits_per_component == 10)
> +		dsc_cfg->line_buf_depth = min(11, drm_dp_dsc_sink_line_buf_depth(dsc_dpcd));
> +
> +	drm_dsc_set_const_params(dsc_cfg);
> +	drm_dsc_set_rc_buf_thresh(dsc_cfg);
> +
> +	ret = drm_dsc_setup_rc_params(dsc_cfg, DRM_DSC_1_1_PRE_SCR);
> +	if (ret) {
> +		dev_err(mhdp->dev, "could not find DSC RC parameters");
> +		return ret;
> +	}
> +
> +	dsc_cfg->initial_scale_value = drm_dsc_initial_scale_value(dsc_cfg);
> +	dsc_cfg->slice_count = mhdp->dsc_params.slice_count;
> +
> +	return drm_dsc_compute_rc_parameters(dsc_cfg);
> +}

Please consider extracting code common with Intel driver.

> +
> +static void cdns_mhdp_write_data_packet(struct cdns_mhdp_device *mhdp, u32 *buf,
> +					int length, int stream_id)
> +{
> +	int i;
> +	u32 reg_val;
> +
> +	for (i = 0; i < length; i++) {
> +		reg_val = buf[i];
> +		writel(reg_val, mhdp->regs + CDNS_SOURCE_PIF_DATA_WR(stream_id));
> +	}
> +}
> +
> +static void cdns_mhdp_write_pps_header(struct cdns_mhdp_device *mhdp, u32 *buf,
> +				       int stream_id)
> +{
> +	writel(SOURCE_PIF_PPS_PPS, mhdp->regs + CDNS_SOURCE_PIF_PPS(stream_id));
> +	writel(*buf, mhdp->regs + CDNS_SOURCE_PIF_PPS_HEADER(stream_id));
> +}
> +
> +static int cdns_mhdp_write_pps_infoframe(struct cdns_mhdp_device *mhdp, int stream_id,
> +					 struct drm_dsc_pps_infoframe *pps_infoframe)
> +{
> +	u32 reg_val;
> +	u32 entry_id = 0;
> +
> +	writel(1, mhdp->regs + CDNS_SOURCE_PIF_FIFO1_FLUSH(stream_id));
> +
> +	cdns_mhdp_write_pps_header(mhdp, (u32 *)&pps_infoframe->pps_header, 0);
> +	cdns_mhdp_write_data_packet(mhdp, (u32 *)&pps_infoframe->pps_payload,
> +				    DP_DSC_PPS_SIZE / 4, 0);

sizeof(foo) / 4

> +
> +	/* Entry ID */
> +	writel(entry_id, mhdp->regs + CDNS_SOURCE_PIF_WR_ADDR(stream_id));
> +
> +	writel(SOURCE_PIF_WR_REQ_HOST_WR, mhdp->regs + CDNS_SOURCE_PIF_WR_REQ(stream_id));
> +
> +	reg_val = SOURCE_PIF_PKT_ALLOC_REG_ACTIVE_IDLE_TYPE |
> +		  SOURCE_PIF_PKT_ALLOC_REG_TYPE_VALID |
> +		  ((DP_SDP_PPS << SOURCE_PIF_PKT_ALLOC_REG_PACKET_TYPE_SHIFT) &
> +		  SOURCE_PIF_PKT_ALLOC_REG_PACKET_TYPE) |
> +		  (entry_id & SOURCE_PIF_PKT_ALLOC_REG_PKT_ALLOC_ADDR);
> +
> +	writel(reg_val, mhdp->regs + CDNS_SOURCE_PIF_PKT_ALLOC_REG(stream_id));
> +
> +	writel
> +	(SOURCE_PIF_PKT_ALLOC_WR_EN_EN, mhdp->regs + CDNS_SOURCE_PIF_PKT_ALLOC_WR_EN(stream_id));

Ugh. Newline after comma, please.

> +
> +	return 0;
> +}
> +
> +int cdns_mhdp_dsc_send_pps_sdp(struct cdns_mhdp_device *mhdp, int stream_id)
> +{
> +	struct drm_dsc_config *dsc_cfg = &mhdp->dsc_config;
> +	struct drm_dsc_pps_infoframe dp_dsc_pps_sdp;
> +
> +	drm_dsc_dp_pps_header_init(&dp_dsc_pps_sdp.pps_header);
> +
> +	drm_dsc_pps_payload_pack(&dp_dsc_pps_sdp.pps_payload, dsc_cfg);
> +
> +	return cdns_mhdp_write_pps_infoframe(mhdp, stream_id, &dp_dsc_pps_sdp);
> +}
> diff --git a/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.h b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.h
> new file mode 100644
> index 000000000000..c44d7e3d0002
> --- /dev/null
> +++ b/drivers/gpu/drm/bridge/cadence/cdns-mhdp8546-dsc.h
> @@ -0,0 +1,285 @@
> +/* SPDX-License-Identifier: GPL-2.0 */
> +/*
> + * Copyright (C) Cadence Design Systems, Inc.
> + *
> + * Author: Swapnil Jakhade <sjakhade@...ence.com>
> + */
> +
> +#ifndef CDNS_MHDP_DSC_H_
> +#define CDNS_MHDP_DSC_H_
> +
> +#include <linux/bitops.h>
> +#include <linux/io.h>
> +#include <linux/minmax.h>
> +#include <drm/display/drm_dsc_helper.h>
> +#include <drm/drm_bridge.h>
> +#include <drm/drm_connector.h>
> +#include <drm/drm_crtc_helper.h>
> +#include <drm/drm_print.h>

Please don't include unnecessary headers through the header file. Use
forward declarations for structs where possible.

> +
> +#define DSC_SUPPORTED_VERSION_MIN			1
> +
> +#define DSC_OUTPUT_BUFFER_MAX_ADDRESS			3667U
> +#define DP_DSC_PPS_SIZE					128
> +
> +#define CDNS_DP_COM_MAIN_CONF				0
> +#define CDNS_DP_COM_MAIN_CONF_SPLIT_PANEL		BIT(0)
> +#define CDNS_DP_COM_MAIN_CONF_MUX_MODE			BIT(1)
> +#define CDNS_DP_COM_MAIN_CONF_MUX_SEL_OUT		BIT(2)
> +#define CDNS_DP_COM_MAIN_CONF_DE_RASTER_EN		BIT(3)
> +#define CDNS_DP_COM_MAIN_CONF_INPUT_MODE		BIT(4)
> +#define CDNS_DP_COM_MAIN_CONF_MUX_EOC_EN		BIT(5)
> +#define CDNS_DP_COM_MAIN_CONF_AUTO_DB_UPDATE		BIT(6)
> +
> +#define CDNS_DP_ENC_MAIN_CONF(d)			(0x120 + ((d) * 0x100))
> +#define CDNS_DP_ENC_MAIN_CONF_INPUT_BPC			GENMASK(1, 0)
> +#define CDNS_DP_ENC_MAIN_CONF_CONVERT_RGB		BIT(2)
> +#define CDNS_DP_ENC_MAIN_CONF_ENABLE_422		BIT(3)
> +#define CDNS_DP_ENC_MAIN_CONF_LINEBUF_DEPTH		GENMASK(7, 4)
> +#define CDNS_DP_ENC_MAIN_CONF_LINEBUF_DEPTH_SHIFT	4
> +#define CDNS_DP_ENC_MAIN_CONF_BITS_PER_PIXEL		GENMASK(17, 8)
> +#define CDNS_DP_ENC_MAIN_CONF_BITS_PER_PIXEL_SHIFT	8
> +#define CDNS_DP_ENC_MAIN_CONF_BLOCK_PRED_EN		BIT(18)
> +#define CDNS_DP_ENC_MAIN_CONF_VIDEO_MODE		BIT(19)
> +#define CDNS_DP_ENC_MAIN_CONF_ICH_RST_EOL		BIT(20)
> +#define CDNS_DP_ENC_MAIN_CONF_INITIAL_LINES		GENMASK(31, 24)
> +#define CDNS_DP_ENC_MAIN_CONF_INITIAL_LINES_SHIFT	24
> +
> +#define CDNS_DP_ENC_PICTURE_SIZE(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x4)
> +#define CDNS_DP_ENC_PICTURE_SIZE_HEIGHT			GENMASK(31, 16)
> +#define CDNS_DP_ENC_PICTURE_SIZE_HEIGHT_SHIFT		16
> +#define CDNS_DP_ENC_PICTURE_SIZE_WIDTH			GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_SLICE_SIZE(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x8)
> +#define CDNS_DP_ENC_SLICE_SIZE_HEIGHT			GENMASK(31, 16)
> +#define CDNS_DP_ENC_SLICE_SIZE_HEIGHT_SHIFT		16
> +#define CDNS_DP_ENC_SLICE_SIZE_WIDTH			GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_MISC_SIZE(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0xc)
> +#define CDNS_DP_ENC_MISC_SIZE_CHUNK_SIZE		GENMASK(31, 16)
> +#define CDNS_DP_ENC_MISC_SIZE_CHUNK_SIZE_SHIFT		16
> +#define CDNS_DP_ENC_MISC_SIZE_OB_MAX_ADDR		GENMASK(15, 2)
> +#define CDNS_DP_ENC_MISC_SIZE_OB_MAX_ADDR_SHIFT		2
> +#define CDNS_DP_ENC_MISC_SIZE_LAST_GRP_SIZE		GENMASK(1, 0)
> +
> +#define CDNS_DP_ENC_HRD_DELAYS(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x10)
> +#define CDNS_DP_ENC_HRD_DELAYS_INIT_DEC_DELAY		GENMASK(31, 16)
> +#define CDNS_DP_ENC_HRD_DELAYS_INIT_DEC_DELAY_SHIFT	16
> +#define CDNS_DP_ENC_HRD_DELAYS_INIT_XMIT_DELAY		GENMASK(9, 0)
> +
> +#define CDNS_DP_ENC_RC_SCALE(d)				(CDNS_DP_ENC_MAIN_CONF(d) + 0x14)
> +#define CDNS_DP_ENC_RC_SCALE_INIT_SCALE_VALUE		GENMASK(5, 0)
> +
> +#define CDNS_DP_ENC_RC_SCALE_INC_DEC(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x18)
> +#define CDNS_DP_ENC_RC_SCALE_DEC_INTERVAL		GENMASK(27, 16)
> +#define CDNS_DP_ENC_RC_SCALE_DEC_INTERVAL_SHIFT		16
> +#define CDNS_DP_ENC_RC_SCALE_INC_INTERVAL		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_RC_OFFSETS_1(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x1c)
> +#define CDNS_DP_ENC_RC_OFFSETS_1_FL_BPG_OFFSET		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_OFFSETS_2(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x20)
> +#define CDNS_DP_ENC_RC_OFFSETS_2_SL_BPG_OFFSET		GENMASK(31, 16)
> +#define CDNS_DP_ENC_RC_OFFSETS_2_SL_BPG_OFFSET_SHIFT	16
> +#define CDNS_DP_ENC_RC_OFFSETS_2_NFL_BPG_OFFSET		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_RC_OFFSETS_3(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x24)
> +#define CDNS_DP_ENC_RC_OFFSETS_3_FINAL_OFFSET		GENMASK(31, 16)
> +#define CDNS_DP_ENC_RC_OFFSETS_3_FINAL_OFFSET_SHIFT	16
> +#define CDNS_DP_ENC_RC_OFFSETS_3_INIT_OFFSET		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_FLATNESS_DETECTION(d)		(CDNS_DP_ENC_MAIN_CONF(d) + 0x28)
> +#define CDNS_DP_ENC_FLATNESS_DETECTION_THRESH		GENMASK(17, 10)
> +#define CDNS_DP_ENC_FLATNESS_DETECTION_THRESH_SHIFT	10
> +#define CDNS_DP_ENC_FLATNESS_DETECTION_MAX_QP		GENMASK(9, 5)
> +#define CDNS_DP_ENC_FLATNESS_DETECTION_MAX_QP_SHIFT	5
> +#define CDNS_DP_ENC_FLATNESS_DETECTION_MIN_QP		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_MODEL_SIZE(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x2c)
> +#define CDNS_DP_ENC_RC_MODEL_SIZE_RC_MODEL_SIZE		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_RC_CONFIG(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x30)
> +#define CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_LO		GENMASK(27, 24)
> +#define CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_LO_SHIFT	24
> +#define CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_HI		GENMASK(23, 20)
> +#define CDNS_DP_ENC_RC_CONFIG_TGT_OFFSET_HI_SHIFT	20
> +#define CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_1		GENMASK(17, 13)
> +#define CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_1_SHIFT	13
> +#define CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_0		GENMASK(12, 8)
> +#define CDNS_DP_ENC_RC_CONFIG_QUANT_INC_LIM_0_SHIFT	8
> +#define CDNS_DP_ENC_RC_CONFIG_EDGE_FACTOR		GENMASK(3, 0)
> +
> +#define CDNS_DP_ENC_RC_BUF_THRESH_0(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x34)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_3		GENMASK(31, 24)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_2		GENMASK(23, 16)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_1		GENMASK(15, 8)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_0_THRESH_0		GENMASK(7, 0)
> +
> +#define CDNS_DP_ENC_RC_BUF_THRESH_1(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x38)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_7		GENMASK(31, 24)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_6		GENMASK(23, 16)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_5		GENMASK(15, 8)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_1_THRESH_4		GENMASK(7, 0)
> +
> +#define CDNS_DP_ENC_RC_BUF_THRESH_2(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x3c)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_11		GENMASK(31, 24)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_10		GENMASK(23, 16)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_9		GENMASK(15, 8)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_2_THRESH_8		GENMASK(7, 0)
> +
> +#define CDNS_DP_ENC_RC_BUF_THRESH_3(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x40)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_3_THRESH_13		GENMASK(15, 8)
> +#define CDNS_DP_ENC_RC_BUF_THRESH_3_THRESH_12		GENMASK(7, 0)
> +
> +#define CDNS_DP_ENC_RC_BUF_THRESH_THRESH_3_SHIFT	24
> +#define CDNS_DP_ENC_RC_BUF_THRESH_THRESH_2_SHIFT	16
> +#define CDNS_DP_ENC_RC_BUF_THRESH_THRESH_1_SHIFT	8
> +
> +#define CDNS_DP_ENC_RC_MIN_QP_0(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x44)
> +#define CDNS_DP_ENC_RC_MIN_QP_0_RANGE_4			GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_MIN_QP_0_RANGE_3			GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_MIN_QP_0_RANGE_2			GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_MIN_QP_0_RANGE_1			GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_MIN_QP_0_RANGE_0			GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_MIN_QP_1(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x48)
> +#define CDNS_DP_ENC_RC_MIN_QP_1_RANGE_9			GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_MIN_QP_1_RANGE_8			GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_MIN_QP_1_RANGE_7			GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_MIN_QP_1_RANGE_6			GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_MIN_QP_1_RANGE_5			GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_MIN_QP_2(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x4c)
> +#define CDNS_DP_ENC_RC_MIN_QP_2_RANGE_14		GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_MIN_QP_2_RANGE_13		GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_MIN_QP_2_RANGE_12		GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_MIN_QP_2_RANGE_11		GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_MIN_QP_2_RANGE_10		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_MAX_QP_0(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x50)
> +#define CDNS_DP_ENC_RC_MAX_QP_0_RANGE_4			GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_MAX_QP_0_RANGE_3			GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_MAX_QP_0_RANGE_2			GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_MAX_QP_0_RANGE_1			GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_MAX_QP_0_RANGE_0			GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_MAX_QP_1(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x54)
> +#define CDNS_DP_ENC_RC_MAX_QP_1_RANGE_9			GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_MAX_QP_1_RANGE_8			GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_MAX_QP_1_RANGE_7			GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_MAX_QP_1_RANGE_6			GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_MAX_QP_1_RANGE_5			GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_MAX_QP_2(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0x58)
> +#define CDNS_DP_ENC_RC_MAX_QP_2_RANGE_14		GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_MAX_QP_2_RANGE_13		GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_MAX_QP_2_RANGE_12		GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_MAX_QP_2_RANGE_11		GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_MAX_QP_2_RANGE_10		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_QP_RANGE_4_SHIFT			20
> +#define CDNS_DP_ENC_RC_QP_RANGE_3_SHIFT			15
> +#define CDNS_DP_ENC_RC_QP_RANGE_2_SHIFT			10
> +#define CDNS_DP_ENC_RC_QP_RANGE_1_SHIFT			5
> +
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0(d)		(CDNS_DP_ENC_MAIN_CONF(d) + 0x5c)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_4		GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_3		GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_2		GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_1		GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_0		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_1(d)		(CDNS_DP_ENC_MAIN_CONF(d) + 0x60)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_9		GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_8		GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_7		GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_6		GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_5		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_2(d)		(CDNS_DP_ENC_MAIN_CONF(d) + 0x64)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_14		GENMASK(24, 20)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_13		GENMASK(19, 15)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_12		GENMASK(14, 10)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_11		GENMASK(9, 5)
> +#define CDNS_DP_ENC_RC_RANGE_BPG_OFFSETS_0_10		GENMASK(4, 0)
> +
> +#define CDNS_DP_ENC_DPI_CTRL_OUT_DELAY(d)		(CDNS_DP_ENC_MAIN_CONF(d) + 0x68)
> +#define CDNS_DP_ENC_DPI_CTRL_OUT_DELAY_CYCLES		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_GENERAL_STATUS(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0xa0)
> +#define CDNS_DP_ENC_GENERAL_STATUS_OUT_BUFF_FULL_1	BIT(6)
> +#define CDNS_DP_ENC_GENERAL_STATUS_OUT_BUFF_FULL_0	BIT(5)
> +#define CDNS_DP_ENC_GENERAL_STATUS_OUT_BUFF_EMPTY_1	BIT(4)
> +#define CDNS_DP_ENC_GENERAL_STATUS_OUT_BUFF_EMPTY_0	BIT(3)
> +#define CDNS_DP_ENC_GENERAL_STATUS_FRAME_DONE		BIT(2)
> +#define CDNS_DP_ENC_GENERAL_STATUS_FRAME_STARTED	BIT(1)
> +#define CDNS_DP_ENC_GENERAL_STATUS_CE			BIT(0)
> +
> +#define CDNS_DP_ENC_HSLICE_STATUS(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0xa4)
> +#define CDNS_DP_ENC_HSLICE_STATUS_ACTUAL_SLICE		GENMASK(31, 16)
> +#define CDNS_DP_ENC_HSLICE_STATUS_ACTUAL_LINE		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_OUT_STATUS(d)			(CDNS_DP_ENC_MAIN_CONF(d) + 0xa8)
> +#define CDNS_DP_ENC_OUT_STATUS_ACTUAL_SLICE		GENMASK(31, 16)
> +#define CDNS_DP_ENC_OUT_STATUS_ACTUAL_LINE		GENMASK(15, 0)
> +
> +#define CDNS_DP_ENC_INT_STAT(d)				(CDNS_DP_ENC_MAIN_CONF(d) + 0xac)
> +#define CDNS_DP_ENC_INT_STAT_OUT_BUFF_FULL_1		BIT(10)
> +#define CDNS_DP_ENC_INT_STAT_OUT_BUFF_FULL_0		BIT(9)
> +#define CDNS_DP_ENC_INT_STAT_OUT_BUFF_EMPTY_1		BIT(8)
> +#define CDNS_DP_ENC_INT_STAT_OUT_BUFF_EMPTY_0		BIT(7)
> +#define CDNS_DP_ENC_INT_STAT_FRAME_DONE			BIT(6)
> +#define CDNS_DP_ENC_INT_STAT_FRAME_STARTED		BIT(5)
> +#define CDNS_DP_ENC_INT_STAT_CE				BIT(4)
> +#define CDNS_DP_ENC_INT_STAT_RC_BUFF_OVER_1		BIT(3)
> +#define CDNS_DP_ENC_INT_STAT_RC_BUFF_OVER_0		BIT(2)
> +#define CDNS_DP_ENC_INT_STAT_OUT_BUFF_UND_1		BIT(1)
> +#define CDNS_DP_ENC_INT_STAT_OUT_BUFF_UND_0		BIT(0)
> +
> +#define CDNS_DP_ENC_INT_CLR(d)				(CDNS_DP_ENC_MAIN_CONF(d) + 0xb0)
> +#define CDNS_DP_ENC_INT_CLR_OUT_BUFF_FULL_1		BIT(10)
> +#define CDNS_DP_ENC_INT_CLR_OUT_BUFF_FULL_0		BIT(9)
> +#define CDNS_DP_ENC_INT_CLR_OUT_BUFF_EMPTY_1		BIT(8)
> +#define CDNS_DP_ENC_INT_CLR_OUT_BUFF_EMPTY_0		BIT(7)
> +#define CDNS_DP_ENC_INT_CLR_FRAME_DONE			BIT(6)
> +#define CDNS_DP_ENC_INT_CLR_FRAME_STARTED		BIT(5)
> +#define CDNS_DP_ENC_INT_CLR_CE				BIT(4)
> +#define CDNS_DP_ENC_INT_CLR_RC_BUFF_OVER_1		BIT(3)
> +#define CDNS_DP_ENC_INT_CLR_RC_BUFF_OVER_0		BIT(2)
> +#define CDNS_DP_ENC_INT_CLR_OUT_BUFF_UND_1		BIT(1)
> +#define CDNS_DP_ENC_INT_CLR_OUT_BUFF_UND_0		BIT(0)
> +
> +#define CDNS_DP_ENC_INT_MASK(d)				(CDNS_DP_ENC_MAIN_CONF(d) + 0xb4)
> +#define CDNS_DP_ENC_INT_MASK_OUT_BUFF_FULL_1		BIT(10)
> +#define CDNS_DP_ENC_INT_MASK_OUT_BUFF_FULL_0		BIT(9)
> +#define CDNS_DP_ENC_INT_MASK_OUT_BUFF_EMPTY_1		BIT(8)
> +#define CDNS_DP_ENC_INT_MASK_OUT_BUFF_EMPTY_0		BIT(7)
> +#define CDNS_DP_ENC_INT_MASK_FRAME_DONE			BIT(6)
> +#define CDNS_DP_ENC_INT_MASK_FRAME_STARTED		BIT(5)
> +#define CDNS_DP_ENC_INT_MASK_CE				BIT(4)
> +#define CDNS_DP_ENC_INT_MASK_RC_BUFF_OVER_1		BIT(3)
> +#define CDNS_DP_ENC_INT_MASK_RC_BUFF_OVER_0		BIT(2)
> +#define CDNS_DP_ENC_INT_MASK_OUT_BUFF_UND_1		BIT(1)
> +#define CDNS_DP_ENC_INT_MASK_OUT_BUFF_UND_0		BIT(0)
> +
> +#define CDNS_DP_ENC_INT_TEST(d)				(CDNS_DP_ENC_MAIN_CONF(d) + 0xb8)
> +#define CDNS_DP_ENC_INT_TEST_OUT_BUFF_FULL_1		BIT(10)
> +#define CDNS_DP_ENC_INT_TEST_OUT_BUFF_FULL_0		BIT(9)
> +#define CDNS_DP_ENC_INT_TEST_OUT_BUFF_EMPTY_1		BIT(8)
> +#define CDNS_DP_ENC_INT_TEST_OUT_BUFF_EMPTY_0		BIT(7)
> +#define CDNS_DP_ENC_INT_TEST_FRAME_DONE			BIT(6)
> +#define CDNS_DP_ENC_INT_TEST_FRAME_STARTED		BIT(5)
> +#define CDNS_DP_ENC_INT_TEST_CE				BIT(4)
> +#define CDNS_DP_ENC_INT_TEST_RC_BUFF_OVER_1		BIT(3)
> +#define CDNS_DP_ENC_INT_TEST_RC_BUFF_OVER_0		BIT(2)
> +#define CDNS_DP_ENC_INT_TEST_OUT_BUFF_UND_1		BIT(1)
> +#define CDNS_DP_ENC_INT_TEST_OUT_BUFF_UND_0		BIT(0)
> +
> +void cdns_mhdp_dsc_write_config(struct cdns_mhdp_device *mhdp);
> +void cdns_mhdp_dsc_write_enc_config(struct cdns_mhdp_device *mhdp, int stream_id,
> +				    const struct drm_display_mode *mode);
> +int cdns_mhdp_dsc_sink_support(struct cdns_mhdp_device *mhdp);
> +int cdns_mhdp_compute_dsc_params(struct cdns_mhdp_device *mhdp);
> +int cdns_mhdp_dsc_send_pps_sdp(struct cdns_mhdp_device *mhdp, int stream_id);
> +
> +#endif /* MHDP_DSC_H_ */
> -- 
> 2.34.1
> 

-- 
With best wishes
Dmitry

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ