Sorry, this patch has been send twice by error. Patrice
On 06/27/2018 10:49 AM, Patrice Chotard wrote: > From: Patrick Delaunay <patrick.delau...@st.com> > > Correctly manage the SDMMC reset and card cycle power > to fully handle the power cycle added in the MMC uclass > and avoid issue with level-shifter with some uSDCARD. > > 3 states managed in driver: > 1/ reset: SDMMC disable, signal HiZ > 2/ power-cycle: SDMMC disable, signals drive to 0 > 3/ power-on: SDMMC enabled > > Signed-off-by: Patrick Delaunay <patrick.delau...@st.com> > Signed-off-by: Patrice Chotard <patrice.chot...@st.com> > --- > > drivers/mmc/stm32_sdmmc2.c | 78 > +++++++++++++++++++++++++++++++++++++++------- > 1 file changed, 67 insertions(+), 11 deletions(-) > > diff --git a/drivers/mmc/stm32_sdmmc2.c b/drivers/mmc/stm32_sdmmc2.c > index e8292c438d9f..a36612dd937e 100644 > --- a/drivers/mmc/stm32_sdmmc2.c > +++ b/drivers/mmc/stm32_sdmmc2.c > @@ -56,7 +56,10 @@ struct stm32_sdmmc2_ctx { > #define SDMMC_IDMABASE0 0x58 /* SDMMC DMA buffer 0 base > address */ > > /* SDMMC_POWER register */ > -#define SDMMC_POWER_PWRCTRL GENMASK(1, 0) > +#define SDMMC_POWER_PWRCTRL_MASK GENMASK(1, 0) > +#define SDMMC_POWER_PWRCTRL_OFF 0 > +#define SDMMC_POWER_PWRCTRL_CYCLE 2 > +#define SDMMC_POWER_PWRCTRL_ON 3 > #define SDMMC_POWER_VSWITCH BIT(2) > #define SDMMC_POWER_VSWITCHEN BIT(3) > #define SDMMC_POWER_DIRPOL BIT(4) > @@ -440,23 +443,74 @@ retry_cmd: > return ret; > } > > -static void stm32_sdmmc2_pwron(struct stm32_sdmmc2_priv *priv) > +/* > + * Reset the SDMMC with the RCC.SDMMCxRST register bit. > + * This will reset the SDMMC to the reset state and the CPSM and DPSM > + * to the Idle state. SDMMC is disabled, Signals Hiz. > + */ > +static void stm32_sdmmc2_reset(struct stm32_sdmmc2_priv *priv) > { > /* Reset */ > reset_assert(&priv->reset_ctl); > udelay(2); > reset_deassert(&priv->reset_ctl); > > - udelay(1000); > + /* init the needed SDMMC register after reset */ > + writel(priv->pwr_reg_msk, priv->base + SDMMC_POWER); > +} > + > +/* > + * Set the SDMMC in power-cycle state. > + * This will make that the SDMMC_D[7:0], > + * SDMMC_CMD and SDMMC_CK are driven low, to prevent the card from being > + * supplied through the signal lines. > + */ > +static void stm32_sdmmc2_pwrcycle(struct stm32_sdmmc2_priv *priv) > +{ > + if ((readl(priv->base + SDMMC_POWER) & SDMMC_POWER_PWRCTRL_MASK) == > + SDMMC_POWER_PWRCTRL_CYCLE) > + return; > > - /* Set Power State to ON */ > - writel(SDMMC_POWER_PWRCTRL | priv->pwr_reg_msk, priv->base + > SDMMC_POWER); > + stm32_sdmmc2_reset(priv); > + writel(SDMMC_POWER_PWRCTRL_CYCLE | priv->pwr_reg_msk, > + priv->base + SDMMC_POWER); > +} > + > +/* > + * set the SDMMC state Power-on: the card is clocked > + * manage the SDMMC state control: > + * Reset => Power-Cycle => Power-Off => Power > + * PWRCTRL=10 PWCTRL=00 PWCTRL=11 > + */ > +static void stm32_sdmmc2_pwron(struct stm32_sdmmc2_priv *priv) > +{ > + u32 pwrctrl = > + readl(priv->base + SDMMC_POWER) & SDMMC_POWER_PWRCTRL_MASK; > + > + if (pwrctrl == SDMMC_POWER_PWRCTRL_ON) > + return; > + > + /* warning: same PWRCTRL value after reset and for power-off state > + * it is the reset state here = the only managed by the driver > + */ > + if (pwrctrl == SDMMC_POWER_PWRCTRL_OFF) { > + writel(SDMMC_POWER_PWRCTRL_CYCLE | priv->pwr_reg_msk, > + priv->base + SDMMC_POWER); > + } > > /* > - * 1ms: required power up waiting time before starting the > - * SD initialization sequence > + * the remaining case is SDMMC_POWER_PWRCTRL_CYCLE > + * switch to Power-Off state: SDMCC disable, signals drive 1 > */ > - udelay(1000); > + writel(SDMMC_POWER_PWRCTRL_OFF | priv->pwr_reg_msk, > + priv->base + SDMMC_POWER); > + > + /* After the 1ms delay set the SDMMC to power-on */ > + mdelay(1); > + writel(SDMMC_POWER_PWRCTRL_ON | priv->pwr_reg_msk, > + priv->base + SDMMC_POWER); > + > + /* during the first 74 SDMMC_CK cycles the SDMMC is still disabled. */ > } > > #define IS_RISING_EDGE(reg) (reg & SDMMC_CLKCR_NEGEDGE ? 0 : 1) > @@ -464,8 +518,6 @@ static int stm32_sdmmc2_set_ios(struct udevice *dev) > { > struct mmc *mmc = mmc_get_mmc_dev(dev); > struct stm32_sdmmc2_priv *priv = dev_get_priv(dev); > - struct stm32_sdmmc2_plat *plat = dev_get_platdata(dev); > - struct mmc_config *cfg = &plat->cfg; > u32 desired = mmc->clock; > u32 sys_clock = clk_get_rate(&priv->clk); > u32 clk = 0; > @@ -473,7 +525,9 @@ static int stm32_sdmmc2_set_ios(struct udevice *dev) > debug("%s: bus_with = %d, clock = %d\n", __func__, > mmc->bus_width, mmc->clock); > > - if ((mmc->bus_width == 1) && (desired == cfg->f_min)) > + if (mmc->clk_disable) > + stm32_sdmmc2_pwrcycle(priv); > + else > stm32_sdmmc2_pwron(priv); > > /* > @@ -577,6 +631,8 @@ static int stm32_sdmmc2_probe(struct udevice *dev) > > upriv->mmc = &plat->mmc; > > + /* SDMMC init */ > + stm32_sdmmc2_reset(priv); > return 0; > > clk_disable: > _______________________________________________ U-Boot mailing list U-Boot@lists.denx.de https://lists.denx.de/listinfo/u-boot