Commit bea2b6a4 authored by Niklas Söderlund's avatar Niklas Söderlund Committed by Mauro Carvalho Chehab

media: rcar-vin: Add support for suspend and resume

Add support for suspend and resume by stopping and starting the video
pipeline while still retaining all buffers given to the driver by
user-space and internally allocated ones, this gives the application a
seamless experience.

Buffers are never returned to user-space unprocessed so user-space don't
notice when suspending. When resuming the driver restarts the capture
session using the internal scratch buffer, this happens before
user-space is unfrozen, this leads to speedy response times once the
application resumes its execution.

As the entire pipeline is stopped on suspend all subdevices in use are
also stopped, and if they enter a shutdown state when not streaming
(such as the R-Car CSI-2 driver) they too will be suspended and resumed
in sync with the VIN driver.

To be able to do keep track of which VINs should be resumed a new
internal state SUSPENDED is added to recode this.
Signed-off-by: default avatarNiklas Söderlund <niklas.soderlund+renesas@ragnatech.se>
Reviewed-by: default avatarJacopo Mondi <jacopo+renesas@jmondi.org>
Signed-off-by: default avatarHans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+huawei@kernel.org>
parent 63a71dd8
...@@ -918,6 +918,54 @@ static int rvin_mc_init(struct rvin_dev *vin) ...@@ -918,6 +918,54 @@ static int rvin_mc_init(struct rvin_dev *vin)
return ret; return ret;
} }
/* -----------------------------------------------------------------------------
* Suspend / Resume
*/
static int __maybe_unused rvin_suspend(struct device *dev)
{
struct rvin_dev *vin = dev_get_drvdata(dev);
if (vin->state != RUNNING)
return 0;
rvin_stop_streaming(vin);
vin->state = SUSPENDED;
return 0;
}
static int __maybe_unused rvin_resume(struct device *dev)
{
struct rvin_dev *vin = dev_get_drvdata(dev);
if (vin->state != SUSPENDED)
return 0;
/*
* Restore group master CHSEL setting.
*
* This needs to be done by every VIN resuming not only the master
* as we don't know if and in which order the master VINs will
* be resumed.
*/
if (vin->info->use_mc) {
unsigned int master_id = rvin_group_id_to_master(vin->id);
struct rvin_dev *master = vin->group->vin[master_id];
int ret;
if (WARN_ON(!master))
return -ENODEV;
ret = rvin_set_channel_routing(master, master->chsel);
if (ret)
return ret;
}
return rvin_start_streaming(vin);
}
/* ----------------------------------------------------------------------------- /* -----------------------------------------------------------------------------
* Platform Device Driver * Platform Device Driver
*/ */
...@@ -1405,9 +1453,12 @@ static int rcar_vin_remove(struct platform_device *pdev) ...@@ -1405,9 +1453,12 @@ static int rcar_vin_remove(struct platform_device *pdev)
return 0; return 0;
} }
static SIMPLE_DEV_PM_OPS(rvin_pm_ops, rvin_suspend, rvin_resume);
static struct platform_driver rcar_vin_driver = { static struct platform_driver rcar_vin_driver = {
.driver = { .driver = {
.name = "rcar-vin", .name = "rcar-vin",
.pm = &rvin_pm_ops,
.of_match_table = rvin_of_id_table, .of_match_table = rvin_of_id_table,
}, },
.probe = rcar_vin_probe, .probe = rcar_vin_probe,
......
...@@ -53,12 +53,14 @@ enum rvin_csi_id { ...@@ -53,12 +53,14 @@ enum rvin_csi_id {
* STARTING - Capture starting up * STARTING - Capture starting up
* RUNNING - Operation in progress have buffers * RUNNING - Operation in progress have buffers
* STOPPING - Stopping operation * STOPPING - Stopping operation
* SUSPENDED - Capture is suspended
*/ */
enum rvin_dma_state { enum rvin_dma_state {
STOPPED = 0, STOPPED = 0,
STARTING, STARTING,
RUNNING, RUNNING,
STOPPING, STOPPING,
SUSPENDED,
}; };
/** /**
......
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