Commit cf4cebce authored by Peichen Huang's avatar Peichen Huang Committed by Alex Deucher

drm/amd/display: Restructure dpia link training

[WHY]
We intend to consolidate dp tunneling and conventional dp link training.

[HOW]
1. Use the same link training entry for both dp and dpia
2. Move SET_CONFIG of non-transparent mode to dmub side
3. Add set_tps_notification dmub_cmd to notify tps request for
   non-transparent dpia link training
4. Check dpcd request result and abort link training early if dpia
   aux tunneling fails
5. Add option to avoid affect old product
6. Separately handle wait_time_microsec for dpia
Reviewed-by: default avatarCruise Hung <cruise.hung@amd.com>
Reviewed-by: default avatarGeorge Shen <george.shen@amd.com>
Reviewed-by: default avatarMeenakshikumar Somasundaram <meenakshikumar.somasundaram@amd.com>
Signed-off-by: default avatarPeichen Huang <PeiChen.Huang@amd.com>
Signed-off-by: default avatarAlex Hung <alex.hung@amd.com>
Tested-by: default avatarDaniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent ae510080
...@@ -5755,6 +5755,27 @@ enum dc_status dc_process_dmub_set_mst_slots(const struct dc *dc, ...@@ -5755,6 +5755,27 @@ enum dc_status dc_process_dmub_set_mst_slots(const struct dc *dc,
return DC_OK; return DC_OK;
} }
/**
* dc_process_dmub_dpia_set_tps_notification - Submits tps notification
*
* @dc: [in] dc structure
* @link_index: [in] link index
* @ts: [in] request tps
*
* Submits set_tps_notification command to dmub via inbox message
*/
void dc_process_dmub_dpia_set_tps_notification(const struct dc *dc, uint32_t link_index, uint8_t tps)
{
union dmub_rb_cmd cmd = {0};
cmd.set_tps_notification.header.type = DMUB_CMD__DPIA;
cmd.set_tps_notification.header.sub_type = DMUB_CMD__DPIA_SET_TPS_NOTIFICATION;
cmd.set_tps_notification.tps_notification.instance = dc->links[link_index]->ddc_hw_inst;
cmd.set_tps_notification.tps_notification.tps = tps;
dc_wake_and_execute_dmub_cmd(dc->ctx, &cmd, DM_DMUB_WAIT_TYPE_WAIT);
}
/** /**
* dc_process_dmub_dpia_hpd_int_enable - Submits DPIA DPD interruption * dc_process_dmub_dpia_hpd_int_enable - Submits DPIA DPD interruption
* *
......
...@@ -462,6 +462,7 @@ struct dc_config { ...@@ -462,6 +462,7 @@ struct dc_config {
bool support_edp0_on_dp1; bool support_edp0_on_dp1;
unsigned int enable_fpo_flicker_detection; unsigned int enable_fpo_flicker_detection;
bool disable_hbr_audio_dp2; bool disable_hbr_audio_dp2;
bool consolidated_dpia_dp_lt;
}; };
enum visual_confirm { enum visual_confirm {
...@@ -762,7 +763,8 @@ union dpia_debug_options { ...@@ -762,7 +763,8 @@ union dpia_debug_options {
uint32_t disable_mst_dsc_work_around:1; /* bit 3 */ uint32_t disable_mst_dsc_work_around:1; /* bit 3 */
uint32_t enable_force_tbt3_work_around:1; /* bit 4 */ uint32_t enable_force_tbt3_work_around:1; /* bit 4 */
uint32_t disable_usb4_pm_support:1; /* bit 5 */ uint32_t disable_usb4_pm_support:1; /* bit 5 */
uint32_t reserved:26; uint32_t enable_consolidated_dpia_dp_lt:1; /* bit 6 */
uint32_t reserved:25;
} bits; } bits;
uint32_t raw; uint32_t raw;
}; };
...@@ -2525,6 +2527,8 @@ enum dc_status dc_process_dmub_set_mst_slots(const struct dc *dc, ...@@ -2525,6 +2527,8 @@ enum dc_status dc_process_dmub_set_mst_slots(const struct dc *dc,
uint8_t mst_alloc_slots, uint8_t mst_alloc_slots,
uint8_t *mst_slots_in_use); uint8_t *mst_slots_in_use);
void dc_process_dmub_dpia_set_tps_notification(const struct dc *dc, uint32_t link_index, uint8_t tps);
void dc_process_dmub_dpia_hpd_int_enable(const struct dc *dc, void dc_process_dmub_dpia_hpd_int_enable(const struct dc *dc,
uint32_t hpd_int_enable); uint32_t hpd_int_enable);
......
...@@ -50,8 +50,31 @@ static void update_dpia_stream_allocation_table(struct dc_link *link, ...@@ -50,8 +50,31 @@ static void update_dpia_stream_allocation_table(struct dc_link *link,
DC_LOG_MST("dpia : status[%d]: alloc_slots[%d]: used_slots[%d]\n", DC_LOG_MST("dpia : status[%d]: alloc_slots[%d]: used_slots[%d]\n",
status, mst_alloc_slots, prev_mst_slots_in_use); status, mst_alloc_slots, prev_mst_slots_in_use);
ASSERT(link_enc); if (link_enc)
link_enc->funcs->update_mst_stream_allocation_table(link_enc, table); link_enc->funcs->update_mst_stream_allocation_table(link_enc, table);
}
static void set_dio_dpia_link_test_pattern(struct dc_link *link,
const struct link_resource *link_res,
struct encoder_set_dp_phy_pattern_param *tp_params)
{
if (tp_params->dp_phy_pattern != DP_TEST_PATTERN_VIDEO_MODE)
return;
struct link_encoder *link_enc = link_enc_cfg_get_link_enc(link);
if (!link_enc)
return;
link_enc->funcs->dp_set_phy_pattern(link_enc, tp_params);
link->dc->link_srv->dp_trace_source_sequence(link, DPCD_SOURCE_SEQ_AFTER_SET_SOURCE_PATTERN);
}
static void set_dio_dpia_lane_settings(struct dc_link *link,
const struct link_resource *link_res,
const struct dc_link_settings *link_settings,
const struct dc_lane_settings lane_settings[LANE_COUNT_DP_MAX])
{
} }
static const struct link_hwss dpia_link_hwss = { static const struct link_hwss dpia_link_hwss = {
...@@ -65,8 +88,8 @@ static const struct link_hwss dpia_link_hwss = { ...@@ -65,8 +88,8 @@ static const struct link_hwss dpia_link_hwss = {
.ext = { .ext = {
.set_throttled_vcp_size = set_dio_throttled_vcp_size, .set_throttled_vcp_size = set_dio_throttled_vcp_size,
.enable_dp_link_output = enable_dio_dp_link_output, .enable_dp_link_output = enable_dio_dp_link_output,
.set_dp_link_test_pattern = set_dio_dp_link_test_pattern, .set_dp_link_test_pattern = set_dio_dpia_link_test_pattern,
.set_dp_lane_settings = set_dio_dp_lane_settings, .set_dp_lane_settings = set_dio_dpia_lane_settings,
.update_stream_allocation_table = update_dpia_stream_allocation_table, .update_stream_allocation_table = update_dpia_stream_allocation_table,
}, },
}; };
......
...@@ -515,6 +515,41 @@ bool dp_is_interlane_aligned(union lane_align_status_updated align_status) ...@@ -515,6 +515,41 @@ bool dp_is_interlane_aligned(union lane_align_status_updated align_status)
return align_status.bits.INTERLANE_ALIGN_DONE == 1; return align_status.bits.INTERLANE_ALIGN_DONE == 1;
} }
bool dp_check_interlane_aligned(union lane_align_status_updated align_status,
struct dc_link *link,
uint8_t retries)
{
/* Take into consideration corner case for DP 1.4a LL Compliance CTS as USB4
* has to share encoders unlike DP and USBC
*/
return (dp_is_interlane_aligned(align_status) ||
(link->skip_fallback_on_link_loss && retries));
}
uint32_t dp_get_eq_aux_rd_interval(
const struct dc_link *link,
const struct link_training_settings *lt_settings,
uint32_t offset,
uint8_t retries)
{
if (link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA) {
if (offset == 0 && retries == 1 && lt_settings->lttpr_mode == LTTPR_MODE_NON_TRANSPARENT)
return max(lt_settings->eq_pattern_time, (uint32_t) DPIA_CLK_SYNC_DELAY);
else
return dpia_get_eq_aux_rd_interval(link, lt_settings, offset);
} else if (is_repeater(lt_settings, offset))
return dp_translate_training_aux_read_interval(
link->dpcd_caps.lttpr_caps.aux_rd_interval[offset - 1]);
else
return lt_settings->eq_pattern_time;
}
bool dp_check_dpcd_reqeust_status(const struct dc_link *link,
enum dc_status status)
{
return (status != DC_OK && link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA);
}
enum link_training_result dp_check_link_loss_status( enum link_training_result dp_check_link_loss_status(
struct dc_link *link, struct dc_link *link,
const struct link_training_settings *link_training_setting) const struct link_training_settings *link_training_setting)
...@@ -973,13 +1008,17 @@ void repeater_training_done(struct dc_link *link, uint32_t offset) ...@@ -973,13 +1008,17 @@ void repeater_training_done(struct dc_link *link, uint32_t offset)
dpcd_pattern.v1_4.TRAINING_PATTERN_SET); dpcd_pattern.v1_4.TRAINING_PATTERN_SET);
} }
static void dpcd_exit_training_mode(struct dc_link *link, enum dp_link_encoding encoding) static enum link_training_result dpcd_exit_training_mode(struct dc_link *link, enum dp_link_encoding encoding)
{ {
enum dc_status status;
uint8_t sink_status = 0; uint8_t sink_status = 0;
uint8_t i; uint8_t i;
/* clear training pattern set */ /* clear training pattern set */
dpcd_set_training_pattern(link, DP_TRAINING_PATTERN_VIDEOIDLE); status = dpcd_set_training_pattern(link, DP_TRAINING_PATTERN_VIDEOIDLE);
if (dp_check_dpcd_reqeust_status(link, status))
return LINK_TRAINING_ABORT;
if (encoding == DP_128b_132b_ENCODING) { if (encoding == DP_128b_132b_ENCODING) {
/* poll for intra-hop disable */ /* poll for intra-hop disable */
...@@ -990,6 +1029,8 @@ static void dpcd_exit_training_mode(struct dc_link *link, enum dp_link_encoding ...@@ -990,6 +1029,8 @@ static void dpcd_exit_training_mode(struct dc_link *link, enum dp_link_encoding
fsleep(1000); fsleep(1000);
} }
} }
return LINK_TRAINING_SUCCESS;
} }
enum dc_status dpcd_configure_channel_coding(struct dc_link *link, enum dc_status dpcd_configure_channel_coding(struct dc_link *link,
...@@ -1013,17 +1054,18 @@ enum dc_status dpcd_configure_channel_coding(struct dc_link *link, ...@@ -1013,17 +1054,18 @@ enum dc_status dpcd_configure_channel_coding(struct dc_link *link,
return status; return status;
} }
void dpcd_set_training_pattern( enum dc_status dpcd_set_training_pattern(
struct dc_link *link, struct dc_link *link,
enum dc_dp_training_pattern training_pattern) enum dc_dp_training_pattern training_pattern)
{ {
enum dc_status status;
union dpcd_training_pattern dpcd_pattern = {0}; union dpcd_training_pattern dpcd_pattern = {0};
dpcd_pattern.v1_4.TRAINING_PATTERN_SET = dpcd_pattern.v1_4.TRAINING_PATTERN_SET =
dp_training_pattern_to_dpcd_training_pattern( dp_training_pattern_to_dpcd_training_pattern(
link, training_pattern); link, training_pattern);
core_link_write_dpcd( status = core_link_write_dpcd(
link, link,
DP_TRAINING_PATTERN_SET, DP_TRAINING_PATTERN_SET,
&dpcd_pattern.raw, &dpcd_pattern.raw,
...@@ -1033,6 +1075,8 @@ void dpcd_set_training_pattern( ...@@ -1033,6 +1075,8 @@ void dpcd_set_training_pattern(
__func__, __func__,
DP_TRAINING_PATTERN_SET, DP_TRAINING_PATTERN_SET,
dpcd_pattern.v1_4.TRAINING_PATTERN_SET); dpcd_pattern.v1_4.TRAINING_PATTERN_SET);
return status;
} }
enum dc_status dpcd_set_link_settings( enum dc_status dpcd_set_link_settings(
...@@ -1185,6 +1229,13 @@ void dpcd_set_lt_pattern_and_lane_settings( ...@@ -1185,6 +1229,13 @@ void dpcd_set_lt_pattern_and_lane_settings(
dpcd_lt_buffer[DP_TRAINING_PATTERN_SET - DP_TRAINING_PATTERN_SET] dpcd_lt_buffer[DP_TRAINING_PATTERN_SET - DP_TRAINING_PATTERN_SET]
= dpcd_pattern.raw; = dpcd_pattern.raw;
if (link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA)
dpia_set_tps_notification(
link,
lt_settings,
dpcd_pattern.v1_4.TRAINING_PATTERN_SET,
offset);
if (is_repeater(lt_settings, offset)) { if (is_repeater(lt_settings, offset)) {
DC_LOG_HW_LINK_TRAINING("%s\n LTTPR Repeater ID: %d\n 0x%X pattern = %x\n", DC_LOG_HW_LINK_TRAINING("%s\n LTTPR Repeater ID: %d\n 0x%X pattern = %x\n",
__func__, __func__,
...@@ -1455,7 +1506,8 @@ static enum link_training_result dp_transition_to_video_idle( ...@@ -1455,7 +1506,8 @@ static enum link_training_result dp_transition_to_video_idle(
*/ */
if (link->connector_signal != SIGNAL_TYPE_EDP && status == LINK_TRAINING_SUCCESS) { if (link->connector_signal != SIGNAL_TYPE_EDP && status == LINK_TRAINING_SUCCESS) {
msleep(5); msleep(5);
status = dp_check_link_loss_status(link, lt_settings); if (!link->skip_fallback_on_link_loss)
status = dp_check_link_loss_status(link, lt_settings);
} }
return status; return status;
} }
...@@ -1521,7 +1573,9 @@ enum link_training_result dp_perform_link_training( ...@@ -1521,7 +1573,9 @@ enum link_training_result dp_perform_link_training(
ASSERT(0); ASSERT(0);
/* exit training mode */ /* exit training mode */
dpcd_exit_training_mode(link, encoding); if ((dpcd_exit_training_mode(link, encoding) != LINK_TRAINING_SUCCESS || status == LINK_TRAINING_ABORT) &&
link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA)
dpia_training_abort(link, &lt_settings, 0);
/* switch to video idle */ /* switch to video idle */
if ((status == LINK_TRAINING_SUCCESS) || !skip_video_pattern) if ((status == LINK_TRAINING_SUCCESS) || !skip_video_pattern)
...@@ -1599,8 +1653,7 @@ bool perform_link_training_with_retries( ...@@ -1599,8 +1653,7 @@ bool perform_link_training_with_retries(
dp_perform_link_training_skip_aux(link, &pipe_ctx->link_res, &cur_link_settings); dp_perform_link_training_skip_aux(link, &pipe_ctx->link_res, &cur_link_settings);
return true; return true;
} else { } else {
/** @todo Consolidate USB4 DP and DPx.x training. */ if (!link->dc->config.consolidated_dpia_dp_lt && link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA) {
if (link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA) {
status = dpia_perform_link_training( status = dpia_perform_link_training(
link, link,
&pipe_ctx->link_res, &pipe_ctx->link_res,
...@@ -1629,8 +1682,17 @@ bool perform_link_training_with_retries( ...@@ -1629,8 +1682,17 @@ bool perform_link_training_with_retries(
dp_trace_lt_total_count_increment(link, false); dp_trace_lt_total_count_increment(link, false);
dp_trace_lt_result_update(link, status, false); dp_trace_lt_result_update(link, status, false);
dp_trace_set_lt_end_timestamp(link, false); dp_trace_set_lt_end_timestamp(link, false);
if (status == LINK_TRAINING_SUCCESS && !is_link_bw_low) if (status == LINK_TRAINING_SUCCESS && !is_link_bw_low) {
// Update verified link settings to current one
// Because DPIA LT might fallback to lower link setting.
if (link->ep_type == DISPLAY_ENDPOINT_USB4_DPIA &&
stream->signal == SIGNAL_TYPE_DISPLAY_PORT_MST) {
link->verified_link_cap.link_rate = link->cur_link_settings.link_rate;
link->verified_link_cap.lane_count = link->cur_link_settings.lane_count;
dm_helpers_dp_mst_update_branch_bandwidth(link->ctx, link);
}
return true; return true;
}
} }
fail_count++; fail_count++;
......
...@@ -55,7 +55,7 @@ void dp_set_hw_test_pattern( ...@@ -55,7 +55,7 @@ void dp_set_hw_test_pattern(
uint8_t *custom_pattern, uint8_t *custom_pattern,
uint32_t custom_pattern_size); uint32_t custom_pattern_size);
void dpcd_set_training_pattern( enum dc_status dpcd_set_training_pattern(
struct dc_link *link, struct dc_link *link,
enum dc_dp_training_pattern training_pattern); enum dc_dp_training_pattern training_pattern);
...@@ -182,4 +182,18 @@ uint32_t dp_translate_training_aux_read_interval( ...@@ -182,4 +182,18 @@ uint32_t dp_translate_training_aux_read_interval(
uint8_t dp_get_nibble_at_index(const uint8_t *buf, uint8_t dp_get_nibble_at_index(const uint8_t *buf,
uint32_t index); uint32_t index);
bool dp_check_interlane_aligned(union lane_align_status_updated align_status,
struct dc_link *link,
uint8_t retries);
uint32_t dp_get_eq_aux_rd_interval(
const struct dc_link *link,
const struct link_training_settings *lt_settings,
uint32_t offset,
uint8_t retries);
bool dp_check_dpcd_reqeust_status(const struct dc_link *link,
enum dc_status status);
#endif /* __DC_LINK_DP_TRAINING_H__ */ #endif /* __DC_LINK_DP_TRAINING_H__ */
...@@ -157,6 +157,7 @@ enum link_training_result perform_8b_10b_clock_recovery_sequence( ...@@ -157,6 +157,7 @@ enum link_training_result perform_8b_10b_clock_recovery_sequence(
struct link_training_settings *lt_settings, struct link_training_settings *lt_settings,
uint32_t offset) uint32_t offset)
{ {
enum dc_status status;
uint32_t retries_cr; uint32_t retries_cr;
uint32_t retry_count; uint32_t retry_count;
uint32_t wait_time_microsec; uint32_t wait_time_microsec;
...@@ -216,7 +217,7 @@ enum link_training_result perform_8b_10b_clock_recovery_sequence( ...@@ -216,7 +217,7 @@ enum link_training_result perform_8b_10b_clock_recovery_sequence(
/* 4. Read lane status and requested drive /* 4. Read lane status and requested drive
* settings as set by the sink * settings as set by the sink
*/ */
dp_get_lane_status_and_lane_adjust( status = dp_get_lane_status_and_lane_adjust(
link, link,
lt_settings, lt_settings,
dpcd_lane_status, dpcd_lane_status,
...@@ -224,6 +225,9 @@ enum link_training_result perform_8b_10b_clock_recovery_sequence( ...@@ -224,6 +225,9 @@ enum link_training_result perform_8b_10b_clock_recovery_sequence(
dpcd_lane_adjust, dpcd_lane_adjust,
offset); offset);
if (dp_check_dpcd_reqeust_status(link, status))
return LINK_TRAINING_ABORT;
/* 5. check CR done*/ /* 5. check CR done*/
if (dp_is_cr_done(lane_count, dpcd_lane_status)) { if (dp_is_cr_done(lane_count, dpcd_lane_status)) {
DC_LOG_HW_LINK_TRAINING("%s: Clock recovery OK\n", __func__); DC_LOG_HW_LINK_TRAINING("%s: Clock recovery OK\n", __func__);
...@@ -273,6 +277,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence( ...@@ -273,6 +277,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence(
struct link_training_settings *lt_settings, struct link_training_settings *lt_settings,
uint32_t offset) uint32_t offset)
{ {
enum dc_status status;
enum dc_dp_training_pattern tr_pattern; enum dc_dp_training_pattern tr_pattern;
uint32_t retries_ch_eq; uint32_t retries_ch_eq;
uint32_t wait_time_microsec; uint32_t wait_time_microsec;
...@@ -308,12 +313,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence( ...@@ -308,12 +313,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence(
dpcd_set_lane_settings(link, lt_settings, offset); dpcd_set_lane_settings(link, lt_settings, offset);
/* 3. wait for receiver to lock-on*/ /* 3. wait for receiver to lock-on*/
wait_time_microsec = lt_settings->eq_pattern_time; wait_time_microsec = dp_get_eq_aux_rd_interval(link, lt_settings, offset, retries_ch_eq);
if (is_repeater(lt_settings, offset))
wait_time_microsec =
dp_translate_training_aux_read_interval(
link->dpcd_caps.lttpr_caps.aux_rd_interval[offset - 1]);
dp_wait_for_training_aux_rd_interval( dp_wait_for_training_aux_rd_interval(
link, link,
...@@ -322,7 +322,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence( ...@@ -322,7 +322,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence(
/* 4. Read lane status and requested /* 4. Read lane status and requested
* drive settings as set by the sink*/ * drive settings as set by the sink*/
dp_get_lane_status_and_lane_adjust( status = dp_get_lane_status_and_lane_adjust(
link, link,
lt_settings, lt_settings,
dpcd_lane_status, dpcd_lane_status,
...@@ -330,6 +330,9 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence( ...@@ -330,6 +330,9 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence(
dpcd_lane_adjust, dpcd_lane_adjust,
offset); offset);
if (dp_check_dpcd_reqeust_status(link, status))
return LINK_TRAINING_ABORT;
/* 5. check CR done*/ /* 5. check CR done*/
if (!dp_is_cr_done(lane_count, dpcd_lane_status)) if (!dp_is_cr_done(lane_count, dpcd_lane_status))
return dpcd_lane_status[0].bits.CR_DONE_0 ? return dpcd_lane_status[0].bits.CR_DONE_0 ?
...@@ -339,7 +342,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence( ...@@ -339,7 +342,7 @@ enum link_training_result perform_8b_10b_channel_equalization_sequence(
/* 6. check CHEQ done*/ /* 6. check CHEQ done*/
if (dp_is_ch_eq_done(lane_count, dpcd_lane_status) && if (dp_is_ch_eq_done(lane_count, dpcd_lane_status) &&
dp_is_symbol_locked(lane_count, dpcd_lane_status) && dp_is_symbol_locked(lane_count, dpcd_lane_status) &&
dp_is_interlane_aligned(dpcd_lane_status_updated)) dp_check_interlane_aligned(dpcd_lane_status_updated, link, retries_ch_eq))
return LINK_TRAINING_SUCCESS; return LINK_TRAINING_SUCCESS;
/* 7. update VS/PE/PC2 in lt_settings*/ /* 7. update VS/PE/PC2 in lt_settings*/
......
...@@ -43,9 +43,6 @@ ...@@ -43,9 +43,6 @@
#define DC_LOGGER \ #define DC_LOGGER \
link->ctx->logger link->ctx->logger
/* The approximate time (us) it takes to transmit 9 USB4 DP clock sync packets. */
#define DPIA_CLK_SYNC_DELAY 16000
/* Extend interval between training status checks for manual testing. */ /* Extend interval between training status checks for manual testing. */
#define DPIA_DEBUG_EXTENDED_AUX_RD_INTERVAL_US 60000000 #define DPIA_DEBUG_EXTENDED_AUX_RD_INTERVAL_US 60000000
...@@ -566,28 +563,6 @@ static enum link_training_result dpia_training_cr_phase( ...@@ -566,28 +563,6 @@ static enum link_training_result dpia_training_cr_phase(
return result; return result;
} }
/* Return status read interval during equalization phase. */
static uint32_t dpia_get_eq_aux_rd_interval(
const struct dc_link *link,
const struct link_training_settings *lt_settings,
uint32_t hop)
{
uint32_t wait_time_microsec;
if (hop == DPRX)
wait_time_microsec = lt_settings->eq_pattern_time;
else
wait_time_microsec =
dp_translate_training_aux_read_interval(
link->dpcd_caps.lttpr_caps.aux_rd_interval[hop - 1]);
/* Check debug option for extending aux read interval. */
if (link->dc->debug.dpia_debug.bits.extend_aux_rd_interval)
wait_time_microsec = DPIA_DEBUG_EXTENDED_AUX_RD_INTERVAL_US;
return wait_time_microsec;
}
/* Execute equalization phase of link training for specified hop in display /* Execute equalization phase of link training for specified hop in display
* path in non-transparent mode: * path in non-transparent mode:
* - driver issues both DPCD and SET_CONFIG transactions. * - driver issues both DPCD and SET_CONFIG transactions.
...@@ -936,6 +911,22 @@ static enum link_training_result dpia_training_end( ...@@ -936,6 +911,22 @@ static enum link_training_result dpia_training_end(
return result; return result;
} }
/* Return status read interval during equalization phase. */
uint32_t dpia_get_eq_aux_rd_interval(
const struct dc_link *link,
const struct link_training_settings *lt_settings,
uint32_t hop)
{
/* Check debug option for extending aux read interval. */
if (link->dc->debug.dpia_debug.bits.extend_aux_rd_interval)
return DPIA_DEBUG_EXTENDED_AUX_RD_INTERVAL_US;
else if (hop == DPRX)
return lt_settings->eq_pattern_time;
else
return dp_translate_training_aux_read_interval(
link->dpcd_caps.lttpr_caps.aux_rd_interval[hop - 1]);
}
/* When aborting training of specified hop in display path, clean up by: /* When aborting training of specified hop in display path, clean up by:
* - Attempting to clear DPCD TRAINING_PATTERN_SET, LINK_BW_SET and LANE_COUNT_SET. * - Attempting to clear DPCD TRAINING_PATTERN_SET, LINK_BW_SET and LANE_COUNT_SET.
* - Sending SET_CONFIG(SET_LINK) with lane count and link rate set to 0. * - Sending SET_CONFIG(SET_LINK) with lane count and link rate set to 0.
...@@ -943,7 +934,7 @@ static enum link_training_result dpia_training_end( ...@@ -943,7 +934,7 @@ static enum link_training_result dpia_training_end(
* @param link DPIA link being trained. * @param link DPIA link being trained.
* @param hop Hop in display path. DPRX = 0. * @param hop Hop in display path. DPRX = 0.
*/ */
static void dpia_training_abort( void dpia_training_abort(
struct dc_link *link, struct dc_link *link,
struct link_training_settings *lt_settings, struct link_training_settings *lt_settings,
uint32_t hop) uint32_t hop)
...@@ -968,7 +959,26 @@ static void dpia_training_abort( ...@@ -968,7 +959,26 @@ static void dpia_training_abort(
core_link_write_dpcd(link, dpcd_tps_offset, &data, 1); core_link_write_dpcd(link, dpcd_tps_offset, &data, 1);
core_link_write_dpcd(link, DP_LINK_BW_SET, &data, 1); core_link_write_dpcd(link, DP_LINK_BW_SET, &data, 1);
core_link_write_dpcd(link, DP_LANE_COUNT_SET, &data, 1); core_link_write_dpcd(link, DP_LANE_COUNT_SET, &data, 1);
core_link_send_set_config(link, DPIA_SET_CFG_SET_LINK, data);
if (!link->dc->config.consolidated_dpia_dp_lt)
core_link_send_set_config(link, DPIA_SET_CFG_SET_LINK, data);
}
void dpia_set_tps_notification(
struct dc_link *link,
const struct link_training_settings *lt_settings,
uint8_t pattern,
uint32_t hop)
{
uint8_t repeater_cnt = 0; /* Number of hops/repeaters in display path. */
if (lt_settings->lttpr_mode != LTTPR_MODE_NON_TRANSPARENT || pattern == DPCD_TRAINING_PATTERN_VIDEOIDLE)
return;
repeater_cnt = dp_parse_lttpr_repeater_count(link->dpcd_caps.lttpr_caps.phy_repeater_cnt);
if (hop != repeater_cnt)
dc_process_dmub_dpia_set_tps_notification(link->ctx->dc, link->link_index, pattern);
} }
enum link_training_result dpia_perform_link_training( enum link_training_result dpia_perform_link_training(
......
...@@ -28,6 +28,9 @@ ...@@ -28,6 +28,9 @@
#define __DC_LINK_DP_TRAINING_DPIA_H__ #define __DC_LINK_DP_TRAINING_DPIA_H__
#include "link_dp_training.h" #include "link_dp_training.h"
/* The approximate time (us) it takes to transmit 9 USB4 DP clock sync packets. */
#define DPIA_CLK_SYNC_DELAY 16000
/* Train DP tunneling link for USB4 DPIA display endpoint. /* Train DP tunneling link for USB4 DPIA display endpoint.
* DPIA equivalent of dc_link_dp_perfrorm_link_training. * DPIA equivalent of dc_link_dp_perfrorm_link_training.
* Aborts link training upon detection of sink unplug. * Aborts link training upon detection of sink unplug.
...@@ -38,4 +41,20 @@ enum link_training_result dpia_perform_link_training( ...@@ -38,4 +41,20 @@ enum link_training_result dpia_perform_link_training(
const struct dc_link_settings *link_setting, const struct dc_link_settings *link_setting,
bool skip_video_pattern); bool skip_video_pattern);
void dpia_training_abort(
struct dc_link *link,
struct link_training_settings *lt_settings,
uint32_t hop);
uint32_t dpia_get_eq_aux_rd_interval(
const struct dc_link *link,
const struct link_training_settings *lt_settings,
uint32_t hop);
void dpia_set_tps_notification(
struct dc_link *link,
const struct link_training_settings *lt_settings,
uint8_t pattern,
uint32_t offset);
#endif /* __DC_LINK_DP_TRAINING_DPIA_H__ */ #endif /* __DC_LINK_DP_TRAINING_DPIA_H__ */
...@@ -300,6 +300,7 @@ struct dmub_srv_hw_params { ...@@ -300,6 +300,7 @@ struct dmub_srv_hw_params {
enum dmub_ips_disable_type disable_ips; enum dmub_ips_disable_type disable_ips;
bool disallow_phy_access; bool disallow_phy_access;
bool disable_sldo_opt; bool disable_sldo_opt;
bool enable_non_transparent_setconfig;
}; };
/** /**
......
...@@ -682,7 +682,7 @@ union dmub_fw_boot_options { ...@@ -682,7 +682,7 @@ union dmub_fw_boot_options {
uint32_t gpint_scratch8: 1; /* 1 if GPINT is in scratch8*/ uint32_t gpint_scratch8: 1; /* 1 if GPINT is in scratch8*/
uint32_t usb4_cm_version: 1; /**< 1 CM support */ uint32_t usb4_cm_version: 1; /**< 1 CM support */
uint32_t dpia_hpd_int_enable_supported: 1; /* 1 if dpia hpd int enable supported */ uint32_t dpia_hpd_int_enable_supported: 1; /* 1 if dpia hpd int enable supported */
uint32_t reserved0: 1; uint32_t enable_non_transparent_setconfig: 1; /* 1 if dpia use conventional dp lt flow*/
uint32_t disable_clk_ds: 1; /* 1 if disallow dispclk_ds and dppclk_ds*/ uint32_t disable_clk_ds: 1; /* 1 if disallow dispclk_ds and dppclk_ds*/
uint32_t disable_timeout_recovery : 1; /* 1 if timeout recovery should be disabled */ uint32_t disable_timeout_recovery : 1; /* 1 if timeout recovery should be disabled */
uint32_t ips_pg_disable: 1; /* 1 to disable ONO domains power gating*/ uint32_t ips_pg_disable: 1; /* 1 to disable ONO domains power gating*/
...@@ -1308,6 +1308,7 @@ enum dmub_cmd_dpia_type { ...@@ -1308,6 +1308,7 @@ enum dmub_cmd_dpia_type {
DMUB_CMD__DPIA_DIG1_DPIA_CONTROL = 0, DMUB_CMD__DPIA_DIG1_DPIA_CONTROL = 0,
DMUB_CMD__DPIA_SET_CONFIG_ACCESS = 1, DMUB_CMD__DPIA_SET_CONFIG_ACCESS = 1,
DMUB_CMD__DPIA_MST_ALLOC_SLOTS = 2, DMUB_CMD__DPIA_MST_ALLOC_SLOTS = 2,
DMUB_CMD__DPIA_SET_TPS_NOTIFICATION = 3,
}; };
/* DMUB_OUT_CMD__DPIA_NOTIFICATION command types. */ /* DMUB_OUT_CMD__DPIA_NOTIFICATION command types. */
...@@ -2138,6 +2139,24 @@ struct dmub_rb_cmd_set_mst_alloc_slots { ...@@ -2138,6 +2139,24 @@ struct dmub_rb_cmd_set_mst_alloc_slots {
struct dmub_cmd_mst_alloc_slots_control_data mst_slots_control; /* mst slots control */ struct dmub_cmd_mst_alloc_slots_control_data mst_slots_control; /* mst slots control */
}; };
/**
* Data passed from driver to FW in a DMUB_CMD__SET_TPS_NOTIFICATION command.
*/
struct dmub_cmd_tps_notification_data {
uint8_t instance; /* DPIA instance */
uint8_t tps; /* requested training pattern */
uint8_t reserved1;
uint8_t reserved2;
};
/**
* DMUB command structure for SET_TPS_NOTIFICATION command.
*/
struct dmub_rb_cmd_set_tps_notification {
struct dmub_cmd_header header; /* header */
struct dmub_cmd_tps_notification_data tps_notification; /* set tps_notification data */
};
/** /**
* DMUB command structure for DPIA HPD int enable control. * DMUB command structure for DPIA HPD int enable control.
*/ */
...@@ -5304,6 +5323,10 @@ union dmub_rb_cmd { ...@@ -5304,6 +5323,10 @@ union dmub_rb_cmd {
* Definition of a DMUB_CMD__DPIA_MST_ALLOC_SLOTS command. * Definition of a DMUB_CMD__DPIA_MST_ALLOC_SLOTS command.
*/ */
struct dmub_rb_cmd_set_mst_alloc_slots set_mst_alloc_slots; struct dmub_rb_cmd_set_mst_alloc_slots set_mst_alloc_slots;
/**
* Definition of a DMUB_CMD__DPIA_SET_TPS_NOTIFICATION command.
*/
struct dmub_rb_cmd_set_tps_notification set_tps_notification;
/** /**
* Definition of a DMUB_CMD__EDID_CEA command. * Definition of a DMUB_CMD__EDID_CEA command.
*/ */
......
...@@ -425,6 +425,7 @@ void dmub_dcn35_enable_dmub_boot_options(struct dmub_srv *dmub, const struct dmu ...@@ -425,6 +425,7 @@ void dmub_dcn35_enable_dmub_boot_options(struct dmub_srv *dmub, const struct dmu
boot_options.bits.ips_disable = params->disable_ips; boot_options.bits.ips_disable = params->disable_ips;
boot_options.bits.ips_sequential_ono = params->ips_sequential_ono; boot_options.bits.ips_sequential_ono = params->ips_sequential_ono;
boot_options.bits.disable_sldo_opt = params->disable_sldo_opt; boot_options.bits.disable_sldo_opt = params->disable_sldo_opt;
boot_options.bits.enable_non_transparent_setconfig = params->enable_non_transparent_setconfig;
REG_WRITE(DMCUB_SCRATCH14, boot_options.all); REG_WRITE(DMCUB_SCRATCH14, boot_options.all);
} }
......
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