Commit 152e0bf6 authored by Hugues Fruchet's avatar Hugues Fruchet Committed by Mauro Carvalho Chehab

media: stm32-dcmi: add power saving support

Implements runtime & system sleep power management ops.
Signed-off-by: default avatarHugues Fruchet <hugues.fruchet@st.com>
Signed-off-by: default avatarHans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+samsung@kernel.org>
parent f11552d0
...@@ -22,7 +22,9 @@ ...@@ -22,7 +22,9 @@
#include <linux/of.h> #include <linux/of.h>
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/of_graph.h> #include <linux/of_graph.h>
#include <linux/pinctrl/consumer.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/pm_runtime.h>
#include <linux/reset.h> #include <linux/reset.h>
#include <linux/videodev2.h> #include <linux/videodev2.h>
...@@ -567,9 +569,9 @@ static int dcmi_start_streaming(struct vb2_queue *vq, unsigned int count) ...@@ -567,9 +569,9 @@ static int dcmi_start_streaming(struct vb2_queue *vq, unsigned int count)
u32 val = 0; u32 val = 0;
int ret; int ret;
ret = clk_enable(dcmi->mclk); ret = pm_runtime_get_sync(dcmi->dev);
if (ret) { if (ret) {
dev_err(dcmi->dev, "%s: Failed to start streaming, cannot enable clock\n", dev_err(dcmi->dev, "%s: Failed to start streaming, cannot get sync\n",
__func__); __func__);
goto err_release_buffers; goto err_release_buffers;
} }
...@@ -579,7 +581,7 @@ static int dcmi_start_streaming(struct vb2_queue *vq, unsigned int count) ...@@ -579,7 +581,7 @@ static int dcmi_start_streaming(struct vb2_queue *vq, unsigned int count)
if (ret && ret != -ENOIOCTLCMD) { if (ret && ret != -ENOIOCTLCMD) {
dev_err(dcmi->dev, "%s: Failed to start streaming, subdev streamon error", dev_err(dcmi->dev, "%s: Failed to start streaming, subdev streamon error",
__func__); __func__);
goto err_disable_clock; goto err_pm_put;
} }
spin_lock_irq(&dcmi->irqlock); spin_lock_irq(&dcmi->irqlock);
...@@ -664,8 +666,8 @@ static int dcmi_start_streaming(struct vb2_queue *vq, unsigned int count) ...@@ -664,8 +666,8 @@ static int dcmi_start_streaming(struct vb2_queue *vq, unsigned int count)
err_subdev_streamoff: err_subdev_streamoff:
v4l2_subdev_call(dcmi->entity.subdev, video, s_stream, 0); v4l2_subdev_call(dcmi->entity.subdev, video, s_stream, 0);
err_disable_clock: err_pm_put:
clk_disable(dcmi->mclk); pm_runtime_put(dcmi->dev);
err_release_buffers: err_release_buffers:
spin_lock_irq(&dcmi->irqlock); spin_lock_irq(&dcmi->irqlock);
...@@ -717,7 +719,7 @@ static void dcmi_stop_streaming(struct vb2_queue *vq) ...@@ -717,7 +719,7 @@ static void dcmi_stop_streaming(struct vb2_queue *vq)
/* Stop all pending DMA operations */ /* Stop all pending DMA operations */
dmaengine_terminate_all(dcmi->dma_chan); dmaengine_terminate_all(dcmi->dma_chan);
clk_disable(dcmi->mclk); pm_runtime_put(dcmi->dev);
if (dcmi->errors_count) if (dcmi->errors_count)
dev_warn(dcmi->dev, "Some errors found while streaming: errors=%d (overrun=%d), buffers=%d\n", dev_warn(dcmi->dev, "Some errors found while streaming: errors=%d (overrun=%d), buffers=%d\n",
...@@ -1707,12 +1709,6 @@ static int dcmi_probe(struct platform_device *pdev) ...@@ -1707,12 +1709,6 @@ static int dcmi_probe(struct platform_device *pdev)
return -EPROBE_DEFER; return -EPROBE_DEFER;
} }
ret = clk_prepare(mclk);
if (ret) {
dev_err(&pdev->dev, "Unable to prepare mclk %p\n", mclk);
goto err_dma_release;
}
spin_lock_init(&dcmi->irqlock); spin_lock_init(&dcmi->irqlock);
mutex_init(&dcmi->lock); mutex_init(&dcmi->lock);
init_completion(&dcmi->complete); init_completion(&dcmi->complete);
...@@ -1728,7 +1724,7 @@ static int dcmi_probe(struct platform_device *pdev) ...@@ -1728,7 +1724,7 @@ static int dcmi_probe(struct platform_device *pdev)
/* Initialize the top-level structure */ /* Initialize the top-level structure */
ret = v4l2_device_register(&pdev->dev, &dcmi->v4l2_dev); ret = v4l2_device_register(&pdev->dev, &dcmi->v4l2_dev);
if (ret) if (ret)
goto err_clk_unprepare; goto err_dma_release;
dcmi->vdev = video_device_alloc(); dcmi->vdev = video_device_alloc();
if (!dcmi->vdev) { if (!dcmi->vdev) {
...@@ -1788,14 +1784,15 @@ static int dcmi_probe(struct platform_device *pdev) ...@@ -1788,14 +1784,15 @@ static int dcmi_probe(struct platform_device *pdev)
dev_info(&pdev->dev, "Probe done\n"); dev_info(&pdev->dev, "Probe done\n");
platform_set_drvdata(pdev, dcmi); platform_set_drvdata(pdev, dcmi);
pm_runtime_enable(&pdev->dev);
return 0; return 0;
err_device_release: err_device_release:
video_device_release(dcmi->vdev); video_device_release(dcmi->vdev);
err_device_unregister: err_device_unregister:
v4l2_device_unregister(&dcmi->v4l2_dev); v4l2_device_unregister(&dcmi->v4l2_dev);
err_clk_unprepare:
clk_unprepare(dcmi->mclk);
err_dma_release: err_dma_release:
dma_release_channel(dcmi->dma_chan); dma_release_channel(dcmi->dma_chan);
...@@ -1806,20 +1803,72 @@ static int dcmi_remove(struct platform_device *pdev) ...@@ -1806,20 +1803,72 @@ static int dcmi_remove(struct platform_device *pdev)
{ {
struct stm32_dcmi *dcmi = platform_get_drvdata(pdev); struct stm32_dcmi *dcmi = platform_get_drvdata(pdev);
pm_runtime_disable(&pdev->dev);
v4l2_async_notifier_unregister(&dcmi->notifier); v4l2_async_notifier_unregister(&dcmi->notifier);
v4l2_device_unregister(&dcmi->v4l2_dev); v4l2_device_unregister(&dcmi->v4l2_dev);
clk_unprepare(dcmi->mclk);
dma_release_channel(dcmi->dma_chan); dma_release_channel(dcmi->dma_chan);
return 0; return 0;
} }
static __maybe_unused int dcmi_runtime_suspend(struct device *dev)
{
struct stm32_dcmi *dcmi = dev_get_drvdata(dev);
clk_disable_unprepare(dcmi->mclk);
return 0;
}
static __maybe_unused int dcmi_runtime_resume(struct device *dev)
{
struct stm32_dcmi *dcmi = dev_get_drvdata(dev);
int ret;
ret = clk_prepare_enable(dcmi->mclk);
if (ret)
dev_err(dev, "%s: Failed to prepare_enable clock\n", __func__);
return ret;
}
static __maybe_unused int dcmi_suspend(struct device *dev)
{
/* disable clock */
pm_runtime_force_suspend(dev);
/* change pinctrl state */
pinctrl_pm_select_sleep_state(dev);
return 0;
}
static __maybe_unused int dcmi_resume(struct device *dev)
{
/* restore pinctl default state */
pinctrl_pm_select_default_state(dev);
/* clock enable */
pm_runtime_force_resume(dev);
return 0;
}
static const struct dev_pm_ops dcmi_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS(dcmi_suspend, dcmi_resume)
SET_RUNTIME_PM_OPS(dcmi_runtime_suspend,
dcmi_runtime_resume, NULL)
};
static struct platform_driver stm32_dcmi_driver = { static struct platform_driver stm32_dcmi_driver = {
.probe = dcmi_probe, .probe = dcmi_probe,
.remove = dcmi_remove, .remove = dcmi_remove,
.driver = { .driver = {
.name = DRV_NAME, .name = DRV_NAME,
.of_match_table = of_match_ptr(stm32_dcmi_of_match), .of_match_table = of_match_ptr(stm32_dcmi_of_match),
.pm = &dcmi_pm_ops,
}, },
}; };
......
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