Hi,

On 02/18/2014 04:37 PM, Maxime Ripard wrote:

<snip>

+
+       for (i = 0; i < data->sg_len; i++) {
+               pdes[i].config = SDXC_IDMAC_DES0_CH | SDXC_IDMAC_DES0_OWN |
+                                SDXC_IDMAC_DES0_DIC;
+
+               if (data->sg[i].length == max_len)
+                       pdes[i].buf_size = 0; /* 0 == max_len */
+               else
+                       pdes[i].buf_size = data->sg[i].length;
+
+               pdes[i].buf_addr_ptr1 = sg_dma_address(&data->sg[i]);
+               pdes[i].buf_addr_ptr2 = (u32)&pdes_pa[i + 1];
+       }
+
+       pdes[0].config |= SDXC_IDMAC_DES0_FD;
+       pdes[i - 1].config = SDXC_IDMAC_DES0_OWN | SDXC_IDMAC_DES0_LD;
+
+       wmb(); /* Ensure idma_des hit main mem before we start the idmac */

wmb ensure the proper ordering of the instructions, not flushing the
caches like what your comment implies.

Since I put that comment there, allow me to explain. A modern ARM
cpu core has 2 or more units handling stores. One for regular
memory stores, and one for io-mem stores. Regular mem stores can
be re-ordered, io stores cannot. Normally there is no "syncing"
between the 2 store units. Cache flushing is not an issue here
since the memory holding the descriptors for the idma controller
is allocated cache coherent, which on arm means it is not cached.

What is an issue here is the io-store starting the idmac hitting
the io-mem before the descriptors hit the main-mem, the wmb()
ensures this does not happen.


+static int sunxi_mmc_prepare_dma(struct sunxi_mmc_host *smc_host,
+                                struct mmc_data *data)
+{
+       u32 dma_len;
+       u32 i;
+       u32 temp;
+       struct scatterlist *sg;
+
+       dma_len = dma_map_sg(mmc_dev(smc_host->mmc), data->sg, data->sg_len,
+                            sunxi_mmc_get_dma_dir(data));
+       if (dma_len == 0) {
+               dev_err(mmc_dev(smc_host->mmc), "dma_map_sg failed\n");
+               return -ENOMEM;
+       }
+
+       for_each_sg(data->sg, sg, data->sg_len, i) {
+               if (sg->offset & 3 || sg->length & 3) {
+                       dev_err(mmc_dev(smc_host->mmc),
+                               "unaligned scatterlist: os %x length %d\n",
+                               sg->offset, sg->length);
+                       return -EINVAL;
+               }
+       }
+
+       sunxi_mmc_init_idma_des(smc_host, data);
+
+       temp = mci_readl(smc_host, REG_GCTRL);
+       temp |= SDXC_DMA_ENABLE_BIT;
+       mci_writel(smc_host, REG_GCTRL, temp);
+       temp |= SDXC_DMA_RESET;
+       mci_writel(smc_host, REG_GCTRL, temp);

Does it really need to be done in two steps?

We don't know, so this is probably best left as is.



(Newline)

+       mci_writel(smc_host, REG_DMAC, SDXC_IDMAC_SOFT_RESET);
+
+       if (!(data->flags & MMC_DATA_WRITE))
+               mci_writel(smc_host, REG_IDIE, SDXC_IDMAC_RECEIVE_INTERRUPT);
+
+       mci_writel(smc_host, REG_DMAC, SDXC_IDMAC_FIX_BURST | 
SDXC_IDMAC_IDMA_ON);
+
+       return 0;
+}
+
+static void sunxi_mmc_send_manual_stop(struct sunxi_mmc_host *host,
+                                      struct mmc_request *req)
+{
+       u32 cmd_val = SDXC_START | SDXC_RESP_EXPIRE | SDXC_STOP_ABORT_CMD
+                       | SDXC_CHECK_RESPONSE_CRC | MMC_STOP_TRANSMISSION;
+       u32 ri = 0;
+       unsigned long expire = jiffies + msecs_to_jiffies(1000);
+
+       mci_writel(host, REG_CARG, 0);
+       mci_writel(host, REG_CMDR, cmd_val);
+
+       do {
+               ri = mci_readl(host, REG_RINTR);
+       } while (!(ri & (SDXC_COMMAND_DONE | SDXC_INTERRUPT_ERROR_BIT)) &&
+                time_before(jiffies, expire));
+
+       if (ri & SDXC_INTERRUPT_ERROR_BIT) {
+               dev_err(mmc_dev(host->mmc), "send stop command failed\n");
+               if (req->stop)
+                       req->stop->resp[0] = -ETIMEDOUT;
+       } else {
+               if (req->stop)
+                       req->stop->resp[0] = mci_readl(host, REG_RESP0);
+       }
+
+       mci_writel(host, REG_RINTR, 0xffff);
+}
+
+static void sunxi_mmc_dump_errinfo(struct sunxi_mmc_host *smc_host)
+{
+       struct mmc_command *cmd = smc_host->mrq->cmd;
+       struct mmc_data *data = smc_host->mrq->data;
+
+       /* For some cmds timeout is normal with sd/mmc cards */
+       if ((smc_host->int_sum & SDXC_INTERRUPT_ERROR_BIT) == SDXC_RESP_TIMEOUT 
&&
+                       (cmd->opcode == SD_IO_SEND_OP_COND || cmd->opcode == 
SD_IO_RW_DIRECT))
+               return;
+
+       dev_err(mmc_dev(smc_host->mmc),

I'd rather put it at a debug loglevel.

Erm, this only happens if something is seriously wrong.


+       /* Make sure the controller is in a sane state before enabling irqs */
+       ret = sunxi_mmc_init_host(host->mmc);
+       if (ret)
+               return ret;
+
+       host->irq = platform_get_irq(pdev, 0);
+       ret = devm_request_irq(&pdev->dev, host->irq, sunxi_mmc_irq, 0,
+                              "sunxi-mmc", host);
+       if (ret == 0)
+               disable_irq(host->irq);

The disable_irq is useless here. Just exit.

No it is not note the ret == 0, this is not an error handling path!

This is done under an if because we want to do the sunxi_mmc_exit_host
regardless of the request_irq succeeding or not.


+
+       /* And put it back in reset */
+       sunxi_mmc_exit_host(host);

Hu? If it's in reset, how can it generate some IRQs?

Yes, that is why we do the whole dance of init controller, get irq,
disable irq, drop it back in reset (until the mmc subsys does a power on
of the mmc card / sdio dev).

Sometime the controller asserts the irq in reset for some reason, so
without the dance as soon as we do the devm_request_irq we get an irq,
and worse, not only do we get an irq, we cannot clear it since writing to
the interrupt status register does not work when the controller is in reset,
so we get stuck re-entering the irq handler.


+       return ret;
+}
+
+static int sunxi_mmc_probe(struct platform_device *pdev)
+{
+       struct sunxi_mmc_host *host;
+       struct mmc_host *mmc;
+       int ret;
+
+       mmc = mmc_alloc_host(sizeof(struct sunxi_mmc_host), &pdev->dev);
+       if (!mmc) {
+               dev_err(&pdev->dev, "mmc alloc host failed\n");
+               return -ENOMEM;
+       }
+
+       ret = mmc_of_parse(mmc);
+       if (ret)
+               goto error_free_host;
+
+       host = mmc_priv(mmc);
+       host->mmc = mmc;
+       spin_lock_init(&host->lock);
+       tasklet_init(&host->tasklet, sunxi_mmc_tasklet, (unsigned long)host);
+
+       ret = sunxi_mmc_resource_request(host, pdev);
+       if (ret)
+               goto error_free_host;
+
+       host->sg_cpu = dma_alloc_coherent(&pdev->dev, PAGE_SIZE,
+                                         &host->sg_dma, GFP_KERNEL);
+       if (!host->sg_cpu) {
+               dev_err(&pdev->dev, "Failed to allocate DMA descriptor mem\n");
+               ret = -ENOMEM;
+               goto error_free_host;
+       }
+
+       mmc->ops             = &sunxi_mmc_ops;
+       mmc->max_blk_count   = 8192;
+       mmc->max_blk_size    = 4096;
+       mmc->max_segs                = PAGE_SIZE / sizeof(struct 
sunxi_idma_des);
+       mmc->max_seg_size    = (1 << host->idma_des_size_bits);
+       mmc->max_req_size    = mmc->max_seg_size * mmc->max_segs;
+       /* 400kHz ~ 50MHz */
+       mmc->f_min           =   400000;
+       mmc->f_max           = 50000000;
+       /* available voltages */
+       if (!IS_ERR(host->vmmc))
+               mmc->ocr_avail = mmc_regulator_get_ocrmask(host->vmmc);
+       else
+               mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34;
+
+       mmc->caps = MMC_CAP_MMC_HIGHSPEED | MMC_CAP_SD_HIGHSPEED |
+               MMC_CAP_UHS_SDR12 | MMC_CAP_UHS_SDR25 | MMC_CAP_UHS_SDR50 |
+               MMC_CAP_UHS_DDR50 | MMC_CAP_SDIO_IRQ | MMC_CAP_DRIVER_TYPE_A;
+       mmc->caps2 = MMC_CAP2_NO_PRESCAN_POWERUP;
+
+       ret = mmc_add_host(mmc);
+
+       if (ret)
+               goto error_free_dma;
+
+       dev_info(&pdev->dev, "base:0x%p irq:%u\n", host->reg_base, host->irq);
+       platform_set_drvdata(pdev, mmc);

This should be before the registration. Otherwise, you're racy.

Nope, we only need this to get the data on sunxi_mmc_remove, everywhere
else the data is found through the mmc-host struct.


+       return 0;
+
+error_free_dma:
+       dma_free_coherent(&pdev->dev, PAGE_SIZE, host->sg_cpu, host->sg_dma);
+error_free_host:
+       mmc_free_host(mmc);
+       return ret;
+}
+
+static int sunxi_mmc_remove(struct platform_device *pdev)
+{
+       struct mmc_host *mmc = platform_get_drvdata(pdev);
+       struct sunxi_mmc_host *host = mmc_priv(mmc);
+
+       mmc_remove_host(mmc);
+       sunxi_mmc_exit_host(host);
+       tasklet_disable(&host->tasklet);
+       dma_free_coherent(&pdev->dev, PAGE_SIZE, host->sg_cpu, host->sg_dma);
+       mmc_free_host(mmc);
+
+       return 0;
+}
+
+static struct platform_driver sunxi_mmc_driver = {
+       .driver = {
+               .name   = "sunxi-mmc",
+               .owner  = THIS_MODULE,
+               .of_match_table = of_match_ptr(sunxi_mmc_of_match),
+       },
+       .probe          = sunxi_mmc_probe,
+       .remove         = sunxi_mmc_remove,
+};
+module_platform_driver(sunxi_mmc_driver);
+
+MODULE_DESCRIPTION("Allwinner's SD/MMC Card Controller Driver");
+MODULE_LICENSE("GPL v2");
+MODULE_AUTHOR("David Lanzend?rfer <[email protected]>");
+MODULE_ALIAS("platform:sunxi-mmc");
diff --git a/drivers/mmc/host/sunxi-mmc.h b/drivers/mmc/host/sunxi-mmc.h
new file mode 100644
index 0000000..75eaa02
--- /dev/null
+++ b/drivers/mmc/host/sunxi-mmc.h
@@ -0,0 +1,239 @@
+/*
+ * Driver for sunxi SD/MMC host controllers
+ * (C) Copyright 2007-2011 Reuuimlla Technology Co., Ltd.
+ * (C) Copyright 2007-2011 Aaron Maoye <[email protected]>
+ * (C) Copyright 2013-2014 O2S GmbH <www.o2s.ch>
+ * (C) Copyright 2013-2014 David Lanzend?rfer <[email protected]>
+ * (C) Copyright 2013-2014 Hans de Goede <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of
+ * the License, or (at your option) any later version.
+ */
+
+#ifndef __SUNXI_MMC_H__
+#define __SUNXI_MMC_H__
+
+/* register offset definitions */
+#define SDXC_REG_GCTRL (0x00) /* SMC Global Control Register */
+#define SDXC_REG_CLKCR (0x04) /* SMC Clock Control Register */
+#define SDXC_REG_TMOUT (0x08) /* SMC Time Out Register */
+#define SDXC_REG_WIDTH (0x0C) /* SMC Bus Width Register */
+#define SDXC_REG_BLKSZ (0x10) /* SMC Block Size Register */
+#define SDXC_REG_BCNTR (0x14) /* SMC Byte Count Register */
+#define SDXC_REG_CMDR  (0x18) /* SMC Command Register */
+#define SDXC_REG_CARG  (0x1C) /* SMC Argument Register */
+#define SDXC_REG_RESP0 (0x20) /* SMC Response Register 0 */
+#define SDXC_REG_RESP1 (0x24) /* SMC Response Register 1 */
+#define SDXC_REG_RESP2 (0x28) /* SMC Response Register 2 */
+#define SDXC_REG_RESP3 (0x2C) /* SMC Response Register 3 */
+#define SDXC_REG_IMASK (0x30) /* SMC Interrupt Mask Register */
+#define SDXC_REG_MISTA (0x34) /* SMC Masked Interrupt Status Register */
+#define SDXC_REG_RINTR (0x38) /* SMC Raw Interrupt Status Register */
+#define SDXC_REG_STAS  (0x3C) /* SMC Status Register */
+#define SDXC_REG_FTRGL (0x40) /* SMC FIFO Threshold Watermark Registe */
+#define SDXC_REG_FUNS  (0x44) /* SMC Function Select Register */
+#define SDXC_REG_CBCR  (0x48) /* SMC CIU Byte Count Register */
+#define SDXC_REG_BBCR  (0x4C) /* SMC BIU Byte Count Register */
+#define SDXC_REG_DBGC  (0x50) /* SMC Debug Enable Register */
+#define SDXC_REG_HWRST (0x78) /* SMC Card Hardware Reset for Register */
+#define SDXC_REG_DMAC  (0x80) /* SMC IDMAC Control Register */
+#define SDXC_REG_DLBA  (0x84) /* SMC IDMAC Descriptor List Base Addre */
+#define SDXC_REG_IDST  (0x88) /* SMC IDMAC Status Register */
+#define SDXC_REG_IDIE  (0x8C) /* SMC IDMAC Interrupt Enable Register */
+#define SDXC_REG_CHDA  (0x90)
+#define SDXC_REG_CBDA  (0x94)
+
+#define mci_readl(host, reg) \
+       readl((host)->reg_base + SDXC_##reg)
+#define mci_writel(host, reg, value) \
+       writel((value), (host)->reg_base + SDXC_##reg)

Please use some inline functions here.

+/* global control register bits */
+#define SDXC_SOFT_RESET                BIT(0)
+#define SDXC_FIFO_RESET                BIT(1)
+#define SDXC_DMA_RESET         BIT(2)
+#define SDXC_HARDWARE_RESET            
(SDXC_SOFT_RESET|SDXC_FIFO_RESET|SDXC_DMA_RESET)
+#define SDXC_INTERRUPT_ENABLE_BIT              BIT(4)
+#define SDXC_DMA_ENABLE_BIT            BIT(5)
+#define SDXC_DEBOUNCE_ENABLE_BIT       BIT(8)
+#define SDXC_POSEDGE_LATCH_DATA        BIT(9)
+#define SDXC_DDR_MODE          BIT(10)
+#define SDXC_MEMORY_ACCESS_DONE        BIT(29)
+#define SDXC_ACCESS_DONE_DIRECT        BIT(30)
+#define SDXC_ACCESS_BY_AHB     BIT(31)
+#define SDXC_ACCESS_BY_DMA     (0U << 31)

Isn't it 0?

Yes, but is is the inverse of ACCESS_BY_DMA, which
this makes much clearer then just 0 does.


+/* clock control bits */
+#define SDXC_CARD_CLOCK_ON             BIT(16)
+#define SDXC_LOW_POWER_ON              BIT(17)
+/* bus width */
+#define SDXC_WIDTH1            (0)
+#define SDXC_WIDTH4            (1)
+#define SDXC_WIDTH8            (2)
+/* smc command bits */
+#define SDXC_RESP_EXPIRE               BIT(6)
+#define SDXC_LONG_RESPONSE             BIT(7)
+#define SDXC_CHECK_RESPONSE_CRC        BIT(8)
+#define SDXC_DATA_EXPIRE               BIT(9)
+#define SDXC_WRITE             BIT(10)
+#define SDXC_SEQUENCE_MODE             BIT(11)
+#define SDXC_SEND_AUTO_STOP    BIT(12)
+#define SDXC_WAIT_PRE_OVER     BIT(13)
+#define SDXC_STOP_ABORT_CMD    BIT(14)
+#define SDXC_SEND_INIT_SEQUENCE        BIT(15)
+#define SDXC_UPCLK_ONLY                BIT(21)
+#define SDXC_READ_CEATA_DEV            BIT(22)
+#define SDXC_CCS_EXPIRE                BIT(23)
+#define SDXC_ENABLE_BIT_BOOT           BIT(24)
+#define SDXC_ALT_BOOT_OPTIONS          BIT(25)
+#define SDXC_BOOT_ACK_EXPIRE           BIT(26)
+#define SDXC_BOOT_ABORT                BIT(27)
+#define SDXC_VOLTAGE_SWITCH            BIT(28)
+#define SDXC_USE_HOLD_REGISTER         BIT(29)
+#define SDXC_START             BIT(31)
+/* interrupt bits */
+#define SDXC_RESP_ERROR                BIT(1)
+#define SDXC_COMMAND_DONE              BIT(2)
+#define SDXC_DATA_OVER         BIT(3)
+#define SDXC_TX_DATA_REQUEST           BIT(4)
+#define SDXC_RX_DATA_REQUEST           BIT(5)
+#define SDXC_RESP_CRC_ERROR            BIT(6)
+#define SDXC_DATA_CRC_ERROR            BIT(7)
+#define SDXC_RESP_TIMEOUT      BIT(8)
+#define SDXC_DATA_TIMEOUT      BIT(9)
+#define SDXC_VOLTAGE_CHANGE_DONE               BIT(10)
+#define SDXC_FIFO_RUN_ERROR            BIT(11)
+#define SDXC_HARD_WARE_LOCKED  BIT(12)
+#define SDXC_START_BIT_ERROR   BIT(13)
+#define SDXC_AUTO_COMMAND_DONE BIT(14)
+#define SDXC_END_BIT_ERROR             BIT(15)
+#define SDXC_SDIO_INTERRUPT            BIT(16)
+#define SDXC_CARD_INSERT               BIT(30)
+#define SDXC_CARD_REMOVE               BIT(31)
+#define SDXC_INTERRUPT_ERROR_BIT               (SDXC_RESP_ERROR | 
SDXC_RESP_CRC_ERROR | \
+                                SDXC_DATA_CRC_ERROR | SDXC_RESP_TIMEOUT | \
+                                SDXC_DATA_TIMEOUT | SDXC_FIFO_RUN_ERROR | \
+                                SDXC_HARD_WARE_LOCKED | SDXC_START_BIT_ERROR | 
\
+                                SDXC_END_BIT_ERROR) /* 0xbbc2 */
+#define SDXC_INTERRUPT_DONE_BIT                (SDXC_AUTO_COMMAND_DONE | 
SDXC_DATA_OVER | \
+                                SDXC_COMMAND_DONE | SDXC_VOLTAGE_CHANGE_DONE)
+/* status */
+#define SDXC_RXWL_FLAG         BIT(0)
+#define SDXC_TXWL_FLAG         BIT(1)
+#define SDXC_FIFO_EMPTY                BIT(2)
+#define SDXC_FIFO_FULL         BIT(3)
+#define SDXC_CARD_PRESENT      BIT(8)
+#define SDXC_CARD_DATA_BUSY    BIT(9)
+#define SDXC_DATA_FSM_BUSY     BIT(10)
+#define SDXC_DMA_REQUEST               BIT(31)
+#define SDXC_FIFO_SIZE         (16)
+/* Function select */
+#define SDXC_CEATA_ON          (0xceaaU << 16)
+#define SDXC_SEND_IRQ_RESPONSE         BIT(0)
+#define SDXC_SDIO_READ_WAIT            BIT(1)
+#define SDXC_ABORT_READ_DATA           BIT(2)
+#define SDXC_SEND_CCSD         BIT(8)
+#define SDXC_SEND_AUTO_STOPCCSD        BIT(9)
+#define SDXC_CEATA_DEV_INTERRUPT_ENABLE_BIT    BIT(10)
+/* IDMA controller bus mod bit field */
+#define SDXC_IDMAC_SOFT_RESET  BIT(0)
+#define SDXC_IDMAC_FIX_BURST   BIT(1)
+#define SDXC_IDMAC_IDMA_ON     BIT(7)
+#define SDXC_IDMAC_REFETCH_DES BIT(31)
+/* IDMA status bit field */
+#define SDXC_IDMAC_TRANSMIT_INTERRUPT  BIT(0)
+#define SDXC_IDMAC_RECEIVE_INTERRUPT   BIT(1)
+#define SDXC_IDMAC_FATAL_BUS_ERROR     BIT(2)
+#define SDXC_IDMAC_DESTINATION_INVALID BIT(4)
+#define SDXC_IDMAC_CARD_ERROR_SUM      BIT(5)
+#define SDXC_IDMAC_NORMAL_INTERRUPT_SUM        BIT(8)
+#define SDXC_IDMAC_ABNORMAL_INTERRUPT_SUM BIT(9)
+#define SDXC_IDMAC_HOST_ABORT_INTERRUPT_TX     BIT(10)
+#define SDXC_IDMAC_HOST_ABORT_INTERRUPT_RX     BIT(10)
+#define SDXC_IDMAC_IDLE                (0U << 13)

Ditto

+#define SDXC_IDMAC_SUSPEND     (1U << 13)
+#define SDXC_IDMAC_DESC_READ   (2U << 13)
+#define SDXC_IDMAC_DESC_CHECK  (3U << 13)
+#define SDXC_IDMAC_READ_REQUEST_WAIT   (4U << 13)
+#define SDXC_IDMAC_WRITE_REQUEST_WAIT  (5U << 13)
+#define SDXC_IDMAC_READ                (6U << 13)
+#define SDXC_IDMAC_WRITE               (7U << 13)
+#define SDXC_IDMAC_DESC_CLOSE  (8U << 13)

Please use BIT as much as possible here.

Erm lets not do that, nor remove the 0 << 13, this are all
values to store in a multi-bit field which lives in bits 13-xx,
changing the values which happen to be power of 2 into BIT
macros is not helpful.



+
+/*
+* If the idma-des-size-bits of property is ie 13, bufsize bits are:
+*  Bits  0-12: buf1 size
+*  Bits 13-25: buf2 size
+*  Bits 26-31: not used
+* Since we only ever set buf1 size, we can simply store it directly.
+*/
+#define SDXC_IDMAC_DES0_DIC    BIT(1)  /* disable interrupt on completion */
+#define SDXC_IDMAC_DES0_LD     BIT(2)  /* last descriptor */
+#define SDXC_IDMAC_DES0_FD     BIT(3)  /* first descriptor */
+#define SDXC_IDMAC_DES0_CH     BIT(4)  /* chain mode */
+#define SDXC_IDMAC_DES0_ER     BIT(5)  /* end of ring */
+#define SDXC_IDMAC_DES0_CES    BIT(30) /* card error summary */
+#define SDXC_IDMAC_DES0_OWN    BIT(31) /* 1-idma owns it, 0-host owns it */

Overall, I prefer to have the registers right beneath the register
declaration. It's easier to spot what bits belong to what registers
that way.

+struct sunxi_idma_des {
+       u32     config;
+       u32     buf_size;
+       u32     buf_addr_ptr1;
+       u32     buf_addr_ptr2;
+};

Some comments on top of this structure would be great.

+struct sunxi_mmc_host {
+       struct mmc_host *mmc;
+       struct regulator *vmmc;
+
+       /* IO mapping base */
+       void __iomem *reg_base;
+
+       spinlock_t lock;
+       struct tasklet_struct tasklet;
+
+       /* clock management */
+       struct clk *clk_ahb;
+       struct clk *clk_mod;
+
+       /* ios information */
+       u32             clk_mod_rate;
+       u32             bus_width;
+       u32             idma_des_size_bits;
+       u32             ddr;
+       u32             voltage_switching;
+
+       /* irq */
+       int             irq;
+       u32             int_sum;
+       u32             sdio_imask;
+
+       /* flags */
+       u32             power_on:1;
+       u32             io_flag:1;
+       u32             wait_dma:1;

bool?

+       dma_addr_t      sg_dma;
+       void            *sg_cpu;
+
+       struct mmc_request *mrq;
+       u32             ferror;
+};

Please be consistent in your spacing, either align the variable names,
or don't, but make a decision :)

+#define MMC_CLK_400K            0
+#define MMC_CLK_25M             1
+#define MMC_CLK_50M             2
+#define MMC_CLK_50MDDR          3
+#define MMC_CLK_50MDDR_8BIT     4
+#define MMC_CLK_100M            5
+#define MMC_CLK_200M            6
+#define MMC_CLK_MOD_NUM         7

Wouldn't an enum be better here?

+
+struct sunxi_mmc_clk_dly {
+       u32 mode;
+       u32 oclk_dly;
+       u32 sclk_dly;
+};

Comments here would be great too.

Thanks for working on this!
Maxime





Regards,

Hans

--
You received this message because you are subscribed to the Google Groups 
"linux-sunxi" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to [email protected].
For more options, visit https://groups.google.com/groups/opt_out.

Reply via email to