We now support the device-model for configuration of the driver
including the interface to the pinctrl, reset and clock frameworks.

Signed-off-by: Philipp Tomsich <philipp.toms...@theobroma-systems.com>
---
 board/sunxi/board.c     |   7 +-
 drivers/mmc/sunxi_mmc.c | 350 ++++++++++++++++++++++++++++++++++++++++++++----
 2 files changed, 331 insertions(+), 26 deletions(-)

diff --git a/board/sunxi/board.c b/board/sunxi/board.c
index 838e89f..810fbd4 100644
--- a/board/sunxi/board.c
+++ b/board/sunxi/board.c
@@ -237,14 +237,14 @@ static void nand_clock_setup(void)
 void board_nand_init(void)
 {
        nand_pinmux_setup();
        nand_clock_setup();
 #ifndef CONFIG_SPL_BUILD
        sunxi_nand_init();
 #endif
 }
 #endif
 
-#ifdef CONFIG_GENERIC_MMC
+#if defined(CONFIG_GENERIC_MMC) && !(defined(CONFIG_DM_MMC) && 
defined(CONFIG_PINCTRL))
 static void mmc_pinmux_setup(int sdc)
 {
        unsigned int pin;
@@ -422,75 +422,76 @@ static void mmc_pinmux_setup(int sdc)
 
 int board_mmc_init(bd_t *bis)
 {
+#if !(defined(CONFIG_DM_MMC) && defined(CONFIG_PINCTRL))
        __maybe_unused struct mmc *mmc0, *mmc1;
        __maybe_unused char buf[512];
        __maybe_unused u32 val;
 
        mmc_pinmux_setup(CONFIG_MMC_SUNXI_SLOT);
        mmc0 = sunxi_mmc_init(CONFIG_MMC_SUNXI_SLOT);
        if (!mmc0)
                return -1;
 
 #if CONFIG_MMC_SUNXI_SLOT_EXTRA != -1
        mmc_pinmux_setup(CONFIG_MMC_SUNXI_SLOT_EXTRA);
        mmc1 = sunxi_mmc_init(CONFIG_MMC_SUNXI_SLOT_EXTRA);
        if (!mmc1)
                return -1;
 #endif
 
 #if !defined(CONFIG_SPL_BUILD) && CONFIG_MMC_SUNXI_SLOT_EXTRA == 2
 #if CONFIG_MACH_SUN6I
        /*
         * the bootdevice is shown in VER_REG in the system controller
         * if U_boot is set then the ROM trys to boot from mmc0 regardless
         * of the BOOT_SEL pins.
         */
 
 #if 0
        val = readl(0x1c00024);
        if ((val & (1<<10))) /* check UBOOT_SEL */
        {
                switch((val & (3<<8))>>8) /* check BOOT_SEL pins */
                {
                        case 0x0: /* SPI0 boot */
                                break;
                        case 0x1: /* eMMC2 boot */
                                /* Check if there is a boot loader on eMMC2
                                 * If not we want to fall back to SD card */
                                if (mmc_init(mmc1) == 0 &&
                                    
mmc1->block_dev.block_read(&mmc1->block_dev, 16, 1, buf) == 1) {
                                        buf[12] = 0;
                                        if (strcmp(&buf[4], "eGON.BT0") == 0) {
                                                /* Boot loader found, swap to 
make eMMC the first device */
                                                mmc0->block_dev.devnum = 1;
                                                mmc1->block_dev.devnum = 0;
                                        }
                                }
                                break;
                        case 0x2: /* SDC2 boot */
                                break;
                        case 0x3: /* NAND Flash boot */
                                break;
                }
        }
 #endif
 
 #else
        /*
         * On systems with an emmc (mmc2), figure out if we are booting from
         * the emmc and if we are make it "mmc dev 0" so that boot.scr, etc.
         * are searched there first. Note we only do this for u-boot proper,
         * not for the SPL, see spl_boot_device().
         */
        if (readb(SPL_ADDR + 0x28) == SUNXI_BOOTED_FROM_MMC2) {
                /* Booting from emmc / mmc2, swap */
                mmc0->block_dev.devnum = 1;
                mmc1->block_dev.devnum = 0;
        }
 #endif
 #endif
-
+#endif
        return 0;
 }
 #endif
 
@@ -854,43 +855,45 @@ static void setup_environment(const void *fdt)
  */
 static void setup_environment(const void *fdt)
 {
+#if !defined(CONFIG_DM_MMC)
        uint8_t mac_addr[6];
        char serial_string[17] = { 0 };
        struct mmc *mmc0;
        struct sunxi_mmc_host {
                unsigned mmc_no;
                uint32_t *mclkreg;
                unsigned fatal_err;
                struct sunxi_mmc *reg;
                struct mmc_config cfg;
        };
 
        mmc0 = find_mmc_device(1);
 
        /* lookup the real device number to get the eMMC */
        if(((struct sunxi_mmc_host*)mmc0->priv)->mmc_no != 2)
                mmc0 = find_mmc_device(0);
 
        if(mmc0->has_init == 0)
                mmc_init(mmc0);
 
        if (!getenv("ethaddr")) {
                /* Non OUI / registered MAC address */
                mac_addr[0] = 0x02;
                mac_addr[1] = (mmc0->cid[0] >> 24) & 0xff;
                mac_addr[2] = (mmc0->cid[2] >> 0) & 0xff;
                mac_addr[3] = (mmc0->cid[3] >> 24) & 0xff;
                mac_addr[4] = (mmc0->cid[3] >> 16) & 0xff;
                mac_addr[5] = (mmc0->cid[3] >> 8) & 0xff;
 
                eth_setenv_enetaddr("ethaddr", mac_addr);
        }
 
        if (!getenv("serial#")) {
                snprintf(serial_string, sizeof(serial_string), "%08x%02x%06x",
                         mmc0->cid[0]>>24,mmc0->cid[2]&0xff,mmc0->cid[3]>>8);
                setenv("serial#", serial_string);
        }
+#endif
 }
 #endif
 
diff --git a/drivers/mmc/sunxi_mmc.c b/drivers/mmc/sunxi_mmc.c
index 46abe4a..8075b9c 100644
--- a/drivers/mmc/sunxi_mmc.c
+++ b/drivers/mmc/sunxi_mmc.c
@@ -1,89 +1,165 @@
 /*
  * (C) Copyright 2007-2011
  * Allwinner Technology Co., Ltd. <www.allwinnertech.com>
  * Aaron <leafy.m...@allwinnertech.com>
  *
+ * (C) 2017 Theobroma Systems Design und Consulting GmbH
+ *
  * MMC driver for allwinner sunxi platform.
  *
  * SPDX-License-Identifier:    GPL-2.0+
  */
 
 #include <common.h>
-#include <errno.h>
-#include <malloc.h>
-#include <mmc.h>
-#include <asm/io.h>
+#include <asm-generic/gpio.h>
 #include <asm/arch/clock.h>
 #include <asm/arch/cpu.h>
 #include <asm/arch/gpio.h>
 #include <asm/arch/mmc.h>
-#include <asm-generic/gpio.h>
+#include <asm/io.h>
+#include <clk.h>
+#include <dm/device.h>
+#include <dt-structs.h>
+#include <errno.h>
+#include <linux/iopoll.h>
+#include <malloc.h>
+#include <mmc.h>
+#include <reset.h>
+
+DECLARE_GLOBAL_DATA_PTR;
+
+struct sunxi_mmc_plat {
+       struct mmc mmc;
+};
 
 struct sunxi_mmc_host {
        unsigned mmc_no;
+#if !defined(CONFIG_DM_MMC)
        uint32_t *mclkreg;
+#endif
        unsigned fatal_err;
        struct sunxi_mmc *reg;
        struct mmc_config cfg;
+       bool cd_inverted;
+#if defined(CONFIG_DM_MMC)
+       struct mmc *mmc;
+       struct gpio_desc cd_gpio;       /* card-detect (optional) */
+       struct gpio_desc pwr_gpio;      /* power-enabled (optional) */
+       struct gpio_desc wp_gpio;       /* write-protect (optional) */
+       bool wp_inverted;
+       struct reset_ctl reset;
+       struct clk ahb_clk_gate;
+       struct clk mmc_clk;
+#else
+       int cd_pin;
+#endif
 };
 
+#if defined(CONFIG_DM_MMC) && defined(CONFIG_DM_MMC_OPS)
+static const struct dm_mmc_ops sunxi_mmc_ops;
+#else
+static const struct mmc_ops sunxi_mmc_ops;
+#endif
+
+#if !defined(CONFIG_DM_MMC)
 /* support 4 mmc hosts */
 struct sunxi_mmc_host mmc_host[4];
 
 static int sunxi_mmc_getcd_gpio(int sdc_no)
 {
        switch (sdc_no) {
        case 0: return sunxi_name_to_gpio(CONFIG_MMC0_CD_PIN);
        case 1: return sunxi_name_to_gpio(CONFIG_MMC1_CD_PIN);
        case 2: return sunxi_name_to_gpio(CONFIG_MMC2_CD_PIN);
+#if !defined(CONFIG_ARCH_SUN50I)  /* only 3 MMC controllers on the A64 */
        case 3: return sunxi_name_to_gpio(CONFIG_MMC3_CD_PIN);
+#endif
        }
        return -EINVAL;
 }
 
 static int mmc_resource_init(int sdc_no)
 {
        struct sunxi_mmc_host *mmchost = &mmc_host[sdc_no];
        struct sunxi_ccm_reg *ccm = (struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
        int cd_pin, ret = 0;
 
        debug("init mmc %d resource\n", sdc_no);
 
        switch (sdc_no) {
        case 0:
                mmchost->reg = (struct sunxi_mmc *)SUNXI_MMC0_BASE;
                mmchost->mclkreg = &ccm->sd0_clk_cfg;
                break;
        case 1:
                mmchost->reg = (struct sunxi_mmc *)SUNXI_MMC1_BASE;
                mmchost->mclkreg = &ccm->sd1_clk_cfg;
                break;
        case 2:
                mmchost->reg = (struct sunxi_mmc *)SUNXI_MMC2_BASE;
                mmchost->mclkreg = &ccm->sd2_clk_cfg;
                break;
+#if !defined(CONFIG_ARCH_SUN50I)  /* only 3 MMC controllers on the A64 */
        case 3:
                mmchost->reg = (struct sunxi_mmc *)SUNXI_MMC3_BASE;
                mmchost->mclkreg = &ccm->sd3_clk_cfg;
                break;
+#endif
        default:
                printf("Wrong mmc number %d\n", sdc_no);
                return -1;
        }
        mmchost->mmc_no = sdc_no;
 
        cd_pin = sunxi_mmc_getcd_gpio(sdc_no);
        if (cd_pin >= 0) {
                ret = gpio_request(cd_pin, "mmc_cd");
                if (!ret) {
                        sunxi_gpio_set_pull(cd_pin, SUNXI_GPIO_PULL_UP);
                        ret = gpio_direction_input(cd_pin);
                }
        }
+       mmchost->cd_pin = cd_pin;
+
+       return ret;
+}
+#endif
+
+#if defined(CONFIG_DM_MMC)
+static int mmc_resource_init_from_udev(struct udevice *dev)
+{
+       struct sunxi_mmc_host *mmchost = dev_get_priv(dev);
+       int ret = 0;
+
+       debug("%s: %s\n", dev->name, __func__);
+
+       switch ((uintptr_t)mmchost->reg) {
+       case SUNXI_MMC0_BASE:
+               mmchost->mmc_no = 0;
+               break;
+       case SUNXI_MMC1_BASE:
+               mmchost->mmc_no = 1;
+               break;
+       case SUNXI_MMC2_BASE:
+               mmchost->mmc_no = 2;
+               break;
+#if !defined(CONFIG_ARCH_SUN50I)  /* only 3 MMC controllers on the A64 */
+       case SUNXI_MMC3_BASE:
+               mmchost->mmc_no = 3;
+               break;
+#endif
+       default:
+               debug("%s: unknown base address %p\n", __func__, mmchost->reg);
+               return -1;
+       }
+
+       debug("%s: mmc_no %d\n", dev->name, mmchost->mmc_no);
 
        return ret;
 }
+#endif
 
+#if !defined(CONFIG_DM_MMC)
 static int mmc_set_mod_clk(struct sunxi_mmc_host *mmchost, unsigned int hz)
 {
        unsigned int pll, pll_hz, div, n, oclk_dly, sclk_dly;
@@ -154,112 +230,156 @@ static int mmc_set_mod_clk(struct sunxi_mmc_host 
*mmchost, unsigned int hz)
        return 0;
 }
 
-static int mmc_clk_io_on(int sdc_no)
+static int mmc_clk_io_on(struct sunxi_mmc_host *mmchost)
 {
-       struct sunxi_mmc_host *mmchost = &mmc_host[sdc_no];
        struct sunxi_ccm_reg *ccm = (struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
+       int sdc_no = mmchost->mmc_no;
 
        debug("init mmc %d clock and io\n", sdc_no);
 
        /* config ahb clock */
        setbits_le32(&ccm->ahb_gate0, 1 << AHB_GATE_OFFSET_MMC(sdc_no));
 
 #ifdef CONFIG_SUNXI_GEN_SUN6I
        /* unassert reset */
        setbits_le32(&ccm->ahb_reset0_cfg, 1 << AHB_RESET_OFFSET_MMC(sdc_no));
 #endif
 #if defined(CONFIG_MACH_SUN9I)
        /* sun9i has a mmc-common module, also set the gate and reset there */
        writel(SUNXI_MMC_COMMON_CLK_GATE | SUNXI_MMC_COMMON_RESET,
               SUNXI_MMC_COMMON_BASE + 4 * sdc_no);
 #endif
 
        return mmc_set_mod_clk(mmchost, 24000000);
 }
+#endif
+
+#if defined(CONFIG_DM_MMC)
+static int mmc_clk_io_on(struct sunxi_mmc_host *mmchost)
+{
+       /* Enable the AHB clock gate */
+       clk_enable(&mmchost->ahb_clk_gate);
+
+       /* Deassert the AHB module reset */
+       reset_deassert(&mmchost->reset);
+
+#if defined(CONFIG_MACH_SUN9I)
+       /* TODO --- covert this to DM */
+       /* sun9i has a mmc-common module, also set the gate and reset there */
+       writel(SUNXI_MMC_COMMON_CLK_GATE | SUNXI_MMC_COMMON_RESET,
+              SUNXI_MMC_COMMON_BASE + 4 * sdc_no);
+#endif
+
+       clk_set_rate(&mmchost->mmc_clk, 24000000);
+       clk_enable(&mmchost->mmc_clk);
+
+       return 0;
+}
+#endif
 
 static int mmc_update_clk(struct mmc *mmc)
 {
        struct sunxi_mmc_host *mmchost = mmc->priv;
        unsigned int cmd;
        unsigned timeout_msecs = 2000;
        unsigned long start = get_timer(0);
 
+       debug("%s: base %p\n", __func__, mmchost->reg);
+
        cmd = SUNXI_MMC_CMD_START |
              SUNXI_MMC_CMD_UPCLK_ONLY |
              SUNXI_MMC_CMD_WAIT_PRE_OVER;
        writel(cmd, &mmchost->reg->cmd);
        while (readl(&mmchost->reg->cmd) & SUNXI_MMC_CMD_START) {
                if (get_timer(start) > timeout_msecs)
                        return -1;
        }
 
        /* clock update sets various irq status bits, clear these */
        writel(readl(&mmchost->reg->rint), &mmchost->reg->rint);
 
        return 0;
 }
 
 static int mmc_config_clock(struct mmc *mmc)
 {
        struct sunxi_mmc_host *mmchost = mmc->priv;
        unsigned rval = readl(&mmchost->reg->clkcr);
 
        /* Disable Clock */
        rval &= ~SUNXI_MMC_CLK_ENABLE;
        writel(rval, &mmchost->reg->clkcr);
        if (mmc_update_clk(mmc))
                return -1;
 
+#if !defined(CONFIG_DM_MMC)
        /* Set mod_clk to new rate */
        if (mmc_set_mod_clk(mmchost, mmc->clock))
                return -1;
+#else
+       if (clk_set_rate(&mmchost->mmc_clk, mmc->clock) == 0)
+               return -1;
+#endif
 
        /* Clear internal divider */
        rval &= ~SUNXI_MMC_CLK_DIVIDER_MASK;
        writel(rval, &mmchost->reg->clkcr);
 
        /* Re-enable Clock */
        rval |= SUNXI_MMC_CLK_ENABLE;
        writel(rval, &mmchost->reg->clkcr);
        if (mmc_update_clk(mmc))
                return -1;
 
        return 0;
 }
 
+#if defined(CONFIG_DM_MMC_OPS)
+static int sunxi_mmc_set_ios(struct udevice *dev)
+{
+       struct mmc *mmc = mmc_get_mmc_dev(dev);
+#else
 static int sunxi_mmc_set_ios(struct mmc *mmc)
 {
+#endif
        struct sunxi_mmc_host *mmchost = mmc->priv;
 
        debug("set ios: bus_width: %x, clock: %d\n",
              mmc->bus_width, mmc->clock);
 
        /* Change clock first */
        if (mmc->clock && mmc_config_clock(mmc) != 0) {
                mmchost->fatal_err = 1;
                return -EINVAL;
        }
 
        /* Change bus width */
        if (mmc->bus_width == 8)
                writel(0x2, &mmchost->reg->width);
        else if (mmc->bus_width == 4)
                writel(0x1, &mmchost->reg->width);
        else
                writel(0x0, &mmchost->reg->width);
 
        return 0;
 }
 
 static int sunxi_mmc_core_init(struct mmc *mmc)
 {
        struct sunxi_mmc_host *mmchost = mmc->priv;
+       uint32_t regval;
+       int ret = 0;
+
+       debug("%s: base %p", __func__, mmchost->reg);
 
        /* Reset controller */
        writel(SUNXI_MMC_GCTRL_RESET, &mmchost->reg->gctrl);
-       udelay(1000);
 
-       return 0;
+       /* Wait for the reset bit (auto-clearing) to deassert */
+       ret = readl_poll_timeout(&mmchost->reg->gctrl, regval,
+                                !(regval & SUNXI_MMC_GCTRL_RESET), 1000000);
+
+       return ret;
 }
 
 static int mmc_trans_data_by_cpu(struct mmc *mmc, struct mmc_data *data)
@@ -317,113 +437,120 @@ static int mmc_rint_wait(struct mmc *mmc, unsigned int 
timeout_msecs,
        return 0;
 }
 
+#if defined(CONFIG_DM_MMC_OPS)
+static int sunxi_mmc_send_cmd(struct udevice *dev, struct mmc_cmd *cmd,
+                             struct mmc_data *data)
+{
+       struct mmc *mmc = mmc_get_mmc_dev(dev);
+#else
 static int sunxi_mmc_send_cmd(struct mmc *mmc, struct mmc_cmd *cmd,
                              struct mmc_data *data)
 {
+#endif
        struct sunxi_mmc_host *mmchost = mmc->priv;
        unsigned int cmdval = SUNXI_MMC_CMD_START;
        unsigned int timeout_msecs;
        int error = 0;
        unsigned int status = 0;
        unsigned int bytecnt = 0;
 
        if (mmchost->fatal_err)
                return -1;
        if (cmd->resp_type & MMC_RSP_BUSY)
                debug("mmc cmd %d check rsp busy\n", cmd->cmdidx);
        if (cmd->cmdidx == 12)
                return 0;
 
        if (!cmd->cmdidx)
                cmdval |= SUNXI_MMC_CMD_SEND_INIT_SEQ;
        if (cmd->resp_type & MMC_RSP_PRESENT)
                cmdval |= SUNXI_MMC_CMD_RESP_EXPIRE;
        if (cmd->resp_type & MMC_RSP_136)
                cmdval |= SUNXI_MMC_CMD_LONG_RESPONSE;
        if (cmd->resp_type & MMC_RSP_CRC)
                cmdval |= SUNXI_MMC_CMD_CHK_RESPONSE_CRC;
 
        if (data) {
                if ((u32)(long)data->dest & 0x3) {
                        error = -1;
                        goto out;
                }
 
                cmdval |= SUNXI_MMC_CMD_DATA_EXPIRE|SUNXI_MMC_CMD_WAIT_PRE_OVER;
                if (data->flags & MMC_DATA_WRITE)
                        cmdval |= SUNXI_MMC_CMD_WRITE;
                if (data->blocks > 1)
                        cmdval |= SUNXI_MMC_CMD_AUTO_STOP;
                writel(data->blocksize, &mmchost->reg->blksz);
                writel(data->blocks * data->blocksize, &mmchost->reg->bytecnt);
        }
 
-       debug("mmc %d, cmd %d(0x%08x), arg 0x%08x\n", mmchost->mmc_no,
+       debug("mmc %p, cmd %d(0x%08x), arg 0x%08x\n", mmchost,
              cmd->cmdidx, cmdval | cmd->cmdidx, cmd->cmdarg);
        writel(cmd->cmdarg, &mmchost->reg->arg);
 
        if (!data)
                writel(cmdval | cmd->cmdidx, &mmchost->reg->cmd);
 
        /*
         * transfer data and check status
         * STATREG[2] : FIFO empty
         * STATREG[3] : FIFO full
         */
        if (data) {
                int ret = 0;
 
                bytecnt = data->blocksize * data->blocks;
                debug("trans data %d bytes\n", bytecnt);
                writel(cmdval | cmd->cmdidx, &mmchost->reg->cmd);
                ret = mmc_trans_data_by_cpu(mmc, data);
                if (ret) {
                        error = readl(&mmchost->reg->rint) & \
                                SUNXI_MMC_RINT_INTERRUPT_ERROR_BIT;
                        error = -ETIMEDOUT;
                        goto out;
                }
        }
 
        error = mmc_rint_wait(mmc, 1000, SUNXI_MMC_RINT_COMMAND_DONE, "cmd");
        if (error)
                goto out;
 
        if (data) {
                timeout_msecs = 120;
                debug("cacl timeout %x msec\n", timeout_msecs);
                error = mmc_rint_wait(mmc, timeout_msecs,
                                      data->blocks > 1 ?
                                      SUNXI_MMC_RINT_AUTO_COMMAND_DONE :
                                      SUNXI_MMC_RINT_DATA_OVER,
                                      "data");
                if (error)
                        goto out;
        }
 
        if (cmd->resp_type & MMC_RSP_BUSY) {
                unsigned long start = get_timer(0);
                timeout_msecs = 2000;
 
                do {
                        status = readl(&mmchost->reg->status);
                        if (get_timer(start) > timeout_msecs) {
                                debug("busy timeout\n");
                                error = -ETIMEDOUT;
                                goto out;
                        }
                } while (status & SUNXI_MMC_STATUS_CARD_DATA_BUSY);
        }
 
        if (cmd->resp_type & MMC_RSP_136) {
                cmd->response[0] = readl(&mmchost->reg->resp3);
                cmd->response[1] = readl(&mmchost->reg->resp2);
                cmd->response[2] = readl(&mmchost->reg->resp1);
                cmd->response[3] = readl(&mmchost->reg->resp0);
                debug("mmc resp 0x%08x 0x%08x 0x%08x 0x%08x\n",
                      cmd->response[3], cmd->response[2],
                      cmd->response[1], cmd->response[0]);
        } else {
                cmd->response[0] = readl(&mmchost->reg->resp0);
                debug("mmc resp 0x%08x\n", cmd->response[0]);
        }
@@ -439,50 +566,225 @@ out:
        return error;
 }
 
+static inline int cdpin_is_valid(struct sunxi_mmc_host *priv)
+{
+#if !defined(CONFIG_DM_MMC)
+       return priv->cd_pin >= 0;
+#else
+       return dm_gpio_is_valid(&priv->cd_gpio);
+#endif
+}
+
+static inline int cdpin_get_value(struct sunxi_mmc_host *priv)
+{
+#if !defined(CONFIG_DM_MMC)
+       return gpio_get_value(priv->cd_pin);
+#else
+       return dm_gpio_get_value(&priv->cd_gpio);
+#endif
+}
+
+#if defined(CONFIG_DM_MMC_OPS)
+static int sunxi_mmc_getcd(struct udevice *dev)
+{
+       struct mmc *mmc = mmc_get_mmc_dev(dev);
+#else
 static int sunxi_mmc_getcd(struct mmc *mmc)
 {
-       struct sunxi_mmc_host *mmchost = mmc->priv;
-       int cd_pin;
+#endif
+       struct sunxi_mmc_host *priv = mmc->priv;
+       int value = 1;
 
-       cd_pin = sunxi_mmc_getcd_gpio(mmchost->mmc_no);
-       if (cd_pin < 0)
-               return 1;
+       if (cdpin_is_valid(priv)) {
+               value = cdpin_get_value(priv);
 
-       return !gpio_get_value(cd_pin);
-}
+               if (priv->cd_inverted)
+                       return !value;
+       }
 
-static const struct mmc_ops sunxi_mmc_ops = {
-       .send_cmd       = sunxi_mmc_send_cmd,
-       .set_ios        = sunxi_mmc_set_ios,
-       .init           = sunxi_mmc_core_init,
-       .getcd          = sunxi_mmc_getcd,
-};
+       return value;
+}
 
+#if !defined(CONFIG_DM_MMC)
 struct mmc *sunxi_mmc_init(int sdc_no)
 {
        struct mmc_config *cfg = &mmc_host[sdc_no].cfg;
 
        memset(&mmc_host[sdc_no], 0, sizeof(struct sunxi_mmc_host));
+       mmc_host[sdc_no].cd_inverted = true;
 
        cfg->name = "SUNXI SD/MMC";
        cfg->ops  = &sunxi_mmc_ops;
 
        cfg->voltages = MMC_VDD_32_33 | MMC_VDD_33_34;
        cfg->host_caps = MMC_MODE_4BIT;
 #if defined(CONFIG_MACH_SUN50I) || defined(CONFIG_MACH_SUN8I)
        if (sdc_no == 2)
                cfg->host_caps = MMC_MODE_8BIT;
 #endif
        cfg->host_caps |= MMC_MODE_HS_52MHz | MMC_MODE_HS;
        cfg->b_max = CONFIG_SYS_MMC_MAX_BLK_COUNT;
 
        cfg->f_min = 400000;
        cfg->f_max = 52000000;
 
        if (mmc_resource_init(sdc_no) != 0)
                return NULL;
 
-       mmc_clk_io_on(sdc_no);
+       mmc_clk_io_on(&mmc_host[sdc_no]);
 
        return mmc_create(cfg, &mmc_host[sdc_no]);
 }
+#endif
+
+#if defined(CONFIG_DM_MMC) && defined(CONFIG_DM_MMC_OPS)
+static const struct dm_mmc_ops sunxi_mmc_ops = {
+       .send_cmd       = sunxi_mmc_send_cmd,
+       .set_ios        = sunxi_mmc_set_ios,
+       .get_cd         = sunxi_mmc_getcd,
+};
+#else
+static const struct mmc_ops sunxi_mmc_ops = {
+       .send_cmd       = sunxi_mmc_send_cmd,
+       .set_ios        = sunxi_mmc_set_ios,
+       .init           = sunxi_mmc_core_init,
+       .getcd          = sunxi_mmc_getcd,
+};
+#endif
+
+#if defined(CONFIG_DM_MMC)
+static int sunxi_mmc_ofdata_to_platdata(struct udevice *dev)
+{
+       return 0;
+}
+
+static int sunxi_mmc_probe(struct udevice *dev)
+{
+       struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
+#if defined(CONFIG_BLK)
+       struct sunxi_mmc_plat *plat = dev_get_platdata(dev);
+#endif
+       struct sunxi_mmc_host *priv = dev_get_priv(dev);
+       struct mmc_config *cfg = &priv->cfg;
+       int bus_width;
+       u32 f_minmax[2];
+
+       priv->reg = (void *)dev_get_addr(dev);
+       cfg->name = "SUNXI SD/MMC";
+#if !(defined(CONFIG_DM_MMC) && defined(CONFIG_DM_MMC_OPS))
+       cfg->ops  = &sunxi_mmc_ops;
+#endif
+       cfg->voltages = MMC_VDD_32_33 | MMC_VDD_33_34;
+       cfg->b_max = CONFIG_SYS_MMC_MAX_BLK_COUNT;
+       cfg->host_caps = MMC_MODE_HS_52MHz | MMC_MODE_HS;
+
+       bus_width = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
+                                  "bus-width", 4);
+       if (bus_width == 8)
+               cfg->host_caps |= MMC_MODE_8BIT;
+       else if (bus_width == 4)
+               cfg->host_caps |= MMC_MODE_4BIT;
+
+       debug("%s: reg %p bus_width %d\n", dev->name, priv->reg, bus_width);
+
+       if (!fdtdec_get_int_array(gd->fdt_blob, dev->of_offset,
+                                 "clock-freq-min-max", f_minmax, 2)) {
+               cfg->f_min = f_minmax[0];
+               cfg->f_max = f_minmax[1];
+       } else {
+               /* use the defaults */
+               cfg->f_min = 400000;
+               cfg->f_max = 52000000;
+       }
+
+       /* Some legacy functionality in our tree still depends on the
+        * mmchost->mmc_no... until we can get rid of this, initialise
+        * it based on the base address of the device.
+        */
+       if (mmc_resource_init_from_udev(dev) != 0)
+               return -EINVAL;
+
+       /* All GPIOs are optional */
+       gpio_request_by_name(dev, "cd-gpios", 0,
+                            &priv->cd_gpio, GPIOD_IS_IN);
+       priv->cd_inverted = fdtdec_get_bool(gd->fdt_blob, dev->of_offset,
+                                           "cd-inverted");
+       gpio_request_by_name(dev, "wp-gpios", 0,
+                            &priv->wp_gpio, GPIOD_IS_IN);
+       priv->wp_inverted = fdtdec_get_bool(gd->fdt_blob, dev->of_offset,
+                                           "wp-inverted");
+       gpio_request_by_name(dev, "power-gpios", 0,
+                            &priv->pwr_gpio, GPIOD_IS_OUT);
+       if (dm_gpio_is_valid(&priv->pwr_gpio))
+               dm_gpio_set_value(&priv->pwr_gpio, 1);
+
+       if (reset_get_by_name(dev, "ahb", &priv->reset)) {
+               error("%s: failed to get 'ahb' reset\n", dev->name);
+               return -EINVAL;
+       }
+
+       if (clk_get_by_name(dev, "ahb", &priv->ahb_clk_gate) ||
+           clk_get_by_name(dev, "mmc", &priv->mmc_clk)) {
+               error("%s: failed to get all required clocks ('ahb', 'mmc')\n",
+                     dev->name);
+               return -EINVAL;
+       }
+
+       mmc_clk_io_on(priv);
+
+#if defined(CONFIG_BLK)
+       priv->mmc = &plat->mmc;
+#else
+       priv->mmc = mmc_create(cfg, priv);
+       if (priv->mmc == NULL)
+               return -1;
+#endif
+       priv->mmc->priv = priv;
+       priv->mmc->dev = dev;
+       priv->mmc->cfg = cfg;
+       priv->mmc->has_init = 0;
+       upriv->mmc = priv->mmc;
+
+#if defined(CONFIG_DM_MMC) && defined(CONFIG_DM_MMC_OPS)
+       sunxi_mmc_core_init(priv->mmc);
+#endif
+       return 0;
+}
+
+#if defined(CONFIG_BLK)
+static int sunxi_mmc_bind(struct udevice *dev)
+{
+       struct sunxi_mmc_plat *plat = dev_get_platdata(dev);
+       struct sunxi_mmc_host *priv = dev_get_priv(dev);
+
+       debug("%s: %s\n", dev->name, __func__);
+
+       /* TODO: To move cfg into plat, we need to change the legacy
+          code, which references through the arrays... */
+
+       return mmc_bind(dev, &plat->mmc, &priv->cfg);
+}
+#endif
+
+static const struct udevice_id sunxi_mmc_ids[] = {
+       { .compatible = "allwinner,sun50i-a64-mmc" },
+       { }
+};
+
+U_BOOT_DRIVER(sunxi_mmc_drv) = {
+       .name           = "sunxi_mmc",
+       .id             = UCLASS_MMC,
+       .of_match       = sunxi_mmc_ids,
+       .ofdata_to_platdata = sunxi_mmc_ofdata_to_platdata,
+       .probe          = sunxi_mmc_probe,
+       .priv_auto_alloc_size = sizeof(struct sunxi_mmc_host),
+       .platdata_auto_alloc_size = sizeof(struct sunxi_mmc_plat),
+#if defined(CONFIG_DM_MMC_OPS)
+       .ops            = &sunxi_mmc_ops,
+#endif
+#if defined(CONFIG_BLK)
+       .bind           = sunxi_mmc_bind,
+#endif
+};
+
+#endif
-- 
1.9.1

_______________________________________________
U-Boot mailing list
U-Boot@lists.denx.de
http://lists.denx.de/mailman/listinfo/u-boot

Reply via email to