Commit 12d01d0b authored by Ulf Hansson's avatar Ulf Hansson Committed by Chris Ball

mmc: core: Add bus_ops for runtime pm callbacks

SDIO is the only protocol that uses runtime pm for the card device
right now. To provide the option for sd and mmc to use runtime pm as
well the bus_ops callback are extended with two new functions. One for
runtime_suspend and one for runtime_resume.

This patch will also implement the callbacks for SDIO to make sure
existing functionality is maintained. It also prepares to move
away from using the mmc_power_restore_host API, since it is not
needed when using runtime PM.
Signed-off-by: default avatarUlf Hansson <ulf.hansson@linaro.org>
Signed-off-by: default avatarChris Ball <cjb@laptop.org>
parent 39b9431b
...@@ -151,15 +151,25 @@ static int mmc_bus_resume(struct device *dev) ...@@ -151,15 +151,25 @@ static int mmc_bus_resume(struct device *dev)
static int mmc_runtime_suspend(struct device *dev) static int mmc_runtime_suspend(struct device *dev)
{ {
struct mmc_card *card = mmc_dev_to_card(dev); struct mmc_card *card = mmc_dev_to_card(dev);
struct mmc_host *host = card->host;
int ret = 0;
if (host->bus_ops->runtime_suspend)
ret = host->bus_ops->runtime_suspend(host);
return mmc_power_save_host(card->host); return ret;
} }
static int mmc_runtime_resume(struct device *dev) static int mmc_runtime_resume(struct device *dev)
{ {
struct mmc_card *card = mmc_dev_to_card(dev); struct mmc_card *card = mmc_dev_to_card(dev);
struct mmc_host *host = card->host;
int ret = 0;
if (host->bus_ops->runtime_resume)
ret = host->bus_ops->runtime_resume(host);
return mmc_power_restore_host(card->host); return ret;
} }
static int mmc_runtime_idle(struct device *dev) static int mmc_runtime_idle(struct device *dev)
......
...@@ -1478,7 +1478,7 @@ void mmc_set_driver_type(struct mmc_host *host, unsigned int drv_type) ...@@ -1478,7 +1478,7 @@ void mmc_set_driver_type(struct mmc_host *host, unsigned int drv_type)
* If a host does all the power sequencing itself, ignore the * If a host does all the power sequencing itself, ignore the
* initial MMC_POWER_UP stage. * initial MMC_POWER_UP stage.
*/ */
static void mmc_power_up(struct mmc_host *host) void mmc_power_up(struct mmc_host *host)
{ {
int bit; int bit;
......
...@@ -22,6 +22,8 @@ struct mmc_bus_ops { ...@@ -22,6 +22,8 @@ struct mmc_bus_ops {
void (*detect)(struct mmc_host *); void (*detect)(struct mmc_host *);
int (*suspend)(struct mmc_host *); int (*suspend)(struct mmc_host *);
int (*resume)(struct mmc_host *); int (*resume)(struct mmc_host *);
int (*runtime_suspend)(struct mmc_host *);
int (*runtime_resume)(struct mmc_host *);
int (*power_save)(struct mmc_host *); int (*power_save)(struct mmc_host *);
int (*power_restore)(struct mmc_host *); int (*power_restore)(struct mmc_host *);
int (*alive)(struct mmc_host *); int (*alive)(struct mmc_host *);
...@@ -44,6 +46,7 @@ int mmc_set_signal_voltage(struct mmc_host *host, int signal_voltage); ...@@ -44,6 +46,7 @@ int mmc_set_signal_voltage(struct mmc_host *host, int signal_voltage);
int __mmc_set_signal_voltage(struct mmc_host *host, int signal_voltage); int __mmc_set_signal_voltage(struct mmc_host *host, int signal_voltage);
void mmc_set_timing(struct mmc_host *host, unsigned int timing); void mmc_set_timing(struct mmc_host *host, unsigned int timing);
void mmc_set_driver_type(struct mmc_host *host, unsigned int drv_type); void mmc_set_driver_type(struct mmc_host *host, unsigned int drv_type);
void mmc_power_up(struct mmc_host *host);
void mmc_power_off(struct mmc_host *host); void mmc_power_off(struct mmc_host *host);
void mmc_power_cycle(struct mmc_host *host); void mmc_power_cycle(struct mmc_host *host);
......
...@@ -1059,11 +1059,27 @@ static int mmc_sdio_power_restore(struct mmc_host *host) ...@@ -1059,11 +1059,27 @@ static int mmc_sdio_power_restore(struct mmc_host *host)
return ret; return ret;
} }
static int mmc_sdio_runtime_suspend(struct mmc_host *host)
{
/* No references to the card, cut the power to it. */
mmc_power_off(host);
return 0;
}
static int mmc_sdio_runtime_resume(struct mmc_host *host)
{
/* Restore power and re-initialize. */
mmc_power_up(host);
return mmc_sdio_power_restore(host);
}
static const struct mmc_bus_ops mmc_sdio_ops = { static const struct mmc_bus_ops mmc_sdio_ops = {
.remove = mmc_sdio_remove, .remove = mmc_sdio_remove,
.detect = mmc_sdio_detect, .detect = mmc_sdio_detect,
.suspend = mmc_sdio_suspend, .suspend = mmc_sdio_suspend,
.resume = mmc_sdio_resume, .resume = mmc_sdio_resume,
.runtime_suspend = mmc_sdio_runtime_suspend,
.runtime_resume = mmc_sdio_runtime_resume,
.power_restore = mmc_sdio_power_restore, .power_restore = mmc_sdio_power_restore,
.alive = mmc_sdio_alive, .alive = mmc_sdio_alive,
}; };
......
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