mmc: Add support for downgrading HS200/HS400 to HS mode
The mmc_select_mode_and_width() function can be called while the card is in HS200/HS400 mode and can be used to downgrade the card to lower mode, e.g. HS. This is used for example by mmc_boot_part_access_chk() which cannot access the card in HS200/HS400 mode and which is in turn called by saveenv if env is in the MMC. In such case, forcing the card clock to legacy frequency cannot work. Instead, the card must be switched to HS mode first, from which it can then be reprogrammed as needed. However, this procedure needs additional code changes, since the current implementation checks whether the card correctly switched to HS mode in mmc_set_card_speed(). The check only expects that the card will be going to HS mode from lower modes, not from higher modes, hence add a parameter which indicates that the HS200/HS400 to HS downgrade is happening. This makes the code send the switch command first, reconfigure the controller next and finally perform the EXT_CSD readback check. The last two steps cannot be done in reverse order as the card is already in HS mode when the clock are being switched on the controller side. Signed-off-by: Marek Vasut <marek.vasut+renesas@gmail.com> Cc: Jaehoon Chung <jh80.chung@samsung.com>
This commit is contained in:
parent
9a878e8f17
commit
b9a2a0e2e9
@ -754,7 +754,8 @@ int mmc_switch(struct mmc *mmc, u8 set, u8 index, u8 value)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if !CONFIG_IS_ENABLED(MMC_TINY)
|
#if !CONFIG_IS_ENABLED(MMC_TINY)
|
||||||
static int mmc_set_card_speed(struct mmc *mmc, enum bus_mode mode)
|
static int mmc_set_card_speed(struct mmc *mmc, enum bus_mode mode,
|
||||||
|
bool hsdowngrade)
|
||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
int speed_bits;
|
int speed_bits;
|
||||||
@ -788,6 +789,20 @@ static int mmc_set_card_speed(struct mmc *mmc, enum bus_mode mode)
|
|||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
|
#if CONFIG_IS_ENABLED(MMC_HS200_SUPPORT) || \
|
||||||
|
CONFIG_IS_ENABLED(MMC_HS400_SUPPORT)
|
||||||
|
/*
|
||||||
|
* In case the eMMC is in HS200/HS400 mode and we are downgrading
|
||||||
|
* to HS mode, the card clock are still running much faster than
|
||||||
|
* the supported HS mode clock, so we can not reliably read out
|
||||||
|
* Extended CSD. Reconfigure the controller to run at HS mode.
|
||||||
|
*/
|
||||||
|
if (hsdowngrade) {
|
||||||
|
mmc_select_mode(mmc, MMC_HS);
|
||||||
|
mmc_set_clock(mmc, mmc_mode2freq(mmc, MMC_HS), false);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
if ((mode == MMC_HS) || (mode == MMC_HS_52)) {
|
if ((mode == MMC_HS) || (mode == MMC_HS_52)) {
|
||||||
/* Now check to see that it worked */
|
/* Now check to see that it worked */
|
||||||
err = mmc_send_ext_csd(mmc, test_csd);
|
err = mmc_send_ext_csd(mmc, test_csd);
|
||||||
@ -1849,7 +1864,7 @@ static int mmc_select_hs400(struct mmc *mmc)
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
/* Set timing to HS200 for tuning */
|
/* Set timing to HS200 for tuning */
|
||||||
err = mmc_set_card_speed(mmc, MMC_HS_200);
|
err = mmc_set_card_speed(mmc, MMC_HS_200, false);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
@ -1865,7 +1880,7 @@ static int mmc_select_hs400(struct mmc *mmc)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Set back to HS */
|
/* Set back to HS */
|
||||||
mmc_set_card_speed(mmc, MMC_HS);
|
mmc_set_card_speed(mmc, MMC_HS, false);
|
||||||
mmc_set_clock(mmc, mmc_mode2freq(mmc, MMC_HS), false);
|
mmc_set_clock(mmc, mmc_mode2freq(mmc, MMC_HS), false);
|
||||||
|
|
||||||
err = mmc_switch(mmc, EXT_CSD_CMD_SET_NORMAL, EXT_CSD_BUS_WIDTH,
|
err = mmc_switch(mmc, EXT_CSD_CMD_SET_NORMAL, EXT_CSD_BUS_WIDTH,
|
||||||
@ -1873,7 +1888,7 @@ static int mmc_select_hs400(struct mmc *mmc)
|
|||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
err = mmc_set_card_speed(mmc, MMC_HS_400);
|
err = mmc_set_card_speed(mmc, MMC_HS_400, false);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
@ -1920,7 +1935,19 @@ static int mmc_select_mode_and_width(struct mmc *mmc, uint card_caps)
|
|||||||
return -ENOTSUPP;
|
return -ENOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
mmc_set_clock(mmc, mmc->legacy_speed, MMC_CLK_ENABLE);
|
#if CONFIG_IS_ENABLED(MMC_HS200_SUPPORT) || \
|
||||||
|
CONFIG_IS_ENABLED(MMC_HS400_SUPPORT)
|
||||||
|
/*
|
||||||
|
* In case the eMMC is in HS200/HS400 mode, downgrade to HS mode
|
||||||
|
* before doing anything else, since a transition from either of
|
||||||
|
* the HS200/HS400 mode directly to legacy mode is not supported.
|
||||||
|
*/
|
||||||
|
if (mmc->selected_mode == MMC_HS_200 ||
|
||||||
|
mmc->selected_mode == MMC_HS_400)
|
||||||
|
mmc_set_card_speed(mmc, MMC_HS, true);
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
mmc_set_clock(mmc, mmc->legacy_speed, MMC_CLK_ENABLE);
|
||||||
|
|
||||||
for_each_mmc_mode_by_pref(card_caps, mwt) {
|
for_each_mmc_mode_by_pref(card_caps, mwt) {
|
||||||
for_each_supported_width(card_caps & mwt->widths,
|
for_each_supported_width(card_caps & mwt->widths,
|
||||||
@ -1952,7 +1979,7 @@ static int mmc_select_mode_and_width(struct mmc *mmc, uint card_caps)
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* configure the bus speed (card) */
|
/* configure the bus speed (card) */
|
||||||
err = mmc_set_card_speed(mmc, mwt->mode);
|
err = mmc_set_card_speed(mmc, mwt->mode, false);
|
||||||
if (err)
|
if (err)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user