aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMarek Szyprowski <m.szyprowski@samsung.com>2010-08-10 21:01:59 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2010-08-11 11:59:05 -0400
commit9bea3c850dbff2296892298614388bdc71ad2170 (patch)
treebc5050601ec6349441ef3d35cbda2775808eee20
parent17866e14f3a4f219e94f1374ece7226479418ff8 (diff)
sdhci: add regulator support
This patch adds support for regulator API to sdhci core driver. Regulators can be used to disable power in suspended state to reduce dissipated energy. Signed-off-by: Marek Szyprowski <m.szyprowski@samsung.com> Signed-off-by: Kyungmin Park <kyungmin.park@samsung.com> Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com> Cc: <linux-mmc@vger.kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--drivers/mmc/host/sdhci.c26
-rw-r--r--drivers/mmc/host/sdhci.h2
2 files changed, 27 insertions, 1 deletions
diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c
index 70001ecb6ebe..c86c14af479f 100644
--- a/drivers/mmc/host/sdhci.c
+++ b/drivers/mmc/host/sdhci.c
@@ -19,6 +19,7 @@
19#include <linux/dma-mapping.h> 19#include <linux/dma-mapping.h>
20#include <linux/slab.h> 20#include <linux/slab.h>
21#include <linux/scatterlist.h> 21#include <linux/scatterlist.h>
22#include <linux/regulator/consumer.h>
22 23
23#include <linux/leds.h> 24#include <linux/leds.h>
24 25
@@ -1608,7 +1609,10 @@ int sdhci_suspend_host(struct sdhci_host *host, pm_message_t state)
1608 1609
1609 free_irq(host->irq, host); 1610 free_irq(host->irq, host);
1610 1611
1611 return 0; 1612 if (host->vmmc)
1613 ret = regulator_disable(host->vmmc);
1614
1615 return ret;
1612} 1616}
1613 1617
1614EXPORT_SYMBOL_GPL(sdhci_suspend_host); 1618EXPORT_SYMBOL_GPL(sdhci_suspend_host);
@@ -1617,6 +1621,13 @@ int sdhci_resume_host(struct sdhci_host *host)
1617{ 1621{
1618 int ret; 1622 int ret;
1619 1623
1624 if (host->vmmc) {
1625 int ret = regulator_enable(host->vmmc);
1626 if (ret)
1627 return ret;
1628 }
1629
1630
1620 if (host->flags & (SDHCI_USE_SDMA | SDHCI_USE_ADMA)) { 1631 if (host->flags & (SDHCI_USE_SDMA | SDHCI_USE_ADMA)) {
1621 if (host->ops->enable_dma) 1632 if (host->ops->enable_dma)
1622 host->ops->enable_dma(host); 1633 host->ops->enable_dma(host);
@@ -1889,6 +1900,14 @@ int sdhci_add_host(struct sdhci_host *host)
1889 if (ret) 1900 if (ret)
1890 goto untasklet; 1901 goto untasklet;
1891 1902
1903 host->vmmc = regulator_get(mmc_dev(mmc), "vmmc");
1904 if (IS_ERR(host->vmmc)) {
1905 printk(KERN_INFO "%s: no vmmc regulator found\n", mmc_hostname(mmc));
1906 host->vmmc = NULL;
1907 } else {
1908 regulator_enable(host->vmmc);
1909 }
1910
1892 sdhci_init(host, 0); 1911 sdhci_init(host, 0);
1893 1912
1894#ifdef CONFIG_MMC_DEBUG 1913#ifdef CONFIG_MMC_DEBUG
@@ -1973,6 +1992,11 @@ void sdhci_remove_host(struct sdhci_host *host, int dead)
1973 tasklet_kill(&host->card_tasklet); 1992 tasklet_kill(&host->card_tasklet);
1974 tasklet_kill(&host->finish_tasklet); 1993 tasklet_kill(&host->finish_tasklet);
1975 1994
1995 if (host->vmmc) {
1996 regulator_disable(host->vmmc);
1997 regulator_put(host->vmmc);
1998 }
1999
1976 kfree(host->adma_desc); 2000 kfree(host->adma_desc);
1977 kfree(host->align_buffer); 2001 kfree(host->align_buffer);
1978 2002
diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h
index c98315c197ed..ff070a309f8d 100644
--- a/drivers/mmc/host/sdhci.h
+++ b/drivers/mmc/host/sdhci.h
@@ -249,6 +249,8 @@ struct sdhci_host {
249 249
250 const struct sdhci_ops *ops; /* Low level hw interface */ 250 const struct sdhci_ops *ops; /* Low level hw interface */
251 251
252 struct regulator *vmmc; /* Power regulator */
253
252 /* Internal data */ 254 /* Internal data */
253 struct mmc_host *mmc; /* MMC structure */ 255 struct mmc_host *mmc; /* MMC structure */
254 u64 dma_mask; /* custom DMA mask */ 256 u64 dma_mask; /* custom DMA mask */