Commit 0de07ccb authored by Lyude Paul's avatar Lyude Paul Committed by Kelsey Skunberg

drm/dp_mst: Increase ACT retry timeout to 3s

BugLink: https://bugs.launchpad.net/bugs/1885932

[ Upstream commit 873a95e0 ]

Currently we only poll for an ACT up to 30 times, with a busy-wait delay
of 100µs between each attempt - giving us a timeout of 2900µs. While
this might seem sensible, it would appear that in certain scenarios it
can take dramatically longer then that for us to receive an ACT. On one
of the EVGA MST hubs that I have available, I observed said hub
sometimes taking longer then a second before signalling the ACT. These
delays mostly seem to occur when previous sideband messages we've sent
are NAKd by the hub, however it wouldn't be particularly surprising if
it's possible to reproduce times like this simply by introducing branch
devices with large LCTs since payload allocations have to take effect on
every downstream device up to the payload's target.

So, instead of just retrying 30 times we poll for the ACT for up to 3ms,
and additionally use usleep_range() to avoid a very long and rude
busy-wait. Note that the previous retry count of 30 appears to have been
arbitrarily chosen, as I can't find any mention of a recommended timeout
or retry count for ACTs in the DisplayPort 2.0 specification. This also
goes for the range we were previously using for udelay(), although I
suspect that was just copied from the recommended delay for link
training on SST devices.

Changes since v1:
* Use readx_poll_timeout() instead of open-coding timeout loop - Sean
  Paul
Changes since v2:
* Increase poll interval to 200us - Sean Paul
* Print status in hex when we timeout waiting for ACT - Sean Paul
Signed-off-by: default avatarLyude Paul <lyude@redhat.com>
Fixes: ad7f8a1f ("drm/helper: add Displayport multi-stream helper (v0.6)")
Cc: Sean Paul <sean@poorly.run>
Cc: <stable@vger.kernel.org> # v3.17+
Reviewed-by: default avatarSean Paul <sean@poorly.run>
Link: https://patchwork.freedesktop.org/patch/msgid/20200406221253.1307209-4-lyude@redhat.comSigned-off-by: default avatarSasha Levin <sashal@kernel.org>
Signed-off-by: default avatarKamal Mostafa <kamal@canonical.com>
Signed-off-by: default avatarKelsey Skunberg <kelsey.skunberg@canonical.com>
parent d0f27731
...@@ -29,6 +29,7 @@ ...@@ -29,6 +29,7 @@
#include <linux/i2c.h> #include <linux/i2c.h>
#include <drm/drm_dp_mst_helper.h> #include <drm/drm_dp_mst_helper.h>
#include <drm/drmP.h> #include <drm/drmP.h>
#include <linux/iopoll.h>
#include <drm/drm_fixed.h> #include <drm/drm_fixed.h>
...@@ -2679,6 +2680,17 @@ static int drm_dp_dpcd_write_payload(struct drm_dp_mst_topology_mgr *mgr, ...@@ -2679,6 +2680,17 @@ static int drm_dp_dpcd_write_payload(struct drm_dp_mst_topology_mgr *mgr,
return ret; return ret;
} }
static int do_get_act_status(struct drm_dp_aux *aux)
{
int ret;
u8 status;
ret = drm_dp_dpcd_readb(aux, DP_PAYLOAD_TABLE_UPDATE_STATUS, &status);
if (ret < 0)
return ret;
return status;
}
/** /**
* drm_dp_check_act_status() - Check ACT handled status. * drm_dp_check_act_status() - Check ACT handled status.
...@@ -2688,30 +2700,28 @@ static int drm_dp_dpcd_write_payload(struct drm_dp_mst_topology_mgr *mgr, ...@@ -2688,30 +2700,28 @@ static int drm_dp_dpcd_write_payload(struct drm_dp_mst_topology_mgr *mgr,
*/ */
int drm_dp_check_act_status(struct drm_dp_mst_topology_mgr *mgr) int drm_dp_check_act_status(struct drm_dp_mst_topology_mgr *mgr)
{ {
int count = 0, ret; /*
u8 status; * There doesn't seem to be any recommended retry count or timeout in
* the MST specification. Since some hubs have been observed to take
do { * over 1 second to update their payload allocations under certain
ret = drm_dp_dpcd_readb(mgr->aux, * conditions, we use a rather large timeout value.
DP_PAYLOAD_TABLE_UPDATE_STATUS, */
&status); const int timeout_ms = 3000;
if (ret < 0) { int ret, status;
DRM_DEBUG_KMS("failed to read payload table status %d\n",
ret); ret = readx_poll_timeout(do_get_act_status, mgr->aux, status,
return ret; status & DP_PAYLOAD_ACT_HANDLED || status < 0,
} 200, timeout_ms * USEC_PER_MSEC);
if (ret < 0 && status >= 0) {
if (status & DP_PAYLOAD_ACT_HANDLED) DRM_DEBUG_KMS("Failed to get ACT after %dms, last status: %02x\n",
break; timeout_ms, status);
count++;
udelay(100);
} while (count < 30);
if (!(status & DP_PAYLOAD_ACT_HANDLED)) {
DRM_DEBUG_KMS("failed to get ACT bit %d after %d retries\n",
status, count);
return -EINVAL; return -EINVAL;
} else if (status < 0) {
DRM_DEBUG_KMS("Failed to read payload table status: %d\n",
status);
return status;
} }
return 0; return 0;
} }
EXPORT_SYMBOL(drm_dp_check_act_status); EXPORT_SYMBOL(drm_dp_check_act_status);
......
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