Commit 61aa7a6f authored by abdoulaye berthe's avatar abdoulaye berthe Committed by Alex Deucher

drm/amd/display: disable lttpr for invalid lttpr caps.

1-Read lttpr caps in 5-bytes
2-Parse caps
3-Validate caps and set lttpr_mode
4-Use hw default timeout when lttpr is disabled.
Signed-off-by: default avatarabdoulaye berthe <abdoulaye.berthe@amd.com>
Reviewed-by: default avatarWenjing Liu <Wenjing.Liu@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 9bffd080
...@@ -1172,7 +1172,7 @@ static void configure_lttpr_mode(struct dc_link *link) ...@@ -1172,7 +1172,7 @@ static void configure_lttpr_mode(struct dc_link *link)
uint8_t repeater_cnt; uint8_t repeater_cnt;
uint32_t aux_interval_address; uint32_t aux_interval_address;
uint8_t repeater_id; uint8_t repeater_id;
enum lttpr_mode repeater_mode = phy_repeater_mode_transparent; uint8_t repeater_mode = DP_PHY_REPEATER_MODE_TRANSPARENT;
core_link_write_dpcd(link, core_link_write_dpcd(link,
DP_PHY_REPEATER_MODE, DP_PHY_REPEATER_MODE,
...@@ -1180,7 +1180,7 @@ static void configure_lttpr_mode(struct dc_link *link) ...@@ -1180,7 +1180,7 @@ static void configure_lttpr_mode(struct dc_link *link)
sizeof(repeater_mode)); sizeof(repeater_mode));
if (!link->is_lttpr_mode_transparent) { if (!link->is_lttpr_mode_transparent) {
repeater_mode = phy_repeater_mode_non_transparent; repeater_mode = DP_PHY_REPEATER_MODE_NON_TRANSPARENT;
core_link_write_dpcd(link, core_link_write_dpcd(link,
DP_PHY_REPEATER_MODE, DP_PHY_REPEATER_MODE,
(uint8_t *)&repeater_mode, (uint8_t *)&repeater_mode,
...@@ -2964,7 +2964,11 @@ static void dp_wa_power_up_0010FA(struct dc_link *link, uint8_t *dpcd_data, ...@@ -2964,7 +2964,11 @@ static void dp_wa_power_up_0010FA(struct dc_link *link, uint8_t *dpcd_data,
static bool retrieve_link_cap(struct dc_link *link) static bool retrieve_link_cap(struct dc_link *link)
{ {
uint8_t dpcd_data[DP_ADAPTER_CAP - DP_DPCD_REV + 1]; /* DP_ADAPTER_CAP - DP_DPCD_REV + 1 == 16 and also DP_DSC_BITS_PER_PIXEL_INC - DP_DSC_SUPPORT + 1 == 16,
* which means size 16 will be good for both of those DPCD register block reads
*/
uint8_t dpcd_data[16];
uint8_t lttpr_dpcd_data[6];
/*Only need to read 1 byte starting from DP_DPRX_FEATURE_ENUMERATION_LIST. /*Only need to read 1 byte starting from DP_DPRX_FEATURE_ENUMERATION_LIST.
*/ */
...@@ -2977,7 +2981,6 @@ static bool retrieve_link_cap(struct dc_link *link) ...@@ -2977,7 +2981,6 @@ static bool retrieve_link_cap(struct dc_link *link)
union dp_downstream_port_present ds_port = { 0 }; union dp_downstream_port_present ds_port = { 0 };
enum dc_status status = DC_ERROR_UNEXPECTED; enum dc_status status = DC_ERROR_UNEXPECTED;
uint32_t read_dpcd_retry_cnt = 3; uint32_t read_dpcd_retry_cnt = 3;
uint32_t prev_timeout_val;
int i; int i;
struct dp_sink_hw_fw_revision dp_hw_fw_revision; struct dp_sink_hw_fw_revision dp_hw_fw_revision;
...@@ -2988,12 +2991,12 @@ static bool retrieve_link_cap(struct dc_link *link) ...@@ -2988,12 +2991,12 @@ static bool retrieve_link_cap(struct dc_link *link)
link->is_lttpr_mode_transparent = true; link->is_lttpr_mode_transparent = true;
if (ext_timeout_support) { if (ext_timeout_support) {
prev_timeout_val =
dc_link_aux_configure_timeout(link->ddc, dc_link_aux_configure_timeout(link->ddc,
LINK_AUX_DEFAULT_EXTENDED_TIMEOUT_PERIOD); LINK_AUX_DEFAULT_EXTENDED_TIMEOUT_PERIOD);
} }
memset(dpcd_data, '\0', sizeof(dpcd_data)); memset(dpcd_data, '\0', sizeof(dpcd_data));
memset(lttpr_dpcd_data, '\0', sizeof(lttpr_dpcd_data));
memset(&down_strm_port_count, memset(&down_strm_port_count,
'\0', sizeof(union down_stream_port_count)); '\0', sizeof(union down_stream_port_count));
memset(&edp_config_cap, '\0', memset(&edp_config_cap, '\0',
...@@ -3026,47 +3029,46 @@ static bool retrieve_link_cap(struct dc_link *link) ...@@ -3026,47 +3029,46 @@ static bool retrieve_link_cap(struct dc_link *link)
} }
if (ext_timeout_support) { if (ext_timeout_support) {
status = core_link_read_dpcd( status = core_link_read_dpcd(
link, link,
DP_PHY_REPEATER_CNT, DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV,
&link->dpcd_caps.lttpr_caps.phy_repeater_cnt, lttpr_dpcd_data,
sizeof(link->dpcd_caps.lttpr_caps.phy_repeater_cnt)); sizeof(lttpr_dpcd_data));
if (link->dpcd_caps.lttpr_caps.phy_repeater_cnt > 0) { link->dpcd_caps.lttpr_caps.revision.raw =
lttpr_dpcd_data[DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV -
DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV];
link->is_lttpr_mode_transparent = false; link->dpcd_caps.lttpr_caps.max_link_rate =
lttpr_dpcd_data[DP_MAX_LINK_RATE_PHY_REPEATER -
DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV];
status = core_link_read_dpcd( link->dpcd_caps.lttpr_caps.phy_repeater_cnt =
link, lttpr_dpcd_data[DP_PHY_REPEATER_CNT -
DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV, DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV];
(uint8_t *)&link->dpcd_caps.lttpr_caps.revision,
sizeof(link->dpcd_caps.lttpr_caps.revision));
status = core_link_read_dpcd( link->dpcd_caps.lttpr_caps.max_lane_count =
link, lttpr_dpcd_data[DP_MAX_LANE_COUNT_PHY_REPEATER -
DP_MAX_LINK_RATE_PHY_REPEATER, DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV];
&link->dpcd_caps.lttpr_caps.max_link_rate,
sizeof(link->dpcd_caps.lttpr_caps.max_link_rate));
status = core_link_read_dpcd( link->dpcd_caps.lttpr_caps.mode =
link, lttpr_dpcd_data[DP_PHY_REPEATER_MODE -
DP_PHY_REPEATER_MODE, DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV];
(uint8_t *)&link->dpcd_caps.lttpr_caps.mode,
sizeof(link->dpcd_caps.lttpr_caps.mode));
status = core_link_read_dpcd( link->dpcd_caps.lttpr_caps.max_ext_timeout =
link, lttpr_dpcd_data[DP_PHY_REPEATER_EXTENDED_WAIT_TIMEOUT -
DP_MAX_LANE_COUNT_PHY_REPEATER, DP_LT_TUNABLE_PHY_REPEATER_FIELD_DATA_STRUCTURE_REV];
&link->dpcd_caps.lttpr_caps.max_lane_count,
sizeof(link->dpcd_caps.lttpr_caps.max_lane_count));
status = core_link_read_dpcd( if (link->dpcd_caps.lttpr_caps.phy_repeater_cnt > 0 &&
link, link->dpcd_caps.lttpr_caps.max_lane_count > 0 &&
DP_PHY_REPEATER_EXTENDED_WAIT_TIMEOUT, link->dpcd_caps.lttpr_caps.max_lane_count <= 4 &&
&link->dpcd_caps.lttpr_caps.max_ext_timeout, link->dpcd_caps.lttpr_caps.revision.raw >= 0x14) {
sizeof(link->dpcd_caps.lttpr_caps.max_ext_timeout)); link->is_lttpr_mode_transparent = false;
} else { } else {
dc_link_aux_configure_timeout(link->ddc, prev_timeout_val); /*No lttpr reset timeout to its default value*/
link->is_lttpr_mode_transparent = true;
dc_link_aux_configure_timeout(link->ddc, LINK_AUX_DEFAULT_TIMEOUT_PERIOD);
} }
} }
......
...@@ -420,20 +420,9 @@ enum link_training_offset { ...@@ -420,20 +420,9 @@ enum link_training_offset {
LTTPR_PHY_REPEATER8 = 8 LTTPR_PHY_REPEATER8 = 8
}; };
enum lttpr_mode {
phy_repeater_mode_transparent = 0x55,
phy_repeater_mode_non_transparent = 0xAA
};
enum lttpr_rev {
lttpr_rev_unknown = 0x0,
lttpr_rev_14 = 0x14,
lttpr_rev_max = 0x20
};
struct dc_lttpr_caps { struct dc_lttpr_caps {
enum lttpr_rev revision; union dpcd_rev revision;
enum lttpr_mode mode; uint8_t mode;
uint8_t max_lane_count; uint8_t max_lane_count;
uint8_t max_link_rate; uint8_t max_link_rate;
uint8_t phy_repeater_cnt; uint8_t phy_repeater_cnt;
......
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
#define LINK_TRAINING_ATTEMPTS 4 #define LINK_TRAINING_ATTEMPTS 4
#define LINK_TRAINING_RETRY_DELAY 50 /* ms */ #define LINK_TRAINING_RETRY_DELAY 50 /* ms */
#define LINK_AUX_DEFAULT_EXTENDED_TIMEOUT_PERIOD 3200 /*us*/ #define LINK_AUX_DEFAULT_EXTENDED_TIMEOUT_PERIOD 3200 /*us*/
#define LINK_AUX_DEFAULT_TIMEOUT_PERIOD 400 /*us*/ #define LINK_AUX_DEFAULT_TIMEOUT_PERIOD 552 /*us*/
struct dc_link; struct dc_link;
struct dc_stream_state; struct dc_stream_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