|
@@ -25,6 +25,7 @@
|
|
|
#include <linux/module.h>
|
|
|
#include <linux/platform_device.h>
|
|
|
#include <linux/slab.h>
|
|
|
+#include <linux/pm_runtime.h>
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
|
|
#include <media/soc_camera.h>
|
|
@@ -388,6 +389,11 @@ static int soc_camera_open(struct file *file)
|
|
|
goto eiciadd;
|
|
|
}
|
|
|
|
|
|
+ pm_runtime_enable(&icd->vdev->dev);
|
|
|
+ ret = pm_runtime_resume(&icd->vdev->dev);
|
|
|
+ if (ret < 0 && ret != -ENOSYS)
|
|
|
+ goto eresume;
|
|
|
+
|
|
|
/*
|
|
|
* Try to configure with default parameters. Notice: this is the
|
|
|
* very first open, so, we cannot race against other calls,
|
|
@@ -409,10 +415,12 @@ static int soc_camera_open(struct file *file)
|
|
|
return 0;
|
|
|
|
|
|
/*
|
|
|
- * First five errors are entered with the .video_lock held
|
|
|
+ * First four errors are entered with the .video_lock held
|
|
|
* and use_count == 1
|
|
|
*/
|
|
|
esfmt:
|
|
|
+ pm_runtime_disable(&icd->vdev->dev);
|
|
|
+eresume:
|
|
|
ici->ops->remove(icd);
|
|
|
eiciadd:
|
|
|
if (icl->power)
|
|
@@ -437,7 +445,11 @@ static int soc_camera_close(struct file *file)
|
|
|
if (!icd->use_count) {
|
|
|
struct soc_camera_link *icl = to_soc_camera_link(icd);
|
|
|
|
|
|
+ pm_runtime_suspend(&icd->vdev->dev);
|
|
|
+ pm_runtime_disable(&icd->vdev->dev);
|
|
|
+
|
|
|
ici->ops->remove(icd);
|
|
|
+
|
|
|
if (icl->power)
|
|
|
icl->power(icd->pdev, 0);
|
|
|
}
|
|
@@ -1319,6 +1331,7 @@ static int video_dev_create(struct soc_camera_device *icd)
|
|
|
*/
|
|
|
static int soc_camera_video_start(struct soc_camera_device *icd)
|
|
|
{
|
|
|
+ struct device_type *type = icd->vdev->dev.type;
|
|
|
int ret;
|
|
|
|
|
|
if (!icd->dev.parent)
|
|
@@ -1335,6 +1348,9 @@ static int soc_camera_video_start(struct soc_camera_device *icd)
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
+ /* Restore device type, possibly set by the subdevice driver */
|
|
|
+ icd->vdev->dev.type = type;
|
|
|
+
|
|
|
return 0;
|
|
|
}
|
|
|
|