|
@@ -10,10 +10,11 @@
|
|
|
* option) any later version.
|
|
|
*/
|
|
|
|
|
|
-#include <linux/device.h>
|
|
|
-#include <linux/io.h>
|
|
|
#include <linux/delay.h>
|
|
|
+#include <linux/device.h>
|
|
|
#include <linux/gpio.h>
|
|
|
+#include <linux/io.h>
|
|
|
+#include <linux/iopoll.h>
|
|
|
|
|
|
#include <drm/bridge/analogix_dp.h>
|
|
|
|
|
@@ -992,10 +993,25 @@ void analogix_dp_enable_psr_crc(struct analogix_dp_device *dp)
|
|
|
writel(PSR_VID_CRC_ENABLE, dp->reg_base + ANALOGIX_DP_CRC_CON);
|
|
|
}
|
|
|
|
|
|
-void analogix_dp_send_psr_spd(struct analogix_dp_device *dp,
|
|
|
- struct edp_vsc_psr *vsc)
|
|
|
+static ssize_t analogix_dp_get_psr_status(struct analogix_dp_device *dp)
|
|
|
+{
|
|
|
+ ssize_t val;
|
|
|
+ u8 status;
|
|
|
+
|
|
|
+ val = drm_dp_dpcd_readb(&dp->aux, DP_PSR_STATUS, &status);
|
|
|
+ if (val < 0) {
|
|
|
+ dev_err(dp->dev, "PSR_STATUS read failed ret=%zd", val);
|
|
|
+ return val;
|
|
|
+ }
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+int analogix_dp_send_psr_spd(struct analogix_dp_device *dp,
|
|
|
+ struct edp_vsc_psr *vsc)
|
|
|
{
|
|
|
unsigned int val;
|
|
|
+ int ret;
|
|
|
+ ssize_t psr_status;
|
|
|
|
|
|
/* don't send info frame */
|
|
|
val = readl(dp->reg_base + ANALOGIX_DP_PKT_SEND_CTL);
|
|
@@ -1036,6 +1052,17 @@ void analogix_dp_send_psr_spd(struct analogix_dp_device *dp,
|
|
|
val = readl(dp->reg_base + ANALOGIX_DP_PKT_SEND_CTL);
|
|
|
val |= IF_EN;
|
|
|
writel(val, dp->reg_base + ANALOGIX_DP_PKT_SEND_CTL);
|
|
|
+
|
|
|
+ ret = readx_poll_timeout(analogix_dp_get_psr_status, dp, psr_status,
|
|
|
+ psr_status >= 0 &&
|
|
|
+ ((vsc->DB1 && psr_status == DP_PSR_SINK_ACTIVE_RFB) ||
|
|
|
+ (!vsc->DB1 && psr_status == DP_PSR_SINK_INACTIVE)), 1500,
|
|
|
+ DP_TIMEOUT_PSR_LOOP_MS * 1000);
|
|
|
+ if (ret) {
|
|
|
+ dev_warn(dp->dev, "Failed to apply PSR %d\n", ret);
|
|
|
+ return ret;
|
|
|
+ }
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
ssize_t analogix_dp_transfer(struct analogix_dp_device *dp,
|