Commit 3083a984 authored by Qingqing Zhuo's avatar Qingqing Zhuo Committed by Alex Deucher

drm/amd/display: Only one display lights up while using MST hub

[Why]
With the addition of dc_lock acquire before dc_link_handle_hpd_rx_irq,
there will be a deadlock situation where commit state sends a request
for payload allocation on MST and wait for HPD to process DOWN_REP.

[How]
Move forward the MST message handling in handle_hpd_rx_irq so that
it will not rely on call to dc_link_handle_hpd_rx_irq.
Signed-off-by: default avatarQingqing Zhuo <qingqing.zhuo@amd.com>
Reviewed-by: default avatarHarry Wentland <Harry.Wentland@amd.com>
Acked-by: default avatarEryk Brol <eryk.brol@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 5a83bf80
...@@ -2563,11 +2563,9 @@ static void handle_hpd_rx_irq(void *param) ...@@ -2563,11 +2563,9 @@ static void handle_hpd_rx_irq(void *param)
bool result = false; bool result = false;
enum dc_connection_type new_connection_type = dc_connection_none; enum dc_connection_type new_connection_type = dc_connection_none;
struct amdgpu_device *adev = drm_to_adev(dev); struct amdgpu_device *adev = drm_to_adev(dev);
#ifdef CONFIG_DRM_AMD_DC_HDCP
union hpd_irq_data hpd_irq_data; union hpd_irq_data hpd_irq_data;
memset(&hpd_irq_data, 0, sizeof(hpd_irq_data)); memset(&hpd_irq_data, 0, sizeof(hpd_irq_data));
#endif
/* /*
* TODO:Temporary add mutex to protect hpd interrupt not have a gpio * TODO:Temporary add mutex to protect hpd interrupt not have a gpio
...@@ -2577,6 +2575,21 @@ static void handle_hpd_rx_irq(void *param) ...@@ -2577,6 +2575,21 @@ static void handle_hpd_rx_irq(void *param)
if (dc_link->type != dc_connection_mst_branch) if (dc_link->type != dc_connection_mst_branch)
mutex_lock(&aconnector->hpd_lock); mutex_lock(&aconnector->hpd_lock);
read_hpd_rx_irq_data(dc_link, &hpd_irq_data);
if ((dc_link->cur_link_settings.lane_count != LANE_COUNT_UNKNOWN) ||
(dc_link->type == dc_connection_mst_branch)) {
if (hpd_irq_data.bytes.device_service_irq.bits.UP_REQ_MSG_RDY) {
result = true;
dm_handle_hpd_rx_irq(aconnector);
goto out;
} else if (hpd_irq_data.bytes.device_service_irq.bits.DOWN_REP_MSG_RDY) {
result = false;
dm_handle_hpd_rx_irq(aconnector);
goto out;
}
}
mutex_lock(&adev->dm.dc_lock); mutex_lock(&adev->dm.dc_lock);
#ifdef CONFIG_DRM_AMD_DC_HDCP #ifdef CONFIG_DRM_AMD_DC_HDCP
result = dc_link_handle_hpd_rx_irq(dc_link, &hpd_irq_data, NULL); result = dc_link_handle_hpd_rx_irq(dc_link, &hpd_irq_data, NULL);
...@@ -2585,6 +2598,7 @@ static void handle_hpd_rx_irq(void *param) ...@@ -2585,6 +2598,7 @@ static void handle_hpd_rx_irq(void *param)
#endif #endif
mutex_unlock(&adev->dm.dc_lock); mutex_unlock(&adev->dm.dc_lock);
out:
if (result && !is_mst_root_connector) { if (result && !is_mst_root_connector) {
/* Downstream Port status changed. */ /* Downstream Port status changed. */
if (!dc_link_detect_sink(dc_link, &new_connection_type)) if (!dc_link_detect_sink(dc_link, &new_connection_type))
...@@ -2625,9 +2639,6 @@ static void handle_hpd_rx_irq(void *param) ...@@ -2625,9 +2639,6 @@ static void handle_hpd_rx_irq(void *param)
hdcp_handle_cpirq(adev->dm.hdcp_workqueue, aconnector->base.index); hdcp_handle_cpirq(adev->dm.hdcp_workqueue, aconnector->base.index);
} }
#endif #endif
if ((dc_link->cur_link_settings.lane_count != LANE_COUNT_UNKNOWN) ||
(dc_link->type == dc_connection_mst_branch))
dm_handle_hpd_rx_irq(aconnector);
if (dc_link->type != dc_connection_mst_branch) { if (dc_link->type != dc_connection_mst_branch) {
drm_dp_cec_irq(&aconnector->dm_dp_aux.aux); drm_dp_cec_irq(&aconnector->dm_dp_aux.aux);
......
...@@ -1874,7 +1874,7 @@ static struct dc_link_settings get_max_link_cap(struct dc_link *link) ...@@ -1874,7 +1874,7 @@ static struct dc_link_settings get_max_link_cap(struct dc_link *link)
return max_link_cap; return max_link_cap;
} }
static enum dc_status read_hpd_rx_irq_data( enum dc_status read_hpd_rx_irq_data(
struct dc_link *link, struct dc_link *link,
union hpd_irq_data *irq_data) union hpd_irq_data *irq_data)
{ {
......
...@@ -259,6 +259,10 @@ enum dc_status dc_link_reallocate_mst_payload(struct dc_link *link); ...@@ -259,6 +259,10 @@ enum dc_status dc_link_reallocate_mst_payload(struct dc_link *link);
bool dc_link_handle_hpd_rx_irq(struct dc_link *dc_link, bool dc_link_handle_hpd_rx_irq(struct dc_link *dc_link,
union hpd_irq_data *hpd_irq_dpcd_data, bool *out_link_loss); union hpd_irq_data *hpd_irq_dpcd_data, bool *out_link_loss);
enum dc_status read_hpd_rx_irq_data(
struct dc_link *link,
union hpd_irq_data *irq_data);
struct dc_sink_init_data; struct dc_sink_init_data;
struct dc_sink *dc_link_add_remote_sink( struct dc_sink *dc_link_add_remote_sink(
......
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