]> pilppa.com Git - linux-2.6-omap-h63xx.git/commitdiff
Add OMAP MMC driver
authorJuha Yrjola <juha.yrjola@nokia.com>
Mon, 9 May 2005 21:23:25 +0000 (14:23 -0700)
committerTony Lindgren <tony@atomide.com>
Mon, 9 May 2005 21:23:25 +0000 (14:23 -0700)
Adds OMAP MMC driver.

Signed-off-by: Juha Yrjola <juha.yrjola@nokia.com>
Signed-off-by: Tony Lindgren <tony@atomide.com>
drivers/mmc/Kconfig
drivers/mmc/Makefile
drivers/mmc/mmc.c
drivers/mmc/mmc_block.c
drivers/mmc/omap.c [new file with mode: 0644]
drivers/mmc/omap.h [new file with mode: 0644]

index 2e70d74fbdee2d8bffa3a7a4d01de57061d317a1..8141f9c32be5bde1b8595f85d6ed2d540139fa42 100644 (file)
@@ -29,6 +29,23 @@ config MMC_BLOCK
          mount the filesystem. Almost everyone wishing MMC support
          should say Y or M here.
 
+config MMC_BLOCK_BROKEN_RFD
+       boolean "Write work-around for incompatible cards"
+       depends on MMC_BLOCK
+       default n
+       help
+         Say y here if your MMC card fails write operations. Some cards
+         lie about being ready to receive data while they actually are not.
+
+config MMC_BULKTRANSFER
+       bool "Multi-block writes (EXPERIMENTAL)"
+       depends on MMC_BLOCK != n && EXPERIMENTAL
+       default n
+       help
+         By default all writes are done one sector at a time. Enable
+         this option to transfer as large blocks as the host supports.
+         The transfer speed is in most cases doubled.
+
 config MMC_ARMMMCI
        tristate "ARM AMBA Multimedia Card Interface support"
        depends on ARM_AMBA && MMC
@@ -49,6 +66,36 @@ config MMC_PXA
 
          If unsure, say N.
 
+config MMC_OMAP
+       tristate "TI OMAP Multimedia Card Interface support"
+       depends on ARCH_OMAP && MMC
+       select TPS65010 if MACH_OMAP_H2
+       help
+         This selects the TI OMAP Multimedia card Interface.
+         If you have an OMAP board with a Multimedia Card slot,
+         say Y or M here.
+
+         If unsure, say N.
+
+config MMC_OMAP16XX_BLOCK1
+       boolean "First MMC block on OMAP16XX"
+       depends on ARCH_OMAP16XX && MMC_OMAP
+       default y if MACH_OMAP_H2 || MACH_OMAP_H3
+       help
+         This enables the first of two MMC blocks on OMAP1610 multimedia
+         processor. You need to enable the correct block to activate your
+         MMC slot.
+
+config MMC_OMAP16XX_BLOCK2
+       boolean "Second MMC block on OMAP16XX"
+       depends on ARCH_OMAP16XX && MMC_OMAP
+       default n if MACH_OMAP_H2 || MACH_OMAP_H3
+       default y
+       help
+         This enables the second of two MMC blocks on OMAP1610 multimedia
+         processor. You need to enable the correct block to activate your
+         MMC slot.
+
 config MMC_WBSD
        tristate "Winbond W83L51xD SD/MMC Card Interface support"
        depends on MMC && ISA && ISA_DMA_API
index 89510c2086c783b4e77d55b1de5301c38c2d36f9..1dcb234b065c2c87a41a538b83ebf4308a2f2d90 100644 (file)
@@ -18,5 +18,6 @@ obj-$(CONFIG_MMC_BLOCK)               += mmc_block.o
 obj-$(CONFIG_MMC_ARMMMCI)      += mmci.o
 obj-$(CONFIG_MMC_PXA)          += pxamci.o
 obj-$(CONFIG_MMC_WBSD)         += wbsd.o
+obj-$(CONFIG_MMC_OMAP)         += omap.o
 
 mmc_core-y := mmc.o mmc_queue.o mmc_sysfs.o
index eeb9f6668e69167044b3b41bf6559479f30dd5ab..4c68df91b7d8faf37c16812563d2850803bfd17b 100644 (file)
@@ -474,13 +474,13 @@ static void mmc_power_up(struct mmc_host *host)
        int bit = fls(host->ocr_avail) - 1;
 
        host->ios.vdd = bit;
+       host->ios.clock = host->f_min;
        host->ios.bus_mode = MMC_BUSMODE_OPENDRAIN;
        host->ios.power_mode = MMC_POWER_UP;
        host->ops->set_ios(host, &host->ios);
 
        mmc_delay(1);
 
-       host->ios.clock = host->f_min;
        host->ios.power_mode = MMC_POWER_ON;
        host->ops->set_ios(host, &host->ios);
 
@@ -512,7 +512,7 @@ static int mmc_send_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr)
 
                if (cmd.resp[0] & MMC_CARD_BUSY || ocr == 0)
                        break;
-
+               mmc_delay(1);
                err = MMC_ERR_TIMEOUT;
 
                mmc_delay(10);
index b5b4a7b1190349236fd42f073eb522da471177d1..fa7dff9a2ba81dfbac50018e41a1f6ab950d7b92 100644 (file)
@@ -54,6 +54,7 @@ struct mmc_blk_data {
 
        unsigned int    usage;
        unsigned int    block_bits;
+       unsigned int    suspended;
 };
 
 static DECLARE_MUTEX(open_lock);
@@ -158,6 +159,19 @@ static int mmc_blk_prep_rq(struct mmc_queue *mq, struct request *req)
                stat = BLKPREP_KILL;
        }
 
+       if (md->suspended) {
+               blk_plug_device(md->queue.queue);
+               stat = BLKPREP_DEFER;
+       }
+
+       /*
+        * Check for excessive requests.
+        */
+       if (req->sector + req->nr_sectors > get_capacity(req->rq_disk)) {
+               printk("bad request size\n");
+               stat = BLKPREP_KILL;
+       }
+
        return stat;
 }
 
@@ -166,9 +180,25 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
        struct mmc_blk_data *md = mq->data;
        struct mmc_card *card = md->queue.card;
        int ret;
+       
+#ifdef CONFIG_MMC_BULKTRANSFER
+       int failsafe;
+#endif
 
        if (mmc_card_claim_host(card))
                goto cmd_err;
+       
+#ifdef CONFIG_MMC_BULKTRANSFER
+       /*
+        * We first try transfering multiple blocks. If this fails
+        * we fall back to single block transfers.
+        *
+        * This gives us good performance when all is well and the
+        * possibility to determine which sector fails when all
+        * is not well.
+        */
+       failsafe = 0;
+#endif
 
        do {
                struct mmc_blk_request brq;
@@ -188,14 +218,32 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
                brq.stop.arg = 0;
                brq.stop.flags = MMC_RSP_R1B;
 
+#ifdef CONFIG_MMC_BULKTRANSFER         
+               /*
+                * A multi-block transfer failed. Falling back to single
+                * blocks.
+                */
+               if (failsafe)
+                       brq.data.blocks = 1;
+               
+#else
+               /*
+                * Writes are done one sector at a time.
+                */
+               if (rq_data_dir(req) != READ)
+                       brq.data.blocks = 1;
+#endif
+               
+               ret = 1;
+
                if (rq_data_dir(req) == READ) {
                        brq.cmd.opcode = brq.data.blocks > 1 ? MMC_READ_MULTIPLE_BLOCK : MMC_READ_SINGLE_BLOCK;
                        brq.data.flags |= MMC_DATA_READ;
                } else {
-                       brq.cmd.opcode = MMC_WRITE_BLOCK;
+                       brq.cmd.opcode = brq.data.blocks > 1 ? MMC_WRITE_MULTIPLE_BLOCK :
+                               MMC_WRITE_BLOCK;
                        brq.cmd.flags = MMC_RSP_R1B;
                        brq.data.flags |= MMC_DATA_WRITE;
-                       brq.data.blocks = 1;
                }
                brq.mrq.stop = brq.data.blocks > 1 ? &brq.stop : NULL;
 
@@ -206,19 +254,19 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
                if (brq.cmd.error) {
                        printk(KERN_ERR "%s: error %d sending read/write command\n",
                               req->rq_disk->disk_name, brq.cmd.error);
-                       goto cmd_err;
+                       goto cmd_fail;
                }
 
                if (brq.data.error) {
                        printk(KERN_ERR "%s: error %d transferring data\n",
                               req->rq_disk->disk_name, brq.data.error);
-                       goto cmd_err;
+                       goto cmd_fail;
                }
 
                if (brq.stop.error) {
                        printk(KERN_ERR "%s: error %d sending stop command\n",
                               req->rq_disk->disk_name, brq.stop.error);
-                       goto cmd_err;
+                       goto cmd_fail;
                }
 
                do {
@@ -231,8 +279,15 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
                        if (err) {
                                printk(KERN_ERR "%s: error %d requesting status\n",
                                       req->rq_disk->disk_name, err);
-                               goto cmd_err;
+                               goto cmd_fail;
                        }
+#ifdef CONFIG_MMC_BLOCK_BROKEN_RFD
+                       /* Work-around for broken cards setting READY_FOR_DATA
+                        * when not actually ready.
+                        */
+                       if (R1_CURRENT_STATE(cmd.resp[0]) == 7)
+                               cmd.resp[0] &= ~R1_READY_FOR_DATA;
+#endif
                } while (!(cmd.resp[0] & R1_READY_FOR_DATA));
 
 #if 0
@@ -257,6 +312,27 @@ static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req)
                        end_that_request_last(req);
                }
                spin_unlock_irq(&md->lock);
+               
+#ifdef CONFIG_MMC_BULKTRANSFER
+               /*
+                * Go back to bulk mode if in failsafe mode.
+                */
+               failsafe = 0;
+#endif
+
+               continue;
+
+ cmd_fail:
+
+#ifdef CONFIG_MMC_BULKTRANSFER
+               if (failsafe)
+                       goto cmd_err;
+               else
+                       failsafe = 1;
+#else
+               goto cmd_err;
+#endif
+
        } while (ret);
 
        mmc_card_release_host(card);
diff --git a/drivers/mmc/omap.c b/drivers/mmc/omap.c
new file mode 100644 (file)
index 0000000..5905f26
--- /dev/null
@@ -0,0 +1,1226 @@
+/*
+ *  linux/drivers/media/mmc/omap.c
+ *
+ *  Copyright (C) 2004 Nokia Corporation
+ *  Written by Tuukka Tikkanen and Juha Yrjölä <juha.yrjola@nokia.com>
+ *  Pin multiplexing and Innovator support by Tony Lindgren <tony@atomide.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ */
+
+#include <linux/config.h>
+#include <linux/module.h>
+#include <linux/moduleparam.h>
+#include <linux/init.h>
+#include <linux/ioport.h>
+#include <linux/device.h>
+#include <linux/interrupt.h>
+#include <linux/dma-mapping.h>
+#include <linux/delay.h>
+#include <linux/spinlock.h>
+#include <linux/timer.h>
+#include <linux/mmc/host.h>
+#include <linux/mmc/protocol.h>
+#include <linux/mmc/card.h>
+
+#include <asm/io.h>
+#include <asm/irq.h>
+#include <asm/scatterlist.h>
+#include <asm/mach-types.h>
+
+#include <asm/arch/board.h>
+#include <asm/arch/gpio.h>
+#include <asm/arch/dma.h>
+#include <asm/arch/mux.h>
+#include <asm/arch/fpga.h>
+#include <asm/arch/tps65010.h>
+
+#include <asm/hardware/clock.h>
+
+#include "omap.h"
+
+#define DRIVER_NAME "mmci-omap"
+
+#ifdef CONFIG_MMC_DEBUG
+//#define DBG(x...)    printk(KERN_DEBUG x)
+#define DBG(x...)      printk(x)
+#else
+#define DBG(x...)      do { } while (0)
+#endif
+
+/* Specifies how often in millisecs to poll for card status changes
+ * when the cover switch is open */
+#define OMAP_MMC_SWITCH_POLL_DELAY     500
+
+static s16 mmc1_power_pin = -1,
+          mmc2_power_pin = -1;
+static s16 mmc1_switch_pin = -1,
+          mmc2_switch_pin = -1;
+
+static int mmc_omap_enable_poll = 1;
+
+struct mmc_omap_host {
+       int                     initialized;
+       int                     suspended;
+       struct mmc_request *    mrq;
+       struct mmc_command *    cmd;
+       struct mmc_data *       data;
+       struct mmc_host *       mmc;
+       struct device *         dev;
+       unsigned char           id; /* 1610 has 2 MMC blocks */
+       struct clk *            clk;
+       u32                     base;
+       int irq;
+       unsigned char           bus_mode;
+       unsigned int            dma_len;
+#define OMAP_MMC_DATADIR_NONE  0
+#define OMAP_MMC_DATADIR_READ  1
+#define OMAP_MMC_DATADIR_WRITE 2
+       unsigned char           datadir;
+       u16 * buffer;
+       u32 bytesleft;
+       int                     power_pin;
+
+       int                     use_dma, dma_ch;
+       struct completion       dma_completion;
+
+       int                     switch_pin;
+       struct work_struct      switch_work;
+       struct timer_list       switch_timer;
+       int                     switch_last_state;
+
+       unsigned char           sd_support;
+};
+
+static inline int
+mmc_omap_cover_is_open(struct mmc_omap_host *host)
+{
+       if (host->switch_pin < 0)
+               return 0;
+        return omap_get_gpio_datain(host->switch_pin);
+}
+
+static ssize_t
+mmc_omap_show_cover_switch(struct device *dev, char *buf)
+{
+       struct mmc_omap_host *host = dev_get_drvdata(dev);
+
+       return sprintf(buf, "%s\n", mmc_omap_cover_is_open(host) ? "open" : "closed");
+}
+
+static DEVICE_ATTR(cover_switch, S_IRUGO, mmc_omap_show_cover_switch, NULL);
+
+static ssize_t
+mmc_omap_show_enable_poll(struct device *dev, char *buf)
+{
+       return snprintf(buf, PAGE_SIZE, "%d\n", mmc_omap_enable_poll);
+}
+
+static ssize_t
+mmc_omap_store_enable_poll(struct device *dev, const char *buf, size_t size)
+{
+       int enable_poll;
+
+       if (sscanf(buf, "%10d", &enable_poll) != 1)
+               return -EINVAL;
+
+       if (enable_poll != mmc_omap_enable_poll) {
+               struct mmc_omap_host *host = dev_get_drvdata(dev);
+
+               mmc_omap_enable_poll = enable_poll;
+               if (enable_poll && host->switch_pin >= 0)
+                       schedule_work(&host->switch_work);
+       }
+       return size;
+}
+
+static DEVICE_ATTR(enable_poll, 0664,
+                  mmc_omap_show_enable_poll, mmc_omap_store_enable_poll);
+
+static void
+mmc_omap_start_command(struct mmc_omap_host *host, struct mmc_command *cmd)
+{
+       u32 cmdreg;
+       u32 resptype;
+       u32 cmdtype;
+
+       DBG("MMC%d: CMD%d, argument 0x%08x", host->id, cmd->opcode, cmd->arg);
+       if (cmd->flags & MMC_RSP_SHORT)
+               DBG(", 32-bit response");
+       if (cmd->flags & MMC_RSP_LONG)
+               DBG(", 128-bit response");
+       if (cmd->flags & MMC_RSP_CRC)
+               DBG(", CRC");
+       if (cmd->flags & MMC_RSP_BUSY)
+               DBG(", busy notification");
+       DBG("\n");
+
+       host->cmd = cmd;
+
+       resptype = 0;
+       cmdtype = 0;
+
+       /* Protocol layer does not provide response type,
+        * but our hardware needs to know exact type, not just size!
+        */
+       switch (cmd->flags & MMC_RSP_MASK) {
+       case MMC_RSP_NONE:
+               /* resp 0 */
+               break;
+       case MMC_RSP_SHORT:
+               /* resp 1, resp 1b */
+               /* OR resp 3!! (assume this if bus is set opendrain) */
+               if (host->bus_mode == MMC_BUSMODE_OPENDRAIN)
+                       resptype = 3;
+               else
+                       resptype = 1;
+               break;
+       case MMC_RSP_LONG:
+               /* resp 2 */
+               resptype = 2;
+               break;
+       }
+
+       /* Protocol layer does not provide command type, but our hardware
+        * needs it!
+        * any data transfer means adtc type (but that information is not
+        * in command structure, so we flagged it into host struct.)
+        * However, telling bc, bcr and ac apart based on response is
+        * not foolproof:
+        * CMD0  = bc  = resp0  CMD15 = ac  = resp0
+        * CMD2  = bcr = resp2  CMD10 = ac  = resp2
+        *
+        * Resolve to best guess with some exception testing:
+        * resp0 -> bc, except CMD15 = ac
+        * rest are ac, except if opendrain
+        */
+       if (host->datadir) {
+               cmdtype = OMAP_MMC_CMDTYPE_ADTC;
+       } else if (resptype == 0 && cmd->opcode != 15) {
+               cmdtype = OMAP_MMC_CMDTYPE_BC;
+       } else if (host->bus_mode == MMC_BUSMODE_OPENDRAIN) {
+               cmdtype = OMAP_MMC_CMDTYPE_BCR;
+       } else {
+               cmdtype = OMAP_MMC_CMDTYPE_AC;
+       }
+
+       cmdreg = cmd->opcode | (resptype << 8) | (cmdtype << 12);
+
+       if (host->bus_mode == MMC_BUSMODE_OPENDRAIN)
+               cmdreg |= 1 << 6;
+
+       if (cmd->flags & MMC_RSP_BUSY)
+               cmdreg |= 1 << 11;
+
+       if (host->datadir == OMAP_MMC_DATADIR_READ)
+               cmdreg |= 1 << 15;
+
+       clk_enable(host->clk);
+
+       OMAP_MMC_WRITE(host->base, CTO, 200);
+       OMAP_MMC_WRITE(host->base, ARGL, cmd->arg & 0xffff);
+       OMAP_MMC_WRITE(host->base, ARGH, cmd->arg >> 16);
+       OMAP_MMC_WRITE(host->base, IE,
+                      OMAP_MMC_STAT_A_EMPTY    | OMAP_MMC_STAT_A_FULL    |
+                      OMAP_MMC_STAT_CMD_CRC    | OMAP_MMC_STAT_CMD_TOUT  |
+                      OMAP_MMC_STAT_DATA_CRC   | OMAP_MMC_STAT_DATA_TOUT |
+                      OMAP_MMC_STAT_END_OF_CMD | OMAP_MMC_STAT_CARD_ERR  |
+                      OMAP_MMC_STAT_END_OF_DATA);
+       OMAP_MMC_WRITE(host->base, CMD, cmdreg);
+}
+
+static void
+mmc_omap_xfer_done(struct mmc_omap_host *host, struct mmc_data *data)
+{
+       host->data = NULL;
+       host->datadir = OMAP_MMC_DATADIR_NONE;
+
+       if (data->error == MMC_ERR_NONE)
+               data->bytes_xfered += data->blocks * (1<<data->blksz_bits);
+       else {
+               int dma_ch = host->dma_ch;
+
+               /* We got an error, let's free the DMA channel if it's
+                * still allocated. */
+               if (dma_ch != -1) {
+                       host->dma_ch = -1;
+                       omap_free_dma(dma_ch);
+               }
+       }
+
+       dma_unmap_sg(mmc_dev(host->mmc), data->sg, host->dma_len,
+                    host->datadir);
+       host->dma_len = 0;
+
+       clk_disable(host->clk);
+
+       if (!data->stop) {
+               host->mrq = NULL;
+               mmc_request_done(host->mmc, data->mrq);
+               return;
+       }
+
+       mmc_omap_start_command(host, data->stop);
+}
+
+
+static void
+mmc_omap_cmd_done(struct mmc_omap_host *host, struct mmc_command *cmd)
+{
+       host->cmd = NULL;
+
+       switch (cmd->flags & MMC_RSP_MASK) {
+       case MMC_RSP_NONE:
+               /* resp 0 */
+               break;
+       case MMC_RSP_SHORT:
+               /* response types 1, 1b, 3, 4, 5, 6 */
+               cmd->resp[0] =
+                       OMAP_MMC_READ(host->base, RSP6) |
+                       (OMAP_MMC_READ(host->base, RSP7) << 16);
+               DBG("MMC%d: Response %08x\n", host->id, cmd->resp[0]);
+               break;
+       case MMC_RSP_LONG:
+               /* response type 2 */
+               cmd->resp[3] =
+                       OMAP_MMC_READ(host->base, RSP0) |
+                       (OMAP_MMC_READ(host->base, RSP1) << 16);
+               cmd->resp[2] =
+                       OMAP_MMC_READ(host->base, RSP2) |
+                       (OMAP_MMC_READ(host->base, RSP3) << 16);
+               cmd->resp[1] =
+                       OMAP_MMC_READ(host->base, RSP4) |
+                       (OMAP_MMC_READ(host->base, RSP5) << 16);
+               cmd->resp[0] =
+                       OMAP_MMC_READ(host->base, RSP6) |
+                       (OMAP_MMC_READ(host->base, RSP7) << 16);
+               DBG("MMC%d: Response %08x %08x %08x %08x\n", host->id,
+                   cmd->resp[0], cmd->resp[1],
+                   cmd->resp[2], cmd->resp[3]);
+               break;
+       }
+
+       if (host->data == NULL || cmd->error != MMC_ERR_NONE) {
+               DBG("MMC%d: End request, err %x\n", host->id, cmd->error);
+               host->mrq = NULL;
+               clk_disable(host->clk);
+               mmc_request_done(host->mmc, cmd->mrq);
+       }
+}
+
+static irqreturn_t mmc_omap_irq(int irq, void *dev_id, struct pt_regs *regs)
+{
+       struct mmc_omap_host * host = (struct mmc_omap_host *)dev_id;
+       u16 status;
+       int end_command;
+       int end_transfer;
+       int ii;
+
+       if (host->cmd == NULL && host->data == NULL) {
+               status = OMAP_MMC_READ(host->base, STAT);
+               printk(KERN_INFO "MMC%d: Spurious interrupt 0x%04x\n", host->id, status);
+               if (status != 0) {
+                       OMAP_MMC_WRITE(host->base, STAT, status);
+                       OMAP_MMC_WRITE(host->base, IE, 0);
+               }
+               return IRQ_HANDLED;
+       }
+
+       end_command = 0;
+       end_transfer = 0;
+
+       while ((status = OMAP_MMC_READ(host->base, STAT)) != 0) {
+               OMAP_MMC_WRITE(host->base, STAT, status); // Reset status bits
+               DBG("\tMMC IRQ %04x (cmd %d)\n", status,
+                   host->cmd != NULL ? host->cmd->opcode : -1);
+
+               if ((status & OMAP_MMC_STAT_A_FULL) ||
+                   ((status & OMAP_MMC_STAT_END_OF_DATA) &&
+                    (host->bytesleft > 0))) {
+                       // Buffer almost full
+                       ii = host->bytesleft / 2;
+                       if (ii > 32)
+                               ii = 32;
+                       host->bytesleft -= ii * 2;
+                       while (ii-- > 0)
+                               *host->buffer++ = OMAP_MMC_READ(host->base, DATA);
+               }
+
+               if (status & OMAP_MMC_STAT_A_EMPTY) {
+                       // Buffer almost empty
+                       ii = host->bytesleft / 2;
+                       if (ii > 32)
+                               ii = 32;
+                       host->bytesleft -= ii * 2;
+                       while (ii-- > 0)
+                               OMAP_MMC_WRITE(host->base, DATA, *host->buffer++);
+               }
+
+               if (status & OMAP_MMC_STAT_END_OF_DATA) {
+                       // Block sent/received
+                       end_transfer = 1;
+               }
+
+               if (status & OMAP_MMC_STAT_DATA_TOUT) {
+                       // Data timeout
+                       printk(KERN_DEBUG "MMC%d: Data timeout\n", host->id);
+                       if (host->data) {
+                               host->data->error |= MMC_ERR_TIMEOUT;
+                               end_transfer = 1;
+                       }
+               }
+
+               if (status & OMAP_MMC_STAT_DATA_CRC) {
+                       // Data CRC error
+                       if (host->data) {
+                               host->data->error |= MMC_ERR_BADCRC;
+                               printk(KERN_DEBUG "MMC%d: Data CRC error, bytes left %d\n",
+                                      host->id, host->bytesleft);
+                               end_transfer = 1;
+                       } else {
+                               printk(KERN_DEBUG "MMC%d: Data CRC error\n",
+                                      host->id);
+                       }
+               }
+
+               if (status & OMAP_MMC_STAT_CMD_TOUT) {
+                       // Command timeout
+                       if (host->cmd) {
+                               /* Timeouts are normal in case of MMC_SEND_STATUS */
+                               if (host->cmd->opcode != MMC_ALL_SEND_CID &&
+                                   host->cmd->opcode != MMC_SEND_OP_COND &&
+                                   !mmc_omap_cover_is_open(host))
+                                       printk(KERN_ERR "MMC%d: Command timeout, CMD%d\n",
+                                              host->id, host->cmd->opcode);
+                               host->cmd->error |= MMC_ERR_TIMEOUT;
+                               end_command = 1;
+                       }
+               }
+
+               if (status & OMAP_MMC_STAT_CMD_CRC) {
+                       // Command CRC error
+                       printk(KERN_ERR "MMC%d: Command CRC error\n", host->id);
+                       if (host->cmd) {
+                               host->cmd->error |= MMC_ERR_BADCRC;
+                               end_command = 1;
+                       }
+               }
+
+               if (status & OMAP_MMC_STAT_OCR_BUSY) {
+                       // OCR Busy
+                       if (host->cmd && host->cmd->opcode != MMC_SEND_OP_COND &&
+                           host->cmd->opcode != MMC_SET_RELATIVE_ADDR) {
+                               printk(KERN_DEBUG "MMC%d: OCR busy error, CMD%d\n",
+                                      host->id, host->cmd->opcode);
+                       }
+               }
+
+               if (status & OMAP_MMC_STAT_CARD_ERR) {
+                       // Card status error
+                       printk(KERN_DEBUG "MMC%d: Card status error (CMD%d)\n",
+                              host->id, host->cmd->opcode);
+                       if (host->cmd) {
+                               host->cmd->error |= MMC_ERR_FAILED;
+                               end_command = 1;
+                       }
+                       if (host->data) {
+                               host->data->error |= MMC_ERR_FAILED;
+                               end_transfer = 1;
+                       }
+               }
+
+               /*
+                * NOTE: On 1610 the END_OF_CMD may come too early when
+                *       starting a write 
+                */
+               if ((status & OMAP_MMC_STAT_END_OF_CMD) &&
+                   (!(status & OMAP_MMC_STAT_A_EMPTY))) {
+                       // End of command phase
+                       end_command = 1;
+               }
+       }
+
+       if (end_command) {
+               mmc_omap_cmd_done(host, host->cmd);
+       }
+       if (end_transfer) {
+               mmc_omap_xfer_done(host, host->data);
+       }
+
+       return IRQ_HANDLED;
+}
+
+static irqreturn_t mmc_omap_switch_irq(int irq, void *dev_id, struct pt_regs *regs)
+{
+       struct mmc_omap_host *host = (struct mmc_omap_host *) dev_id;
+
+       DBG("MMC%d cover is now %s\n", host->id,
+           omap_get_gpio_datain(host->switch_pin) ? "open" : "closed");
+       schedule_work(&host->switch_work);
+
+       return IRQ_HANDLED;
+}
+
+static void mmc_omap_switch_timer(unsigned long arg)
+{
+       struct mmc_omap_host *host = (struct mmc_omap_host *) arg;
+
+       schedule_work(&host->switch_work);
+}
+
+static void mmc_omap_switch_handler(void *data)
+{
+       struct mmc_omap_host *host = (struct mmc_omap_host *) data;
+       struct mmc_card *card;
+       static int complained = 0;
+       int cards = 0, cover_open;
+
+       if (host->switch_pin == -1)
+               return;
+       cover_open = mmc_omap_cover_is_open(host);
+       if (cover_open != host->switch_last_state) {
+               kobject_uevent(&host->dev->kobj, KOBJ_CHANGE, &dev_attr_cover_switch.attr);
+               host->switch_last_state = cover_open;
+       }
+       DBG("MMC cover switch handler started\n");
+       mmc_detect_change(host->mmc);
+       list_for_each_entry(card, &host->mmc->cards, node) {
+               if (mmc_card_present(card))
+                       cards++;
+       }
+       DBG("MMC%d: %d card(s) present\n", host->id, cards);
+       if (mmc_omap_cover_is_open(host)) {
+               if (!complained) {
+                       printk(KERN_INFO "MMC%d: cover is open\n", host->id);
+                       complained = 1;
+               }
+               if (mmc_omap_enable_poll)
+                       mod_timer(&host->switch_timer, jiffies +
+                               msecs_to_jiffies(OMAP_MMC_SWITCH_POLL_DELAY));
+       } else {
+               complained = 0;
+       }
+}
+
+static void mmc_omap_dma_cb(int lch, u16 ch_status, void *data)
+{
+       struct mmc_omap_host *host = (struct mmc_omap_host *) data;
+       int dma_ch;
+
+       /* FIXME: We ignore the possible errors for now. */
+       if (host->dma_ch < 0) {
+               printk(KERN_ERR "MMC%d: DMA callback while DMA not enabled?\n",
+                      host->id);
+               return;
+       }
+       dma_ch = host->dma_ch;
+       host->dma_ch = -1;
+
+       omap_free_dma(dma_ch);
+       complete(&host->dma_completion);
+}
+
+static int mmc_omap_start_dma_transfer(struct mmc_omap_host *host, struct mmc_request *req)
+{
+       int sync_dev, dma_ch, r;
+       const char *dev_name;
+       struct mmc_data *data = req->data;
+
+       /* If for some reason the DMA transfer is still active,
+        * we wait for it to complete. This shouldn't normally happen. */
+       if (host->dma_ch != -1)
+               wait_for_completion(&host->dma_completion);
+
+       init_completion(&host->dma_completion);
+
+       if (!(data->flags & MMC_DATA_WRITE)) {
+               if (host->id == 1) {
+                       sync_dev = OMAP_DMA_MMC_RX;
+                       dev_name = "MMC1 read";
+               } else {
+                       sync_dev = OMAP_DMA_MMC2_RX;
+                       dev_name = "MMC2 read";
+               }
+       } else {
+               if (host->id == 1) {
+                       sync_dev = OMAP_DMA_MMC_TX;
+                       dev_name = "MMC1 write";
+               } else {
+                       sync_dev = OMAP_DMA_MMC2_TX;
+                       dev_name = "MMC2 write";
+               }
+       }
+       r = omap_request_dma(sync_dev, dev_name, mmc_omap_dma_cb, host, &dma_ch);
+       if (r != 0) {
+               printk("MMC%d: omap_request_dma() failed with %d\n", host->id, r);
+               return r;
+       }
+       if (!(data->flags & MMC_DATA_WRITE)) {
+               /* 16 frames/block, 32 bytes/frame */
+               omap_set_dma_src_params(dma_ch, OMAP_DMA_PORT_TIPB,
+                                       OMAP_DMA_AMODE_CONSTANT,
+                                       virt_to_phys((void *) host->base) + OMAP_MMC_REG_DATA);
+               omap_set_dma_dest_params(dma_ch, OMAP_DMA_PORT_EMIFF,
+                                        OMAP_DMA_AMODE_POST_INC,
+                                        data->sg->dma_address);
+               OMAP_MMC_WRITE(host->base, BUF, 0x8f0f);
+       } else {
+               omap_set_dma_dest_params(dma_ch, OMAP_DMA_PORT_TIPB,
+                                        OMAP_DMA_AMODE_CONSTANT,
+                                        virt_to_phys((void *) host->base) + OMAP_MMC_REG_DATA);
+               omap_set_dma_src_params(dma_ch, OMAP_DMA_PORT_EMIFF,
+                                       OMAP_DMA_AMODE_POST_INC,
+                                       data->sg->dma_address);
+               OMAP_MMC_WRITE(host->base, BUF, 0x0f8f);
+       }
+       omap_set_dma_transfer_params(dma_ch, OMAP_DMA_DATA_TYPE_S16, 16,
+                                    16 * data->blocks, OMAP_DMA_SYNC_FRAME);
+       host->dma_ch = dma_ch;
+       omap_start_dma(dma_ch);
+
+        return 0;
+}
+
+static inline void set_cmd_timeout(struct mmc_omap_host *host, struct mmc_request *req)
+{
+       u16 reg;
+
+       reg = OMAP_MMC_READ(host->base, SDIO);
+       reg &= ~(1 << 5);
+       OMAP_MMC_WRITE(host->base, SDIO, reg);
+       /* Set maximum timeout */
+       OMAP_MMC_WRITE(host->base, CTO, 0xff);
+}
+
+static inline void set_data_timeout(struct mmc_omap_host *host, struct mmc_request *req)
+{
+       int timeout;
+       u16 reg;
+       
+       /* Convert ns to clock cycles by assuming 20MHz frequency
+        * 1 cycle at 20MHz = 500 ns
+        */
+       timeout = req->data->timeout_clks + req->data->timeout_ns / 500;
+
+       /* Some cards require more time to do at least the first read operation */
+       timeout = timeout << 4;
+
+       /* Check if we need to use timeout multiplier register */
+       reg = OMAP_MMC_READ(host->base, SDIO);
+       if (timeout > 0xffff) {
+               reg |= (1 << 5);
+               timeout /= 1024;
+       } else
+               reg &= ~(1 << 5);
+       OMAP_MMC_WRITE(host->base, SDIO, reg);
+       OMAP_MMC_WRITE(host->base, DTO, timeout);
+}
+
+static void mmc_omap_prepare_data(struct mmc_omap_host *host, struct mmc_request *req)
+{
+       struct mmc_data *data = req->data;
+
+       host->data = data;
+       if (data == NULL) {
+               host->datadir = OMAP_MMC_DATADIR_NONE;
+               OMAP_MMC_WRITE(host->base, BLEN, 0);
+               OMAP_MMC_WRITE(host->base, NBLK, 0);
+               OMAP_MMC_WRITE(host->base, BUF, 0);
+               set_cmd_timeout(host, req);
+               return;
+       }
+
+       DBG("MMC%d: Data xfer (%s %s), DTO %d cycles + %d ns, %d blocks of %d bytes\n",
+           host->id, (data->flags & MMC_DATA_STREAM) ? "stream" : "block",
+           (data->flags & MMC_DATA_WRITE) ? "write" : "read",
+           data->timeout_clks, data->timeout_ns, data->blocks,
+           1 << data->blksz_bits);
+
+       OMAP_MMC_WRITE(host->base, NBLK, data->blocks - 1);
+       OMAP_MMC_WRITE(host->base, BLEN, (1 << data->blksz_bits) - 1);
+       set_data_timeout(host, req);
+
+       host->datadir = (data->flags & MMC_DATA_WRITE) ?
+               OMAP_MMC_DATADIR_WRITE : OMAP_MMC_DATADIR_READ;
+       host->dma_len = 0;
+
+       host->dma_len = dma_map_sg(mmc_dev(host->mmc), data->sg, data->sg_len,
+                                  host->datadir);
+
+       /* No SG-DMA */
+       if (unlikely(host->dma_len > 1))
+               BUG();
+
+       if (host->use_dma) {
+               if (mmc_omap_start_dma_transfer(host, req) == 0) {
+                       host->buffer = NULL;
+                       host->bytesleft = 0;
+               }
+       } else {
+               /* Revert to CPU copy */
+               OMAP_MMC_WRITE(host->base, BUF, 0x1f1f);
+               host->buffer = page_address(data->sg->page) + data->sg->offset;
+               host->bytesleft = data->blocks * (1 << data->blksz_bits);
+               host->dma_ch = -1;
+       }
+}
+
+static inline int is_broken_card(struct mmc_card *card)
+{
+       int i;
+       struct mmc_cid *c = &card->cid;
+       static const struct broken_card_cid {
+               unsigned int manfid;
+               char prod_name[8];
+               unsigned char hwrev;
+               unsigned char fwrev;
+       } broken_cards[] = {
+               { 0x00150000, "\x30\x30\x30\x30\x30\x30\x15\x00", 0x06, 0x03 },
+       };
+
+       for (i = 0; i < sizeof(broken_cards)/sizeof(broken_cards[0]); i++) {
+               const struct broken_card_cid *b = broken_cards + i;
+
+               if (b->manfid != c->manfid)
+                       continue;
+               if (memcmp(b->prod_name, c->prod_name, sizeof(b->prod_name)) != 0)
+                       continue;
+               if (b->hwrev != c->hwrev || b->fwrev != c->fwrev)
+                       continue;
+               return 1;
+       }
+       return 0;
+}
+
+static void mmc_omap_request(struct mmc_host *mmc, struct mmc_request *req)
+{
+       struct mmc_omap_host *host = mmc_priv(mmc);
+
+       WARN_ON(host->mrq != NULL);
+
+       host->mrq = req;
+
+       /* Some cards (vendor left unnamed to protect the guilty) seem to
+        * require this delay after power-up. Otherwise we'll get mysterious
+        * data timeouts. */
+       if (req->cmd->opcode == MMC_SEND_CSD) {
+               struct mmc_card *card;
+               int broken_present = 0;
+
+               list_for_each_entry(card, &mmc->cards, node) {
+                       if (is_broken_card(card)) {
+                               broken_present = 1;
+                               break;
+                       }
+               }
+               if (broken_present) {
+                       static int complained = 0;
+
+                       if (!complained) {
+                               printk(KERN_WARNING "MMC%d: Broken card workaround enabled\n",
+                                      host->id);
+                               complained = 1;
+                       }
+                       if (in_interrupt()) {
+                               /* This is nasty */
+                                printk(KERN_ERR "Sleeping in IRQ handler, FIXME please!\n");
+                                dump_stack();
+                                mdelay(100);
+                       } else {
+                               set_current_state(TASK_UNINTERRUPTIBLE);
+                               schedule_timeout(100 * HZ / 1000);
+                       }
+               }
+       }
+
+       mmc_omap_prepare_data(host, req);
+       mmc_omap_start_command(host, req->cmd);
+}
+
+static void innovator_fpga_socket_power(int on)
+{
+#if defined(CONFIG_MACH_OMAP_INNOVATOR) && defined(CONFIG_ARCH_OMAP1510)
+
+       if (on) {
+               fpga_write(fpga_read(OMAP1510_FPGA_POWER) | (1 << 3),
+                    OMAP1510_FPGA_POWER);
+       } else {
+               fpga_write(fpga_read(OMAP1510_FPGA_POWER) & ~(1 << 3),
+                    OMAP1510_FPGA_POWER);
+       }
+#endif
+}
+
+/*
+ * Turn the socket power on/off. Innovator uses FPGA, most boards
+ * probably use GPIO.
+ */
+static void mmc_omap_power(struct mmc_omap_host *host, int on)
+{
+       if (on) {
+               if (machine_is_omap_innovator())
+                       innovator_fpga_socket_power(1);
+               else if (machine_is_omap_h2())
+                       tps65010_set_gpio_out_value(GPIO3, HIGH);
+               else if (machine_is_omap_h3())
+                       /* GPIO 4 of TPS65010 sends SD_EN signal */
+                       tps65010_set_gpio_out_value(GPIO4, HIGH);
+               else
+                       if (host->power_pin >= 0)
+                               omap_set_gpio_dataout(host->power_pin, 1);
+       } else {
+               if (machine_is_omap_innovator())
+                       innovator_fpga_socket_power(0);
+               else if (machine_is_omap_h2())
+                       tps65010_set_gpio_out_value(GPIO3, LOW);
+               else if (machine_is_omap_h3())
+                       tps65010_set_gpio_out_value(GPIO4, LOW);
+               else
+                       if (host->power_pin >= 0)
+                               omap_set_gpio_dataout(host->power_pin, 0);
+       }
+}
+
+static void mmc_omap_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
+{
+       struct mmc_omap_host *host = mmc_priv(mmc);
+       int dsor;
+       int realclock, i;
+
+       DBG("MMC%d: set_ios: clock %dHz busmode %d powermode %d Vdd %d.%02d\n",
+           host->id, ios->clock, ios->bus_mode, ios->power_mode,
+           ios->vdd / 100, ios->vdd % 100);
+
+       if (ios->power_mode == MMC_POWER_UP && ios->clock < 400000) {
+               /* Fix for broken stack */
+               realclock = 400000;
+       } else {
+               realclock = ios->clock;
+       }
+
+       if (ios->clock == 0) {
+               dsor = 0;
+       } else {
+               dsor = 48000000 / realclock;
+               if (dsor < 1)
+                       dsor = 1;
+
+               if (48000000 / dsor > realclock)
+                       dsor++;
+
+               if (dsor > 250)
+                       dsor = 250;
+       }
+
+       switch (ios->power_mode) {
+       case MMC_POWER_OFF:
+               mmc_omap_power(host, 0);
+               break;
+       case MMC_POWER_UP:
+       case MMC_POWER_ON:
+               mmc_omap_power(host, 1);
+               dsor |= 1<<11;
+               break;
+       }
+
+       host->bus_mode = ios->bus_mode;
+       clk_enable(host->clk);
+       /* On insanely high arm_per frequencies something sometimes
+        * goes somehow out of sync, and the POW bit is not being set,
+        * which results in the while loop below getting stuck.
+        * Writing to the CON register twice seems to do the trick. */
+       for (i = 0; i < 2; i++)
+               OMAP_MMC_WRITE(host->base, CON, dsor);
+       if (ios->power_mode == MMC_POWER_UP) {
+               /* Send clock cycles, poll completion */
+               OMAP_MMC_WRITE(host->base, IE, 0);
+               OMAP_MMC_WRITE(host->base, STAT, 0xffff);
+               OMAP_MMC_WRITE(host->base, CMD, 1<<7);
+               while (0 == (OMAP_MMC_READ(host->base, STAT) & 1));
+               OMAP_MMC_WRITE(host->base, STAT, 1);
+       }
+       clk_disable(host->clk);
+}
+
+static struct mmc_host_ops mmc_omap_ops = {
+       .request        = mmc_omap_request,
+       .set_ios        = mmc_omap_set_ios,
+};
+
+static int mmc_omap_probe(struct device *dev)
+{
+       struct platform_device *pdev = to_platform_device(dev);
+       struct mmc_host *mmc;
+       struct mmc_omap_host *host = NULL;
+       int ret = 0;
+
+       if (pdev->resource[0].flags != IORESOURCE_MEM
+           || pdev->resource[1].flags != IORESOURCE_IRQ) {
+               printk(KERN_ERR "mmc_omap_probe: invalid resource type\n");
+               return -ENODEV;
+       }
+
+       if (!request_mem_region(pdev->resource[0].start,
+                               pdev->resource[0].end - pdev->resource[0].start + 1, 
+                               pdev->name)) {
+               dev_dbg(&pdev->dev, "request_mem_region failed\n");
+               return -EBUSY;
+       }
+
+       mmc = mmc_alloc_host(sizeof(struct mmc_omap_host), dev);
+       if (!mmc) {
+               ret = -ENOMEM;
+               goto out;
+       }
+
+       host = mmc_priv(mmc);
+       host->mmc = mmc;
+
+       host->id = (pdev->resource[0].start == IO_ADDRESS(OMAP_MMC1_BASE)) ? 1 : 2;
+
+       host->clk = clk_get(dev, (host->id == 1) ? "mmc1_ck" : "mmc2_ck");
+       if (IS_ERR(host->clk)) {
+               ret = PTR_ERR(host->clk);
+               goto out;
+       }
+
+       if (host->id == 1) {
+               omap_cfg_reg(MMC_CMD);
+               omap_cfg_reg(MMC_CLK);
+               omap_cfg_reg(MMC_DAT0);
+               if (cpu_is_omap1710()) {
+                     omap_cfg_reg(M15_1710_MMC_CLKI);
+                     omap_cfg_reg(P19_1710_MMC_CMDDIR);
+                     omap_cfg_reg(P20_1710_MMC_DATDIR0);
+               }
+               if (host->sd_support) {
+                       omap_cfg_reg(MMC_DAT1);
+                       omap_cfg_reg(MMC_DAT2); 
+                       omap_cfg_reg(MMC_DAT3);
+               }
+               host->power_pin = mmc1_power_pin;
+               host->switch_pin = mmc1_switch_pin;
+       } else {
+               omap_cfg_reg(Y8_1610_MMC2_CMD);
+               omap_cfg_reg(Y10_1610_MMC2_CLK);
+               omap_cfg_reg(R18_1610_MMC2_CLKIN);
+               omap_cfg_reg(W8_1610_MMC2_DAT0);
+               if (host->sd_support) {
+                       omap_cfg_reg(V8_1610_MMC2_DAT1);
+                       omap_cfg_reg(W15_1610_MMC2_DAT2);
+                       omap_cfg_reg(R10_1610_MMC2_DAT3);
+               }
+
+               /* These are needed for the level shifter */
+               omap_cfg_reg(V9_1610_MMC2_CMDDIR);
+               omap_cfg_reg(V5_1610_MMC2_DATDIR0);
+               omap_cfg_reg(W19_1610_MMC2_DATDIR1);
+
+               host->power_pin = mmc2_power_pin;
+               host->switch_pin = mmc2_switch_pin;
+
+               /* Feedback clock must be set on OMAP-1710 MMC2 */
+               if (cpu_is_omap1710())
+                       omap_writel(omap_readl(MOD_CONF_CTRL_1) | (1 << 24), 
+                                    MOD_CONF_CTRL_1);
+       }
+
+       host->use_dma = 1;
+       host->dma_ch = -1;
+
+       host->irq = pdev->resource[1].start;
+       host->base = (u32)pdev->resource[0].start;
+
+       mmc->ops = &mmc_omap_ops;
+       mmc->f_min = 400000;
+       mmc->f_max = 24000000;
+       mmc->ocr_avail = MMC_VDD_33_34;
+
+       /* No SG-DMA */
+       mmc->max_phys_segs = 1;
+       mmc->max_seg_size = PAGE_SIZE;
+
+       if (host->power_pin >= 0) {
+               if ((ret = omap_request_gpio(host->power_pin)) != 0) {
+                       printk(KERN_ERR "MMC%d: Unable to get GPIO pin for MMC power\n",
+                              host->id);
+                       goto out;
+               }
+               omap_set_gpio_direction(host->power_pin, 0);
+       }
+
+       ret = request_irq(host->irq, mmc_omap_irq, 0, DRIVER_NAME, host);
+       if (ret)
+               goto out;
+
+       host->dev = dev;
+       dev_set_drvdata(dev, host);
+
+       mmc_add_host(mmc);
+
+       if (host->switch_pin >= 0) {
+               INIT_WORK(&host->switch_work, mmc_omap_switch_handler, host);
+               init_timer(&host->switch_timer);
+               host->switch_timer.function = mmc_omap_switch_timer;
+               host->switch_timer.data = (unsigned long) host;
+               if (omap_request_gpio(host->switch_pin) != 0) {
+                       printk(KERN_WARNING "MMC%d: Unable to get GPIO pin for MMC cover switch\n",
+                              host->id);
+                       host->switch_pin = -1;
+                       goto no_switch;
+               }
+
+               omap_set_gpio_direction(host->switch_pin, 1);
+               omap_set_gpio_edge_ctrl(host->switch_pin, OMAP_GPIO_RISING_EDGE);
+               ret = request_irq(OMAP_GPIO_IRQ(host->switch_pin),
+                                 mmc_omap_switch_irq, 0, DRIVER_NAME, host);
+               if (ret) {
+                       printk(KERN_WARNING "MMC%d: Unable to get IRQ for MMC cover switch\n",
+                              host->id);
+                       omap_free_gpio(host->switch_pin);
+                       host->switch_pin = -1;
+                       goto no_switch;
+               }
+               ret = device_create_file(dev, &dev_attr_cover_switch);
+               if (ret == 0) {
+                       ret = device_create_file(dev, &dev_attr_enable_poll);
+                       if (ret != 0)
+                               device_remove_file(dev, &dev_attr_cover_switch);
+               }
+               if (ret) {
+                       printk(KERN_WARNING "MMC%d: Unable to create sysfs attributes\n", 
+                              host->id);
+                       free_irq(OMAP_GPIO_IRQ(host->switch_pin), host);
+                       omap_free_gpio(host->switch_pin);
+                       host->switch_pin = -1;
+                       goto no_switch;
+               }
+               if (mmc_omap_enable_poll && mmc_omap_cover_is_open(host))
+                       schedule_work(&host->switch_work);
+       }
+no_switch:
+       return 0;
+out:
+       /* FIXME: Free other resources too. */
+       if (host) {
+               if (host->clk && !IS_ERR(host->clk))
+                       clk_put(host->clk);
+               mmc_free_host(host->mmc);
+       }
+       return ret;
+}
+
+static int mmc_omap_remove(struct device *dev)
+{
+       struct platform_device *pdev = to_platform_device(dev);
+       struct mmc_omap_host *host = dev_get_drvdata(dev);
+
+       dev_set_drvdata(dev, NULL);
+
+       if (host) {
+               mmc_remove_host(host->mmc);
+               free_irq(host->irq, host);
+               mmc_omap_power(host, 0);
+
+               if (host->power_pin >= 0)
+                       omap_free_gpio(host->power_pin);
+               if (host->switch_pin >= 0) {
+                       device_remove_file(dev, &dev_attr_enable_poll);
+                       device_remove_file(dev, &dev_attr_cover_switch);
+                       free_irq(OMAP_GPIO_IRQ(host->switch_pin), host);
+                       omap_free_gpio(host->switch_pin);
+                       host->switch_pin = -1;
+                       del_timer_sync(&host->switch_timer);
+                       flush_scheduled_work();
+               }
+               if (host->clk && !IS_ERR(host->clk))
+                       clk_put(host->clk);
+               mmc_free_host(host->mmc);
+       }
+
+       release_mem_region(pdev->resource[0].start, 
+                          pdev->resource[0].end - pdev->resource[0].start + 1);
+
+       return 0;
+}
+
+#ifdef CONFIG_PM
+static int mmc_omap_suspend(struct device *dev, u32 state, u32 level)
+{
+       int ret = 0;
+       struct mmc_omap_host *host = dev_get_drvdata(dev);
+
+       if (host && host->suspended)
+               return 0;
+
+       if (!irqs_disabled())
+               return -EAGAIN;
+
+       if (host) {
+               ret = mmc_suspend_host(host->mmc, state);
+               if (ret == 0)
+                       host->suspended = 1;
+       }
+       return ret;
+}
+
+static int mmc_omap_resume(struct device *dev, u32 level)
+{
+       int ret = 0;
+       struct mmc_omap_host *host = dev_get_drvdata(dev);
+
+       if (host && !host->suspended)
+               return 0;
+
+       if (host) {
+               ret = mmc_resume_host(host->mmc);
+               if (ret == 0)
+                       host->suspended = 0;
+       }
+
+       return ret;
+}
+#else
+#define mmc_omap_suspend       NULL
+#define mmc_omap_resume                NULL
+#endif
+
+static void mmc_release(struct device *dev)
+{
+       /* Nothing to release? */
+}
+
+static struct resource mmc1_resources[] = {
+       {
+               .start          = IO_ADDRESS(OMAP_MMC1_BASE),
+               .end            = IO_ADDRESS(OMAP_MMC1_BASE) + 0x7f,
+               .flags          = IORESOURCE_MEM,
+       },
+       {
+               .start          = INT_MMC,
+               .flags          = IORESOURCE_IRQ,
+       },
+};
+
+static struct resource mmc2_resources[] = {
+       {
+               .start          = IO_ADDRESS(OMAP_MMC2_BASE),
+               .end            = IO_ADDRESS(OMAP_MMC2_BASE) + 0x7f,
+               .flags          = IORESOURCE_MEM,
+       },
+       {
+               .start          = INT_1610_MMC2,
+               .flags          = IORESOURCE_IRQ,
+       },
+};
+
+
+static u64 mmc_dmamask = 0xffffffff;
+
+static struct platform_device mmc_omap_device1 = {
+       .name           = "mmci-omap",
+       .id             = 1,
+       .dev = {
+               .release        = mmc_release,
+               .dma_mask       = &mmc_dmamask,
+       },
+       .num_resources  = ARRAY_SIZE(&mmc1_resources),
+       .resource       = mmc1_resources,
+};
+
+static struct platform_device mmc_omap_device2 = {
+       .name           = "mmci-omap",
+       .id             = 2,
+       .dev = {
+               .release        = mmc_release,
+               .dma_mask       = &mmc_dmamask,
+       },
+       .num_resources  = ARRAY_SIZE(&mmc2_resources),
+       .resource       = mmc2_resources,
+};
+
+static struct device_driver mmc_omap_driver = {
+       .name           = "mmci-omap",
+       .bus            = &platform_bus_type,
+       .probe          = mmc_omap_probe,
+       .remove         = mmc_omap_remove,
+       .suspend        = mmc_omap_suspend,
+       .resume         = mmc_omap_resume,
+};
+
+static int enable_blocks = 0;
+
+static int __init mmc_omap_init(void)
+{
+       int ret;
+       const struct omap_mmc_config *minfo;
+
+       minfo = omap_get_config(OMAP_TAG_MMC, struct omap_mmc_config);
+       if (minfo != NULL) {
+               enable_blocks = minfo->mmc_blocks;
+               if (enable_blocks & 1) {
+                       mmc1_power_pin = minfo->mmc1_power_pin;
+                       mmc1_switch_pin = minfo->mmc1_switch_pin;
+               }
+               if (enable_blocks & 2) {
+                       mmc2_power_pin = minfo->mmc2_power_pin;
+                       mmc2_switch_pin = minfo->mmc2_switch_pin;
+               }
+       } else {
+#if defined(CONFIG_ARCH_OMAP1510) || defined(CONFIG_MMC_OMAP16XX_BLOCK1)
+               enable_blocks |= 1;
+#endif
+#if defined(CONFIG_MMC_OMAP16XX_BLOCK2)
+               enable_blocks |= 2;
+#endif
+       }
+       if (enable_blocks == 0) {
+               printk(KERN_INFO "OMAP MMC driver not loaded\n");
+               return -ENODEV;
+       }
+
+       if (enable_blocks & 1) {
+               ret = platform_device_register(&mmc_omap_device1);
+               if (ret != 0)
+                       return -ENODEV;
+       }
+
+       if (enable_blocks & 2) {
+               ret = platform_device_register(&mmc_omap_device2);
+               if (ret != 0)
+                       goto free1;
+       }
+
+       ret = driver_register(&mmc_omap_driver);
+       if (ret == 0)
+               return 0;
+
+       if (enable_blocks & 2)
+               platform_device_unregister(&mmc_omap_device2);
+
+free1:
+       if (enable_blocks & 1)
+               platform_device_unregister(&mmc_omap_device1);
+
+       return -ENODEV;
+}
+
+static void __exit mmc_omap_exit(void)
+{
+       driver_unregister(&mmc_omap_driver);
+
+       if (enable_blocks & 2)
+               platform_device_unregister(&mmc_omap_device2);
+
+       if (enable_blocks & 1)
+               platform_device_unregister(&mmc_omap_device1);
+}
+
+module_init(mmc_omap_init);
+module_exit(mmc_omap_exit);
+
+MODULE_DESCRIPTION("OMAP Multimedia Card driver");
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("Juha Yrjölä");
diff --git a/drivers/mmc/omap.h b/drivers/mmc/omap.h
new file mode 100644 (file)
index 0000000..9ab89b7
--- /dev/null
@@ -0,0 +1,58 @@
+#ifndef        DRIVERS_MEDIA_MMC_OMAP_H
+#define        DRIVERS_MEDIA_MMC_OMAP_H
+
+#define        OMAP_MMC1_BASE  0xfffb7800
+#define        OMAP_MMC2_BASE  0xfffb7c00
+
+#define        OMAP_MMC_REG_CMD        0x00
+#define        OMAP_MMC_REG_ARGL       0x04
+#define        OMAP_MMC_REG_ARGH       0x08
+#define        OMAP_MMC_REG_CON        0x0c
+#define        OMAP_MMC_REG_STAT       0x10
+#define        OMAP_MMC_REG_IE         0x14
+#define        OMAP_MMC_REG_CTO        0x18
+#define        OMAP_MMC_REG_DTO        0x1c
+#define        OMAP_MMC_REG_DATA       0x20
+#define        OMAP_MMC_REG_BLEN       0x24
+#define        OMAP_MMC_REG_NBLK       0x28
+#define        OMAP_MMC_REG_BUF        0x2c
+#define OMAP_MMC_REG_SDIO      0x34
+#define        OMAP_MMC_REG_REV        0x3c
+#define        OMAP_MMC_REG_RSP0       0x40
+#define        OMAP_MMC_REG_RSP1       0x44
+#define        OMAP_MMC_REG_RSP2       0x48
+#define        OMAP_MMC_REG_RSP3       0x4c
+#define        OMAP_MMC_REG_RSP4       0x50
+#define        OMAP_MMC_REG_RSP5       0x54
+#define        OMAP_MMC_REG_RSP6       0x58
+#define        OMAP_MMC_REG_RSP7       0x5c
+#define        OMAP_MMC_REG_IOSR       0x60
+#define        OMAP_MMC_REG_SYSC       0x64
+#define        OMAP_MMC_REG_SYSS       0x68
+
+#define        OMAP_MMC_STAT_CARD_ERR          (1 << 14)
+#define        OMAP_MMC_STAT_CARD_IRQ          (1 << 13)
+#define        OMAP_MMC_STAT_OCR_BUSY          (1 << 12)
+#define        OMAP_MMC_STAT_A_EMPTY           (1 << 11)
+#define        OMAP_MMC_STAT_A_FULL            (1 << 10)
+#define        OMAP_MMC_STAT_CMD_CRC           (1 <<  8)
+#define        OMAP_MMC_STAT_CMD_TOUT          (1 <<  7)
+#define        OMAP_MMC_STAT_DATA_CRC          (1 <<  6)
+#define        OMAP_MMC_STAT_DATA_TOUT         (1 <<  5)
+#define        OMAP_MMC_STAT_END_BUSY          (1 <<  4)
+#define        OMAP_MMC_STAT_END_OF_DATA       (1 <<  3)
+#define        OMAP_MMC_STAT_CARD_BUSY         (1 <<  2)
+#define        OMAP_MMC_STAT_END_OF_CMD        (1 <<  0)
+
+#define OMAP_MMC_READ(base, reg)       __raw_readw((base) + OMAP_MMC_REG_##reg)
+#define OMAP_MMC_WRITE(base, reg, val) __raw_writew((val), (base) + OMAP_MMC_REG_##reg)
+
+/*
+ * Command types
+ */
+#define OMAP_MMC_CMDTYPE_BC    0
+#define OMAP_MMC_CMDTYPE_BCR   1
+#define OMAP_MMC_CMDTYPE_AC    2
+#define OMAP_MMC_CMDTYPE_ADTC  3
+
+#endif