Commit 99077041 authored by Anthony Koo's avatar Anthony Koo Committed by Alex Deucher

drm/amd/display: Prevent PSR from being enabled if initialization fails

[Why]
PSR_SET command is sent to the microcontroller in order to initialize
parameters needed for PSR feature, such as telling the microcontroller
which pipe is driving the PSR supported panel. When this command is
skipped or fails, the microcontroller may program the wrong thing if
driver tries to enable PSR.

[How]
If PSR_SET fails, do not set psr_enable flag to indicate the feature is
not yet initialized.
Signed-off-by: default avatarAnthony Koo <Anthony.Koo@amd.com>
Reviewed-by: default avatarAric Cyr <Aric.Cyr@amd.com>
Acked-by: default avatarBhawanpreet Lakha <Bhawanpreet.Lakha@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent f358b39d
...@@ -150,7 +150,7 @@ static void dce_dmcu_set_psr_enable(struct dmcu *dmcu, bool enable, bool wait) ...@@ -150,7 +150,7 @@ static void dce_dmcu_set_psr_enable(struct dmcu *dmcu, bool enable, bool wait)
} }
} }
static void dce_dmcu_setup_psr(struct dmcu *dmcu, static bool dce_dmcu_setup_psr(struct dmcu *dmcu,
struct dc_link *link, struct dc_link *link,
struct psr_context *psr_context) struct psr_context *psr_context)
{ {
...@@ -261,6 +261,8 @@ static void dce_dmcu_setup_psr(struct dmcu *dmcu, ...@@ -261,6 +261,8 @@ static void dce_dmcu_setup_psr(struct dmcu *dmcu,
/* notifyDMCUMsg */ /* notifyDMCUMsg */
REG_UPDATE(MASTER_COMM_CNTL_REG, MASTER_COMM_INTERRUPT, 1); REG_UPDATE(MASTER_COMM_CNTL_REG, MASTER_COMM_INTERRUPT, 1);
return true;
} }
static bool dce_is_dmcu_initialized(struct dmcu *dmcu) static bool dce_is_dmcu_initialized(struct dmcu *dmcu)
...@@ -545,24 +547,25 @@ static void dcn10_dmcu_set_psr_enable(struct dmcu *dmcu, bool enable, bool wait) ...@@ -545,24 +547,25 @@ static void dcn10_dmcu_set_psr_enable(struct dmcu *dmcu, bool enable, bool wait)
* least a few frames. Should never hit the max retry assert below. * least a few frames. Should never hit the max retry assert below.
*/ */
if (wait == true) { if (wait == true) {
for (retryCount = 0; retryCount <= 1000; retryCount++) { for (retryCount = 0; retryCount <= 1000; retryCount++) {
dcn10_get_dmcu_psr_state(dmcu, &psr_state); dcn10_get_dmcu_psr_state(dmcu, &psr_state);
if (enable) { if (enable) {
if (psr_state != 0) if (psr_state != 0)
break; break;
} else { } else {
if (psr_state == 0) if (psr_state == 0)
break; break;
}
udelay(500);
} }
udelay(500);
}
/* assert if max retry hit */ /* assert if max retry hit */
ASSERT(retryCount <= 1000); if (retryCount >= 1000)
ASSERT(0);
} }
} }
static void dcn10_dmcu_setup_psr(struct dmcu *dmcu, static bool dcn10_dmcu_setup_psr(struct dmcu *dmcu,
struct dc_link *link, struct dc_link *link,
struct psr_context *psr_context) struct psr_context *psr_context)
{ {
...@@ -577,7 +580,7 @@ static void dcn10_dmcu_setup_psr(struct dmcu *dmcu, ...@@ -577,7 +580,7 @@ static void dcn10_dmcu_setup_psr(struct dmcu *dmcu,
/* If microcontroller is not running, do nothing */ /* If microcontroller is not running, do nothing */
if (dmcu->dmcu_state != DMCU_RUNNING) if (dmcu->dmcu_state != DMCU_RUNNING)
return; return false;
link->link_enc->funcs->psr_program_dp_dphy_fast_training(link->link_enc, link->link_enc->funcs->psr_program_dp_dphy_fast_training(link->link_enc,
psr_context->psrExitLinkTrainingRequired); psr_context->psrExitLinkTrainingRequired);
...@@ -677,6 +680,11 @@ static void dcn10_dmcu_setup_psr(struct dmcu *dmcu, ...@@ -677,6 +680,11 @@ static void dcn10_dmcu_setup_psr(struct dmcu *dmcu,
/* notifyDMCUMsg */ /* notifyDMCUMsg */
REG_UPDATE(MASTER_COMM_CNTL_REG, MASTER_COMM_INTERRUPT, 1); REG_UPDATE(MASTER_COMM_CNTL_REG, MASTER_COMM_INTERRUPT, 1);
/* waitDMCUReadyForCmd */
REG_WAIT(MASTER_COMM_CNTL_REG, MASTER_COMM_INTERRUPT, 0, 1, 10000);
return true;
} }
static void dcn10_psr_wait_loop( static void dcn10_psr_wait_loop(
......
...@@ -48,7 +48,7 @@ struct dmcu_funcs { ...@@ -48,7 +48,7 @@ struct dmcu_funcs {
const char *src, const char *src,
unsigned int bytes); unsigned int bytes);
void (*set_psr_enable)(struct dmcu *dmcu, bool enable, bool wait); void (*set_psr_enable)(struct dmcu *dmcu, bool enable, bool wait);
void (*setup_psr)(struct dmcu *dmcu, bool (*setup_psr)(struct dmcu *dmcu,
struct dc_link *link, struct dc_link *link,
struct psr_context *psr_context); struct psr_context *psr_context);
void (*get_psr_state)(struct dmcu *dmcu, uint32_t *psr_state); void (*get_psr_state)(struct dmcu *dmcu, uint32_t *psr_state);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment