mmc: core: add helpers mmc_regulator_enable/disable_vqmmc
authorHeiner Kallweit <hkallweit1@gmail.com>
Sat, 11 Mar 2023 22:39:55 +0000 (23:39 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 8 Dec 2023 07:51:19 +0000 (08:51 +0100)
[ Upstream commit 8d91f3f8ae57e6292142ca89f322e90fa0d6ac02 ]

There's a number of drivers (e.g. dw_mmc, meson-gx, mmci, sunxi) using
the same mechanism and a private flag vqmmc_enabled to deal with
enabling/disabling the vqmmc regulator.

Move this to the core and create new helpers mmc_regulator_enable_vqmmc
and mmc_regulator_disable_vqmmc.

Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
Acked-by: Martin Blumenstingl <martin.blumenstingl@googlemail.com>
Link: https://lore.kernel.org/r/71586432-360f-9b92-17f6-b05a8a971bc2@gmail.com
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Stable-dep-of: 477865af60b2 ("mmc: sdhci-sprd: Fix vqmmc not shutting down after the card was pulled")
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/mmc/core/regulator.c
include/linux/mmc/host.h

index 609201a467ef954e64c9ebc71c57a0f3a7b976d1..4dcbc2281d2b5386a64e821ab4e9d70e5742f596 100644 (file)
@@ -271,3 +271,44 @@ int mmc_regulator_get_supply(struct mmc_host *mmc)
        return 0;
 }
 EXPORT_SYMBOL_GPL(mmc_regulator_get_supply);
+
+/**
+ * mmc_regulator_enable_vqmmc - enable VQMMC regulator for a host
+ * @mmc: the host to regulate
+ *
+ * Returns 0 or errno. Enables the regulator for vqmmc.
+ * Keeps track of the enable status for ensuring that calls to
+ * regulator_enable/disable are balanced.
+ */
+int mmc_regulator_enable_vqmmc(struct mmc_host *mmc)
+{
+       int ret = 0;
+
+       if (!IS_ERR(mmc->supply.vqmmc) && !mmc->vqmmc_enabled) {
+               ret = regulator_enable(mmc->supply.vqmmc);
+               if (ret < 0)
+                       dev_err(mmc_dev(mmc), "enabling vqmmc regulator failed\n");
+               else
+                       mmc->vqmmc_enabled = true;
+       }
+
+       return ret;
+}
+EXPORT_SYMBOL_GPL(mmc_regulator_enable_vqmmc);
+
+/**
+ * mmc_regulator_disable_vqmmc - disable VQMMC regulator for a host
+ * @mmc: the host to regulate
+ *
+ * Returns 0 or errno. Disables the regulator for vqmmc.
+ * Keeps track of the enable status for ensuring that calls to
+ * regulator_enable/disable are balanced.
+ */
+void mmc_regulator_disable_vqmmc(struct mmc_host *mmc)
+{
+       if (!IS_ERR(mmc->supply.vqmmc) && mmc->vqmmc_enabled) {
+               regulator_disable(mmc->supply.vqmmc);
+               mmc->vqmmc_enabled = false;
+       }
+}
+EXPORT_SYMBOL_GPL(mmc_regulator_disable_vqmmc);
index 8fdd3cf971a30f59684de944697d97a7a0d013e9..8f918f9a1228d2c9e41ec734f1415886badbe974 100644 (file)
@@ -450,6 +450,7 @@ struct mmc_host {
        unsigned int            retune_paused:1; /* re-tuning is temporarily disabled */
        unsigned int            retune_crc_disable:1; /* don't trigger retune upon crc */
        unsigned int            can_dma_map_merge:1; /* merging can be used */
+       unsigned int            vqmmc_enabled:1; /* vqmmc regulator is enabled */
 
        int                     rescan_disable; /* disable card detection */
        int                     rescan_entered; /* used with nonremovable devices */
@@ -597,6 +598,8 @@ static inline int mmc_regulator_set_vqmmc(struct mmc_host *mmc,
 #endif
 
 int mmc_regulator_get_supply(struct mmc_host *mmc);
+int mmc_regulator_enable_vqmmc(struct mmc_host *mmc);
+void mmc_regulator_disable_vqmmc(struct mmc_host *mmc);
 
 static inline int mmc_card_is_removable(struct mmc_host *host)
 {