Move ULPS handling into the DSI host controller, so that we
no longer need a custom API for the DSI client.

Signed-off-by: Sebastian Reichel <sebastian.reic...@collabora.com>
---
 .../gpu/drm/omapdrm/displays/panel-dsi-cm.c   | 273 +-----------------
 drivers/gpu/drm/omapdrm/dss/dsi.c             |  61 +++-
 drivers/gpu/drm/omapdrm/dss/omapdss.h         |   2 -
 3 files changed, 62 insertions(+), 274 deletions(-)

diff --git a/drivers/gpu/drm/omapdrm/displays/panel-dsi-cm.c 
b/drivers/gpu/drm/omapdrm/displays/panel-dsi-cm.c
index 239a48910db7..1e33cf7c1448 100644
--- a/drivers/gpu/drm/omapdrm/displays/panel-dsi-cm.c
+++ b/drivers/gpu/drm/omapdrm/displays/panel-dsi-cm.c
@@ -16,7 +16,6 @@
 #include <linux/module.h>
 #include <linux/sched/signal.h>
 #include <linux/slab.h>
-#include <linux/workqueue.h>
 #include <linux/of_device.h>
 #include <linux/regulator/consumer.h>
 
@@ -69,21 +68,13 @@ struct panel_drv_data {
 
        bool intro_printed;
 
-       struct workqueue_struct *workqueue;
-
        bool ulps_enabled;
-       unsigned int ulps_timeout;
-       struct delayed_work ulps_work;
 };
 
 #define to_panel_data(p) container_of(p, struct panel_drv_data, dssdev)
 
 static int _dsicm_enable_te(struct panel_drv_data *ddata, bool enable);
 
-static int dsicm_panel_reset(struct panel_drv_data *ddata);
-
-static void dsicm_ulps_work(struct work_struct *work);
-
 static void dsicm_bl_power(struct panel_drv_data *ddata, bool enable)
 {
        struct backlight_device *backlight;
@@ -207,94 +198,6 @@ static int dsicm_set_update_window(struct panel_drv_data 
*ddata,
        return 0;
 }
 
-static void dsicm_queue_ulps_work(struct panel_drv_data *ddata)
-{
-       if (ddata->ulps_timeout > 0)
-               queue_delayed_work(ddata->workqueue, &ddata->ulps_work,
-                               msecs_to_jiffies(ddata->ulps_timeout));
-}
-
-static void dsicm_cancel_ulps_work(struct panel_drv_data *ddata)
-{
-       cancel_delayed_work(&ddata->ulps_work);
-}
-
-static int dsicm_enter_ulps(struct panel_drv_data *ddata)
-{
-       struct omap_dss_device *src = ddata->src;
-       int r;
-
-       if (ddata->ulps_enabled)
-               return 0;
-
-       dsicm_cancel_ulps_work(ddata);
-
-       r = _dsicm_enable_te(ddata, false);
-       if (r)
-               goto err;
-
-       src->ops->dsi.ulps(src, true);
-
-       ddata->ulps_enabled = true;
-
-       return 0;
-
-err:
-       dev_err(&ddata->dsi->dev, "enter ULPS failed");
-       dsicm_panel_reset(ddata);
-
-       ddata->ulps_enabled = false;
-
-       dsicm_queue_ulps_work(ddata);
-
-       return r;
-}
-
-static int dsicm_exit_ulps(struct panel_drv_data *ddata)
-{
-       struct omap_dss_device *src = ddata->src;
-       int r;
-
-       if (!ddata->ulps_enabled)
-               return 0;
-
-       src->ops->dsi.ulps(src, false);
-       ddata->dsi->mode_flags &= ~MIPI_DSI_MODE_LPM;
-
-       r = _dsicm_enable_te(ddata, ddata->te_enabled);
-       if (r) {
-               dev_err(&ddata->dsi->dev, "failed to re-enable TE");
-               goto err2;
-       }
-
-       dsicm_queue_ulps_work(ddata);
-
-       ddata->ulps_enabled = false;
-
-       return 0;
-
-err2:
-       dev_err(&ddata->dsi->dev, "failed to exit ULPS");
-
-       r = dsicm_panel_reset(ddata);
-       if (!r)
-               ddata->ulps_enabled = false;
-
-       dsicm_queue_ulps_work(ddata);
-
-       return r;
-}
-
-static int dsicm_wake_up(struct panel_drv_data *ddata)
-{
-       if (ddata->ulps_enabled)
-               return dsicm_exit_ulps(ddata);
-
-       dsicm_cancel_ulps_work(ddata);
-       dsicm_queue_ulps_work(ddata);
-       return 0;
-}
-
 static int dsicm_bl_update_status(struct backlight_device *dev)
 {
        struct panel_drv_data *ddata = dev_get_drvdata(&dev->dev);
@@ -312,11 +215,8 @@ static int dsicm_bl_update_status(struct backlight_device 
*dev)
        mutex_lock(&ddata->lock);
 
        if (ddata->enabled) {
-               r = dsicm_wake_up(ddata);
-               if (!r) {
-                       r = dsicm_dcs_write_1(ddata,
-                               MIPI_DCS_SET_DISPLAY_BRIGHTNESS, level);
-               }
+               r = dsicm_dcs_write_1(ddata, MIPI_DCS_SET_DISPLAY_BRIGHTNESS,
+                                     level);
        }
 
        mutex_unlock(&ddata->lock);
@@ -343,18 +243,12 @@ static ssize_t dsicm_num_errors_show(struct device *dev,
 {
        struct panel_drv_data *ddata = dev_get_drvdata(dev);
        u8 errors = 0;
-       int r;
+       int r = -ENODEV;
 
        mutex_lock(&ddata->lock);
 
-       if (ddata->enabled) {
-               r = dsicm_wake_up(ddata);
-               if (!r)
-                       r = dsicm_dcs_read_1(ddata, DCS_READ_NUM_ERRORS,
-                                       &errors);
-       } else {
-               r = -ENODEV;
-       }
+       if (ddata->enabled)
+               r = dsicm_dcs_read_1(ddata, DCS_READ_NUM_ERRORS, &errors);
 
        mutex_unlock(&ddata->lock);
 
@@ -369,17 +263,12 @@ static ssize_t dsicm_hw_revision_show(struct device *dev,
 {
        struct panel_drv_data *ddata = dev_get_drvdata(dev);
        u8 id1, id2, id3;
-       int r;
+       int r = -ENODEV;
 
        mutex_lock(&ddata->lock);
 
-       if (ddata->enabled) {
-               r = dsicm_wake_up(ddata);
-               if (!r)
-                       r = dsicm_get_id(ddata, &id1, &id2, &id3);
-       } else {
-               r = -ENODEV;
-       }
+       if (ddata->enabled)
+               r = dsicm_get_id(ddata, &id1, &id2, &id3);
 
        mutex_unlock(&ddata->lock);
 
@@ -389,103 +278,12 @@ static ssize_t dsicm_hw_revision_show(struct device *dev,
        return snprintf(buf, PAGE_SIZE, "%02x.%02x.%02x\n", id1, id2, id3);
 }
 
-static ssize_t dsicm_store_ulps(struct device *dev,
-               struct device_attribute *attr,
-               const char *buf, size_t count)
-{
-       struct panel_drv_data *ddata = dev_get_drvdata(dev);
-       unsigned long t;
-       int r;
-
-       r = kstrtoul(buf, 0, &t);
-       if (r)
-               return r;
-
-       mutex_lock(&ddata->lock);
-
-       if (ddata->enabled) {
-               if (t)
-                       r = dsicm_enter_ulps(ddata);
-               else
-                       r = dsicm_wake_up(ddata);
-       }
-
-       mutex_unlock(&ddata->lock);
-
-       if (r)
-               return r;
-
-       return count;
-}
-
-static ssize_t dsicm_show_ulps(struct device *dev,
-               struct device_attribute *attr,
-               char *buf)
-{
-       struct panel_drv_data *ddata = dev_get_drvdata(dev);
-       unsigned int t;
-
-       mutex_lock(&ddata->lock);
-       t = ddata->ulps_enabled;
-       mutex_unlock(&ddata->lock);
-
-       return snprintf(buf, PAGE_SIZE, "%u\n", t);
-}
-
-static ssize_t dsicm_store_ulps_timeout(struct device *dev,
-               struct device_attribute *attr,
-               const char *buf, size_t count)
-{
-       struct panel_drv_data *ddata = dev_get_drvdata(dev);
-       unsigned long t;
-       int r;
-
-       r = kstrtoul(buf, 0, &t);
-       if (r)
-               return r;
-
-       mutex_lock(&ddata->lock);
-       ddata->ulps_timeout = t;
-
-       if (ddata->enabled) {
-               /* dsicm_wake_up will restart the timer */
-               r = dsicm_wake_up(ddata);
-       }
-
-       mutex_unlock(&ddata->lock);
-
-       if (r)
-               return r;
-
-       return count;
-}
-
-static ssize_t dsicm_show_ulps_timeout(struct device *dev,
-               struct device_attribute *attr,
-               char *buf)
-{
-       struct panel_drv_data *ddata = dev_get_drvdata(dev);
-       unsigned int t;
-
-       mutex_lock(&ddata->lock);
-       t = ddata->ulps_timeout;
-       mutex_unlock(&ddata->lock);
-
-       return snprintf(buf, PAGE_SIZE, "%u\n", t);
-}
-
 static DEVICE_ATTR(num_dsi_errors, S_IRUGO, dsicm_num_errors_show, NULL);
 static DEVICE_ATTR(hw_revision, S_IRUGO, dsicm_hw_revision_show, NULL);
-static DEVICE_ATTR(ulps, S_IRUGO | S_IWUSR,
-               dsicm_show_ulps, dsicm_store_ulps);
-static DEVICE_ATTR(ulps_timeout, S_IRUGO | S_IWUSR,
-               dsicm_show_ulps_timeout, dsicm_store_ulps_timeout);
 
 static struct attribute *dsicm_attrs[] = {
        &dev_attr_num_dsi_errors.attr,
        &dev_attr_hw_revision.attr,
-       &dev_attr_ulps.attr,
-       &dev_attr_ulps_timeout.attr,
        NULL,
 };
 
@@ -621,15 +419,6 @@ static void dsicm_power_off(struct panel_drv_data *ddata)
        ddata->enabled = false;
 }
 
-static int dsicm_panel_reset(struct panel_drv_data *ddata)
-{
-       dev_err(&ddata->dsi->dev, "performing LCD reset\n");
-
-       dsicm_power_off(ddata);
-       dsicm_hw_reset(ddata);
-       return dsicm_power_on(ddata);
-}
-
 static int dsicm_connect(struct omap_dss_device *src,
                         struct omap_dss_device *dst)
 {
@@ -671,17 +460,12 @@ static void dsicm_enable(struct omap_dss_device *dssdev)
 static void dsicm_disable(struct omap_dss_device *dssdev)
 {
        struct panel_drv_data *ddata = to_panel_data(dssdev);
-       int r;
 
        dsicm_bl_power(ddata, false);
 
        mutex_lock(&ddata->lock);
 
-       dsicm_cancel_ulps_work(ddata);
-
-       r = dsicm_wake_up(ddata);
-       if (!r)
-               dsicm_power_off(ddata);
+       dsicm_power_off(ddata);
 
        mutex_unlock(&ddata->lock);
 }
@@ -705,10 +489,6 @@ static int dsicm_update(struct omap_dss_device *dssdev,
 
        mutex_lock(&ddata->lock);
 
-       r = dsicm_wake_up(ddata);
-       if (r)
-               goto err;
-
        if (!ddata->enabled) {
                r = 0;
                goto err;
@@ -748,24 +528,6 @@ static int _dsicm_enable_te(struct panel_drv_data *ddata, 
bool enable)
        return r;
 }
 
-static void dsicm_ulps_work(struct work_struct *work)
-{
-       struct panel_drv_data *ddata = container_of(work, struct panel_drv_data,
-                       ulps_work.work);
-       struct omap_dss_device *dssdev = &ddata->dssdev;
-
-       mutex_lock(&ddata->lock);
-
-       if (dssdev->state != OMAP_DSS_DISPLAY_ACTIVE || !ddata->enabled) {
-               mutex_unlock(&ddata->lock);
-               return;
-       }
-
-       dsicm_enter_ulps(ddata);
-
-       mutex_unlock(&ddata->lock);
-}
-
 static int dsicm_get_modes(struct omap_dss_device *dssdev,
                           struct drm_connector *connector)
 {
@@ -865,7 +627,7 @@ static int dsicm_probe_of(struct mipi_dsi_device *dsi)
                ddata->use_dsi_backlight = true;
        }
 
-       /* TODO: ulps */
+       /* TODO: ulps_enabled */
 
        return 0;
 }
@@ -913,13 +675,6 @@ static int dsicm_probe(struct mipi_dsi_device *dsi)
 
        mutex_init(&ddata->lock);
 
-       ddata->workqueue = create_singlethread_workqueue("dsicm_wq");
-       if (!ddata->workqueue) {
-               r = -ENOMEM;
-               goto err_reg;
-       }
-       INIT_DELAYED_WORK(&ddata->ulps_work, dsicm_ulps_work);
-
        dsicm_hw_reset(ddata);
 
        if (ddata->use_dsi_backlight) {
@@ -950,6 +705,9 @@ static int dsicm_probe(struct mipi_dsi_device *dsi)
        dsi->hs_rate = 300000000;
        dsi->lp_rate = 10000000;
 
+       if (ddata->ulps_enabled)
+               dsi->mode_flags |= MIPI_DSI_MODE_ULPS_IDLE;
+
        r = mipi_dsi_attach(dsi);
        if (r < 0)
                goto err_dsi_attach;
@@ -959,8 +717,6 @@ static int dsicm_probe(struct mipi_dsi_device *dsi)
 err_dsi_attach:
        sysfs_remove_group(&dsi->dev.kobj, &dsicm_attr_group);
 err_bl:
-       destroy_workqueue(ddata->workqueue);
-err_reg:
        if (ddata->extbldev)
                put_device(&ddata->extbldev->dev);
 
@@ -987,9 +743,6 @@ static int __exit dsicm_remove(struct mipi_dsi_device *dsi)
        if (ddata->extbldev)
                put_device(&ddata->extbldev->dev);
 
-       dsicm_cancel_ulps_work(ddata);
-       destroy_workqueue(ddata->workqueue);
-
        /* reset, to be sure that the panel is in a valid state */
        dsicm_hw_reset(ddata);
 
diff --git a/drivers/gpu/drm/omapdrm/dss/dsi.c 
b/drivers/gpu/drm/omapdrm/dss/dsi.c
index 684d52079e89..12d661ad37ad 100644
--- a/drivers/gpu/drm/omapdrm/dss/dsi.c
+++ b/drivers/gpu/drm/omapdrm/dss/dsi.c
@@ -210,6 +210,8 @@ struct dsi_reg { u16 module; u16 idx; };
 typedef void (*omap_dsi_isr_t) (void *arg, u32 mask);
 struct dsi_data;
 
+static void dsi_set_ulps_auto(struct dsi_data *dsi, bool enable);
+
 static int dsi_display_init_dispc(struct dsi_data *dsi);
 static void dsi_display_uninit_dispc(struct dsi_data *dsi);
 
@@ -380,6 +382,9 @@ struct dsi_data {
 
        bool te_enabled;
        bool ulps_enabled;
+       bool ulps_auto_idle;
+
+       struct delayed_work ulps_work;
 
        void (*framedone_callback)(int, void *);
        void *framedone_data;
@@ -3801,6 +3806,7 @@ static void dsi_handle_framedone(struct dsi_data *dsi, 
int error)
                REG_FLD_MOD(dsi, DSI_TIMING2, 1, 15, 15); /* LP_RX_TO */
        }
 
+       dsi_set_ulps_auto(dsi, true);
        dsi_bus_unlock(dsi);
 
        dsi->framedone_callback(error, dsi->framedone_data);
@@ -3863,6 +3869,7 @@ static int dsi_update(struct omap_dss_device *dssdev, int 
channel,
        struct dsi_data *dsi = to_dsi_data(dssdev);
 
        dsi_bus_lock(dsi);
+       dsi_set_ulps_auto(dsi, false);
 
        dsi->update_channel = channel;
        dsi->framedone_callback = callback;
@@ -4124,16 +4131,6 @@ static void dsi_display_disable(struct omap_dss_device 
*dssdev)
        dsi_display_ulps_disable(dsi, true, false);
 }
 
-static void dsi_ulps(struct omap_dss_device *dssdev, bool enable)
-{
-       struct dsi_data *dsi = to_dsi_data(dssdev);
-       DSSDBG("dsi_ulps\n");
-       if (enable)
-               dsi_display_ulps_disable(dsi, false, true);
-       else
-               dsi_display_ulps_enable(dsi);
-}
-
 static int dsi_enable_te(struct dsi_data *dsi, bool enable)
 {
        dsi->te_enabled = enable;
@@ -4148,6 +4145,40 @@ static int dsi_enable_te(struct dsi_data *dsi, bool 
enable)
        return 0;
 }
 
+static void omap_dsi_ulps_work_callback(struct work_struct *work)
+{
+       struct dsi_data *dsi = container_of(work, struct dsi_data,
+                                           ulps_work.work);
+
+       dsi_bus_lock(dsi);
+
+       dsi_enable_te(dsi, false);
+
+       dsi_display_ulps_disable(dsi, false, true);
+
+       dsi_bus_unlock(dsi);
+}
+
+static void dsi_set_ulps_auto(struct dsi_data *dsi, bool enable)
+{
+       WARN_ON(!dsi_bus_is_locked(dsi));
+
+       if (!dsi->ulps_auto_idle)
+               return;
+
+       if (enable) {
+               schedule_delayed_work(&dsi->ulps_work, msecs_to_jiffies(250));
+       } else {
+               cancel_delayed_work_sync(&dsi->ulps_work);
+
+               if (!dsi->ulps_enabled)
+                       return;
+
+               dsi_display_ulps_enable(dsi);
+               dsi_enable_te(dsi, true);
+       }
+}
+
 #ifdef PRINT_VERBOSE_VM_TIMINGS
 static void print_dsi_vm(const char *str,
                const struct omap_dss_dsi_videomode_timings *t)
@@ -4798,7 +4829,9 @@ static ssize_t omap_dsi_host_transfer(struct 
mipi_dsi_host *host,
        int r;
 
        dsi_bus_lock(dsi);
+       dsi_set_ulps_auto(dsi, false);
        r = _omap_dsi_host_transfer(dsi, msg);
+       dsi_set_ulps_auto(dsi, true);
        dsi_bus_unlock(dsi);
 
        return r;
@@ -4838,8 +4871,6 @@ static const struct omap_dss_device_ops dsi_ops = {
        .disable = dsi_display_disable,
 
        .dsi = {
-               .ulps = dsi_ulps,
-
                .set_config = dsi_set_config,
 
                .enable_video_output = dsi_enable_video_output,
@@ -4958,6 +4989,12 @@ int omap_dsi_host_attach(struct mipi_dsi_host *host,
        dsi->vc[channel].dest = client;
        dsi->pix_fmt = client->format;
 
+       INIT_DEFERRABLE_WORK(&dsi->ulps_work,
+                            omap_dsi_ulps_work_callback);
+
+       dsi->ulps_auto_idle = !!(client->mode_flags & MIPI_DSI_MODE_ULPS_IDLE);
+       dsi_set_ulps_auto(dsi, true);
+
        return 0;
 }
 
diff --git a/drivers/gpu/drm/omapdrm/dss/omapdss.h 
b/drivers/gpu/drm/omapdrm/dss/omapdss.h
index 484bdb01871f..e2643623b5f2 100644
--- a/drivers/gpu/drm/omapdrm/dss/omapdss.h
+++ b/drivers/gpu/drm/omapdrm/dss/omapdss.h
@@ -281,8 +281,6 @@ struct omap_dss_writeback_info {
 
 struct omapdss_dsi_ops {
        /* bus configuration */
-       void (*ulps)(struct omap_dss_device *dssdev, bool enable);
-
        int (*set_config)(struct omap_dss_device *dssdev,
                        const struct omap_dss_dsi_config *cfg);
 
-- 
2.25.0

_______________________________________________
dri-devel mailing list
dri-devel@lists.freedesktop.org
https://lists.freedesktop.org/mailman/listinfo/dri-devel

Reply via email to