[PATCH 3/3] mmc: core: Add 'fixed-emmc-driver-type-hs{200,400}'

From: Eugeniu Rosca
Date: Tue Nov 05 2019 - 00:51:47 EST


Add support for two more DT bindings, which stem from the need to
implement below real-life requirement shared by eMMC vendor:

---snip---
Use "drive strength" value of 4 or 1 for HS400 or 0 for HS200.
---snip---

Inspire from [Y] and [Z] during implementation and testing (H3ULCB-KF).
Below decision matrix is intended as function of user's input:

[0] [2] [4]
[0] hs200:[0] hs200:[2] hs200:[0]
hs400:[0] hs400:[0] hs400:[4]
[2] hs200:[2] hs200:[2] hs200:[2]
hs400:[0] hs400:[R] hs400:[4]
[4] hs200:[0] hs200:[2] hs200:[R]
hs400:[4] hs400:[4] hs400:[4]

[0] "fixed-emmc-driver-type"
[2] "fixed-emmc-driver-type-hs200"
[4] "fixed-emmc-driver-type-hs400"
[R] RAW/ECSD drive strength as implemented in
commit cc4f414c885cd0 ("mmc: mmc: Add driver strength selection")
[Y] commit 6186d06c519e21 ("mmc: parse new binding for eMMC fixed driver type")
[Z] https://www.elinux.org/Tests:eMMC-fixed-drive-strength

Cc: Wolfram Sang <wsa+renesas@xxxxxxxxxxxxxxxxxxxx>
Signed-off-by: Eugeniu Rosca <erosca@xxxxxxxxxxxxxx>
---
drivers/mmc/core/host.c | 4 ++++
drivers/mmc/core/mmc.c | 19 ++++++++++++++++---
include/linux/mmc/host.h | 2 ++
3 files changed, 22 insertions(+), 3 deletions(-)

diff --git a/drivers/mmc/core/host.c b/drivers/mmc/core/host.c
index 54abfdfc69ba..2a3d3b542e0d 100644
--- a/drivers/mmc/core/host.c
+++ b/drivers/mmc/core/host.c
@@ -336,6 +336,8 @@ int mmc_of_parse(struct mmc_host *host)

/* Must be after "non-removable" check */
mmc_of_read_drv_type(host, "fixed-emmc-driver-type", &host->fixed_drv_type);
+ mmc_of_read_drv_type(host, "fixed-emmc-driver-type-hs200", &host->fixed_drv_type_hs200);
+ mmc_of_read_drv_type(host, "fixed-emmc-driver-type-hs400", &host->fixed_drv_type_hs400);

host->dsr_req = !device_property_read_u32(dev, "dsr", &host->dsr);
if (host->dsr_req && (host->dsr & ~0xffff)) {
@@ -455,6 +457,8 @@ struct mmc_host *mmc_alloc_host(int extra, struct device *dev)
host->max_blk_count = PAGE_SIZE / 512;

host->fixed_drv_type = -EINVAL;
+ host->fixed_drv_type_hs200 = -EINVAL;
+ host->fixed_drv_type_hs400 = -EINVAL;
host->ios.power_delay_ms = 10;

return host;
diff --git a/drivers/mmc/core/mmc.c b/drivers/mmc/core/mmc.c
index c8804895595f..89e6fb9aedeb 100644
--- a/drivers/mmc/core/mmc.c
+++ b/drivers/mmc/core/mmc.c
@@ -62,6 +62,8 @@ static const unsigned int taac_mant[] = {
__res & __mask; \
})

+static void mmc_select_driver_type(struct mmc_card *card, int timing);
+
/*
* Given the decoded CSD structure, decode the raw CID to our CID structure.
*/
@@ -1192,6 +1194,8 @@ static int mmc_select_hs400(struct mmc_card *card)
return err;
}

+ mmc_select_driver_type(card, EXT_CSD_TIMING_HS400);
+
/* Switch card to HS400 */
val = EXT_CSD_TIMING_HS400 |
card->drive_strength << EXT_CSD_DRV_STR_SHIFT;
@@ -1270,6 +1274,8 @@ int mmc_hs400_to_hs200(struct mmc_card *card)
if (err)
goto out_err;

+ mmc_select_driver_type(card, EXT_CSD_TIMING_HS200);
+
/* Switch HS to HS200 */
val = EXT_CSD_TIMING_HS200 |
card->drive_strength << EXT_CSD_DRV_STR_SHIFT;
@@ -1304,10 +1310,17 @@ int mmc_hs400_to_hs200(struct mmc_card *card)
return err;
}

-static void mmc_select_driver_type(struct mmc_card *card)
+static void mmc_select_driver_type(struct mmc_card *card, int timing)
{
int card_drv_type, drive_strength, drv_type = 0;
int fixed_drv_type = card->host->fixed_drv_type;
+ int fixed_drv_type_hs200 = card->host->fixed_drv_type_hs200;
+ int fixed_drv_type_hs400 = card->host->fixed_drv_type_hs400;
+
+ if (fixed_drv_type_hs200 >= 0 && timing == EXT_CSD_TIMING_HS200)
+ fixed_drv_type = fixed_drv_type_hs200;
+ else if (fixed_drv_type_hs400 >= 0 && timing == EXT_CSD_TIMING_HS400)
+ fixed_drv_type = fixed_drv_type_hs400;

card_drv_type = card->ext_csd.raw_driver_strength |
mmc_driver_type_mask(0);
@@ -1385,7 +1398,7 @@ static int mmc_select_hs400es(struct mmc_card *card)
goto out_err;
}

- mmc_select_driver_type(card);
+ mmc_select_driver_type(card, EXT_CSD_TIMING_HS400);

/* Switch card to HS400 */
val = EXT_CSD_TIMING_HS400 |
@@ -1445,7 +1458,7 @@ static int mmc_select_hs200(struct mmc_card *card)
if (err)
return err;

- mmc_select_driver_type(card);
+ mmc_select_driver_type(card, EXT_CSD_TIMING_HS200);

/*
* Set the bus width(4 or 8) with host's support and
diff --git a/include/linux/mmc/host.h b/include/linux/mmc/host.h
index ba703384bea0..6960ba98810a 100644
--- a/include/linux/mmc/host.h
+++ b/include/linux/mmc/host.h
@@ -371,6 +371,8 @@ struct mmc_host {
#define MMC_CAP2_MERGE_CAPABLE (1 << 26) /* Host can merge a segment over the segment size */

int fixed_drv_type; /* fixed driver type for non-removable media */
+ int fixed_drv_type_hs200; /* HS200-specific fixed_drv_type */
+ int fixed_drv_type_hs400; /* HS400-specific fixed_drv_type */

mmc_pm_flag_t pm_caps; /* supported pm features */

--
2.23.0