aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/gpu/drm/bridge/analogix
diff options
context:
space:
mode:
authorYakir Yang <ykk@rock-chips.com>2016-02-15 06:11:15 -0500
committerYakir Yang <ykk@rock-chips.com>2016-04-04 22:13:07 -0400
commit5cff007c58b7103a7d6bb203a74eb3eb416c10e0 (patch)
tree87449e8a7ab7125949f8cc07defb17ced9965f06 /drivers/gpu/drm/bridge/analogix
parent0d0abd894ead85d20d7febe603f570d4fdd8111a (diff)
drm: bridge: analogix/dp: try force hpd after plug in lookup failed
Some edp screen do not have hpd signal, so we can't just return failed when hpd plug in detect failed. This is an hardware property, so we need add a devicetree property "analogix,need-force-hpd" to indicate this sutiation. Acked-by: Rob Herring <robh@kernel.org> Tested-by: Caesar Wang <wxt@rock-chips.com> Tested-by: Douglas Anderson <dianders@chromium.org> Tested-by: Heiko Stuebner <heiko@sntech.de> Tested-by: Javier Martinez Canillas <javier@osg.samsung.com> Signed-off-by: Yakir Yang <ykk@rock-chips.com>
Diffstat (limited to 'drivers/gpu/drm/bridge/analogix')
-rw-r--r--drivers/gpu/drm/bridge/analogix/analogix_dp_core.c35
-rw-r--r--drivers/gpu/drm/bridge/analogix/analogix_dp_core.h2
-rw-r--r--drivers/gpu/drm/bridge/analogix/analogix_dp_reg.c9
3 files changed, 41 insertions, 5 deletions
diff --git a/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c b/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c
index a62e8d07b06a..13986b7b9a61 100644
--- a/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c
+++ b/drivers/gpu/drm/bridge/analogix/analogix_dp_core.c
@@ -62,15 +62,38 @@ static int analogix_dp_detect_hpd(struct analogix_dp_device *dp)
62{ 62{
63 int timeout_loop = 0; 63 int timeout_loop = 0;
64 64
65 while (analogix_dp_get_plug_in_status(dp) != 0) { 65 while (timeout_loop < DP_TIMEOUT_LOOP_COUNT) {
66 if (analogix_dp_get_plug_in_status(dp) == 0)
67 return 0;
68
66 timeout_loop++; 69 timeout_loop++;
67 if (timeout_loop > DP_TIMEOUT_LOOP_COUNT) {
68 dev_err(dp->dev, "failed to get hpd plug status\n");
69 return -ETIMEDOUT;
70 }
71 usleep_range(10, 11); 70 usleep_range(10, 11);
72 } 71 }
73 72
73 /*
74 * Some edp screen do not have hpd signal, so we can't just
75 * return failed when hpd plug in detect failed, DT property
76 * "force-hpd" would indicate whether driver need this.
77 */
78 if (!dp->force_hpd)
79 return -ETIMEDOUT;
80
81 /*
82 * The eDP TRM indicate that if HPD_STATUS(RO) is 0, AUX CH
83 * will not work, so we need to give a force hpd action to
84 * set HPD_STATUS manually.
85 */
86 dev_dbg(dp->dev, "failed to get hpd plug status, try to force hpd\n");
87
88 analogix_dp_force_hpd(dp);
89
90 if (analogix_dp_get_plug_in_status(dp) != 0) {
91 dev_err(dp->dev, "failed to get hpd plug in status\n");
92 return -EINVAL;
93 }
94
95 dev_dbg(dp->dev, "success to get plug in status after force hpd\n");
96
74 return 0; 97 return 0;
75} 98}
76 99
@@ -1291,6 +1314,8 @@ int analogix_dp_bind(struct device *dev, struct drm_device *drm_dev,
1291 if (IS_ERR(dp->reg_base)) 1314 if (IS_ERR(dp->reg_base))
1292 return PTR_ERR(dp->reg_base); 1315 return PTR_ERR(dp->reg_base);
1293 1316
1317 dp->force_hpd = of_property_read_bool(dev->of_node, "force-hpd");
1318
1294 dp->hpd_gpio = of_get_named_gpio(dev->of_node, "hpd-gpios", 0); 1319 dp->hpd_gpio = of_get_named_gpio(dev->of_node, "hpd-gpios", 0);
1295 if (!gpio_is_valid(dp->hpd_gpio)) 1320 if (!gpio_is_valid(dp->hpd_gpio))
1296 dp->hpd_gpio = of_get_named_gpio(dev->of_node, 1321 dp->hpd_gpio = of_get_named_gpio(dev->of_node,
diff --git a/drivers/gpu/drm/bridge/analogix/analogix_dp_core.h b/drivers/gpu/drm/bridge/analogix/analogix_dp_core.h
index 02c0ecffa804..22aff07c1995 100644
--- a/drivers/gpu/drm/bridge/analogix/analogix_dp_core.h
+++ b/drivers/gpu/drm/bridge/analogix/analogix_dp_core.h
@@ -154,6 +154,7 @@ struct analogix_dp_device {
154 struct phy *phy; 154 struct phy *phy;
155 int dpms_mode; 155 int dpms_mode;
156 int hpd_gpio; 156 int hpd_gpio;
157 bool force_hpd;
157 158
158 struct analogix_dp_plat_data *plat_data; 159 struct analogix_dp_plat_data *plat_data;
159}; 160};
@@ -174,6 +175,7 @@ void analogix_dp_set_analog_power_down(struct analogix_dp_device *dp,
174 bool enable); 175 bool enable);
175void analogix_dp_init_analog_func(struct analogix_dp_device *dp); 176void analogix_dp_init_analog_func(struct analogix_dp_device *dp);
176void analogix_dp_init_hpd(struct analogix_dp_device *dp); 177void analogix_dp_init_hpd(struct analogix_dp_device *dp);
178void analogix_dp_force_hpd(struct analogix_dp_device *dp);
177enum dp_irq_type analogix_dp_get_irq_type(struct analogix_dp_device *dp); 179enum dp_irq_type analogix_dp_get_irq_type(struct analogix_dp_device *dp);
178void analogix_dp_clear_hotplug_interrupts(struct analogix_dp_device *dp); 180void analogix_dp_clear_hotplug_interrupts(struct analogix_dp_device *dp);
179void analogix_dp_reset_aux(struct analogix_dp_device *dp); 181void analogix_dp_reset_aux(struct analogix_dp_device *dp);
diff --git a/drivers/gpu/drm/bridge/analogix/analogix_dp_reg.c b/drivers/gpu/drm/bridge/analogix/analogix_dp_reg.c
index 1e24b3722ff1..cba3ffd88649 100644
--- a/drivers/gpu/drm/bridge/analogix/analogix_dp_reg.c
+++ b/drivers/gpu/drm/bridge/analogix/analogix_dp_reg.c
@@ -365,6 +365,15 @@ void analogix_dp_init_hpd(struct analogix_dp_device *dp)
365 writel(reg, dp->reg_base + ANALOGIX_DP_SYS_CTL_3); 365 writel(reg, dp->reg_base + ANALOGIX_DP_SYS_CTL_3);
366} 366}
367 367
368void analogix_dp_force_hpd(struct analogix_dp_device *dp)
369{
370 u32 reg;
371
372 reg = readl(dp->reg_base + ANALOGIX_DP_SYS_CTL_3);
373 reg = (F_HPD | HPD_CTRL);
374 writel(reg, dp->reg_base + ANALOGIX_DP_SYS_CTL_3);
375}
376
368enum dp_irq_type analogix_dp_get_irq_type(struct analogix_dp_device *dp) 377enum dp_irq_type analogix_dp_get_irq_type(struct analogix_dp_device *dp)
369{ 378{
370 u32 reg; 379 u32 reg;