提交 f006325c 编写于 作者: L Laurent Pinchart 提交者: Tomi Valkeinen

drm/omap: Move HPD disconnection handling to omap_connector

On HDMI outputs, CEC support requires notification of HPD signal
deassertion. The HPD signal can be handled by various omap_dss_device
instances in the pipeline, and all of them forward HPD events to the
OMAP4 internal HDMI encoder.

Knowledge of the DSS internals need to be removed from the
omap_dss_device instances in order to migrate to drm_bridge. To do so,
move HPD handling for CEC to the omap_connector.
Signed-off-by: NLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: NSebastian Reichel <sebastian.reichel@collabora.co.uk>
Signed-off-by: NTomi Valkeinen <tomi.valkeinen@ti.com>
上级 18412b66
...@@ -137,13 +137,8 @@ static int hdmic_read_edid(struct omap_dss_device *dssdev, ...@@ -137,13 +137,8 @@ static int hdmic_read_edid(struct omap_dss_device *dssdev,
static bool hdmic_detect(struct omap_dss_device *dssdev) static bool hdmic_detect(struct omap_dss_device *dssdev)
{ {
struct panel_drv_data *ddata = to_panel_data(dssdev); struct panel_drv_data *ddata = to_panel_data(dssdev);
struct omap_dss_device *src = dssdev->src;
bool connected;
connected = gpiod_get_value_cansleep(ddata->hpd_gpio); return gpiod_get_value_cansleep(ddata->hpd_gpio);
if (!connected && src->ops->hdmi.lost_hotplug)
src->ops->hdmi.lost_hotplug(src);
return connected;
} }
static void hdmic_register_hpd_cb(struct omap_dss_device *dssdev, static void hdmic_register_hpd_cb(struct omap_dss_device *dssdev,
......
...@@ -130,13 +130,8 @@ static int tpd_read_edid(struct omap_dss_device *dssdev, ...@@ -130,13 +130,8 @@ static int tpd_read_edid(struct omap_dss_device *dssdev,
static bool tpd_detect(struct omap_dss_device *dssdev) static bool tpd_detect(struct omap_dss_device *dssdev)
{ {
struct panel_drv_data *ddata = to_panel_data(dssdev); struct panel_drv_data *ddata = to_panel_data(dssdev);
struct omap_dss_device *src = dssdev->src;
bool connected;
connected = gpiod_get_value_cansleep(ddata->hpd_gpio); return gpiod_get_value_cansleep(ddata->hpd_gpio);
if (!connected && src->ops->hdmi.lost_hotplug)
src->ops->hdmi.lost_hotplug(src);
return connected;
} }
static void tpd_register_hpd_cb(struct omap_dss_device *dssdev, static void tpd_register_hpd_cb(struct omap_dss_device *dssdev,
......
...@@ -34,6 +34,22 @@ struct omap_connector { ...@@ -34,6 +34,22 @@ struct omap_connector {
bool hdmi_mode; bool hdmi_mode;
}; };
static void omap_connector_hpd_notify(struct drm_connector *connector,
struct omap_dss_device *src,
enum drm_connector_status status)
{
if (status == connector_status_disconnected) {
/*
* If the source is an HDMI encoder, notify it of disconnection.
* This is required to let the HDMI encoder reset any internal
* state related to connection status, such as the CEC address.
*/
if (src && src->type == OMAP_DISPLAY_TYPE_HDMI &&
src->ops->hdmi.lost_hotplug)
src->ops->hdmi.lost_hotplug(src);
}
}
static void omap_connector_hpd_cb(void *cb_data, static void omap_connector_hpd_cb(void *cb_data,
enum drm_connector_status status) enum drm_connector_status status)
{ {
...@@ -47,7 +63,11 @@ static void omap_connector_hpd_cb(void *cb_data, ...@@ -47,7 +63,11 @@ static void omap_connector_hpd_cb(void *cb_data,
connector->status = status; connector->status = status;
mutex_unlock(&dev->mode_config.mutex); mutex_unlock(&dev->mode_config.mutex);
if (old_status != status) if (old_status == status)
return;
omap_connector_hpd_notify(connector, omap_connector->hpd, status);
drm_kms_helper_hotplug_event(dev); drm_kms_helper_hotplug_event(dev);
} }
...@@ -103,10 +123,11 @@ static enum drm_connector_status omap_connector_detect( ...@@ -103,10 +123,11 @@ static enum drm_connector_status omap_connector_detect(
OMAP_DSS_DEVICE_OP_DETECT); OMAP_DSS_DEVICE_OP_DETECT);
if (dssdev) { if (dssdev) {
if (dssdev->ops->detect(dssdev)) status = dssdev->ops->detect(dssdev)
status = connector_status_connected; ? connector_status_connected
else : connector_status_disconnected;
status = connector_status_disconnected;
omap_connector_hpd_notify(connector, dssdev->src, status);
} else { } else {
switch (omap_connector->dssdev->type) { switch (omap_connector->dssdev->type) {
case OMAP_DISPLAY_TYPE_DPI: case OMAP_DISPLAY_TYPE_DPI:
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册