drm/amd/display: Add helper for blanking all dp displays
authorLeo (Hanghong) Ma <hanghong.ma@amd.com>
Thu, 4 Nov 2021 20:51:54 +0000 (16:51 -0400)
committerAlex Deucher <alexander.deucher@amd.com>
Wed, 17 Nov 2021 21:58:05 +0000 (16:58 -0500)
[Why & How]
1. The code to blank all dp display have been called many times,
so add helpers in dc_link to make it more concise.
2. Add some check to fix the dmesg errors at boot and resume from S3
on dcn3.1 during DQE's promotion test.

Reviewed-by: Alvin Lee <alvin.lee2@amd.com>
Reviewed-by: Wesley Chalmers <Wesley.Chalmers@amd.com>
Reviewed-by: Aric Cyr <Aric.Cyr@amd.com>
Acked-by: Anson Jacob <Anson.Jacob@amd.com>
Tested-by: Daniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: Leo (Hanghong) Ma <hanghong.ma@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
drivers/gpu/drm/amd/display/dc/core/dc_link.c
drivers/gpu/drm/amd/display/dc/dc_link.h
drivers/gpu/drm/amd/display/dc/dce110/dce110_hw_sequencer.c
drivers/gpu/drm/amd/display/dc/dcn10/dcn10_hw_sequencer.c
drivers/gpu/drm/amd/display/dc/dcn30/dcn30_hwseq.c
drivers/gpu/drm/amd/display/dc/dcn31/dcn31_hwseq.c

index 60544788e911ee15969e0cefa1aeb630cf1f3f44..01c1f518eab0a6a12dce9d51d82e8e02d0563cde 100644 (file)
@@ -1999,6 +1999,57 @@ static enum dc_status enable_link_dp_mst(
        return enable_link_dp(state, pipe_ctx);
 }
 
+void dc_link_blank_all_dp_displays(struct dc *dc)
+{
+       unsigned int i;
+       uint8_t dpcd_power_state = '\0';
+       enum dc_status status = DC_ERROR_UNEXPECTED;
+
+       for (i = 0; i < dc->link_count; i++) {
+               if ((dc->links[i]->connector_signal != SIGNAL_TYPE_DISPLAY_PORT) ||
+                       (dc->links[i]->priv == NULL) || (dc->links[i]->local_sink == NULL))
+                       continue;
+
+               /* DP 2.0 spec requires that we read LTTPR caps first */
+               dp_retrieve_lttpr_cap(dc->links[i]);
+               /* if any of the displays are lit up turn them off */
+               status = core_link_read_dpcd(dc->links[i], DP_SET_POWER,
+                                                       &dpcd_power_state, sizeof(dpcd_power_state));
+
+               if (status == DC_OK && dpcd_power_state == DP_POWER_STATE_D0)
+                       dc_link_blank_dp_stream(dc->links[i], true);
+       }
+
+}
+
+void dc_link_blank_dp_stream(struct dc_link *link, bool hw_init)
+{
+       unsigned int j;
+       struct dc  *dc = link->ctx->dc;
+       enum signal_type signal = link->connector_signal;
+
+       if ((signal == SIGNAL_TYPE_EDP) ||
+               (signal == SIGNAL_TYPE_DISPLAY_PORT)) {
+               if (link->ep_type == DISPLAY_ENDPOINT_PHY &&
+                       link->link_enc->funcs->get_dig_frontend &&
+                       link->link_enc->funcs->is_dig_enabled(link->link_enc)) {
+                       unsigned int fe = link->link_enc->funcs->get_dig_frontend(link->link_enc);
+
+                       if (fe != ENGINE_ID_UNKNOWN)
+                               for (j = 0; j < dc->res_pool->stream_enc_count; j++) {
+                                       if (fe == dc->res_pool->stream_enc[j]->id) {
+                                               dc->res_pool->stream_enc[j]->funcs->dp_blank(link,
+                                                                       dc->res_pool->stream_enc[j]);
+                                               break;
+                                       }
+                               }
+               }
+
+               if ((!link->wa_flags.dp_keep_receiver_powered) || hw_init)
+                       dp_receiver_power_ctrl(link, false);
+       }
+}
+
 static bool get_ext_hdmi_settings(struct pipe_ctx *pipe_ctx,
                enum engine_id eng_id,
                struct ext_hdmi_settings *settings)
index 180ecd860296b250fe40f6d040d6a2a33bb88817..669c162c0c02a969fd5b89df9235a4f2eca4e929 100644 (file)
@@ -287,6 +287,10 @@ bool dc_link_setup_psr(struct dc_link *dc_link,
 
 void dc_link_get_psr_residency(const struct dc_link *link, uint32_t *residency);
 
+void dc_link_blank_all_dp_displays(struct dc *dc);
+
+void dc_link_blank_dp_stream(struct dc_link *link, bool hw_init);
+
 /* Request DC to detect if there is a Panel connected.
  * boot - If this call is during initial boot.
  * Return false for any type of detection failure or MST detection
index 24e47df526f6a08c3a56556d1d030c2834f00a71..e7e2aa46218d55551ad551443d77fcbcb1a82dd3 100644 (file)
@@ -1655,30 +1655,12 @@ static enum dc_status apply_single_controller_ctx_to_hw(
 
 static void power_down_encoders(struct dc *dc)
 {
-       int i, j;
+       int i;
 
        for (i = 0; i < dc->link_count; i++) {
                enum signal_type signal = dc->links[i]->connector_signal;
 
-               if ((signal == SIGNAL_TYPE_EDP) ||
-                       (signal == SIGNAL_TYPE_DISPLAY_PORT)) {
-                       if (dc->links[i]->link_enc->funcs->get_dig_frontend &&
-                               dc->links[i]->link_enc->funcs->is_dig_enabled(dc->links[i]->link_enc)) {
-                               unsigned int fe = dc->links[i]->link_enc->funcs->get_dig_frontend(
-                                                                       dc->links[i]->link_enc);
-
-                               for (j = 0; j < dc->res_pool->stream_enc_count; j++) {
-                                       if (fe == dc->res_pool->stream_enc[j]->id) {
-                                               dc->res_pool->stream_enc[j]->funcs->dp_blank(dc->links[i],
-                                                                       dc->res_pool->stream_enc[j]);
-                                               break;
-                                       }
-                               }
-                       }
-
-                       if (!dc->links[i]->wa_flags.dp_keep_receiver_powered)
-                               dp_receiver_power_ctrl(dc->links[i], false);
-               }
+               dc_link_blank_dp_stream(dc->links[i], false);
 
                if (signal != SIGNAL_TYPE_EDP)
                        signal = SIGNAL_TYPE_NONE;
index 0b788d794fb334e0fc9e66fc67e919936bbec6f1..135e2b58cc7339a790fb529960e1022f6dd4ebea 100644 (file)
@@ -1366,7 +1366,7 @@ void dcn10_init_pipes(struct dc *dc, struct dc_state *context)
 
 void dcn10_init_hw(struct dc *dc)
 {
-       int i, j;
+       int i;
        struct abm *abm = dc->res_pool->abm;
        struct dmcu *dmcu = dc->res_pool->dmcu;
        struct dce_hwseq *hws = dc->hwseq;
@@ -1468,43 +1468,8 @@ void dcn10_init_hw(struct dc *dc)
                dmub_enable_outbox_notification(dc);
 
        /* we want to turn off all dp displays before doing detection */
-       if (dc->config.power_down_display_on_boot) {
-               uint8_t dpcd_power_state = '\0';
-               enum dc_status status = DC_ERROR_UNEXPECTED;
-
-               for (i = 0; i < dc->link_count; i++) {
-                       if (dc->links[i]->connector_signal != SIGNAL_TYPE_DISPLAY_PORT)
-                               continue;
-
-                       /* DP 2.0 requires that LTTPR Caps be read first */
-                       dp_retrieve_lttpr_cap(dc->links[i]);
-
-                       /*
-                        * If any of the displays are lit up turn them off.
-                        * The reason is that some MST hubs cannot be turned off
-                        * completely until we tell them to do so.
-                        * If not turned off, then displays connected to MST hub
-                        * won't light up.
-                        */
-                       status = core_link_read_dpcd(dc->links[i], DP_SET_POWER,
-                                                       &dpcd_power_state, sizeof(dpcd_power_state));
-                       if (status == DC_OK && dpcd_power_state == DP_POWER_STATE_D0) {
-                               /* blank dp stream before power off receiver*/
-                               if (dc->links[i]->link_enc->funcs->get_dig_frontend) {
-                                       unsigned int fe = dc->links[i]->link_enc->funcs->get_dig_frontend(dc->links[i]->link_enc);
-
-                                       for (j = 0; j < dc->res_pool->stream_enc_count; j++) {
-                                               if (fe == dc->res_pool->stream_enc[j]->id) {
-                                                       dc->res_pool->stream_enc[j]->funcs->dp_blank(dc->links[i],
-                                                                               dc->res_pool->stream_enc[j]);
-                                                       break;
-                                               }
-                                       }
-                               }
-                               dp_receiver_power_ctrl(dc->links[i], false);
-                       }
-               }
-       }
+       if (dc->config.power_down_display_on_boot)
+               dc_link_blank_all_dp_displays(dc);
 
        /* If taking control over from VBIOS, we may want to optimize our first
         * mode set, so we need to skip powering down pipes until we know which
index df27171166044f2d6eb388b16eca87bfa965a849..3e99bb9c70ab8e3dcaa099cad2c45655302e857d 100644 (file)
@@ -437,7 +437,7 @@ void dcn30_init_hw(struct dc *dc)
        struct dce_hwseq *hws = dc->hwseq;
        struct dc_bios *dcb = dc->ctx->dc_bios;
        struct resource_pool *res_pool = dc->res_pool;
-       int i, j;
+       int i;
        int edp_num;
        uint32_t backlight = MAX_BACKLIGHT_LEVEL;
 
@@ -534,41 +534,8 @@ void dcn30_init_hw(struct dc *dc)
                        hws->funcs.dsc_pg_control(hws, res_pool->dscs[i]->inst, false);
 
        /* we want to turn off all dp displays before doing detection */
-       if (dc->config.power_down_display_on_boot) {
-               uint8_t dpcd_power_state = '\0';
-               enum dc_status status = DC_ERROR_UNEXPECTED;
-
-               for (i = 0; i < dc->link_count; i++) {
-                       if (dc->links[i]->connector_signal != SIGNAL_TYPE_DISPLAY_PORT)
-                               continue;
-                       /* DP 2.0 states that LTTPR regs must be read first */
-                       dp_retrieve_lttpr_cap(dc->links[i]);
-
-                       /* if any of the displays are lit up turn them off */
-                       status = core_link_read_dpcd(dc->links[i], DP_SET_POWER,
-                                                    &dpcd_power_state, sizeof(dpcd_power_state));
-                       if (status == DC_OK && dpcd_power_state == DP_POWER_STATE_D0) {
-                               /* blank dp stream before power off receiver*/
-                               if (dc->links[i]->link_enc->funcs->get_dig_frontend) {
-                                       unsigned int fe;
-
-                                       fe = dc->links[i]->link_enc->funcs->get_dig_frontend(
-                                                                               dc->links[i]->link_enc);
-                                       if (fe == ENGINE_ID_UNKNOWN)
-                                               continue;
-
-                                       for (j = 0; j < dc->res_pool->stream_enc_count; j++) {
-                                               if (fe == dc->res_pool->stream_enc[j]->id) {
-                                                       dc->res_pool->stream_enc[j]->funcs->dp_blank(dc->links[i],
-                                                                               dc->res_pool->stream_enc[j]);
-                                                       break;
-                                               }
-                                       }
-                               }
-                               dp_receiver_power_ctrl(dc->links[i], false);
-                       }
-               }
-       }
+       if (dc->config.power_down_display_on_boot)
+               dc_link_blank_all_dp_displays(dc);
 
        /* If taking control over from VBIOS, we may want to optimize our first
         * mode set, so we need to skip powering down pipes until we know which
index 5dd1ce9ddb539afb2aa1dbfb16f6550086402ecd..bdaeb6ebdacb0f9a28e693ae293b91c15d047161 100644 (file)
@@ -112,7 +112,7 @@ void dcn31_init_hw(struct dc *dc)
        struct dc_bios *dcb = dc->ctx->dc_bios;
        struct resource_pool *res_pool = dc->res_pool;
        uint32_t backlight = MAX_BACKLIGHT_LEVEL;
-       int i, j;
+       int i;
 
        if (dc->clk_mgr && dc->clk_mgr->funcs->init_clocks)
                dc->clk_mgr->funcs->init_clocks(dc->clk_mgr);
@@ -202,40 +202,8 @@ void dcn31_init_hw(struct dc *dc)
                dmub_enable_outbox_notification(dc);
 
        /* we want to turn off all dp displays before doing detection */
-       if (dc->config.power_down_display_on_boot) {
-               uint8_t dpcd_power_state = '\0';
-               enum dc_status status = DC_ERROR_UNEXPECTED;
-
-               for (i = 0; i < dc->link_count; i++) {
-                       if (dc->links[i]->connector_signal != SIGNAL_TYPE_DISPLAY_PORT)
-                               continue;
-
-                       /* if any of the displays are lit up turn them off */
-                       status = core_link_read_dpcd(dc->links[i], DP_SET_POWER,
-                                                    &dpcd_power_state, sizeof(dpcd_power_state));
-                       if (status == DC_OK && dpcd_power_state == DP_POWER_STATE_D0) {
-                               /* blank dp stream before power off receiver*/
-                               if (dc->links[i]->ep_type == DISPLAY_ENDPOINT_PHY &&
-                                               dc->links[i]->link_enc->funcs->get_dig_frontend) {
-                                       unsigned int fe;
-
-                                       fe = dc->links[i]->link_enc->funcs->get_dig_frontend(
-                                                                               dc->links[i]->link_enc);
-                                       if (fe == ENGINE_ID_UNKNOWN)
-                                               continue;
-
-                                       for (j = 0; j < dc->res_pool->stream_enc_count; j++) {
-                                               if (fe == dc->res_pool->stream_enc[j]->id) {
-                                                       dc->res_pool->stream_enc[j]->funcs->dp_blank(dc->links[i],
-                                                                               dc->res_pool->stream_enc[j]);
-                                                       break;
-                                               }
-                                       }
-                               }
-                               dp_receiver_power_ctrl(dc->links[i], false);
-                       }
-               }
-       }
+       if (dc->config.power_down_display_on_boot)
+               dc_link_blank_all_dp_displays(dc);
 
        /* If taking control over from VBIOS, we may want to optimize our first
         * mode set, so we need to skip powering down pipes until we know which