aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMark Brown <broonie@linaro.org>2013-07-28 09:47:02 -0400
committerMark Brown <broonie@linaro.org>2013-07-29 12:59:20 -0400
commit49834de234f3cf592c3d242c889ca603db8e7050 (patch)
tree5a8563efa8626d982039a3fe9b570b7e61edf7bc
parentbafe886936a8982f5780330c901889a37bba7d4c (diff)
spi: Provide core support for runtime PM during transfers
Most SPI drivers that implement runtime PM support use identical code to do so: they acquire a runtime PM lock in prepare_transfer_hardware() and then they release it in unprepare_transfer_hardware(). The variations in this are mostly missing error checking and the choice to use autosuspend. Since these runtime PM calls are normally the only thing in the prepare and unprepare callbacks and the autosuspend API transparently does the right thing on devices with autosuspend disabled factor all of this out into the core with a flag to enable the behaviour. Signed-off-by: Mark Brown <broonie@linaro.org> Reviewed-by: Stephen Warren <swarren@nvidia.com> Acked-by: Linus Walleij <linus.walleij@linaro.org>
-rw-r--r--drivers/spi/spi.c16
-rw-r--r--include/linux/spi/spi.h5
2 files changed, 21 insertions, 0 deletions
diff --git a/drivers/spi/spi.c b/drivers/spi/spi.c
index 978dda2c5239..361cced68069 100644
--- a/drivers/spi/spi.c
+++ b/drivers/spi/spi.c
@@ -553,6 +553,10 @@ static void spi_pump_messages(struct kthread_work *work)
553 master->unprepare_transfer_hardware(master)) 553 master->unprepare_transfer_hardware(master))
554 dev_err(&master->dev, 554 dev_err(&master->dev,
555 "failed to unprepare transfer hardware\n"); 555 "failed to unprepare transfer hardware\n");
556 if (master->auto_runtime_pm) {
557 pm_runtime_mark_last_busy(master->dev.parent);
558 pm_runtime_put_autosuspend(master->dev.parent);
559 }
556 return; 560 return;
557 } 561 }
558 562
@@ -572,11 +576,23 @@ static void spi_pump_messages(struct kthread_work *work)
572 master->busy = true; 576 master->busy = true;
573 spin_unlock_irqrestore(&master->queue_lock, flags); 577 spin_unlock_irqrestore(&master->queue_lock, flags);
574 578
579 if (!was_busy && master->auto_runtime_pm) {
580 ret = pm_runtime_get_sync(master->dev.parent);
581 if (ret < 0) {
582 dev_err(&master->dev, "Failed to power device: %d\n",
583 ret);
584 return;
585 }
586 }
587
575 if (!was_busy && master->prepare_transfer_hardware) { 588 if (!was_busy && master->prepare_transfer_hardware) {
576 ret = master->prepare_transfer_hardware(master); 589 ret = master->prepare_transfer_hardware(master);
577 if (ret) { 590 if (ret) {
578 dev_err(&master->dev, 591 dev_err(&master->dev,
579 "failed to prepare transfer hardware\n"); 592 "failed to prepare transfer hardware\n");
593
594 if (master->auto_runtime_pm)
595 pm_runtime_put(master->dev.parent);
580 return; 596 return;
581 } 597 }
582 } 598 }
diff --git a/include/linux/spi/spi.h b/include/linux/spi/spi.h
index 28e440be1c07..bf0204c00053 100644
--- a/include/linux/spi/spi.h
+++ b/include/linux/spi/spi.h
@@ -254,6 +254,9 @@ static inline void spi_unregister_driver(struct spi_driver *sdrv)
254 * @busy: message pump is busy 254 * @busy: message pump is busy
255 * @running: message pump is running 255 * @running: message pump is running
256 * @rt: whether this queue is set to run as a realtime task 256 * @rt: whether this queue is set to run as a realtime task
257 * @auto_runtime_pm: the core should ensure a runtime PM reference is held
258 * while the hardware is prepared, using the parent
259 * device for the spidev
257 * @prepare_transfer_hardware: a message will soon arrive from the queue 260 * @prepare_transfer_hardware: a message will soon arrive from the queue
258 * so the subsystem requests the driver to prepare the transfer hardware 261 * so the subsystem requests the driver to prepare the transfer hardware
259 * by issuing this call 262 * by issuing this call
@@ -374,11 +377,13 @@ struct spi_master {
374 bool busy; 377 bool busy;
375 bool running; 378 bool running;
376 bool rt; 379 bool rt;
380 bool auto_runtime_pm;
377 381
378 int (*prepare_transfer_hardware)(struct spi_master *master); 382 int (*prepare_transfer_hardware)(struct spi_master *master);
379 int (*transfer_one_message)(struct spi_master *master, 383 int (*transfer_one_message)(struct spi_master *master,
380 struct spi_message *mesg); 384 struct spi_message *mesg);
381 int (*unprepare_transfer_hardware)(struct spi_master *master); 385 int (*unprepare_transfer_hardware)(struct spi_master *master);
386
382 /* gpio chip select */ 387 /* gpio chip select */
383 int *cs_gpios; 388 int *cs_gpios;
384}; 389};