Commit 18412b66 authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Tomi Valkeinen

drm/omap: Merge HPD enable operation with HPD callback registration

The omap_dss_device .enable_hpd() and .disable_hpd() are used to enable
and disable hot-plug detection at omapdrm probe and remove time. This is
required to avoid reporting hot-plug detection events before the DRM
infrastructure is ready to accept them, as that could result in crashes
or other malfunction.

Hot-plug event reporting is conditioned by both HPD being enabled
through the .enable_hpd() operation and by the HPD callback being
registered though the .register_hpd_cb() operation. We thus don't need a
separate enable operation if we can guarantee that callbacks won't be
registered too early.

HPD callbacks are registered at connector initialization time, which is
too early to start reporting HPD events. There's however nothing
blocking a move of callback registration to a later time when the
omapdrm driver calls the HPD enable operations. Do so, and remove the
HPD enable operation completely from omap_dss_device drivers.
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: default avatarSebastian Reichel <sebastian.reichel@collabora.co.uk>
Signed-off-by: default avatarTomi Valkeinen <tomi.valkeinen@ti.com>
parent a21a8f3c
...@@ -234,30 +234,6 @@ static void dvic_unregister_hpd_cb(struct omap_dss_device *dssdev) ...@@ -234,30 +234,6 @@ static void dvic_unregister_hpd_cb(struct omap_dss_device *dssdev)
mutex_unlock(&ddata->hpd_lock); mutex_unlock(&ddata->hpd_lock);
} }
static void dvic_enable_hpd(struct omap_dss_device *dssdev)
{
struct panel_drv_data *ddata = to_panel_data(dssdev);
if (!ddata->hpd_gpio)
return;
mutex_lock(&ddata->hpd_lock);
ddata->hpd_enabled = true;
mutex_unlock(&ddata->hpd_lock);
}
static void dvic_disable_hpd(struct omap_dss_device *dssdev)
{
struct panel_drv_data *ddata = to_panel_data(dssdev);
if (!ddata->hpd_gpio)
return;
mutex_lock(&ddata->hpd_lock);
ddata->hpd_enabled = false;
mutex_unlock(&ddata->hpd_lock);
}
static const struct omap_dss_device_ops dvic_ops = { static const struct omap_dss_device_ops dvic_ops = {
.connect = dvic_connect, .connect = dvic_connect,
.disconnect = dvic_disconnect, .disconnect = dvic_disconnect,
...@@ -274,8 +250,6 @@ static const struct omap_dss_device_ops dvic_ops = { ...@@ -274,8 +250,6 @@ static const struct omap_dss_device_ops dvic_ops = {
.register_hpd_cb = dvic_register_hpd_cb, .register_hpd_cb = dvic_register_hpd_cb,
.unregister_hpd_cb = dvic_unregister_hpd_cb, .unregister_hpd_cb = dvic_unregister_hpd_cb,
.enable_hpd = dvic_enable_hpd,
.disable_hpd = dvic_disable_hpd,
}; };
static irqreturn_t dvic_hpd_isr(int irq, void *data) static irqreturn_t dvic_hpd_isr(int irq, void *data)
......
...@@ -37,7 +37,6 @@ struct panel_drv_data { ...@@ -37,7 +37,6 @@ struct panel_drv_data {
struct omap_dss_device dssdev; struct omap_dss_device dssdev;
void (*hpd_cb)(void *cb_data, enum drm_connector_status status); void (*hpd_cb)(void *cb_data, enum drm_connector_status status);
void *hpd_cb_data; void *hpd_cb_data;
bool hpd_enabled;
struct mutex hpd_lock; struct mutex hpd_lock;
struct device *dev; struct device *dev;
...@@ -170,30 +169,6 @@ static void hdmic_unregister_hpd_cb(struct omap_dss_device *dssdev) ...@@ -170,30 +169,6 @@ static void hdmic_unregister_hpd_cb(struct omap_dss_device *dssdev)
mutex_unlock(&ddata->hpd_lock); mutex_unlock(&ddata->hpd_lock);
} }
static void hdmic_enable_hpd(struct omap_dss_device *dssdev)
{
struct panel_drv_data *ddata = to_panel_data(dssdev);
if (!ddata->hpd_gpio)
return;
mutex_lock(&ddata->hpd_lock);
ddata->hpd_enabled = true;
mutex_unlock(&ddata->hpd_lock);
}
static void hdmic_disable_hpd(struct omap_dss_device *dssdev)
{
struct panel_drv_data *ddata = to_panel_data(dssdev);
if (!ddata->hpd_gpio)
return;
mutex_lock(&ddata->hpd_lock);
ddata->hpd_enabled = false;
mutex_unlock(&ddata->hpd_lock);
}
static int hdmic_set_hdmi_mode(struct omap_dss_device *dssdev, bool hdmi_mode) static int hdmic_set_hdmi_mode(struct omap_dss_device *dssdev, bool hdmi_mode)
{ {
struct omap_dss_device *src = dssdev->src; struct omap_dss_device *src = dssdev->src;
...@@ -224,8 +199,6 @@ static const struct omap_dss_device_ops hdmic_ops = { ...@@ -224,8 +199,6 @@ static const struct omap_dss_device_ops hdmic_ops = {
.detect = hdmic_detect, .detect = hdmic_detect,
.register_hpd_cb = hdmic_register_hpd_cb, .register_hpd_cb = hdmic_register_hpd_cb,
.unregister_hpd_cb = hdmic_unregister_hpd_cb, .unregister_hpd_cb = hdmic_unregister_hpd_cb,
.enable_hpd = hdmic_enable_hpd,
.disable_hpd = hdmic_disable_hpd,
.hdmi = { .hdmi = {
.set_hdmi_mode = hdmic_set_hdmi_mode, .set_hdmi_mode = hdmic_set_hdmi_mode,
...@@ -238,7 +211,7 @@ static irqreturn_t hdmic_hpd_isr(int irq, void *data) ...@@ -238,7 +211,7 @@ static irqreturn_t hdmic_hpd_isr(int irq, void *data)
struct panel_drv_data *ddata = data; struct panel_drv_data *ddata = data;
mutex_lock(&ddata->hpd_lock); mutex_lock(&ddata->hpd_lock);
if (ddata->hpd_enabled && ddata->hpd_cb) { if (ddata->hpd_cb) {
enum drm_connector_status status; enum drm_connector_status status;
if (hdmic_detect(&ddata->dssdev)) if (hdmic_detect(&ddata->dssdev))
......
...@@ -23,7 +23,6 @@ struct panel_drv_data { ...@@ -23,7 +23,6 @@ struct panel_drv_data {
struct omap_dss_device dssdev; struct omap_dss_device dssdev;
void (*hpd_cb)(void *cb_data, enum drm_connector_status status); void (*hpd_cb)(void *cb_data, enum drm_connector_status status);
void *hpd_cb_data; void *hpd_cb_data;
bool hpd_enabled;
struct mutex hpd_lock; struct mutex hpd_lock;
struct gpio_desc *ct_cp_hpd_gpio; struct gpio_desc *ct_cp_hpd_gpio;
...@@ -163,24 +162,6 @@ static void tpd_unregister_hpd_cb(struct omap_dss_device *dssdev) ...@@ -163,24 +162,6 @@ static void tpd_unregister_hpd_cb(struct omap_dss_device *dssdev)
mutex_unlock(&ddata->hpd_lock); mutex_unlock(&ddata->hpd_lock);
} }
static void tpd_enable_hpd(struct omap_dss_device *dssdev)
{
struct panel_drv_data *ddata = to_panel_data(dssdev);
mutex_lock(&ddata->hpd_lock);
ddata->hpd_enabled = true;
mutex_unlock(&ddata->hpd_lock);
}
static void tpd_disable_hpd(struct omap_dss_device *dssdev)
{
struct panel_drv_data *ddata = to_panel_data(dssdev);
mutex_lock(&ddata->hpd_lock);
ddata->hpd_enabled = false;
mutex_unlock(&ddata->hpd_lock);
}
static int tpd_set_infoframe(struct omap_dss_device *dssdev, static int tpd_set_infoframe(struct omap_dss_device *dssdev,
const struct hdmi_avi_infoframe *avi) const struct hdmi_avi_infoframe *avi)
{ {
...@@ -208,8 +189,6 @@ static const struct omap_dss_device_ops tpd_ops = { ...@@ -208,8 +189,6 @@ static const struct omap_dss_device_ops tpd_ops = {
.detect = tpd_detect, .detect = tpd_detect,
.register_hpd_cb = tpd_register_hpd_cb, .register_hpd_cb = tpd_register_hpd_cb,
.unregister_hpd_cb = tpd_unregister_hpd_cb, .unregister_hpd_cb = tpd_unregister_hpd_cb,
.enable_hpd = tpd_enable_hpd,
.disable_hpd = tpd_disable_hpd,
.hdmi = { .hdmi = {
.set_infoframe = tpd_set_infoframe, .set_infoframe = tpd_set_infoframe,
...@@ -222,7 +201,7 @@ static irqreturn_t tpd_hpd_isr(int irq, void *data) ...@@ -222,7 +201,7 @@ static irqreturn_t tpd_hpd_isr(int irq, void *data)
struct panel_drv_data *ddata = data; struct panel_drv_data *ddata = data;
mutex_lock(&ddata->hpd_lock); mutex_lock(&ddata->hpd_lock);
if (ddata->hpd_enabled && ddata->hpd_cb) { if (ddata->hpd_cb) {
enum drm_connector_status status; enum drm_connector_status status;
if (tpd_detect(&ddata->dssdev)) if (tpd_detect(&ddata->dssdev))
......
...@@ -377,8 +377,6 @@ struct omap_dss_device_ops { ...@@ -377,8 +377,6 @@ struct omap_dss_device_ops {
enum drm_connector_status status), enum drm_connector_status status),
void *cb_data); void *cb_data);
void (*unregister_hpd_cb)(struct omap_dss_device *dssdev); void (*unregister_hpd_cb)(struct omap_dss_device *dssdev);
void (*enable_hpd)(struct omap_dss_device *dssdev);
void (*disable_hpd)(struct omap_dss_device *dssdev);
int (*read_edid)(struct omap_dss_device *dssdev, u8 *buf, int len); int (*read_edid)(struct omap_dss_device *dssdev, u8 *buf, int len);
......
...@@ -30,6 +30,7 @@ ...@@ -30,6 +30,7 @@
struct omap_connector { struct omap_connector {
struct drm_connector base; struct drm_connector base;
struct omap_dss_device *dssdev; struct omap_dss_device *dssdev;
struct omap_dss_device *hpd;
bool hdmi_mode; bool hdmi_mode;
}; };
...@@ -50,6 +51,25 @@ static void omap_connector_hpd_cb(void *cb_data, ...@@ -50,6 +51,25 @@ static void omap_connector_hpd_cb(void *cb_data,
drm_kms_helper_hotplug_event(dev); drm_kms_helper_hotplug_event(dev);
} }
void omap_connector_enable_hpd(struct drm_connector *connector)
{
struct omap_connector *omap_connector = to_omap_connector(connector);
struct omap_dss_device *hpd = omap_connector->hpd;
if (hpd)
hpd->ops->register_hpd_cb(hpd, omap_connector_hpd_cb,
omap_connector);
}
void omap_connector_disable_hpd(struct drm_connector *connector)
{
struct omap_connector *omap_connector = to_omap_connector(connector);
struct omap_dss_device *hpd = omap_connector->hpd;
if (hpd)
hpd->ops->unregister_hpd_cb(hpd);
}
bool omap_connector_get_hdmi_mode(struct drm_connector *connector) bool omap_connector_get_hdmi_mode(struct drm_connector *connector)
{ {
struct omap_connector *omap_connector = to_omap_connector(connector); struct omap_connector *omap_connector = to_omap_connector(connector);
...@@ -109,14 +129,15 @@ static enum drm_connector_status omap_connector_detect( ...@@ -109,14 +129,15 @@ static enum drm_connector_status omap_connector_detect(
static void omap_connector_destroy(struct drm_connector *connector) static void omap_connector_destroy(struct drm_connector *connector)
{ {
struct omap_connector *omap_connector = to_omap_connector(connector); struct omap_connector *omap_connector = to_omap_connector(connector);
struct omap_dss_device *dssdev;
DBG("%s", omap_connector->dssdev->name); DBG("%s", omap_connector->dssdev->name);
if (connector->polled == DRM_CONNECTOR_POLL_HPD) { if (omap_connector->hpd) {
dssdev = omap_connector_find_device(connector, struct omap_dss_device *hpd = omap_connector->hpd;
OMAP_DSS_DEVICE_OP_HPD);
dssdev->ops->unregister_hpd_cb(dssdev); hpd->ops->unregister_hpd_cb(hpd);
omapdss_device_put(hpd);
omap_connector->hpd = NULL;
} }
drm_connector_unregister(connector); drm_connector_unregister(connector);
...@@ -298,8 +319,7 @@ struct drm_connector *omap_connector_init(struct drm_device *dev, ...@@ -298,8 +319,7 @@ struct drm_connector *omap_connector_init(struct drm_device *dev,
*/ */
dssdev = omap_connector_find_device(connector, OMAP_DSS_DEVICE_OP_HPD); dssdev = omap_connector_find_device(connector, OMAP_DSS_DEVICE_OP_HPD);
if (dssdev) { if (dssdev) {
dssdev->ops->register_hpd_cb(dssdev, omap_connector_hpd_cb, omap_connector->hpd = omapdss_device_get(dssdev);
omap_connector);
connector->polled = DRM_CONNECTOR_POLL_HPD; connector->polled = DRM_CONNECTOR_POLL_HPD;
} else { } else {
dssdev = omap_connector_find_device(connector, dssdev = omap_connector_find_device(connector,
......
...@@ -33,5 +33,7 @@ struct drm_connector *omap_connector_init(struct drm_device *dev, ...@@ -33,5 +33,7 @@ struct drm_connector *omap_connector_init(struct drm_device *dev,
struct drm_encoder *omap_connector_attached_encoder( struct drm_encoder *omap_connector_attached_encoder(
struct drm_connector *connector); struct drm_connector *connector);
bool omap_connector_get_hdmi_mode(struct drm_connector *connector); bool omap_connector_get_hdmi_mode(struct drm_connector *connector);
void omap_connector_enable_hpd(struct drm_connector *connector);
void omap_connector_disable_hpd(struct drm_connector *connector);
#endif /* __OMAPDRM_CONNECTOR_H__ */ #endif /* __OMAPDRM_CONNECTOR_H__ */
...@@ -375,12 +375,8 @@ static void omap_modeset_enable_external_hpd(struct drm_device *ddev) ...@@ -375,12 +375,8 @@ static void omap_modeset_enable_external_hpd(struct drm_device *ddev)
struct omap_drm_private *priv = ddev->dev_private; struct omap_drm_private *priv = ddev->dev_private;
int i; int i;
for (i = 0; i < priv->num_pipes; i++) { for (i = 0; i < priv->num_pipes; i++)
struct omap_dss_device *display = priv->pipes[i].display; omap_connector_enable_hpd(priv->pipes[i].connector);
if (display->ops->enable_hpd)
display->ops->enable_hpd(display);
}
} }
/* /*
...@@ -391,12 +387,8 @@ static void omap_modeset_disable_external_hpd(struct drm_device *ddev) ...@@ -391,12 +387,8 @@ static void omap_modeset_disable_external_hpd(struct drm_device *ddev)
struct omap_drm_private *priv = ddev->dev_private; struct omap_drm_private *priv = ddev->dev_private;
int i; int i;
for (i = 0; i < priv->num_pipes; i++) { for (i = 0; i < priv->num_pipes; i++)
struct omap_dss_device *display = priv->pipes[i].display; omap_connector_disable_hpd(priv->pipes[i].connector);
if (display->ops->disable_hpd)
display->ops->disable_hpd(display);
}
} }
/* /*
......
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