diff options
| author | Adrian Hunter <adrian.hunter@nokia.com> | 2010-02-15 13:03:34 -0500 |
|---|---|---|
| committer | Tony Lindgren <tony@atomide.com> | 2010-02-15 13:03:34 -0500 |
| commit | e0eb2424469ec2333885672d3db8bd07d322455d (patch) | |
| tree | 987a225999bbed33b1f456ccdf56fd10ed574019 | |
| parent | 6da20c89af64b75302399369a90b9d50c1a87665 (diff) | |
omap_hsmmc: Allow for a shared VccQ
EMMC can have two voltage supplies, Vcc and VccQ
which are implemented in the code as consumer
supplies vmmc and vmmc_aux.
If the regulator that supplies vmmc_aux is shared
with other consumers, then sending it to sleep
will disrupt those consumers. However, the
TWL4030-family regulators may have OFF remapped
to SLEEP, in which case 'regulator_disable()'
will put the regulator to sleep only when all
consumers are disabled - which is the desired
behaviour.
This patch adds a platform data field to allow
that option.
Signed-off-by: Adrian Hunter <adrian.hunter@nokia.com>
Signed-off-by: Tony Lindgren <tony@atomide.com>
| -rw-r--r-- | arch/arm/mach-omap2/hsmmc.c | 3 | ||||
| -rw-r--r-- | arch/arm/mach-omap2/hsmmc.h | 1 | ||||
| -rw-r--r-- | arch/arm/plat-omap/include/plat/mmc.h | 3 | ||||
| -rw-r--r-- | drivers/mmc/host/omap_hsmmc.c | 16 |
4 files changed, 22 insertions, 1 deletions
diff --git a/arch/arm/mach-omap2/hsmmc.c b/arch/arm/mach-omap2/hsmmc.c index e4ab123cd4d0..9ad229594b46 100644 --- a/arch/arm/mach-omap2/hsmmc.c +++ b/arch/arm/mach-omap2/hsmmc.c | |||
| @@ -205,6 +205,9 @@ void __init omap2_hsmmc_init(struct omap2_hsmmc_info *controllers) | |||
| 205 | if (c->no_off) | 205 | if (c->no_off) |
| 206 | mmc->slots[0].no_off = 1; | 206 | mmc->slots[0].no_off = 1; |
| 207 | 207 | ||
| 208 | if (c->vcc_aux_disable_is_sleep) | ||
| 209 | mmc->slots[0].vcc_aux_disable_is_sleep = 1; | ||
| 210 | |||
| 208 | /* NOTE: MMC slots should have a Vcc regulator set up. | 211 | /* NOTE: MMC slots should have a Vcc regulator set up. |
| 209 | * This may be from a TWL4030-family chip, another | 212 | * This may be from a TWL4030-family chip, another |
| 210 | * controllable regulator, or a fixed supply. | 213 | * controllable regulator, or a fixed supply. |
diff --git a/arch/arm/mach-omap2/hsmmc.h b/arch/arm/mach-omap2/hsmmc.h index 2453a7aeaf6f..36f0ba8d89e2 100644 --- a/arch/arm/mach-omap2/hsmmc.h +++ b/arch/arm/mach-omap2/hsmmc.h | |||
| @@ -15,6 +15,7 @@ struct omap2_hsmmc_info { | |||
| 15 | bool nonremovable; /* Nonremovable e.g. eMMC */ | 15 | bool nonremovable; /* Nonremovable e.g. eMMC */ |
| 16 | bool power_saving; /* Try to sleep or power off when possible */ | 16 | bool power_saving; /* Try to sleep or power off when possible */ |
| 17 | bool no_off; /* power_saving and power is not to go off */ | 17 | bool no_off; /* power_saving and power is not to go off */ |
| 18 | bool vcc_aux_disable_is_sleep; /* Regulator off remapped to sleep */ | ||
| 18 | int gpio_cd; /* or -EINVAL */ | 19 | int gpio_cd; /* or -EINVAL */ |
| 19 | int gpio_wp; /* or -EINVAL */ | 20 | int gpio_wp; /* or -EINVAL */ |
| 20 | char *name; /* or NULL for default */ | 21 | char *name; /* or NULL for default */ |
diff --git a/arch/arm/plat-omap/include/plat/mmc.h b/arch/arm/plat-omap/include/plat/mmc.h index b46394955f37..a1bac07c89eb 100644 --- a/arch/arm/plat-omap/include/plat/mmc.h +++ b/arch/arm/plat-omap/include/plat/mmc.h | |||
| @@ -99,6 +99,9 @@ struct omap_mmc_platform_data { | |||
| 99 | /* If using power_saving and the MMC power is not to go off */ | 99 | /* If using power_saving and the MMC power is not to go off */ |
| 100 | unsigned no_off:1; | 100 | unsigned no_off:1; |
| 101 | 101 | ||
| 102 | /* Regulator off remapped to sleep */ | ||
| 103 | unsigned vcc_aux_disable_is_sleep:1; | ||
| 104 | |||
| 102 | int switch_pin; /* gpio (card detect) */ | 105 | int switch_pin; /* gpio (card detect) */ |
| 103 | int gpio_wp; /* gpio (write protect) */ | 106 | int gpio_wp; /* gpio (write protect) */ |
| 104 | 107 | ||
diff --git a/drivers/mmc/host/omap_hsmmc.c b/drivers/mmc/host/omap_hsmmc.c index d2fad587f371..af374771bed0 100644 --- a/drivers/mmc/host/omap_hsmmc.c +++ b/drivers/mmc/host/omap_hsmmc.c | |||
| @@ -347,7 +347,14 @@ static int omap_hsmmc_23_set_sleep(struct device *dev, int slot, int sleep, | |||
| 347 | err = regulator_set_mode(host->vcc, mode); | 347 | err = regulator_set_mode(host->vcc, mode); |
| 348 | if (err) | 348 | if (err) |
| 349 | return err; | 349 | return err; |
| 350 | return regulator_set_mode(host->vcc_aux, mode); | 350 | |
| 351 | if (!mmc_slot(host).vcc_aux_disable_is_sleep) | ||
| 352 | return regulator_set_mode(host->vcc_aux, mode); | ||
| 353 | |||
| 354 | if (sleep) | ||
| 355 | return regulator_disable(host->vcc_aux); | ||
| 356 | else | ||
| 357 | return regulator_enable(host->vcc_aux); | ||
| 351 | } | 358 | } |
| 352 | 359 | ||
| 353 | static int omap_hsmmc_gpio_init(struct omap_mmc_platform_data *pdata) | 360 | static int omap_hsmmc_gpio_init(struct omap_mmc_platform_data *pdata) |
| @@ -1982,6 +1989,13 @@ static int __init omap_hsmmc_probe(struct platform_device *pdev) | |||
| 1982 | else | 1989 | else |
| 1983 | mmc->ops = &omap_hsmmc_ops; | 1990 | mmc->ops = &omap_hsmmc_ops; |
| 1984 | 1991 | ||
| 1992 | /* | ||
| 1993 | * If regulator_disable can only put vcc_aux to sleep then there is | ||
| 1994 | * no off state. | ||
| 1995 | */ | ||
| 1996 | if (mmc_slot(host).vcc_aux_disable_is_sleep) | ||
| 1997 | mmc_slot(host).no_off = 1; | ||
| 1998 | |||
| 1985 | mmc->f_min = 400000; | 1999 | mmc->f_min = 400000; |
| 1986 | mmc->f_max = 52000000; | 2000 | mmc->f_max = 52000000; |
| 1987 | 2001 | ||
