3.2.46-rc1 review patch.  If anyone has any objections, please let me know.

------------------

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

commit bdbc5d0c60f3e9de3eeccf1c1a18bdc11dca62cc upstream.

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 <te...@beam.ltd.uk>
Acked-by: Ludovic Desroches <ludovic.desroc...@atmel.com>
Signed-off-by: Chris Ball <c...@laptop.org>
Signed-off-by: Ben Hutchings <b...@decadent.org.uk>
---
 drivers/mmc/host/atmel-mci.c |   14 ++++++++++----
 1 file changed, 10 insertions(+), 4 deletions(-)

--- 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 atm
        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 a
                        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 a
 
                        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
                        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
                        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;
                        }

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to