Some controller as the ColdFire eshdc may require an endianness
byte swap, because DMA read endianness is not configurable.

Signed-off-by: Angelo Dureghello <ang...@sysam.it>
---
 drivers/mmc/host/sdhci.c | 19 +++++++++++++++++++
 drivers/mmc/host/sdhci.h |  7 +++++++
 2 files changed, 26 insertions(+)

diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c
index 59acf8e3331e..f56ae6f153d4 100644
--- a/drivers/mmc/host/sdhci.c
+++ b/drivers/mmc/host/sdhci.c
@@ -2600,6 +2600,18 @@ static const struct mmc_host_ops sdhci_ops = {
        .card_busy      = sdhci_card_busy,
 };
 
+static void sdhci_be_to_le(char *buff, u32 length)
+{
+       int i, size = length >> 2;
+       u32 *buffer = (u32 *)buff;
+       u32 temp;
+
+       for (i = 0; i < size; i++) {
+               temp = *buffer;
+               *buffer++ = __le32_to_cpu(temp);
+       }
+}
+
 /*****************************************************************************\
  *                                                                           *
  * Request done                                                              *
@@ -2655,6 +2667,13 @@ static bool sdhci_request_done(struct sdhci_host *host)
                                                host->bounce_addr,
                                                host->bounce_buffer_size,
                                                DMA_FROM_DEVICE);
+
+                                       if (host->quirks2 &
+                                           SDHCI_QUIRK2_USE_32BIT_BE_DMA_SWAP)
+                                               sdhci_be_to_le(
+                                                       host->bounce_buffer,
+                                                       length);
+
                                        sg_copy_from_buffer(data->sg,
                                                data->sg_len,
                                                host->bounce_buffer,
diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h
index 199712e7adbb..be08ff1a8c6f 100644
--- a/drivers/mmc/host/sdhci.h
+++ b/drivers/mmc/host/sdhci.h
@@ -482,6 +482,13 @@ struct sdhci_host {
  */
 #define SDHCI_QUIRK2_USE_32BIT_BLK_CNT                 (1<<18)
 
+/*
+ * On some architectures, as ColdFire/m68k, native endianness is big endian,
+ * and the dma buffer is filled in big endian order only (no other options).
+ * So, a swap is needed for these specific cases.
+ */
+#define SDHCI_QUIRK2_USE_32BIT_BE_DMA_SWAP             (1<<19)
+
        int irq;                /* Device IRQ */
        void __iomem *ioaddr;   /* Mapped address */
        char *bounce_buffer;    /* For packing SDMA reads/writes */
-- 
2.20.1

Reply via email to