1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (C) 2013 Red Hat
7 #include <linux/delay.h>
8 #include <linux/gpio/consumer.h>
9 #include <linux/pinctrl/consumer.h>
14 static void msm_hdmi_phy_reset(struct hdmi *hdmi)
18 val = hdmi_read(hdmi, REG_HDMI_PHY_CTRL);
20 if (val & HDMI_PHY_CTRL_SW_RESET_LOW) {
22 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
23 val & ~HDMI_PHY_CTRL_SW_RESET);
26 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
27 val | HDMI_PHY_CTRL_SW_RESET);
30 if (val & HDMI_PHY_CTRL_SW_RESET_PLL_LOW) {
32 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
33 val & ~HDMI_PHY_CTRL_SW_RESET_PLL);
36 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
37 val | HDMI_PHY_CTRL_SW_RESET_PLL);
42 if (val & HDMI_PHY_CTRL_SW_RESET_LOW) {
44 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
45 val | HDMI_PHY_CTRL_SW_RESET);
48 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
49 val & ~HDMI_PHY_CTRL_SW_RESET);
52 if (val & HDMI_PHY_CTRL_SW_RESET_PLL_LOW) {
54 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
55 val | HDMI_PHY_CTRL_SW_RESET_PLL);
58 hdmi_write(hdmi, REG_HDMI_PHY_CTRL,
59 val & ~HDMI_PHY_CTRL_SW_RESET_PLL);
63 static void enable_hpd_clocks(struct hdmi *hdmi, bool enable)
65 const struct hdmi_platform_config *config = hdmi->config;
66 struct device *dev = &hdmi->pdev->dev;
70 for (i = 0; i < config->hpd_clk_cnt; i++) {
71 if (config->hpd_freq && config->hpd_freq[i]) {
72 ret = clk_set_rate(hdmi->hpd_clks[i],
76 "failed to set clk %s (%d)\n",
77 config->hpd_clk_names[i], ret);
80 ret = clk_prepare_enable(hdmi->hpd_clks[i]);
83 "failed to enable hpd clk: %s (%d)\n",
84 config->hpd_clk_names[i], ret);
88 for (i = config->hpd_clk_cnt - 1; i >= 0; i--)
89 clk_disable_unprepare(hdmi->hpd_clks[i]);
93 int msm_hdmi_hpd_enable(struct drm_bridge *bridge)
95 struct hdmi_bridge *hdmi_bridge = to_hdmi_bridge(bridge);
96 struct hdmi *hdmi = hdmi_bridge->hdmi;
97 const struct hdmi_platform_config *config = hdmi->config;
98 struct device *dev = &hdmi->pdev->dev;
103 ret = regulator_bulk_enable(config->hpd_reg_cnt, hdmi->hpd_regs);
105 DRM_DEV_ERROR(dev, "failed to enable hpd regulators: %d\n", ret);
109 ret = pinctrl_pm_select_default_state(dev);
111 DRM_DEV_ERROR(dev, "pinctrl state chg failed: %d\n", ret);
116 gpiod_set_value_cansleep(hdmi->hpd_gpiod, 1);
118 pm_runtime_get_sync(dev);
119 enable_hpd_clocks(hdmi, true);
121 msm_hdmi_set_mode(hdmi, false);
122 msm_hdmi_phy_reset(hdmi);
123 msm_hdmi_set_mode(hdmi, true);
125 hdmi_write(hdmi, REG_HDMI_USEC_REFTIMER, 0x0001001b);
127 /* enable HPD events: */
128 hdmi_write(hdmi, REG_HDMI_HPD_INT_CTRL,
129 HDMI_HPD_INT_CTRL_INT_CONNECT |
130 HDMI_HPD_INT_CTRL_INT_EN);
132 /* set timeout to 4.1ms (max) for hardware debounce */
133 spin_lock_irqsave(&hdmi->reg_lock, flags);
134 hpd_ctrl = hdmi_read(hdmi, REG_HDMI_HPD_CTRL);
135 hpd_ctrl |= HDMI_HPD_CTRL_TIMEOUT(0x1fff);
137 /* Toggle HPD circuit to trigger HPD sense */
138 hdmi_write(hdmi, REG_HDMI_HPD_CTRL,
139 ~HDMI_HPD_CTRL_ENABLE & hpd_ctrl);
140 hdmi_write(hdmi, REG_HDMI_HPD_CTRL,
141 HDMI_HPD_CTRL_ENABLE | hpd_ctrl);
142 spin_unlock_irqrestore(&hdmi->reg_lock, flags);
150 void msm_hdmi_hpd_disable(struct hdmi_bridge *hdmi_bridge)
152 struct hdmi *hdmi = hdmi_bridge->hdmi;
153 const struct hdmi_platform_config *config = hdmi->config;
154 struct device *dev = &hdmi->pdev->dev;
157 /* Disable HPD interrupt */
158 hdmi_write(hdmi, REG_HDMI_HPD_INT_CTRL, 0);
160 msm_hdmi_set_mode(hdmi, false);
162 enable_hpd_clocks(hdmi, false);
165 ret = pinctrl_pm_select_sleep_state(dev);
167 dev_warn(dev, "pinctrl state chg failed: %d\n", ret);
169 ret = regulator_bulk_disable(config->hpd_reg_cnt, hdmi->hpd_regs);
171 dev_warn(dev, "failed to disable hpd regulator: %d\n", ret);
174 void msm_hdmi_hpd_irq(struct drm_bridge *bridge)
176 struct hdmi_bridge *hdmi_bridge = to_hdmi_bridge(bridge);
177 struct hdmi *hdmi = hdmi_bridge->hdmi;
178 uint32_t hpd_int_status, hpd_int_ctrl;
181 hpd_int_status = hdmi_read(hdmi, REG_HDMI_HPD_INT_STATUS);
182 hpd_int_ctrl = hdmi_read(hdmi, REG_HDMI_HPD_INT_CTRL);
184 if ((hpd_int_ctrl & HDMI_HPD_INT_CTRL_INT_EN) &&
185 (hpd_int_status & HDMI_HPD_INT_STATUS_INT)) {
186 bool detected = !!(hpd_int_status & HDMI_HPD_INT_STATUS_CABLE_DETECTED);
188 /* ack & disable (temporarily) HPD events: */
189 hdmi_write(hdmi, REG_HDMI_HPD_INT_CTRL,
190 HDMI_HPD_INT_CTRL_INT_ACK);
192 DBG("status=%04x, ctrl=%04x", hpd_int_status, hpd_int_ctrl);
194 /* detect disconnect if we are connected or visa versa: */
195 hpd_int_ctrl = HDMI_HPD_INT_CTRL_INT_EN;
197 hpd_int_ctrl |= HDMI_HPD_INT_CTRL_INT_CONNECT;
198 hdmi_write(hdmi, REG_HDMI_HPD_INT_CTRL, hpd_int_ctrl);
200 queue_work(hdmi->workq, &hdmi_bridge->hpd_work);
204 static enum drm_connector_status detect_reg(struct hdmi *hdmi)
206 uint32_t hpd_int_status;
208 pm_runtime_get_sync(&hdmi->pdev->dev);
209 enable_hpd_clocks(hdmi, true);
211 hpd_int_status = hdmi_read(hdmi, REG_HDMI_HPD_INT_STATUS);
213 enable_hpd_clocks(hdmi, false);
214 pm_runtime_put(&hdmi->pdev->dev);
216 return (hpd_int_status & HDMI_HPD_INT_STATUS_CABLE_DETECTED) ?
217 connector_status_connected : connector_status_disconnected;
220 #define HPD_GPIO_INDEX 2
221 static enum drm_connector_status detect_gpio(struct hdmi *hdmi)
223 return gpiod_get_value(hdmi->hpd_gpiod) ?
224 connector_status_connected :
225 connector_status_disconnected;
228 enum drm_connector_status msm_hdmi_bridge_detect(
229 struct drm_bridge *bridge)
231 struct hdmi_bridge *hdmi_bridge = to_hdmi_bridge(bridge);
232 struct hdmi *hdmi = hdmi_bridge->hdmi;
233 enum drm_connector_status stat_gpio, stat_reg;
237 * some platforms may not have hpd gpio. Rely only on the status
238 * provided by REG_HDMI_HPD_INT_STATUS in this case.
240 if (!hdmi->hpd_gpiod)
241 return detect_reg(hdmi);
244 stat_gpio = detect_gpio(hdmi);
245 stat_reg = detect_reg(hdmi);
247 if (stat_gpio == stat_reg)
253 /* the status we get from reading gpio seems to be more reliable,
254 * so trust that one the most if we didn't manage to get hdmi and
255 * gpio status to agree:
257 if (stat_gpio != stat_reg) {
258 DBG("HDMI_HPD_INT_STATUS tells us: %d", stat_reg);
259 DBG("hpd gpio tells us: %d", stat_gpio);