On Mon, 2017-03-13 at 16:54 +0530, Shashank Sharma wrote:
> Geminilake platform sports a native HDMI 2.0 controller, and is
> capable of driving pixel-clocks upto 594Mhz. HDMI 2.0 spec
> mendates scrambling for these higher clocks, for reduced RF footprint.
> 
> This patch checks if the monitor supports scrambling, and if required,
> enables it during the modeset.
> 
> V2: Addressed review comments from Ville:
>  - Do not track scrambling status in DRM layer, track somewhere in
>    driver like in intel_crtc_state.
>  - Don't talk to monitor at such a low layer, set monitor scrambling
>    in intel_enable_ddi() before enabling the port.
> 
> V3: Addressed review comments from Jani
>  - In comments, function names, use "sink" instead of "monitor",
>    so that the implementation could be close to the language of
>    HDMI spec.
> 
> V4: Addressed review comment from Maarten
>  - scrambling -> hdmi_scrambling
>  - high_tmds_clock_ratio -> hdmi_high_tmds_clock_ratio
> 
> V5: Addressed review comments from Ville and Ander
>  - Do not modifiy the crtc_state after compute_config. Move all
>    scrambling and tmds_clock_ratio calcutations to compute_config.
>  - While setting scrambling for source/sink, do not check the
>    conditions again, just go by the crtc_state flags. This will
>    simplyfy the condition checks.
> 
> V6: Addressed review comments from Ville
>  - Do not add IS_GLK check in disable/enable function, instead add it
>    in compute_config, while setting state flags.
>  - Remove unnecessary paranthesis.
>  - Simplyfy handle_sink_scrambling function as suggested.
>  - Add readout code for scrambling status in get_ddi_config and add a
>    check for the same in pipe_config_compare.
> 
> V7: Addressed review comments from Ander/Ville
>  - No separate function for source scrambling, make it inline
>  - Align the last line of the macro TRANS_DDI_HDMI_SCRAMBLING_MASK
>  - Do not add platform check while setting source scrambling
>  - Use pipe_config instead of crtc->config to set sink scrambling
>  - To readout scrambling status, Compare with SCRAMBLING_MASK
>    not any of its bits
>  - Remove platform check in intel_pipe_config_compare while checking
>    scrambling status
> 
> V8: Fixed mege conflict, Addressed review comments from Ander
>  - Remove the desciption/comment about scrambling fom the caller, move
>    it to the function
>  - Move the IS_GLK check into scrambling function
>  - Fix alignment
> 
> V9: Fixed review comments from Ville, Ander
>  - Pass the scrambling state variables as bool input to the sink_scrambling
>    function and let the disable call be unconditional.
>  - Fix alignments in function calls and debug messages.
>  - Add kernel doc for function intel_hdmi_handle_sink_scrambling
> 
> V10: Rebase
> Signed-off-by: Shashank Sharma <shashank.sha...@intel.com>

Reviewed-by: Ander Conselvan de Oliveira <conselv...@gmail.com>

> ---
>  drivers/gpu/drm/i915/i915_reg.h      |  7 ++++
>  drivers/gpu/drm/i915/intel_ddi.c     | 23 +++++++++++++
>  drivers/gpu/drm/i915/intel_display.c |  3 ++
>  drivers/gpu/drm/i915/intel_drv.h     | 10 ++++++
>  drivers/gpu/drm/i915/intel_hdmi.c    | 63 
> ++++++++++++++++++++++++++++++++++++
>  5 files changed, 106 insertions(+)
> 
> diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h
> index 19d42e8..a1aae7a 100644
> --- a/drivers/gpu/drm/i915/i915_reg.h
> +++ b/drivers/gpu/drm/i915/i915_reg.h
> @@ -7831,7 +7831,14 @@ enum {
>  #define  TRANS_DDI_EDP_INPUT_B_ONOFF (5<<12)
>  #define  TRANS_DDI_EDP_INPUT_C_ONOFF (6<<12)
>  #define  TRANS_DDI_DP_VC_PAYLOAD_ALLOC       (1<<8)
> +#define  TRANS_DDI_HDMI_SCRAMBLER_CTS_ENABLE (1<<7)
> +#define  TRANS_DDI_HDMI_SCRAMBLER_RESET_FREQ (1<<6)
>  #define  TRANS_DDI_BFI_ENABLE                (1<<4)
> +#define  TRANS_DDI_HIGH_TMDS_CHAR_RATE       (1<<4)
> +#define  TRANS_DDI_HDMI_SCRAMBLING   (1<<0)
> +#define  TRANS_DDI_HDMI_SCRAMBLING_MASK (TRANS_DDI_HDMI_SCRAMBLER_CTS_ENABLE 
> \
> +                                     | TRANS_DDI_HDMI_SCRAMBLER_RESET_FREQ \
> +                                     | TRANS_DDI_HDMI_SCRAMBLING)
>  
>  /* DisplayPort Transport Control */
>  #define _DP_TP_CTL_A                 0x64040
> diff --git a/drivers/gpu/drm/i915/intel_ddi.c 
> b/drivers/gpu/drm/i915/intel_ddi.c
> index ee341ef..169d2b4 100644
> --- a/drivers/gpu/drm/i915/intel_ddi.c
> +++ b/drivers/gpu/drm/i915/intel_ddi.c
> @@ -1309,6 +1309,11 @@ void intel_ddi_enable_transcoder_func(const struct 
> intel_crtc_state *crtc_state)
>                       temp |= TRANS_DDI_MODE_SELECT_HDMI;
>               else
>                       temp |= TRANS_DDI_MODE_SELECT_DVI;
> +
> +             if (crtc_state->hdmi_scrambling)
> +                     temp |= TRANS_DDI_HDMI_SCRAMBLING_MASK;
> +             if (crtc_state->hdmi_high_tmds_clock_ratio)
> +                     temp |= TRANS_DDI_HIGH_TMDS_CHAR_RATE;
>       } else if (type == INTEL_OUTPUT_ANALOG) {
>               temp |= TRANS_DDI_MODE_SELECT_FDI;
>               temp |= (crtc_state->fdi_lanes - 1) << 1;
> @@ -1881,6 +1886,12 @@ static void intel_enable_ddi(struct intel_encoder 
> *intel_encoder,
>       if (type == INTEL_OUTPUT_HDMI) {
>               struct intel_digital_port *intel_dig_port =
>                       enc_to_dig_port(encoder);
> +             bool clock_ratio = pipe_config->hdmi_high_tmds_clock_ratio;
> +             bool scrambling = pipe_config->hdmi_scrambling;
> +
> +             intel_hdmi_handle_sink_scrambling(intel_encoder,
> +                                               conn_state->connector,
> +                                               clock_ratio, scrambling);
>  
>               /* In HDMI/DVI mode, the port width, and swing/emphasis values
>                * are ignored so nothing special needs to be done besides
> @@ -1914,6 +1925,12 @@ static void intel_disable_ddi(struct intel_encoder 
> *intel_encoder,
>       if (old_crtc_state->has_audio)
>               intel_audio_codec_disable(intel_encoder);
>  
> +     if (type == INTEL_OUTPUT_HDMI) {
> +             intel_hdmi_handle_sink_scrambling(intel_encoder,
> +                                               old_conn_state->connector,
> +                                               false, false);
> +     }
> +
>       if (type == INTEL_OUTPUT_EDP) {
>               struct intel_dp *intel_dp = enc_to_intel_dp(encoder);
>  
> @@ -2040,6 +2057,12 @@ void intel_ddi_get_config(struct intel_encoder 
> *encoder,
>  
>               if (intel_hdmi->infoframe_enabled(&encoder->base, pipe_config))
>                       pipe_config->has_infoframe = true;
> +
> +             if ((temp & TRANS_DDI_HDMI_SCRAMBLING_MASK) ==
> +                     TRANS_DDI_HDMI_SCRAMBLING_MASK)
> +                     pipe_config->hdmi_scrambling = true;
> +             if (temp & TRANS_DDI_HIGH_TMDS_CHAR_RATE)
> +                     pipe_config->hdmi_high_tmds_clock_ratio = true;
>               /* fall through */
>       case TRANS_DDI_MODE_SELECT_DVI:
>               pipe_config->lane_count = 4;
> diff --git a/drivers/gpu/drm/i915/intel_display.c 
> b/drivers/gpu/drm/i915/intel_display.c
> index 5526a19..5860ea1 100644
> --- a/drivers/gpu/drm/i915/intel_display.c
> +++ b/drivers/gpu/drm/i915/intel_display.c
> @@ -11674,6 +11674,9 @@ static void __printf(3, 4)
>       if ((INTEL_GEN(dev_priv) < 8 && !IS_HASWELL(dev_priv)) ||
>           IS_VALLEYVIEW(dev_priv) || IS_CHERRYVIEW(dev_priv))
>               PIPE_CONF_CHECK_I(limited_color_range);
> +
> +     PIPE_CONF_CHECK_I(hdmi_scrambling);
> +     PIPE_CONF_CHECK_I(hdmi_high_tmds_clock_ratio);
>       PIPE_CONF_CHECK_I(has_infoframe);
>  
>       PIPE_CONF_CHECK_I(has_audio);
> diff --git a/drivers/gpu/drm/i915/intel_drv.h 
> b/drivers/gpu/drm/i915/intel_drv.h
> index 578f7d2..62d767e 100644
> --- a/drivers/gpu/drm/i915/intel_drv.h
> +++ b/drivers/gpu/drm/i915/intel_drv.h
> @@ -729,6 +729,12 @@ struct intel_crtc_state {
>  
>       /* bitmask of visible planes (enum plane_id) */
>       u8 active_planes;
> +
> +     /* HDMI scrambling status */
> +     bool hdmi_scrambling;
> +
> +     /* HDMI High TMDS char rate ratio */
> +     bool hdmi_high_tmds_clock_ratio;
>  };
>  
>  struct intel_crtc {
> @@ -1616,6 +1622,10 @@ void intel_hdmi_init_connector(struct 
> intel_digital_port *intel_dig_port,
>  bool intel_hdmi_compute_config(struct intel_encoder *encoder,
>                              struct intel_crtc_state *pipe_config,
>                              struct drm_connector_state *conn_state);
> +void intel_hdmi_handle_sink_scrambling(struct intel_encoder *intel_encoder,
> +                                    struct drm_connector *connector,
> +                                    bool high_tmds_clock_ratio,
> +                                    bool scrambling);
>  void intel_dp_dual_mode_set_tmds_output(struct intel_hdmi *hdmi, bool 
> enable);
>  
>  
> diff --git a/drivers/gpu/drm/i915/intel_hdmi.c 
> b/drivers/gpu/drm/i915/intel_hdmi.c
> index c2184f7..23b3387 100644
> --- a/drivers/gpu/drm/i915/intel_hdmi.c
> +++ b/drivers/gpu/drm/i915/intel_hdmi.c
> @@ -34,6 +34,7 @@
>  #include <drm/drm_atomic_helper.h>
>  #include <drm/drm_crtc.h>
>  #include <drm/drm_edid.h>
> +#include <drm/drm_scdc_helper.h>
>  #include "intel_drv.h"
>  #include <drm/i915_drm.h>
>  #include <drm/intel_lpe_audio.h>
> @@ -1316,6 +1317,7 @@ bool intel_hdmi_compute_config(struct intel_encoder 
> *encoder,
>       struct intel_hdmi *intel_hdmi = enc_to_intel_hdmi(&encoder->base);
>       struct drm_i915_private *dev_priv = to_i915(encoder->base.dev);
>       struct drm_display_mode *adjusted_mode = 
> &pipe_config->base.adjusted_mode;
> +     struct drm_scdc *scdc = &conn_state->connector->display_info.hdmi.scdc;
>       int clock_8bpc = pipe_config->base.adjusted_mode.crtc_clock;
>       int clock_12bpc = clock_8bpc * 3 / 2;
>       int desired_bpp;
> @@ -1385,6 +1387,16 @@ bool intel_hdmi_compute_config(struct intel_encoder 
> *encoder,
>  
>       pipe_config->lane_count = 4;
>  
> +     if (scdc->scrambling.supported && IS_GEMINILAKE(dev_priv)) {
> +             if (scdc->scrambling.low_rates)
> +                     pipe_config->hdmi_scrambling = true;
> +
> +             if (pipe_config->port_clock > 340000) {
> +                     pipe_config->hdmi_scrambling = true;
> +                     pipe_config->hdmi_high_tmds_clock_ratio = true;
> +             }
> +     }
> +
>       return true;
>  }
>  
> @@ -1794,6 +1806,57 @@ static void intel_hdmi_destroy(struct drm_connector 
> *connector)
>       intel_hdmi->aspect_ratio = HDMI_PICTURE_ASPECT_NONE;
>  }
>  
> +/*
> + * intel_hdmi_handle_sink_scrambling: handle sink scrambling/clock ratio 
> setup
> + * @encoder: intel_encoder
> + * @connector: drm_connector
> + * @high_tmds_clock_ratio = bool to indicate if the function needs to set
> + *  or reset the high tmds clock ratio for scrambling
> + * @scrambling: bool to Indicate if the function needs to set or reset
> + *  sink scrambling
> + *
> + * This function handles scrambling on HDMI 2.0 capable sinks.
> + * If required clock rate is > 340 Mhz && scrambling is supported by sink
> + * it enables scrambling. This should be called before enabling the HDMI
> + * 2.0 port, as the sink can choose to disable the scrambling if it doesn't
> + * detect a scrambled clock within 100 ms.
> + */
> +void intel_hdmi_handle_sink_scrambling(struct intel_encoder *encoder,
> +                                    struct drm_connector *connector,
> +                                    bool high_tmds_clock_ratio,
> +                                    bool scrambling)
> +{
> +     struct intel_hdmi *intel_hdmi = enc_to_intel_hdmi(&encoder->base);
> +     struct drm_i915_private *dev_priv = connector->dev->dev_private;
> +     struct drm_scrambling *sink_scrambling =
> +                             &connector->display_info.hdmi.scdc.scrambling;
> +     struct i2c_adapter *adptr = intel_gmbus_get_adapter(dev_priv,
> +                                                        intel_hdmi->ddc_bus);
> +     bool ret;
> +
> +     if (!sink_scrambling->supported)
> +             return;
> +
> +     DRM_DEBUG_KMS("Setting sink scrambling for enc:%s connector:%s\n",
> +                   encoder->base.name, connector->name);
> +
> +     /* Set TMDS bit clock ratio to 1/40 or 1/10 */
> +     ret = drm_scdc_set_high_tmds_clock_ratio(adptr, high_tmds_clock_ratio);
> +     if (!ret) {
> +             DRM_ERROR("Set TMDS ratio failed\n");
> +             return;
> +     }
> +
> +     /* Enable/disable sink scrambling */
> +     ret = drm_scdc_set_scrambling(adptr, scrambling);
> +     if (!ret) {
> +             DRM_ERROR("Set sink scrambling failed\n");
> +             return;
> +     }
> +
> +     DRM_DEBUG_KMS("sink scrambling handled\n");
> +}
> +
>  static u8 intel_hdmi_ddc_pin(struct drm_i915_private *dev_priv,
>                            enum port port)
>  {
_______________________________________________
Intel-gfx mailing list
Intel-gfx@lists.freedesktop.org
https://lists.freedesktop.org/mailman/listinfo/intel-gfx

Reply via email to