public inbox for linux-mmc@vger.kernel.org
 help / color / mirror / Atom feed
* [PATCH] mmc: atmel-mci: pio hang on block errors
@ 2013-03-20  9:32 ludovic.desroches
  2013-04-08 16:02 ` Ludovic Desroches
       [not found] ` <20130408160224.GC6398@ludovic.desroches@atmel.com>
  0 siblings, 2 replies; 3+ messages in thread
From: ludovic.desroches @ 2013-03-20  9:32 UTC (permalink / raw)
  To: linux-mmc, linux-arm-kernel; +Cc: nicolas.ferre, terry, ludovic.desroches

From: Terry Barnaby <terry@beam.ltd.uk>

The driver is doing, by default, multi-block reads. When a block error occurs
card/block.c instigates a single block read: "mmcblk0: retrying using single
block read".
It leaves the sg chain intact and just changes the length attribute for the
first sg entry and the overall sg_len parameter. When atmci_read_data_pio is
called to read the single block of data it ignores the sg_len and expects to
read more than 512 bytes as it sees there are multiple items in the sg list. No
more data comes as the controller has only been commanded to get one block.

Signed-off-by: Terry Barnaby <terry@beam.ltd.uk>
Acked-by: Ludovic Desroches <ludovic.desroches@atmel.com>
Cc: stable <stable@vger.kernel.org> # 3.2+
---
 drivers/mmc/host/atmel-mci.c | 14 ++++++++++----
 1 file changed, 10 insertions(+), 4 deletions(-)

diff --git a/drivers/mmc/host/atmel-mci.c b/drivers/mmc/host/atmel-mci.c
index 0932024..d98e68e 100644
--- a/drivers/mmc/host/atmel-mci.c
+++ b/drivers/mmc/host/atmel-mci.c
@@ -163,6 +163,7 @@ struct atmel_mci {
 	void __iomem		*regs;
 
 	struct scatterlist	*sg;
+	unsigned int		sg_len;
 	unsigned int		pio_offset;
 
 	struct atmel_mci_slot	*cur_slot;
@@ -751,6 +752,7 @@ static u32 atmci_prepare_data(struct atmel_mci *host, struct mmc_data *data)
 	data->error = -EINPROGRESS;
 
 	host->sg = data->sg;
+	host->sg_len = data->sg_len;
 	host->data = data;
 	host->data_chan = NULL;
 
@@ -1573,7 +1575,8 @@ static void atmci_read_data_pio(struct atmel_mci *host)
 			if (offset == sg->length) {
 				flush_dcache_page(sg_page(sg));
 				host->sg = sg = sg_next(sg);
-				if (!sg)
+				host->sg_len--;
+				if (!sg || !host->sg_len)
 					goto done;
 
 				offset = 0;
@@ -1586,7 +1589,8 @@ static void atmci_read_data_pio(struct atmel_mci *host)
 
 			flush_dcache_page(sg_page(sg));
 			host->sg = sg = sg_next(sg);
-			if (!sg)
+			host->sg_len--;
+			if (!sg || !host->sg_len)
 				goto done;
 
 			offset = 4 - remaining;
@@ -1640,7 +1644,8 @@ static void atmci_write_data_pio(struct atmel_mci *host)
 			nbytes += 4;
 			if (offset == sg->length) {
 				host->sg = sg = sg_next(sg);
-				if (!sg)
+				host->sg_len--;
+				if (!sg || !host->sg_len)
 					goto done;
 
 				offset = 0;
@@ -1654,7 +1659,8 @@ static void atmci_write_data_pio(struct atmel_mci *host)
 			nbytes += remaining;
 
 			host->sg = sg = sg_next(sg);
-			if (!sg) {
+			host->sg_len--;
+			if (!sg || !host->sg_len) {
 				atmci_writel(host, ATMCI_TDR, value);
 				goto done;
 			}
-- 
1.7.11.3


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

* Re: [PATCH] mmc: atmel-mci: pio hang on block errors
  2013-03-20  9:32 [PATCH] mmc: atmel-mci: pio hang on block errors ludovic.desroches
@ 2013-04-08 16:02 ` Ludovic Desroches
       [not found] ` <20130408160224.GC6398@ludovic.desroches@atmel.com>
  1 sibling, 0 replies; 3+ messages in thread
From: Ludovic Desroches @ 2013-04-08 16:02 UTC (permalink / raw)
  To: linux-mmc, linux-arm-kernel, cjb; +Cc: terry, nicolas.ferre

Hi Chris,

Could you take this fix? If there are no objections of course.

Regards

Ludovic

On Wed, Mar 20, 2013 at 10:32:26AM +0100, ludovic.desroches@atmel.com wrote:
> From: Terry Barnaby <terry@beam.ltd.uk>
> 
> The driver is doing, by default, multi-block reads. When a block error occurs
> card/block.c instigates a single block read: "mmcblk0: retrying using single
> block read".
> It leaves the sg chain intact and just changes the length attribute for the
> first sg entry and the overall sg_len parameter. When atmci_read_data_pio is
> called to read the single block of data it ignores the sg_len and expects to
> read more than 512 bytes as it sees there are multiple items in the sg list. No
> more data comes as the controller has only been commanded to get one block.
> 
> Signed-off-by: Terry Barnaby <terry@beam.ltd.uk>
> Acked-by: Ludovic Desroches <ludovic.desroches@atmel.com>
> Cc: stable <stable@vger.kernel.org> # 3.2+
> ---
>  drivers/mmc/host/atmel-mci.c | 14 ++++++++++----
>  1 file changed, 10 insertions(+), 4 deletions(-)
> 
> diff --git a/drivers/mmc/host/atmel-mci.c b/drivers/mmc/host/atmel-mci.c
> index 0932024..d98e68e 100644
> --- a/drivers/mmc/host/atmel-mci.c
> +++ b/drivers/mmc/host/atmel-mci.c
> @@ -163,6 +163,7 @@ struct atmel_mci {
>  	void __iomem		*regs;
>  
>  	struct scatterlist	*sg;
> +	unsigned int		sg_len;
>  	unsigned int		pio_offset;
>  
>  	struct atmel_mci_slot	*cur_slot;
> @@ -751,6 +752,7 @@ static u32 atmci_prepare_data(struct atmel_mci *host, struct mmc_data *data)
>  	data->error = -EINPROGRESS;
>  
>  	host->sg = data->sg;
> +	host->sg_len = data->sg_len;
>  	host->data = data;
>  	host->data_chan = NULL;
>  
> @@ -1573,7 +1575,8 @@ static void atmci_read_data_pio(struct atmel_mci *host)
>  			if (offset == sg->length) {
>  				flush_dcache_page(sg_page(sg));
>  				host->sg = sg = sg_next(sg);
> -				if (!sg)
> +				host->sg_len--;
> +				if (!sg || !host->sg_len)
>  					goto done;
>  
>  				offset = 0;
> @@ -1586,7 +1589,8 @@ static void atmci_read_data_pio(struct atmel_mci *host)
>  
>  			flush_dcache_page(sg_page(sg));
>  			host->sg = sg = sg_next(sg);
> -			if (!sg)
> +			host->sg_len--;
> +			if (!sg || !host->sg_len)
>  				goto done;
>  
>  			offset = 4 - remaining;
> @@ -1640,7 +1644,8 @@ static void atmci_write_data_pio(struct atmel_mci *host)
>  			nbytes += 4;
>  			if (offset == sg->length) {
>  				host->sg = sg = sg_next(sg);
> -				if (!sg)
> +				host->sg_len--;
> +				if (!sg || !host->sg_len)
>  					goto done;
>  
>  				offset = 0;
> @@ -1654,7 +1659,8 @@ static void atmci_write_data_pio(struct atmel_mci *host)
>  			nbytes += remaining;
>  
>  			host->sg = sg = sg_next(sg);
> -			if (!sg) {
> +			host->sg_len--;
> +			if (!sg || !host->sg_len) {
>  				atmci_writel(host, ATMCI_TDR, value);
>  				goto done;
>  			}
> -- 
> 1.7.11.3
> 

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

* Re: [PATCH] mmc: atmel-mci: pio hang on block errors
       [not found] ` <20130408160224.GC6398@ludovic.desroches@atmel.com>
@ 2013-04-08 16:07   ` Chris Ball
  0 siblings, 0 replies; 3+ messages in thread
From: Chris Ball @ 2013-04-08 16:07 UTC (permalink / raw)
  To: linux-mmc; +Cc: linux-arm-kernel, nicolas.ferre, terry

Hi,

On Mon, Apr 08 2013, Ludovic Desroches wrote:
> Could you take this fix? If there are no objections of course.

Thanks for the reminder; pushed to mmc-next now.

- Chris.
-- 
Chris Ball   <cjb@laptop.org>   <http://printf.net/>
One Laptop Per Child

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

end of thread, other threads:[~2013-04-08 16:08 UTC | newest]

Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2013-03-20  9:32 [PATCH] mmc: atmel-mci: pio hang on block errors ludovic.desroches
2013-04-08 16:02 ` Ludovic Desroches
     [not found] ` <20130408160224.GC6398@ludovic.desroches@atmel.com>
2013-04-08 16:07   ` Chris Ball

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox