linux-mmc.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH 0/4] mmc: core: hw_reset changes
@ 2014-10-24 12:46 Johan Rudholm
  2014-10-24 12:46 ` [PATCH 1/4] mmc: core: use mmc_send_status to check hw_reset Johan Rudholm
                   ` (4 more replies)
  0 siblings, 5 replies; 12+ messages in thread
From: Johan Rudholm @ 2014-10-24 12:46 UTC (permalink / raw)
  To: linux-mmc, Chris Ball, Ulf Hansson
  Cc: Adrian Hunter, Guennadi Liakhovetski, David Lanzendörfer,
	Jesper Nilsson, Johan Rudholm

Make the mmc_hw_reset routines more generic, so we can easily add a
power cycle of SD cards as well. Also simplify the (e)MMC specific
parts of the reset code.

As I don't have an eMMC device myself, much less one with a reset line,
I'd be very happy if someone could help me test the code with an eMMC?

Johan Rudholm (4):
  mmc: core: use mmc_send_status to check hw_reset
  mmc: core: use mmc_power_up in hw_reset
  mmc: core: make hw_reset generic
  mmc: sd: add power_reset callback

 drivers/mmc/card/block.c    |    2 +-
 drivers/mmc/card/mmc_test.c |    5 +--
 drivers/mmc/core/core.c     |   73 ++++++++-----------------------------------
 drivers/mmc/core/core.h     |    4 ++
 drivers/mmc/core/mmc.c      |   40 +++++++++++++++++++++++
 drivers/mmc/core/sd.c       |   16 +++++++++
 include/linux/mmc/core.h    |    5 +--
 7 files changed, 77 insertions(+), 68 deletions(-)

-- 
1.7.2.5


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

* [PATCH 1/4] mmc: core: use mmc_send_status to check hw_reset
  2014-10-24 12:46 [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
@ 2014-10-24 12:46 ` Johan Rudholm
  2014-10-24 12:46 ` [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset Johan Rudholm
                   ` (3 subsequent siblings)
  4 siblings, 0 replies; 12+ messages in thread
From: Johan Rudholm @ 2014-10-24 12:46 UTC (permalink / raw)
  To: linux-mmc, Chris Ball, Ulf Hansson
  Cc: Adrian Hunter, Guennadi Liakhovetski, David Lanzendörfer,
	Jesper Nilsson, Johan Rudholm

Signed-off-by: Johan Rudholm <johanru@axis.com>
---
 drivers/mmc/core/core.c |   10 ++--------
 1 files changed, 2 insertions(+), 8 deletions(-)

diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
index 7dc0c85..5d215ee 100644
--- a/drivers/mmc/core/core.c
+++ b/drivers/mmc/core/core.c
@@ -2265,15 +2265,9 @@ static int mmc_do_hw_reset(struct mmc_host *host, int check)
 
 	/* If the reset has happened, then a status command will fail */
 	if (check) {
-		struct mmc_command cmd = {0};
-		int err;
+		u32 status;
 
-		cmd.opcode = MMC_SEND_STATUS;
-		if (!mmc_host_is_spi(card->host))
-			cmd.arg = card->rca << 16;
-		cmd.flags = MMC_RSP_SPI_R2 | MMC_RSP_R1 | MMC_CMD_AC;
-		err = mmc_wait_for_cmd(card->host, &cmd, 0);
-		if (!err) {
+		if (!mmc_send_status(card, &status)) {
 			mmc_host_clk_release(host);
 			return -ENOSYS;
 		}
-- 
1.7.2.5


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

* [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset
  2014-10-24 12:46 [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
  2014-10-24 12:46 ` [PATCH 1/4] mmc: core: use mmc_send_status to check hw_reset Johan Rudholm
@ 2014-10-24 12:46 ` Johan Rudholm
  2014-11-03  9:21   ` Adrian Hunter
  2014-10-24 12:46 ` [PATCH 3/4] mmc: core: make hw_reset generic Johan Rudholm
                   ` (2 subsequent siblings)
  4 siblings, 1 reply; 12+ messages in thread
From: Johan Rudholm @ 2014-10-24 12:46 UTC (permalink / raw)
  To: linux-mmc, Chris Ball, Ulf Hansson
  Cc: Adrian Hunter, Guennadi Liakhovetski, David Lanzendörfer,
	Jesper Nilsson, Johan Rudholm

The steps performed in mmc_do_hw_reset for eMMC:s after the hardware
reset are very similar to those performed by mmc_power_up, so do a call
to this function instead.

Signed-off-by: Johan Rudholm <johanru@axis.com>
---
 drivers/mmc/core/core.c |   11 +----------
 1 files changed, 1 insertions(+), 10 deletions(-)

diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
index 5d215ee..d56e222 100644
--- a/drivers/mmc/core/core.c
+++ b/drivers/mmc/core/core.c
@@ -2273,16 +2273,7 @@ static int mmc_do_hw_reset(struct mmc_host *host, int check)
 		}
 	}
 
-	if (mmc_host_is_spi(host)) {
-		host->ios.chip_select = MMC_CS_HIGH;
-		host->ios.bus_mode = MMC_BUSMODE_PUSHPULL;
-	} else {
-		host->ios.chip_select = MMC_CS_DONTCARE;
-		host->ios.bus_mode = MMC_BUSMODE_OPENDRAIN;
-	}
-	host->ios.bus_width = MMC_BUS_WIDTH_1;
-	host->ios.timing = MMC_TIMING_LEGACY;
-	mmc_set_ios(host);
+	mmc_power_up(host, card->ocr);
 
 	mmc_host_clk_release(host);
 
-- 
1.7.2.5


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

* [PATCH 3/4] mmc: core: make hw_reset generic
  2014-10-24 12:46 [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
  2014-10-24 12:46 ` [PATCH 1/4] mmc: core: use mmc_send_status to check hw_reset Johan Rudholm
  2014-10-24 12:46 ` [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset Johan Rudholm
@ 2014-10-24 12:46 ` Johan Rudholm
  2014-11-03  9:20   ` Adrian Hunter
  2014-10-24 12:46 ` [PATCH 4/4] mmc: sd: add power_reset callback Johan Rudholm
  2014-11-03  8:46 ` [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
  4 siblings, 1 reply; 12+ messages in thread
From: Johan Rudholm @ 2014-10-24 12:46 UTC (permalink / raw)
  To: linux-mmc, Chris Ball, Ulf Hansson
  Cc: Adrian Hunter, Guennadi Liakhovetski, David Lanzendörfer,
	Jesper Nilsson, Johan Rudholm

Move the (e)MMC specific hw_reset code from core.c into mmc.c and call
it from the new bus_ops member power_reset. This also lets us add code
for reseting SD cards as well.

Rename the mmc_hw_reset* functions into mmc_reset*, since what they
now actually do depends on the device type (and it may be something else
than doing a hw_reset).

Signed-off-by: Johan Rudholm <johanru@axis.com>
---
 drivers/mmc/card/block.c    |    2 +-
 drivers/mmc/card/mmc_test.c |    5 +---
 drivers/mmc/core/core.c     |   58 +++++++++---------------------------------
 drivers/mmc/core/core.h     |    4 +++
 drivers/mmc/core/mmc.c      |   40 +++++++++++++++++++++++++++++
 include/linux/mmc/core.h    |    5 +--
 6 files changed, 61 insertions(+), 53 deletions(-)

diff --git a/drivers/mmc/card/block.c b/drivers/mmc/card/block.c
index 452782b..8c4fa46 100644
--- a/drivers/mmc/card/block.c
+++ b/drivers/mmc/card/block.c
@@ -1001,7 +1001,7 @@ static int mmc_blk_reset(struct mmc_blk_data *md, struct mmc_host *host,
 		return -EEXIST;
 
 	md->reset_done |= type;
-	err = mmc_hw_reset(host);
+	err = mmc_reset(host);
 	/* Ensure we switch back to the correct partition */
 	if (err != -EOPNOTSUPP) {
 		struct mmc_blk_data *main_md = mmc_get_drvdata(host->card);
diff --git a/drivers/mmc/card/mmc_test.c b/drivers/mmc/card/mmc_test.c
index 0c0fc52..599c683 100644
--- a/drivers/mmc/card/mmc_test.c
+++ b/drivers/mmc/card/mmc_test.c
@@ -2339,7 +2339,7 @@ static int mmc_test_hw_reset(struct mmc_test_card *test)
 	struct mmc_host *host = card->host;
 	int err;
 
-	err = mmc_hw_reset_check(host);
+	err = mmc_reset_check(host);
 	if (!err)
 		return RESULT_OK;
 
@@ -2349,9 +2349,6 @@ static int mmc_test_hw_reset(struct mmc_test_card *test)
 	if (err != -EOPNOTSUPP)
 		return err;
 
-	if (!mmc_can_reset(card))
-		return RESULT_UNSUP_CARD;
-
 	return RESULT_UNSUP_HOST;
 }
 
diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
index d56e222..486cda8 100644
--- a/drivers/mmc/core/core.c
+++ b/drivers/mmc/core/core.c
@@ -2232,65 +2232,33 @@ static void mmc_hw_reset_for_init(struct mmc_host *host)
 	mmc_host_clk_release(host);
 }
 
-int mmc_can_reset(struct mmc_card *card)
+/* Reset card in a bus-specific way */
+static int mmc_do_reset(struct mmc_host *host, int check)
 {
-	u8 rst_n_function;
-
-	if (!mmc_card_mmc(card))
-		return 0;
-	rst_n_function = card->ext_csd.rst_n_function;
-	if ((rst_n_function & EXT_CSD_RST_N_EN_MASK) != EXT_CSD_RST_N_ENABLED)
-		return 0;
-	return 1;
-}
-EXPORT_SYMBOL(mmc_can_reset);
-
-static int mmc_do_hw_reset(struct mmc_host *host, int check)
-{
-	struct mmc_card *card = host->card;
-
-	if (!(host->caps & MMC_CAP_HW_RESET) || !host->ops->hw_reset)
-		return -EOPNOTSUPP;
-
-	if (!card)
-		return -EINVAL;
+	int ret;
 
-	if (!mmc_can_reset(card))
+	if (!host->bus_ops || !host->bus_ops->power_reset ||
+			host->bus_ops->power_reset(host, MMC_POWER_RESET_TEST))
 		return -EOPNOTSUPP;
 
-	mmc_host_clk_hold(host);
-	mmc_set_clock(host, host->f_init);
-
-	host->ops->hw_reset(host);
-
-	/* If the reset has happened, then a status command will fail */
-	if (check) {
-		u32 status;
-
-		if (!mmc_send_status(card, &status)) {
-			mmc_host_clk_release(host);
-			return -ENOSYS;
-		}
-	}
-
-	mmc_power_up(host, card->ocr);
+	ret = host->bus_ops->power_reset(host, check);
 
-	mmc_host_clk_release(host);
+	pr_warning("%s: tried to reset card (%d)\n", mmc_hostname(host), ret);
 
 	return host->bus_ops->power_restore(host);
 }
 
-int mmc_hw_reset(struct mmc_host *host)
+int mmc_reset(struct mmc_host *host)
 {
-	return mmc_do_hw_reset(host, 0);
+	return mmc_do_reset(host, MMC_POWER_RESET_RESET);
 }
-EXPORT_SYMBOL(mmc_hw_reset);
+EXPORT_SYMBOL(mmc_reset);
 
-int mmc_hw_reset_check(struct mmc_host *host)
+int mmc_reset_check(struct mmc_host *host)
 {
-	return mmc_do_hw_reset(host, 1);
+	return mmc_do_reset(host, MMC_POWER_RESET_CHECK);
 }
-EXPORT_SYMBOL(mmc_hw_reset_check);
+EXPORT_SYMBOL(mmc_reset_check);
 
 static int mmc_rescan_try_freq(struct mmc_host *host, unsigned freq)
 {
diff --git a/drivers/mmc/core/core.h b/drivers/mmc/core/core.h
index 443a584..6a0420b 100644
--- a/drivers/mmc/core/core.h
+++ b/drivers/mmc/core/core.h
@@ -25,6 +25,10 @@ struct mmc_bus_ops {
 	int (*runtime_resume)(struct mmc_host *);
 	int (*power_save)(struct mmc_host *);
 	int (*power_restore)(struct mmc_host *);
+	int (*power_reset)(struct mmc_host *, int);
+#define MMC_POWER_RESET_RESET	0
+#define MMC_POWER_RESET_TEST	1
+#define MMC_POWER_RESET_CHECK	2
 	int (*alive)(struct mmc_host *);
 	int (*shutdown)(struct mmc_host *);
 };
diff --git a/drivers/mmc/core/mmc.c b/drivers/mmc/core/mmc.c
index 793c6f7..ac5192c 100644
--- a/drivers/mmc/core/mmc.c
+++ b/drivers/mmc/core/mmc.c
@@ -1762,6 +1762,45 @@ static int mmc_runtime_resume(struct mmc_host *host)
 	return 0;
 }
 
+static int mmc_power_reset(struct mmc_host *host, int test)
+{
+	struct mmc_card *card = host->card;
+	u8 rst_n_function;
+
+	if (!card)
+		return -EINVAL;
+
+	if (!(host->caps & MMC_CAP_HW_RESET) || !host->ops->hw_reset)
+		return -EOPNOTSUPP;
+
+	rst_n_function = card->ext_csd.rst_n_function;
+	if ((rst_n_function & EXT_CSD_RST_N_EN_MASK) != EXT_CSD_RST_N_ENABLED)
+		return -EOPNOTSUPP;
+
+	if (test == MMC_POWER_RESET_TEST)
+		return 0;
+
+	mmc_host_clk_hold(host);
+	mmc_set_clock(host, host->f_init);
+
+	host->ops->hw_reset(host);
+
+	if (test == MMC_POWER_RESET_CHECK) {
+		u32 status;
+
+		if (!mmc_send_status(card, &status)) {
+			mmc_host_clk_release(host);
+			return -ENOSYS;
+		}
+	}
+
+	mmc_power_up(host, card->ocr);
+
+	mmc_host_clk_release(host);
+
+	return 0;
+}
+
 static int mmc_power_restore(struct mmc_host *host)
 {
 	int ret;
@@ -1781,6 +1820,7 @@ static const struct mmc_bus_ops mmc_ops = {
 	.runtime_suspend = mmc_runtime_suspend,
 	.runtime_resume = mmc_runtime_resume,
 	.power_restore = mmc_power_restore,
+	.power_reset = mmc_power_reset,
 	.alive = mmc_alive,
 	.shutdown = mmc_shutdown,
 };
diff --git a/include/linux/mmc/core.h b/include/linux/mmc/core.h
index f206e29..4f7a76e 100644
--- a/include/linux/mmc/core.h
+++ b/include/linux/mmc/core.h
@@ -180,9 +180,8 @@ extern unsigned int mmc_calc_max_discard(struct mmc_card *card);
 extern int mmc_set_blocklen(struct mmc_card *card, unsigned int blocklen);
 extern int mmc_set_blockcount(struct mmc_card *card, unsigned int blockcount,
 			      bool is_rel_write);
-extern int mmc_hw_reset(struct mmc_host *host);
-extern int mmc_hw_reset_check(struct mmc_host *host);
-extern int mmc_can_reset(struct mmc_card *card);
+extern int mmc_reset(struct mmc_host *host);
+extern int mmc_reset_check(struct mmc_host *host);
 
 extern void mmc_set_data_timeout(struct mmc_data *, const struct mmc_card *);
 extern unsigned int mmc_align_data_size(struct mmc_card *, unsigned int);
-- 
1.7.2.5


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

* [PATCH 4/4] mmc: sd: add power_reset callback
  2014-10-24 12:46 [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
                   ` (2 preceding siblings ...)
  2014-10-24 12:46 ` [PATCH 3/4] mmc: core: make hw_reset generic Johan Rudholm
@ 2014-10-24 12:46 ` Johan Rudholm
  2014-11-03  8:46 ` [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
  4 siblings, 0 replies; 12+ messages in thread
From: Johan Rudholm @ 2014-10-24 12:46 UTC (permalink / raw)
  To: linux-mmc, Chris Ball, Ulf Hansson
  Cc: Adrian Hunter, Guennadi Liakhovetski, David Lanzendörfer,
	Jesper Nilsson, Johan Rudholm

Enable power cycle and re-initialization of SD cards via the mmc_reset
function. Power cycling a buggy SD card sometimes helps it get back on
track.

Signed-off-by: Johan Rudholm <johanru@axis.com>
---
 drivers/mmc/core/sd.c |   16 ++++++++++++++++
 1 files changed, 16 insertions(+), 0 deletions(-)

diff --git a/drivers/mmc/core/sd.c b/drivers/mmc/core/sd.c
index 0c44510..9ad46b2 100644
--- a/drivers/mmc/core/sd.c
+++ b/drivers/mmc/core/sd.c
@@ -1177,6 +1177,21 @@ static int mmc_sd_runtime_resume(struct mmc_host *host)
 	return 0;
 }
 
+static int mmc_sd_power_reset(struct mmc_host *host, int test)
+{
+	struct mmc_card *card = host->card;
+
+	if (!card)
+		return -EINVAL;
+
+	if (test == MMC_POWER_RESET_TEST)
+		return 0;
+
+	mmc_power_cycle(host, card->ocr);
+
+	return 0;
+}
+
 static int mmc_sd_power_restore(struct mmc_host *host)
 {
 	int ret;
@@ -1196,6 +1211,7 @@ static const struct mmc_bus_ops mmc_sd_ops = {
 	.suspend = mmc_sd_suspend,
 	.resume = mmc_sd_resume,
 	.power_restore = mmc_sd_power_restore,
+	.power_reset = mmc_sd_power_reset,
 	.alive = mmc_sd_alive,
 	.shutdown = mmc_sd_suspend,
 };
-- 
1.7.2.5


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

* Re: [PATCH 0/4] mmc: core: hw_reset changes
  2014-10-24 12:46 [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
                   ` (3 preceding siblings ...)
  2014-10-24 12:46 ` [PATCH 4/4] mmc: sd: add power_reset callback Johan Rudholm
@ 2014-11-03  8:46 ` Johan Rudholm
  4 siblings, 0 replies; 12+ messages in thread
From: Johan Rudholm @ 2014-11-03  8:46 UTC (permalink / raw)
  To: Johan Rudholm
  Cc: linux-mmc@vger.kernel.org, Chris Ball, Ulf Hansson, Adrian Hunter,
	Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson

Hi all,

any comments on this patch set?

//Johan

2014-10-24 14:46 GMT+02:00 Johan Rudholm <johan.rudholm@axis.com>:
> Make the mmc_hw_reset routines more generic, so we can easily add a
> power cycle of SD cards as well. Also simplify the (e)MMC specific
> parts of the reset code.
>
> As I don't have an eMMC device myself, much less one with a reset line,
> I'd be very happy if someone could help me test the code with an eMMC?
>
> Johan Rudholm (4):
>   mmc: core: use mmc_send_status to check hw_reset
>   mmc: core: use mmc_power_up in hw_reset
>   mmc: core: make hw_reset generic
>   mmc: sd: add power_reset callback
>
>  drivers/mmc/card/block.c    |    2 +-
>  drivers/mmc/card/mmc_test.c |    5 +--
>  drivers/mmc/core/core.c     |   73 ++++++++-----------------------------------
>  drivers/mmc/core/core.h     |    4 ++
>  drivers/mmc/core/mmc.c      |   40 +++++++++++++++++++++++
>  drivers/mmc/core/sd.c       |   16 +++++++++
>  include/linux/mmc/core.h    |    5 +--
>  7 files changed, 77 insertions(+), 68 deletions(-)
>
> --
> 1.7.2.5
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-mmc" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

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

* Re: [PATCH 3/4] mmc: core: make hw_reset generic
  2014-10-24 12:46 ` [PATCH 3/4] mmc: core: make hw_reset generic Johan Rudholm
@ 2014-11-03  9:20   ` Adrian Hunter
  2014-11-03 10:19     ` Johan Rudholm
  0 siblings, 1 reply; 12+ messages in thread
From: Adrian Hunter @ 2014-11-03  9:20 UTC (permalink / raw)
  To: Johan Rudholm, linux-mmc, Chris Ball, Ulf Hansson
  Cc: Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson,
	Johan Rudholm

On 24/10/14 15:46, Johan Rudholm wrote:
> Move the (e)MMC specific hw_reset code from core.c into mmc.c and call
> it from the new bus_ops member power_reset. This also lets us add code

power_reset is not a good name because it does not necessarily have
anything to do with power.  I am not sure why you don't stick with
hw_reset.

> for reseting SD cards as well.
> 
> Rename the mmc_hw_reset* functions into mmc_reset*, since what they
> now actually do depends on the device type (and it may be something else
> than doing a hw_reset).

I don't follow your reasoning about the rename.  Cycling the power is
a kind of hardware reset isn't it.

> 
> Signed-off-by: Johan Rudholm <johanru@axis.com>
> ---
>  drivers/mmc/card/block.c    |    2 +-
>  drivers/mmc/card/mmc_test.c |    5 +---
>  drivers/mmc/core/core.c     |   58 +++++++++---------------------------------
>  drivers/mmc/core/core.h     |    4 +++
>  drivers/mmc/core/mmc.c      |   40 +++++++++++++++++++++++++++++
>  include/linux/mmc/core.h    |    5 +--
>  6 files changed, 61 insertions(+), 53 deletions(-)
> 
> diff --git a/drivers/mmc/card/block.c b/drivers/mmc/card/block.c
> index 452782b..8c4fa46 100644
> --- a/drivers/mmc/card/block.c
> +++ b/drivers/mmc/card/block.c
> @@ -1001,7 +1001,7 @@ static int mmc_blk_reset(struct mmc_blk_data *md, struct mmc_host *host,
>  		return -EEXIST;
>  
>  	md->reset_done |= type;
> -	err = mmc_hw_reset(host);
> +	err = mmc_reset(host);
>  	/* Ensure we switch back to the correct partition */
>  	if (err != -EOPNOTSUPP) {
>  		struct mmc_blk_data *main_md = mmc_get_drvdata(host->card);
> diff --git a/drivers/mmc/card/mmc_test.c b/drivers/mmc/card/mmc_test.c
> index 0c0fc52..599c683 100644
> --- a/drivers/mmc/card/mmc_test.c
> +++ b/drivers/mmc/card/mmc_test.c
> @@ -2339,7 +2339,7 @@ static int mmc_test_hw_reset(struct mmc_test_card *test)
>  	struct mmc_host *host = card->host;
>  	int err;
>  
> -	err = mmc_hw_reset_check(host);
> +	err = mmc_reset_check(host);
>  	if (!err)
>  		return RESULT_OK;
>  
> @@ -2349,9 +2349,6 @@ static int mmc_test_hw_reset(struct mmc_test_card *test)
>  	if (err != -EOPNOTSUPP)
>  		return err;
>  
> -	if (!mmc_can_reset(card))
> -		return RESULT_UNSUP_CARD;
> -
>  	return RESULT_UNSUP_HOST;
>  }
>  
> diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
> index d56e222..486cda8 100644
> --- a/drivers/mmc/core/core.c
> +++ b/drivers/mmc/core/core.c
> @@ -2232,65 +2232,33 @@ static void mmc_hw_reset_for_init(struct mmc_host *host)
>  	mmc_host_clk_release(host);
>  }
>  
> -int mmc_can_reset(struct mmc_card *card)
> +/* Reset card in a bus-specific way */
> +static int mmc_do_reset(struct mmc_host *host, int check)
>  {
> -	u8 rst_n_function;
> -
> -	if (!mmc_card_mmc(card))
> -		return 0;
> -	rst_n_function = card->ext_csd.rst_n_function;
> -	if ((rst_n_function & EXT_CSD_RST_N_EN_MASK) != EXT_CSD_RST_N_ENABLED)
> -		return 0;
> -	return 1;
> -}
> -EXPORT_SYMBOL(mmc_can_reset);
> -
> -static int mmc_do_hw_reset(struct mmc_host *host, int check)
> -{
> -	struct mmc_card *card = host->card;
> -
> -	if (!(host->caps & MMC_CAP_HW_RESET) || !host->ops->hw_reset)
> -		return -EOPNOTSUPP;
> -
> -	if (!card)
> -		return -EINVAL;
> +	int ret;
>  
> -	if (!mmc_can_reset(card))
> +	if (!host->bus_ops || !host->bus_ops->power_reset ||
> +			host->bus_ops->power_reset(host, MMC_POWER_RESET_TEST))
>  		return -EOPNOTSUPP;
>  
> -	mmc_host_clk_hold(host);
> -	mmc_set_clock(host, host->f_init);
> -
> -	host->ops->hw_reset(host);
> -
> -	/* If the reset has happened, then a status command will fail */
> -	if (check) {
> -		u32 status;
> -
> -		if (!mmc_send_status(card, &status)) {
> -			mmc_host_clk_release(host);
> -			return -ENOSYS;
> -		}
> -	}
> -
> -	mmc_power_up(host, card->ocr);
> +	ret = host->bus_ops->power_reset(host, check);
>  
> -	mmc_host_clk_release(host);
> +	pr_warning("%s: tried to reset card (%d)\n", mmc_hostname(host), ret);
>  
>  	return host->bus_ops->power_restore(host);
>  }
>  
> -int mmc_hw_reset(struct mmc_host *host)
> +int mmc_reset(struct mmc_host *host)
>  {
> -	return mmc_do_hw_reset(host, 0);
> +	return mmc_do_reset(host, MMC_POWER_RESET_RESET);
>  }
> -EXPORT_SYMBOL(mmc_hw_reset);
> +EXPORT_SYMBOL(mmc_reset);
>  
> -int mmc_hw_reset_check(struct mmc_host *host)
> +int mmc_reset_check(struct mmc_host *host)
>  {
> -	return mmc_do_hw_reset(host, 1);
> +	return mmc_do_reset(host, MMC_POWER_RESET_CHECK);
>  }
> -EXPORT_SYMBOL(mmc_hw_reset_check);
> +EXPORT_SYMBOL(mmc_reset_check);
>  
>  static int mmc_rescan_try_freq(struct mmc_host *host, unsigned freq)
>  {
> diff --git a/drivers/mmc/core/core.h b/drivers/mmc/core/core.h
> index 443a584..6a0420b 100644
> --- a/drivers/mmc/core/core.h
> +++ b/drivers/mmc/core/core.h
> @@ -25,6 +25,10 @@ struct mmc_bus_ops {
>  	int (*runtime_resume)(struct mmc_host *);
>  	int (*power_save)(struct mmc_host *);
>  	int (*power_restore)(struct mmc_host *);
> +	int (*power_reset)(struct mmc_host *, int);
> +#define MMC_POWER_RESET_RESET	0
> +#define MMC_POWER_RESET_TEST	1
> +#define MMC_POWER_RESET_CHECK	2
>  	int (*alive)(struct mmc_host *);
>  	int (*shutdown)(struct mmc_host *);
>  };
> diff --git a/drivers/mmc/core/mmc.c b/drivers/mmc/core/mmc.c
> index 793c6f7..ac5192c 100644
> --- a/drivers/mmc/core/mmc.c
> +++ b/drivers/mmc/core/mmc.c
> @@ -1762,6 +1762,45 @@ static int mmc_runtime_resume(struct mmc_host *host)
>  	return 0;
>  }
>  
> +static int mmc_power_reset(struct mmc_host *host, int test)
> +{
> +	struct mmc_card *card = host->card;
> +	u8 rst_n_function;
> +
> +	if (!card)
> +		return -EINVAL;
> +
> +	if (!(host->caps & MMC_CAP_HW_RESET) || !host->ops->hw_reset)
> +		return -EOPNOTSUPP;
> +
> +	rst_n_function = card->ext_csd.rst_n_function;
> +	if ((rst_n_function & EXT_CSD_RST_N_EN_MASK) != EXT_CSD_RST_N_ENABLED)
> +		return -EOPNOTSUPP;
> +
> +	if (test == MMC_POWER_RESET_TEST)
> +		return 0;
> +
> +	mmc_host_clk_hold(host);
> +	mmc_set_clock(host, host->f_init);
> +
> +	host->ops->hw_reset(host);
> +
> +	if (test == MMC_POWER_RESET_CHECK) {
> +		u32 status;
> +
> +		if (!mmc_send_status(card, &status)) {
> +			mmc_host_clk_release(host);
> +			return -ENOSYS;
> +		}
> +	}
> +
> +	mmc_power_up(host, card->ocr);
> +
> +	mmc_host_clk_release(host);
> +
> +	return 0;
> +}
> +
>  static int mmc_power_restore(struct mmc_host *host)
>  {
>  	int ret;
> @@ -1781,6 +1820,7 @@ static const struct mmc_bus_ops mmc_ops = {
>  	.runtime_suspend = mmc_runtime_suspend,
>  	.runtime_resume = mmc_runtime_resume,
>  	.power_restore = mmc_power_restore,
> +	.power_reset = mmc_power_reset,
>  	.alive = mmc_alive,
>  	.shutdown = mmc_shutdown,
>  };
> diff --git a/include/linux/mmc/core.h b/include/linux/mmc/core.h
> index f206e29..4f7a76e 100644
> --- a/include/linux/mmc/core.h
> +++ b/include/linux/mmc/core.h
> @@ -180,9 +180,8 @@ extern unsigned int mmc_calc_max_discard(struct mmc_card *card);
>  extern int mmc_set_blocklen(struct mmc_card *card, unsigned int blocklen);
>  extern int mmc_set_blockcount(struct mmc_card *card, unsigned int blockcount,
>  			      bool is_rel_write);
> -extern int mmc_hw_reset(struct mmc_host *host);
> -extern int mmc_hw_reset_check(struct mmc_host *host);
> -extern int mmc_can_reset(struct mmc_card *card);
> +extern int mmc_reset(struct mmc_host *host);
> +extern int mmc_reset_check(struct mmc_host *host);
>  
>  extern void mmc_set_data_timeout(struct mmc_data *, const struct mmc_card *);
>  extern unsigned int mmc_align_data_size(struct mmc_card *, unsigned int);
> 


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

* Re: [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset
  2014-10-24 12:46 ` [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset Johan Rudholm
@ 2014-11-03  9:21   ` Adrian Hunter
  2014-11-03 10:13     ` Johan Rudholm
  0 siblings, 1 reply; 12+ messages in thread
From: Adrian Hunter @ 2014-11-03  9:21 UTC (permalink / raw)
  To: Johan Rudholm, linux-mmc, Chris Ball, Ulf Hansson
  Cc: Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson,
	Johan Rudholm

On 24/10/14 15:46, Johan Rudholm wrote:
> The steps performed in mmc_do_hw_reset for eMMC:s after the hardware
> reset are very similar to those performed by mmc_power_up, so do a call

They perform different functions.  mmc_power_up() does nothing if the card
is powered up.  To share the common code you need to create a function for
the common functionality which is setting the initial state e.g.

int mmc_set_initial_state(struct mmc_host *host)
{
	...
}

And call that from mmc_do_hw_reset and mmc_power_up.

> to this function instead.
> 
> Signed-off-by: Johan Rudholm <johanru@axis.com>
> ---
>  drivers/mmc/core/core.c |   11 +----------
>  1 files changed, 1 insertions(+), 10 deletions(-)
> 
> diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
> index 5d215ee..d56e222 100644
> --- a/drivers/mmc/core/core.c
> +++ b/drivers/mmc/core/core.c
> @@ -2273,16 +2273,7 @@ static int mmc_do_hw_reset(struct mmc_host *host, int check)
>  		}
>  	}
>  
> -	if (mmc_host_is_spi(host)) {
> -		host->ios.chip_select = MMC_CS_HIGH;
> -		host->ios.bus_mode = MMC_BUSMODE_PUSHPULL;
> -	} else {
> -		host->ios.chip_select = MMC_CS_DONTCARE;
> -		host->ios.bus_mode = MMC_BUSMODE_OPENDRAIN;
> -	}
> -	host->ios.bus_width = MMC_BUS_WIDTH_1;
> -	host->ios.timing = MMC_TIMING_LEGACY;
> -	mmc_set_ios(host);
> +	mmc_power_up(host, card->ocr);
>  
>  	mmc_host_clk_release(host);
>  
> 


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

* Re: [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset
  2014-11-03  9:21   ` Adrian Hunter
@ 2014-11-03 10:13     ` Johan Rudholm
  2014-11-03 10:59       ` Adrian Hunter
  0 siblings, 1 reply; 12+ messages in thread
From: Johan Rudholm @ 2014-11-03 10:13 UTC (permalink / raw)
  To: Adrian Hunter
  Cc: Johan Rudholm, linux-mmc@vger.kernel.org, Chris Ball, Ulf Hansson,
	Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson

2014-11-03 10:21 GMT+01:00 Adrian Hunter <adrian.hunter@intel.com>:
> On 24/10/14 15:46, Johan Rudholm wrote:
>> The steps performed in mmc_do_hw_reset for eMMC:s after the hardware
>> reset are very similar to those performed by mmc_power_up, so do a call
>
> They perform different functions.  mmc_power_up() does nothing if the card
> is powered up.  To share the common code you need to create a function for
> the common functionality which is setting the initial state e.g.
>
> int mmc_set_initial_state(struct mmc_host *host)
> {
>         ...
> }
>
> And call that from mmc_do_hw_reset and mmc_power_up.

Thank you for the comment. What do you think of doing a complete
mmc_power_cycle after having performed a (successful) reset? In this
way we can simplify the SD card reset and we don't have to break out
the common code from mmc_power_up. Naming discussion apart, something
like this:

diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
index 486cda8..941f631 100644
--- a/drivers/mmc/core/core.c
+++ b/drivers/mmc/core/core.c
@@ -2236,12 +2236,15 @@ static void mmc_hw_reset_for_init(struct mmc_host *host)
 static int mmc_do_reset(struct mmc_host *host, int check)
 {
        int ret;
+       struct mmc_card *card = host->card;

        if (!host->bus_ops || !host->bus_ops->power_reset ||
                        host->bus_ops->power_reset(host, MMC_POWER_RESET_TEST))
                return -EOPNOTSUPP;

        ret = host->bus_ops->power_reset(host, check);
+       if (!ret && card)
+               mmc_power_cycle(host, card->ocr);

        pr_warning("%s: tried to reset card (%d)\n", mmc_hostname(host), ret);

diff --git a/drivers/mmc/core/mmc.c b/drivers/mmc/core/mmc.c
index ac5192c..900133a 100644
--- a/drivers/mmc/core/mmc.c
+++ b/drivers/mmc/core/mmc.c
@@ -1794,8 +1794,6 @@ static int mmc_power_reset(struct mmc_host
*host, int test)
                }
        }

-       mmc_power_up(host, card->ocr);
-
        mmc_host_clk_release(host);

        return 0;
diff --git a/drivers/mmc/core/sd.c b/drivers/mmc/core/sd.c
index 9ad46b2..d5b8ee7 100644
--- a/drivers/mmc/core/sd.c
+++ b/drivers/mmc/core/sd.c
@@ -1187,8 +1187,6 @@ static int mmc_sd_power_reset(struct mmc_host
*host, int test)
        if (test == MMC_POWER_RESET_TEST)
                return 0;

-       mmc_power_cycle(host, card->ocr);
-
        return 0;
 }

//Johan

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

* Re: [PATCH 3/4] mmc: core: make hw_reset generic
  2014-11-03  9:20   ` Adrian Hunter
@ 2014-11-03 10:19     ` Johan Rudholm
  2014-11-03 10:56       ` Adrian Hunter
  0 siblings, 1 reply; 12+ messages in thread
From: Johan Rudholm @ 2014-11-03 10:19 UTC (permalink / raw)
  To: Adrian Hunter
  Cc: Johan Rudholm, linux-mmc@vger.kernel.org, Chris Ball, Ulf Hansson,
	Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson

2014-11-03 10:20 GMT+01:00 Adrian Hunter <adrian.hunter@intel.com>:
> On 24/10/14 15:46, Johan Rudholm wrote:
>> Move the (e)MMC specific hw_reset code from core.c into mmc.c and call
>> it from the new bus_ops member power_reset. This also lets us add code
>
> power_reset is not a good name because it does not necessarily have
> anything to do with power.  I am not sure why you don't stick with
> hw_reset.
>
>> for reseting SD cards as well.
>>
>> Rename the mmc_hw_reset* functions into mmc_reset*, since what they
>> now actually do depends on the device type (and it may be something else
>> than doing a hw_reset).
>
> I don't follow your reasoning about the rename.  Cycling the power is
> a kind of hardware reset isn't it.

Since we finalize the reset by calling bus_ops->power_restore, I was
looking for something symmetrical. Ulf mentioned perhaps extending
bus_ops->power_save, but I couldn't find a reasonable way of doing
this. Do you propose simply calling it bus_ops->hw_reset ?

//Johan

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

* Re: [PATCH 3/4] mmc: core: make hw_reset generic
  2014-11-03 10:19     ` Johan Rudholm
@ 2014-11-03 10:56       ` Adrian Hunter
  0 siblings, 0 replies; 12+ messages in thread
From: Adrian Hunter @ 2014-11-03 10:56 UTC (permalink / raw)
  To: Johan Rudholm
  Cc: Johan Rudholm, linux-mmc@vger.kernel.org, Chris Ball, Ulf Hansson,
	Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson

On 03/11/14 12:19, Johan Rudholm wrote:
> 2014-11-03 10:20 GMT+01:00 Adrian Hunter <adrian.hunter@intel.com>:
>> On 24/10/14 15:46, Johan Rudholm wrote:
>>> Move the (e)MMC specific hw_reset code from core.c into mmc.c and call
>>> it from the new bus_ops member power_reset. This also lets us add code
>>
>> power_reset is not a good name because it does not necessarily have
>> anything to do with power.  I am not sure why you don't stick with
>> hw_reset.
>>
>>> for reseting SD cards as well.
>>>
>>> Rename the mmc_hw_reset* functions into mmc_reset*, since what they
>>> now actually do depends on the device type (and it may be something else
>>> than doing a hw_reset).
>>
>> I don't follow your reasoning about the rename.  Cycling the power is
>> a kind of hardware reset isn't it.
> 
> Since we finalize the reset by calling bus_ops->power_restore, I was
> looking for something symmetrical. Ulf mentioned perhaps extending
> bus_ops->power_save, but I couldn't find a reasonable way of doing
> this. Do you propose simply calling it bus_ops->hw_reset ?

Yes



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

* Re: [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset
  2014-11-03 10:13     ` Johan Rudholm
@ 2014-11-03 10:59       ` Adrian Hunter
  0 siblings, 0 replies; 12+ messages in thread
From: Adrian Hunter @ 2014-11-03 10:59 UTC (permalink / raw)
  To: Johan Rudholm
  Cc: Johan Rudholm, linux-mmc@vger.kernel.org, Chris Ball, Ulf Hansson,
	Guennadi Liakhovetski, David Lanzendörfer, Jesper Nilsson

On 03/11/14 12:13, Johan Rudholm wrote:
> 2014-11-03 10:21 GMT+01:00 Adrian Hunter <adrian.hunter@intel.com>:
>> On 24/10/14 15:46, Johan Rudholm wrote:
>>> The steps performed in mmc_do_hw_reset for eMMC:s after the hardware
>>> reset are very similar to those performed by mmc_power_up, so do a call
>>
>> They perform different functions.  mmc_power_up() does nothing if the card
>> is powered up.  To share the common code you need to create a function for
>> the common functionality which is setting the initial state e.g.
>>
>> int mmc_set_initial_state(struct mmc_host *host)
>> {
>>         ...
>> }
>>
>> And call that from mmc_do_hw_reset and mmc_power_up.
> 
> Thank you for the comment. What do you think of doing a complete
> mmc_power_cycle after having performed a (successful) reset? In this
> way we can simplify the SD card reset and we don't have to break out
> the common code from mmc_power_up. Naming discussion apart, something
> like this:

I would keep the concept of hw_reset separate from how that is
implemented i.e. whether it has anything to do with power.

> 
> diff --git a/drivers/mmc/core/core.c b/drivers/mmc/core/core.c
> index 486cda8..941f631 100644
> --- a/drivers/mmc/core/core.c
> +++ b/drivers/mmc/core/core.c
> @@ -2236,12 +2236,15 @@ static void mmc_hw_reset_for_init(struct mmc_host *host)
>  static int mmc_do_reset(struct mmc_host *host, int check)
>  {
>         int ret;
> +       struct mmc_card *card = host->card;
> 
>         if (!host->bus_ops || !host->bus_ops->power_reset ||
>                         host->bus_ops->power_reset(host, MMC_POWER_RESET_TEST))
>                 return -EOPNOTSUPP;
> 
>         ret = host->bus_ops->power_reset(host, check);
> +       if (!ret && card)
> +               mmc_power_cycle(host, card->ocr);
> 
>         pr_warning("%s: tried to reset card (%d)\n", mmc_hostname(host), ret);
> 
> diff --git a/drivers/mmc/core/mmc.c b/drivers/mmc/core/mmc.c
> index ac5192c..900133a 100644
> --- a/drivers/mmc/core/mmc.c
> +++ b/drivers/mmc/core/mmc.c
> @@ -1794,8 +1794,6 @@ static int mmc_power_reset(struct mmc_host
> *host, int test)
>                 }
>         }
> 
> -       mmc_power_up(host, card->ocr);
> -
>         mmc_host_clk_release(host);
> 
>         return 0;
> diff --git a/drivers/mmc/core/sd.c b/drivers/mmc/core/sd.c
> index 9ad46b2..d5b8ee7 100644
> --- a/drivers/mmc/core/sd.c
> +++ b/drivers/mmc/core/sd.c
> @@ -1187,8 +1187,6 @@ static int mmc_sd_power_reset(struct mmc_host
> *host, int test)
>         if (test == MMC_POWER_RESET_TEST)
>                 return 0;
> 
> -       mmc_power_cycle(host, card->ocr);
> -
>         return 0;
>  }
> 
> //Johan
> 
> 


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

end of thread, other threads:[~2014-11-03 11:01 UTC | newest]

Thread overview: 12+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-10-24 12:46 [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm
2014-10-24 12:46 ` [PATCH 1/4] mmc: core: use mmc_send_status to check hw_reset Johan Rudholm
2014-10-24 12:46 ` [PATCH 2/4] mmc: core: use mmc_power_up in hw_reset Johan Rudholm
2014-11-03  9:21   ` Adrian Hunter
2014-11-03 10:13     ` Johan Rudholm
2014-11-03 10:59       ` Adrian Hunter
2014-10-24 12:46 ` [PATCH 3/4] mmc: core: make hw_reset generic Johan Rudholm
2014-11-03  9:20   ` Adrian Hunter
2014-11-03 10:19     ` Johan Rudholm
2014-11-03 10:56       ` Adrian Hunter
2014-10-24 12:46 ` [PATCH 4/4] mmc: sd: add power_reset callback Johan Rudholm
2014-11-03  8:46 ` [PATCH 0/4] mmc: core: hw_reset changes Johan Rudholm

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).