All of lore.kernel.org
 help / color / mirror / Atom feed
* [U-Boot] [PATCH] mmc: stm32_sdmmc2: update pwron management
@ 2018-06-27  8:49 Patrice Chotard
  2018-06-27  9:10 ` Patrice CHOTARD
  2018-07-20 12:35 ` [U-Boot] " Tom Rini
  0 siblings, 2 replies; 4+ messages in thread
From: Patrice Chotard @ 2018-06-27  8:49 UTC (permalink / raw)
  To: u-boot

From: Patrick Delaunay <patrick.delaunay@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.delaunay@st.com>
Signed-off-by: Patrice Chotard <patrice.chotard@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:
-- 
1.9.1

^ permalink raw reply related	[flat|nested] 4+ messages in thread

* [U-Boot] [PATCH] mmc: stm32_sdmmc2: update pwron management
  2018-06-27  8:49 [U-Boot] [PATCH] mmc: stm32_sdmmc2: update pwron management Patrice Chotard
@ 2018-06-27  9:10 ` Patrice CHOTARD
  2018-07-20 12:35 ` [U-Boot] " Tom Rini
  1 sibling, 0 replies; 4+ messages in thread
From: Patrice CHOTARD @ 2018-06-27  9:10 UTC (permalink / raw)
  To: u-boot

Sorry, this patch has been send twice by error.

Patrice

On 06/27/2018 10:49 AM, Patrice Chotard wrote:
> From: Patrick Delaunay <patrick.delaunay@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.delaunay@st.com>
> Signed-off-by: Patrice Chotard <patrice.chotard@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:
> 

^ permalink raw reply	[flat|nested] 4+ messages in thread

* [U-Boot] mmc: stm32_sdmmc2: update pwron management
  2018-06-27  8:49 [U-Boot] [PATCH] mmc: stm32_sdmmc2: update pwron management Patrice Chotard
  2018-06-27  9:10 ` Patrice CHOTARD
@ 2018-07-20 12:35 ` Tom Rini
  1 sibling, 0 replies; 4+ messages in thread
From: Tom Rini @ 2018-07-20 12:35 UTC (permalink / raw)
  To: u-boot

On Wed, Jun 27, 2018 at 10:49:31AM +0200, Patrice Chotard wrote:

> From: Patrick Delaunay <patrick.delaunay@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.delaunay@st.com>
> Signed-off-by: Patrice Chotard <patrice.chotard@st.com>

Applied to u-boot/master, thanks!

-- 
Tom
-------------- next part --------------
A non-text attachment was scrubbed...
Name: signature.asc
Type: application/pgp-signature
Size: 819 bytes
Desc: not available
URL: <http://lists.denx.de/pipermail/u-boot/attachments/20180720/31f3aca0/attachment.sig>

^ permalink raw reply	[flat|nested] 4+ messages in thread

* [U-Boot] [PATCH] mmc: stm32_sdmmc2: update pwron management
@ 2018-06-27  8:15 Patrice Chotard
  0 siblings, 0 replies; 4+ messages in thread
From: Patrice Chotard @ 2018-06-27  8:15 UTC (permalink / raw)
  To: u-boot

From: Patrick Delaunay <patrick.delaunay@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.delaunay@st.com>
Signed-off-by: Patrice Chotard <patrice.chotard@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:
-- 
1.9.1

^ permalink raw reply related	[flat|nested] 4+ messages in thread

end of thread, other threads:[~2018-07-20 12:35 UTC | newest]

Thread overview: 4+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2018-06-27  8:49 [U-Boot] [PATCH] mmc: stm32_sdmmc2: update pwron management Patrice Chotard
2018-06-27  9:10 ` Patrice CHOTARD
2018-07-20 12:35 ` [U-Boot] " Tom Rini
  -- strict thread matches above, loose matches on Subject: below --
2018-06-27  8:15 [U-Boot] [PATCH] " Patrice Chotard

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.