提交 0b7421f0 编写于 作者: A Aurabindo Pillai 提交者: Alex Deucher

drm/amd/display: Old sequence for HUBP blank

New proposed sequence for HUBP blanking causes regressions where the
hardware would fail to enter blank which triggers an assert in the new
sequence. This change brings back the old sequence.

Fixes: 985faf2c ("drm/amd/display: New sequence for HUBP blank")
Signed-off-by: NAurabindo Pillai <aurabindo.pillai@amd.com>
Signed-off-by: NBhawanpreet Lakha <bhawanpreet.lakha@amd.com>
Reviewed-by: NRodrigo Siqueira <Rodrigo.Siqueira@amd.com>
Signed-off-by: NAlex Deucher <alexander.deucher@amd.com>
上级 ea1b8c9b
...@@ -2635,7 +2635,7 @@ static void dcn10_update_dchubp_dpp( ...@@ -2635,7 +2635,7 @@ static void dcn10_update_dchubp_dpp(
hws->funcs.update_plane_addr(dc, pipe_ctx); hws->funcs.update_plane_addr(dc, pipe_ctx);
if (is_pipe_tree_visible(pipe_ctx)) if (is_pipe_tree_visible(pipe_ctx))
dc->hwss.set_hubp_blank(dc, pipe_ctx, false); hubp->funcs->set_blank(hubp, false);
} }
void dcn10_blank_pixel_data( void dcn10_blank_pixel_data(
...@@ -3146,16 +3146,13 @@ void dcn10_setup_stereo(struct pipe_ctx *pipe_ctx, struct dc *dc) ...@@ -3146,16 +3146,13 @@ void dcn10_setup_stereo(struct pipe_ctx *pipe_ctx, struct dc *dc)
return; return;
} }
static struct pipe_ctx *get_pipe_ctx_by_hubp_inst(struct dc_state *context, int mpcc_inst) static struct hubp *get_hubp_by_inst(struct resource_pool *res_pool, int mpcc_inst)
{ {
int i; int i;
for (i = 0; i < MAX_PIPES; i++) { for (i = 0; i < res_pool->pipe_count; i++) {
if (context->res_ctx.pipe_ctx[i].plane_res.hubp if (res_pool->hubps[i]->inst == mpcc_inst)
&& context->res_ctx.pipe_ctx[i].plane_res.hubp->inst == mpcc_inst) { return res_pool->hubps[i];
return &context->res_ctx.pipe_ctx[i];
}
} }
ASSERT(false); ASSERT(false);
return NULL; return NULL;
...@@ -3178,23 +3175,11 @@ void dcn10_wait_for_mpcc_disconnect( ...@@ -3178,23 +3175,11 @@ void dcn10_wait_for_mpcc_disconnect(
for (mpcc_inst = 0; mpcc_inst < MAX_PIPES; mpcc_inst++) { for (mpcc_inst = 0; mpcc_inst < MAX_PIPES; mpcc_inst++) {
if (pipe_ctx->stream_res.opp->mpcc_disconnect_pending[mpcc_inst]) { if (pipe_ctx->stream_res.opp->mpcc_disconnect_pending[mpcc_inst]) {
struct pipe_ctx *restore_bottom_pipe; struct hubp *hubp = get_hubp_by_inst(res_pool, mpcc_inst);
struct pipe_ctx *restore_top_pipe;
struct pipe_ctx *inst_pipe_ctx = get_pipe_ctx_by_hubp_inst(dc->current_state, mpcc_inst);
ASSERT(inst_pipe_ctx);
res_pool->mpc->funcs->wait_for_idle(res_pool->mpc, mpcc_inst); res_pool->mpc->funcs->wait_for_idle(res_pool->mpc, mpcc_inst);
pipe_ctx->stream_res.opp->mpcc_disconnect_pending[mpcc_inst] = false; pipe_ctx->stream_res.opp->mpcc_disconnect_pending[mpcc_inst] = false;
/* hubp->funcs->set_blank(hubp, true);
* Set top and bottom pipes NULL, as we don't want
* to blank those pipes when disconnecting from MPCC
*/
restore_bottom_pipe = inst_pipe_ctx->bottom_pipe;
restore_top_pipe = inst_pipe_ctx->top_pipe;
inst_pipe_ctx->top_pipe = inst_pipe_ctx->bottom_pipe = NULL;
dc->hwss.set_hubp_blank(dc, inst_pipe_ctx, true);
inst_pipe_ctx->top_pipe = restore_top_pipe;
inst_pipe_ctx->bottom_pipe = restore_bottom_pipe;
} }
} }
...@@ -3747,10 +3732,3 @@ void dcn10_get_clock(struct dc *dc, ...@@ -3747,10 +3732,3 @@ void dcn10_get_clock(struct dc *dc,
dc->clk_mgr->funcs->get_clock(dc->clk_mgr, context, clock_type, clock_cfg); dc->clk_mgr->funcs->get_clock(dc->clk_mgr, context, clock_type, clock_cfg);
} }
void dcn10_set_hubp_blank(const struct dc *dc,
struct pipe_ctx *pipe_ctx,
bool blank_enable)
{
pipe_ctx->plane_res.hubp->funcs->set_blank(pipe_ctx->plane_res.hubp, blank_enable);
}
...@@ -204,8 +204,5 @@ void dcn10_wait_for_pending_cleared(struct dc *dc, ...@@ -204,8 +204,5 @@ void dcn10_wait_for_pending_cleared(struct dc *dc,
struct dc_state *context); struct dc_state *context);
void dcn10_set_hdr_multiplier(struct pipe_ctx *pipe_ctx); void dcn10_set_hdr_multiplier(struct pipe_ctx *pipe_ctx);
void dcn10_verify_allow_pstate_change_high(struct dc *dc); void dcn10_verify_allow_pstate_change_high(struct dc *dc);
void dcn10_set_hubp_blank(const struct dc *dc,
struct pipe_ctx *pipe_ctx,
bool blank_enable);
#endif /* __DC_HWSS_DCN10_H__ */ #endif /* __DC_HWSS_DCN10_H__ */
...@@ -79,7 +79,6 @@ static const struct hw_sequencer_funcs dcn10_funcs = { ...@@ -79,7 +79,6 @@ static const struct hw_sequencer_funcs dcn10_funcs = {
.set_backlight_level = dce110_set_backlight_level, .set_backlight_level = dce110_set_backlight_level,
.set_abm_immediate_disable = dce110_set_abm_immediate_disable, .set_abm_immediate_disable = dce110_set_abm_immediate_disable,
.set_pipe = dce110_set_pipe, .set_pipe = dce110_set_pipe,
.set_hubp_blank = dcn10_set_hubp_blank,
}; };
static const struct hwseq_private_funcs dcn10_private_funcs = { static const struct hwseq_private_funcs dcn10_private_funcs = {
......
...@@ -1576,7 +1576,7 @@ static void dcn20_update_dchubp_dpp( ...@@ -1576,7 +1576,7 @@ static void dcn20_update_dchubp_dpp(
if (pipe_ctx->update_flags.bits.enable) if (pipe_ctx->update_flags.bits.enable)
dc->hwss.set_hubp_blank(dc, pipe_ctx, false); hubp->funcs->set_blank(hubp, false);
} }
...@@ -1772,10 +1772,19 @@ void dcn20_post_unlock_program_front_end( ...@@ -1772,10 +1772,19 @@ void dcn20_post_unlock_program_front_end(
for (i = 0; i < dc->res_pool->pipe_count; i++) { for (i = 0; i < dc->res_pool->pipe_count; i++) {
struct pipe_ctx *pipe = &context->res_ctx.pipe_ctx[i]; struct pipe_ctx *pipe = &context->res_ctx.pipe_ctx[i];
struct pipe_ctx *mpcc_pipe;
if (pipe->vtp_locked) { if (pipe->vtp_locked) {
dc->hwss.set_hubp_blank(dc, pipe, true); dc->hwseq->funcs.wait_for_blank_complete(pipe->stream_res.opp);
pipe->plane_res.hubp->funcs->set_blank(pipe->plane_res.hubp, true);
pipe->vtp_locked = false; pipe->vtp_locked = false;
for (mpcc_pipe = pipe->bottom_pipe; mpcc_pipe; mpcc_pipe = mpcc_pipe->bottom_pipe)
mpcc_pipe->plane_res.hubp->funcs->set_blank(mpcc_pipe->plane_res.hubp, true);
for (i = 0; i < dc->res_pool->pipe_count; i++)
if (context->res_ctx.pipe_ctx[i].update_flags.bits.disable)
dc->hwss.disable_plane(dc, &dc->current_state->res_ctx.pipe_ctx[i]);
} }
} }
/* WA to apply WM setting*/ /* WA to apply WM setting*/
......
...@@ -94,7 +94,6 @@ static const struct hw_sequencer_funcs dcn20_funcs = { ...@@ -94,7 +94,6 @@ static const struct hw_sequencer_funcs dcn20_funcs = {
.optimize_timing_for_fsft = dcn20_optimize_timing_for_fsft, .optimize_timing_for_fsft = dcn20_optimize_timing_for_fsft,
#endif #endif
.set_disp_pattern_generator = dcn20_set_disp_pattern_generator, .set_disp_pattern_generator = dcn20_set_disp_pattern_generator,
.set_hubp_blank = dcn10_set_hubp_blank,
}; };
static const struct hwseq_private_funcs dcn20_private_funcs = { static const struct hwseq_private_funcs dcn20_private_funcs = {
......
...@@ -99,7 +99,6 @@ static const struct hw_sequencer_funcs dcn21_funcs = { ...@@ -99,7 +99,6 @@ static const struct hw_sequencer_funcs dcn21_funcs = {
#endif #endif
.is_abm_supported = dcn21_is_abm_supported, .is_abm_supported = dcn21_is_abm_supported,
.set_disp_pattern_generator = dcn20_set_disp_pattern_generator, .set_disp_pattern_generator = dcn20_set_disp_pattern_generator,
.set_hubp_blank = dcn10_set_hubp_blank,
}; };
static const struct hwseq_private_funcs dcn21_private_funcs = { static const struct hwseq_private_funcs dcn21_private_funcs = {
......
...@@ -938,53 +938,6 @@ void dcn30_hardware_release(struct dc *dc) ...@@ -938,53 +938,6 @@ void dcn30_hardware_release(struct dc *dc)
dc->res_pool->hubbub, true, true); dc->res_pool->hubbub, true, true);
} }
void dcn30_set_hubp_blank(const struct dc *dc,
struct pipe_ctx *pipe_ctx,
bool blank_enable)
{
struct pipe_ctx *mpcc_pipe;
struct pipe_ctx *odm_pipe;
if (blank_enable) {
struct plane_resource *plane_res = &pipe_ctx->plane_res;
struct stream_resource *stream_res = &pipe_ctx->stream_res;
/* Wait for enter vblank */
stream_res->tg->funcs->wait_for_state(stream_res->tg, CRTC_STATE_VBLANK);
/* Blank HUBP to allow p-state during blank on all timings */
pipe_ctx->plane_res.hubp->funcs->set_blank(pipe_ctx->plane_res.hubp, true);
/* Confirm hubp in blank */
ASSERT(plane_res->hubp->funcs->hubp_in_blank(plane_res->hubp));
/* Toggle HUBP_DISABLE */
plane_res->hubp->funcs->hubp_soft_reset(plane_res->hubp, true);
plane_res->hubp->funcs->hubp_soft_reset(plane_res->hubp, false);
for (mpcc_pipe = pipe_ctx->bottom_pipe; mpcc_pipe; mpcc_pipe = mpcc_pipe->bottom_pipe) {
mpcc_pipe->plane_res.hubp->funcs->set_blank(mpcc_pipe->plane_res.hubp, true);
/* Confirm hubp in blank */
ASSERT(mpcc_pipe->plane_res.hubp->funcs->hubp_in_blank(mpcc_pipe->plane_res.hubp));
/* Toggle HUBP_DISABLE */
mpcc_pipe->plane_res.hubp->funcs->hubp_soft_reset(mpcc_pipe->plane_res.hubp, true);
mpcc_pipe->plane_res.hubp->funcs->hubp_soft_reset(mpcc_pipe->plane_res.hubp, false);
}
for (odm_pipe = pipe_ctx->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe) {
odm_pipe->plane_res.hubp->funcs->set_blank(odm_pipe->plane_res.hubp, true);
/* Confirm hubp in blank */
ASSERT(odm_pipe->plane_res.hubp->funcs->hubp_in_blank(odm_pipe->plane_res.hubp));
/* Toggle HUBP_DISABLE */
odm_pipe->plane_res.hubp->funcs->hubp_soft_reset(odm_pipe->plane_res.hubp, true);
odm_pipe->plane_res.hubp->funcs->hubp_soft_reset(odm_pipe->plane_res.hubp, false);
}
} else {
pipe_ctx->plane_res.hubp->funcs->set_blank(pipe_ctx->plane_res.hubp, false);
for (mpcc_pipe = pipe_ctx->bottom_pipe; mpcc_pipe; mpcc_pipe = mpcc_pipe->bottom_pipe)
mpcc_pipe->plane_res.hubp->funcs->set_blank(mpcc_pipe->plane_res.hubp, false);
for (odm_pipe = pipe_ctx->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe)
odm_pipe->plane_res.hubp->funcs->set_blank(odm_pipe->plane_res.hubp, false);
}
}
void dcn30_set_disp_pattern_generator(const struct dc *dc, void dcn30_set_disp_pattern_generator(const struct dc *dc,
struct pipe_ctx *pipe_ctx, struct pipe_ctx *pipe_ctx,
enum controller_dp_test_pattern test_pattern, enum controller_dp_test_pattern test_pattern,
...@@ -994,6 +947,7 @@ void dcn30_set_disp_pattern_generator(const struct dc *dc, ...@@ -994,6 +947,7 @@ void dcn30_set_disp_pattern_generator(const struct dc *dc,
int width, int height, int offset) int width, int height, int offset)
{ {
struct stream_resource *stream_res = &pipe_ctx->stream_res; struct stream_resource *stream_res = &pipe_ctx->stream_res;
struct pipe_ctx *mpcc_pipe;
if (test_pattern != CONTROLLER_DP_TEST_PATTERN_VIDEOMODE) { if (test_pattern != CONTROLLER_DP_TEST_PATTERN_VIDEOMODE) {
pipe_ctx->vtp_locked = false; pipe_ctx->vtp_locked = false;
...@@ -1005,12 +959,20 @@ void dcn30_set_disp_pattern_generator(const struct dc *dc, ...@@ -1005,12 +959,20 @@ void dcn30_set_disp_pattern_generator(const struct dc *dc,
if (stream_res->tg->funcs->is_tg_enabled(stream_res->tg)) { if (stream_res->tg->funcs->is_tg_enabled(stream_res->tg)) {
if (stream_res->tg->funcs->is_locked(stream_res->tg)) if (stream_res->tg->funcs->is_locked(stream_res->tg))
pipe_ctx->vtp_locked = true; pipe_ctx->vtp_locked = true;
else else {
dc->hwss.set_hubp_blank(dc, pipe_ctx, true); /* Blank HUBP to allow p-state during blank on all timings */
pipe_ctx->plane_res.hubp->funcs->set_blank(pipe_ctx->plane_res.hubp, true);
for (mpcc_pipe = pipe_ctx->bottom_pipe; mpcc_pipe; mpcc_pipe = mpcc_pipe->bottom_pipe)
mpcc_pipe->plane_res.hubp->funcs->set_blank(mpcc_pipe->plane_res.hubp, true);
}
} }
} else { } else {
dc->hwss.set_hubp_blank(dc, pipe_ctx, false);
/* turning off DPG */ /* turning off DPG */
pipe_ctx->plane_res.hubp->funcs->set_blank(pipe_ctx->plane_res.hubp, false);
for (mpcc_pipe = pipe_ctx->bottom_pipe; mpcc_pipe; mpcc_pipe = mpcc_pipe->bottom_pipe)
mpcc_pipe->plane_res.hubp->funcs->set_blank(mpcc_pipe->plane_res.hubp, false);
stream_res->opp->funcs->opp_set_disp_pattern_generator(stream_res->opp, test_pattern, color_space, stream_res->opp->funcs->opp_set_disp_pattern_generator(stream_res->opp, test_pattern, color_space,
color_depth, solid_color, width, height, offset); color_depth, solid_color, width, height, offset);
} }
......
...@@ -80,8 +80,4 @@ void dcn30_set_disp_pattern_generator(const struct dc *dc, ...@@ -80,8 +80,4 @@ void dcn30_set_disp_pattern_generator(const struct dc *dc,
const struct tg_color *solid_color, const struct tg_color *solid_color,
int width, int height, int offset); int width, int height, int offset);
void dcn30_set_hubp_blank(const struct dc *dc,
struct pipe_ctx *pipe_ctx,
bool blank_enable);
#endif /* __DC_HWSS_DCN30_H__ */ #endif /* __DC_HWSS_DCN30_H__ */
...@@ -98,7 +98,6 @@ static const struct hw_sequencer_funcs dcn30_funcs = { ...@@ -98,7 +98,6 @@ static const struct hw_sequencer_funcs dcn30_funcs = {
.hardware_release = dcn30_hardware_release, .hardware_release = dcn30_hardware_release,
.set_pipe = dcn21_set_pipe, .set_pipe = dcn21_set_pipe,
.set_disp_pattern_generator = dcn30_set_disp_pattern_generator, .set_disp_pattern_generator = dcn30_set_disp_pattern_generator,
.set_hubp_blank = dcn30_set_hubp_blank,
}; };
static const struct hwseq_private_funcs dcn30_private_funcs = { static const struct hwseq_private_funcs dcn30_private_funcs = {
......
...@@ -98,7 +98,6 @@ static const struct hw_sequencer_funcs dcn301_funcs = { ...@@ -98,7 +98,6 @@ static const struct hw_sequencer_funcs dcn301_funcs = {
.set_abm_immediate_disable = dcn21_set_abm_immediate_disable, .set_abm_immediate_disable = dcn21_set_abm_immediate_disable,
.set_pipe = dcn21_set_pipe, .set_pipe = dcn21_set_pipe,
.set_disp_pattern_generator = dcn30_set_disp_pattern_generator, .set_disp_pattern_generator = dcn30_set_disp_pattern_generator,
.set_hubp_blank = dcn30_set_hubp_blank,
}; };
static const struct hwseq_private_funcs dcn301_private_funcs = { static const struct hwseq_private_funcs dcn301_private_funcs = {
......
...@@ -231,10 +231,6 @@ struct hw_sequencer_funcs { ...@@ -231,10 +231,6 @@ struct hw_sequencer_funcs {
enum dc_color_depth color_depth, enum dc_color_depth color_depth,
const struct tg_color *solid_color, const struct tg_color *solid_color,
int width, int height, int offset); int width, int height, int offset);
void (*set_hubp_blank)(const struct dc *dc,
struct pipe_ctx *pipe_ctx,
bool blank_enable);
}; };
void color_space_to_black_color( void color_space_to_black_color(
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册