|
@@ -17,6 +17,7 @@
|
|
|
*/
|
|
|
|
|
|
#include "linux/component.h"
|
|
|
+#include "linux/pm_runtime.h"
|
|
|
#include "vc4_drv.h"
|
|
|
#include "vc4_regs.h"
|
|
|
|
|
@@ -167,6 +168,29 @@ static void vc4_v3d_init_hw(struct drm_device *dev)
|
|
|
V3D_WRITE(V3D_VPMBASE, 0);
|
|
|
}
|
|
|
|
|
|
+#ifdef CONFIG_PM
|
|
|
+static int vc4_v3d_runtime_suspend(struct device *dev)
|
|
|
+{
|
|
|
+ struct vc4_v3d *v3d = dev_get_drvdata(dev);
|
|
|
+ struct vc4_dev *vc4 = v3d->vc4;
|
|
|
+
|
|
|
+ vc4_irq_uninstall(vc4->dev);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static int vc4_v3d_runtime_resume(struct device *dev)
|
|
|
+{
|
|
|
+ struct vc4_v3d *v3d = dev_get_drvdata(dev);
|
|
|
+ struct vc4_dev *vc4 = v3d->vc4;
|
|
|
+
|
|
|
+ vc4_v3d_init_hw(vc4->dev);
|
|
|
+ vc4_irq_postinstall(vc4->dev);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
static int vc4_v3d_bind(struct device *dev, struct device *master, void *data)
|
|
|
{
|
|
|
struct platform_device *pdev = to_platform_device(dev);
|
|
@@ -179,6 +203,8 @@ static int vc4_v3d_bind(struct device *dev, struct device *master, void *data)
|
|
|
if (!v3d)
|
|
|
return -ENOMEM;
|
|
|
|
|
|
+ dev_set_drvdata(dev, v3d);
|
|
|
+
|
|
|
v3d->pdev = pdev;
|
|
|
|
|
|
v3d->regs = vc4_ioremap_regs(pdev, 0);
|
|
@@ -186,6 +212,7 @@ static int vc4_v3d_bind(struct device *dev, struct device *master, void *data)
|
|
|
return PTR_ERR(v3d->regs);
|
|
|
|
|
|
vc4->v3d = v3d;
|
|
|
+ v3d->vc4 = vc4;
|
|
|
|
|
|
if (V3D_READ(V3D_IDENT0) != V3D_EXPECTED_IDENT0) {
|
|
|
DRM_ERROR("V3D_IDENT0 read 0x%08x instead of 0x%08x\n",
|
|
@@ -207,6 +234,8 @@ static int vc4_v3d_bind(struct device *dev, struct device *master, void *data)
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
+ pm_runtime_enable(dev);
|
|
|
+
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -216,6 +245,8 @@ static void vc4_v3d_unbind(struct device *dev, struct device *master,
|
|
|
struct drm_device *drm = dev_get_drvdata(master);
|
|
|
struct vc4_dev *vc4 = to_vc4_dev(drm);
|
|
|
|
|
|
+ pm_runtime_disable(dev);
|
|
|
+
|
|
|
drm_irq_uninstall(drm);
|
|
|
|
|
|
/* Disable the binner's overflow memory address, so the next
|
|
@@ -228,6 +259,10 @@ static void vc4_v3d_unbind(struct device *dev, struct device *master,
|
|
|
vc4->v3d = NULL;
|
|
|
}
|
|
|
|
|
|
+static const struct dev_pm_ops vc4_v3d_pm_ops = {
|
|
|
+ SET_RUNTIME_PM_OPS(vc4_v3d_runtime_suspend, vc4_v3d_runtime_resume, NULL)
|
|
|
+};
|
|
|
+
|
|
|
static const struct component_ops vc4_v3d_ops = {
|
|
|
.bind = vc4_v3d_bind,
|
|
|
.unbind = vc4_v3d_unbind,
|
|
@@ -255,5 +290,6 @@ struct platform_driver vc4_v3d_driver = {
|
|
|
.driver = {
|
|
|
.name = "vc4_v3d",
|
|
|
.of_match_table = vc4_v3d_dt_match,
|
|
|
+ .pm = &vc4_v3d_pm_ops,
|
|
|
},
|
|
|
};
|