Commit 0df62823 authored by George Shen's avatar George Shen Committed by Alex Deucher

drm/amd/display: Increase the number of retries after AUX DEFER

[Why]
When a timeout occurs after a DEFER, some devices require more retries
than in the case of a regular timeout.

[How]
In a timeout occurrence, check whether a DEFER has occurred before the
timeout and retry MAX_DEFER_RETRIES retries times instead of
MAX_TIMEOUT_RETRIES.
Signed-off-by: default avatarGeorge Shen <george.shen@amd.com>
Reviewed-by: default avatarTony Cheng <Tony.Cheng@amd.com>
Acked-by: default avatarAbdoulaye Berthe <Abdoulaye.Berthe@amd.com>
Acked-by: default avatarLeo Li <sunpeng.li@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 44ce6c3d
...@@ -583,6 +583,8 @@ bool dce_aux_transfer_with_retries(struct ddc_service *ddc, ...@@ -583,6 +583,8 @@ bool dce_aux_transfer_with_retries(struct ddc_service *ddc,
uint8_t reply; uint8_t reply;
bool payload_reply = true; bool payload_reply = true;
enum aux_channel_operation_result operation_result; enum aux_channel_operation_result operation_result;
bool retry_on_defer = false;
int aux_ack_retries = 0, int aux_ack_retries = 0,
aux_defer_retries = 0, aux_defer_retries = 0,
aux_i2c_defer_retries = 0, aux_i2c_defer_retries = 0,
...@@ -613,8 +615,9 @@ bool dce_aux_transfer_with_retries(struct ddc_service *ddc, ...@@ -613,8 +615,9 @@ bool dce_aux_transfer_with_retries(struct ddc_service *ddc,
break; break;
case AUX_TRANSACTION_REPLY_AUX_DEFER: case AUX_TRANSACTION_REPLY_AUX_DEFER:
case AUX_TRANSACTION_REPLY_I2C_OVER_AUX_NACK:
case AUX_TRANSACTION_REPLY_I2C_OVER_AUX_DEFER: case AUX_TRANSACTION_REPLY_I2C_OVER_AUX_DEFER:
retry_on_defer = true;
case AUX_TRANSACTION_REPLY_I2C_OVER_AUX_NACK:
if (++aux_defer_retries >= AUX_MAX_DEFER_RETRIES) { if (++aux_defer_retries >= AUX_MAX_DEFER_RETRIES) {
goto fail; goto fail;
} else { } else {
...@@ -647,15 +650,24 @@ bool dce_aux_transfer_with_retries(struct ddc_service *ddc, ...@@ -647,15 +650,24 @@ bool dce_aux_transfer_with_retries(struct ddc_service *ddc,
break; break;
case AUX_CHANNEL_OPERATION_FAILED_TIMEOUT: case AUX_CHANNEL_OPERATION_FAILED_TIMEOUT:
if (++aux_timeout_retries >= AUX_MAX_TIMEOUT_RETRIES) // Check whether a DEFER had occurred before the timeout.
goto fail; // If so, treat timeout as a DEFER.
else { if (retry_on_defer) {
/* if (++aux_defer_retries >= AUX_MAX_DEFER_RETRIES)
* DP 1.4, 2.8.2: AUX Transaction Response/Reply Timeouts goto fail;
* According to the DP spec there should be 3 retries total else if (payload->defer_delay > 0)
* with a 400us wait inbetween each. Hardware already waits msleep(payload->defer_delay);
* for 550us therefore no wait is required here. } else {
*/ if (++aux_timeout_retries >= AUX_MAX_TIMEOUT_RETRIES)
goto fail;
else {
/*
* DP 1.4, 2.8.2: AUX Transaction Response/Reply Timeouts
* According to the DP spec there should be 3 retries total
* with a 400us wait inbetween each. Hardware already waits
* for 550us therefore no wait is required here.
*/
}
} }
break; break;
......
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