a2b760a601
The argument "override_active_level" made it possible to enforce a specific polarity on the write-protect GPIO line. All callers in the kernel pass "false" to this call after I have converted all drivers to use GPIO machine descriptors, so remove the argument and clean out this. This kind of polarity inversion should be handled by the GPIO descriptor inside the GPIO library if needed. This rids us of one instance of the kludgy calls into the gpiod_get_raw_value() API. Signed-off-by: Linus Walleij <linus.walleij@linaro.org> Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
1851 lines
45 KiB
C
1851 lines
45 KiB
C
/*
|
|
* linux/drivers/mmc/s3cmci.h - Samsung S3C MCI driver
|
|
*
|
|
* Copyright (C) 2004-2006 maintech GmbH, Thomas Kleffel <tk@maintech.de>
|
|
*
|
|
* Current driver maintained by Ben Dooks and Simtec Electronics
|
|
* Copyright (C) 2008 Simtec Electronics <ben-linux@fluff.org>
|
|
*
|
|
* 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/module.h>
|
|
#include <linux/dmaengine.h>
|
|
#include <linux/dma-mapping.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/mmc/host.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/cpufreq.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/seq_file.h>
|
|
#include <linux/gpio.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/irq.h>
|
|
#include <linux/io.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_device.h>
|
|
#include <linux/mmc/slot-gpio.h>
|
|
|
|
#include <plat/gpio-cfg.h>
|
|
#include <mach/dma.h>
|
|
#include <mach/gpio-samsung.h>
|
|
|
|
#include <linux/platform_data/mmc-s3cmci.h>
|
|
|
|
#include "s3cmci.h"
|
|
|
|
#define DRIVER_NAME "s3c-mci"
|
|
|
|
#define S3C2410_SDICON (0x00)
|
|
#define S3C2410_SDIPRE (0x04)
|
|
#define S3C2410_SDICMDARG (0x08)
|
|
#define S3C2410_SDICMDCON (0x0C)
|
|
#define S3C2410_SDICMDSTAT (0x10)
|
|
#define S3C2410_SDIRSP0 (0x14)
|
|
#define S3C2410_SDIRSP1 (0x18)
|
|
#define S3C2410_SDIRSP2 (0x1C)
|
|
#define S3C2410_SDIRSP3 (0x20)
|
|
#define S3C2410_SDITIMER (0x24)
|
|
#define S3C2410_SDIBSIZE (0x28)
|
|
#define S3C2410_SDIDCON (0x2C)
|
|
#define S3C2410_SDIDCNT (0x30)
|
|
#define S3C2410_SDIDSTA (0x34)
|
|
#define S3C2410_SDIFSTA (0x38)
|
|
|
|
#define S3C2410_SDIDATA (0x3C)
|
|
#define S3C2410_SDIIMSK (0x40)
|
|
|
|
#define S3C2440_SDIDATA (0x40)
|
|
#define S3C2440_SDIIMSK (0x3C)
|
|
|
|
#define S3C2440_SDICON_SDRESET (1 << 8)
|
|
#define S3C2410_SDICON_SDIOIRQ (1 << 3)
|
|
#define S3C2410_SDICON_FIFORESET (1 << 1)
|
|
#define S3C2410_SDICON_CLOCKTYPE (1 << 0)
|
|
|
|
#define S3C2410_SDICMDCON_LONGRSP (1 << 10)
|
|
#define S3C2410_SDICMDCON_WAITRSP (1 << 9)
|
|
#define S3C2410_SDICMDCON_CMDSTART (1 << 8)
|
|
#define S3C2410_SDICMDCON_SENDERHOST (1 << 6)
|
|
#define S3C2410_SDICMDCON_INDEX (0x3f)
|
|
|
|
#define S3C2410_SDICMDSTAT_CRCFAIL (1 << 12)
|
|
#define S3C2410_SDICMDSTAT_CMDSENT (1 << 11)
|
|
#define S3C2410_SDICMDSTAT_CMDTIMEOUT (1 << 10)
|
|
#define S3C2410_SDICMDSTAT_RSPFIN (1 << 9)
|
|
|
|
#define S3C2440_SDIDCON_DS_WORD (2 << 22)
|
|
#define S3C2410_SDIDCON_TXAFTERRESP (1 << 20)
|
|
#define S3C2410_SDIDCON_RXAFTERCMD (1 << 19)
|
|
#define S3C2410_SDIDCON_BLOCKMODE (1 << 17)
|
|
#define S3C2410_SDIDCON_WIDEBUS (1 << 16)
|
|
#define S3C2410_SDIDCON_DMAEN (1 << 15)
|
|
#define S3C2410_SDIDCON_STOP (1 << 14)
|
|
#define S3C2440_SDIDCON_DATSTART (1 << 14)
|
|
|
|
#define S3C2410_SDIDCON_XFER_RXSTART (2 << 12)
|
|
#define S3C2410_SDIDCON_XFER_TXSTART (3 << 12)
|
|
|
|
#define S3C2410_SDIDCON_BLKNUM_MASK (0xFFF)
|
|
|
|
#define S3C2410_SDIDSTA_SDIOIRQDETECT (1 << 9)
|
|
#define S3C2410_SDIDSTA_FIFOFAIL (1 << 8)
|
|
#define S3C2410_SDIDSTA_CRCFAIL (1 << 7)
|
|
#define S3C2410_SDIDSTA_RXCRCFAIL (1 << 6)
|
|
#define S3C2410_SDIDSTA_DATATIMEOUT (1 << 5)
|
|
#define S3C2410_SDIDSTA_XFERFINISH (1 << 4)
|
|
#define S3C2410_SDIDSTA_TXDATAON (1 << 1)
|
|
#define S3C2410_SDIDSTA_RXDATAON (1 << 0)
|
|
|
|
#define S3C2440_SDIFSTA_FIFORESET (1 << 16)
|
|
#define S3C2440_SDIFSTA_FIFOFAIL (3 << 14)
|
|
#define S3C2410_SDIFSTA_TFDET (1 << 13)
|
|
#define S3C2410_SDIFSTA_RFDET (1 << 12)
|
|
#define S3C2410_SDIFSTA_COUNTMASK (0x7f)
|
|
|
|
#define S3C2410_SDIIMSK_RESPONSECRC (1 << 17)
|
|
#define S3C2410_SDIIMSK_CMDSENT (1 << 16)
|
|
#define S3C2410_SDIIMSK_CMDTIMEOUT (1 << 15)
|
|
#define S3C2410_SDIIMSK_RESPONSEND (1 << 14)
|
|
#define S3C2410_SDIIMSK_SDIOIRQ (1 << 12)
|
|
#define S3C2410_SDIIMSK_FIFOFAIL (1 << 11)
|
|
#define S3C2410_SDIIMSK_CRCSTATUS (1 << 10)
|
|
#define S3C2410_SDIIMSK_DATACRC (1 << 9)
|
|
#define S3C2410_SDIIMSK_DATATIMEOUT (1 << 8)
|
|
#define S3C2410_SDIIMSK_DATAFINISH (1 << 7)
|
|
#define S3C2410_SDIIMSK_TXFIFOHALF (1 << 4)
|
|
#define S3C2410_SDIIMSK_RXFIFOLAST (1 << 2)
|
|
#define S3C2410_SDIIMSK_RXFIFOHALF (1 << 0)
|
|
|
|
enum dbg_channels {
|
|
dbg_err = (1 << 0),
|
|
dbg_debug = (1 << 1),
|
|
dbg_info = (1 << 2),
|
|
dbg_irq = (1 << 3),
|
|
dbg_sg = (1 << 4),
|
|
dbg_dma = (1 << 5),
|
|
dbg_pio = (1 << 6),
|
|
dbg_fail = (1 << 7),
|
|
dbg_conf = (1 << 8),
|
|
};
|
|
|
|
static const int dbgmap_err = dbg_fail;
|
|
static const int dbgmap_info = dbg_info | dbg_conf;
|
|
static const int dbgmap_debug = dbg_err | dbg_debug;
|
|
|
|
#define dbg(host, channels, args...) \
|
|
do { \
|
|
if (dbgmap_err & channels) \
|
|
dev_err(&host->pdev->dev, args); \
|
|
else if (dbgmap_info & channels) \
|
|
dev_info(&host->pdev->dev, args); \
|
|
else if (dbgmap_debug & channels) \
|
|
dev_dbg(&host->pdev->dev, args); \
|
|
} while (0)
|
|
|
|
static void finalize_request(struct s3cmci_host *host);
|
|
static void s3cmci_send_request(struct mmc_host *mmc);
|
|
static void s3cmci_reset(struct s3cmci_host *host);
|
|
|
|
#ifdef CONFIG_MMC_DEBUG
|
|
|
|
static void dbg_dumpregs(struct s3cmci_host *host, char *prefix)
|
|
{
|
|
u32 con, pre, cmdarg, cmdcon, cmdsta, r0, r1, r2, r3, timer, bsize;
|
|
u32 datcon, datcnt, datsta, fsta, imask;
|
|
|
|
con = readl(host->base + S3C2410_SDICON);
|
|
pre = readl(host->base + S3C2410_SDIPRE);
|
|
cmdarg = readl(host->base + S3C2410_SDICMDARG);
|
|
cmdcon = readl(host->base + S3C2410_SDICMDCON);
|
|
cmdsta = readl(host->base + S3C2410_SDICMDSTAT);
|
|
r0 = readl(host->base + S3C2410_SDIRSP0);
|
|
r1 = readl(host->base + S3C2410_SDIRSP1);
|
|
r2 = readl(host->base + S3C2410_SDIRSP2);
|
|
r3 = readl(host->base + S3C2410_SDIRSP3);
|
|
timer = readl(host->base + S3C2410_SDITIMER);
|
|
bsize = readl(host->base + S3C2410_SDIBSIZE);
|
|
datcon = readl(host->base + S3C2410_SDIDCON);
|
|
datcnt = readl(host->base + S3C2410_SDIDCNT);
|
|
datsta = readl(host->base + S3C2410_SDIDSTA);
|
|
fsta = readl(host->base + S3C2410_SDIFSTA);
|
|
imask = readl(host->base + host->sdiimsk);
|
|
|
|
dbg(host, dbg_debug, "%s CON:[%08x] PRE:[%08x] TMR:[%08x]\n",
|
|
prefix, con, pre, timer);
|
|
|
|
dbg(host, dbg_debug, "%s CCON:[%08x] CARG:[%08x] CSTA:[%08x]\n",
|
|
prefix, cmdcon, cmdarg, cmdsta);
|
|
|
|
dbg(host, dbg_debug, "%s DCON:[%08x] FSTA:[%08x]"
|
|
" DSTA:[%08x] DCNT:[%08x]\n",
|
|
prefix, datcon, fsta, datsta, datcnt);
|
|
|
|
dbg(host, dbg_debug, "%s R0:[%08x] R1:[%08x]"
|
|
" R2:[%08x] R3:[%08x]\n",
|
|
prefix, r0, r1, r2, r3);
|
|
}
|
|
|
|
static void prepare_dbgmsg(struct s3cmci_host *host, struct mmc_command *cmd,
|
|
int stop)
|
|
{
|
|
snprintf(host->dbgmsg_cmd, 300,
|
|
"#%u%s op:%i arg:0x%08x flags:0x08%x retries:%u",
|
|
host->ccnt, (stop ? " (STOP)" : ""),
|
|
cmd->opcode, cmd->arg, cmd->flags, cmd->retries);
|
|
|
|
if (cmd->data) {
|
|
snprintf(host->dbgmsg_dat, 300,
|
|
"#%u bsize:%u blocks:%u bytes:%u",
|
|
host->dcnt, cmd->data->blksz,
|
|
cmd->data->blocks,
|
|
cmd->data->blocks * cmd->data->blksz);
|
|
} else {
|
|
host->dbgmsg_dat[0] = '\0';
|
|
}
|
|
}
|
|
|
|
static void dbg_dumpcmd(struct s3cmci_host *host, struct mmc_command *cmd,
|
|
int fail)
|
|
{
|
|
unsigned int dbglvl = fail ? dbg_fail : dbg_debug;
|
|
|
|
if (!cmd)
|
|
return;
|
|
|
|
if (cmd->error == 0) {
|
|
dbg(host, dbglvl, "CMD[OK] %s R0:0x%08x\n",
|
|
host->dbgmsg_cmd, cmd->resp[0]);
|
|
} else {
|
|
dbg(host, dbglvl, "CMD[ERR %i] %s Status:%s\n",
|
|
cmd->error, host->dbgmsg_cmd, host->status);
|
|
}
|
|
|
|
if (!cmd->data)
|
|
return;
|
|
|
|
if (cmd->data->error == 0) {
|
|
dbg(host, dbglvl, "DAT[OK] %s\n", host->dbgmsg_dat);
|
|
} else {
|
|
dbg(host, dbglvl, "DAT[ERR %i] %s DCNT:0x%08x\n",
|
|
cmd->data->error, host->dbgmsg_dat,
|
|
readl(host->base + S3C2410_SDIDCNT));
|
|
}
|
|
}
|
|
#else
|
|
static void dbg_dumpcmd(struct s3cmci_host *host,
|
|
struct mmc_command *cmd, int fail) { }
|
|
|
|
static void prepare_dbgmsg(struct s3cmci_host *host, struct mmc_command *cmd,
|
|
int stop) { }
|
|
|
|
static void dbg_dumpregs(struct s3cmci_host *host, char *prefix) { }
|
|
|
|
#endif /* CONFIG_MMC_DEBUG */
|
|
|
|
/**
|
|
* s3cmci_host_usedma - return whether the host is using dma or pio
|
|
* @host: The host state
|
|
*
|
|
* Return true if the host is using DMA to transfer data, else false
|
|
* to use PIO mode. Will return static data depending on the driver
|
|
* configuration.
|
|
*/
|
|
static inline bool s3cmci_host_usedma(struct s3cmci_host *host)
|
|
{
|
|
#ifdef CONFIG_MMC_S3C_PIO
|
|
return false;
|
|
#else /* CONFIG_MMC_S3C_DMA */
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
static inline u32 enable_imask(struct s3cmci_host *host, u32 imask)
|
|
{
|
|
u32 newmask;
|
|
|
|
newmask = readl(host->base + host->sdiimsk);
|
|
newmask |= imask;
|
|
|
|
writel(newmask, host->base + host->sdiimsk);
|
|
|
|
return newmask;
|
|
}
|
|
|
|
static inline u32 disable_imask(struct s3cmci_host *host, u32 imask)
|
|
{
|
|
u32 newmask;
|
|
|
|
newmask = readl(host->base + host->sdiimsk);
|
|
newmask &= ~imask;
|
|
|
|
writel(newmask, host->base + host->sdiimsk);
|
|
|
|
return newmask;
|
|
}
|
|
|
|
static inline void clear_imask(struct s3cmci_host *host)
|
|
{
|
|
u32 mask = readl(host->base + host->sdiimsk);
|
|
|
|
/* preserve the SDIO IRQ mask state */
|
|
mask &= S3C2410_SDIIMSK_SDIOIRQ;
|
|
writel(mask, host->base + host->sdiimsk);
|
|
}
|
|
|
|
/**
|
|
* s3cmci_check_sdio_irq - test whether the SDIO IRQ is being signalled
|
|
* @host: The host to check.
|
|
*
|
|
* Test to see if the SDIO interrupt is being signalled in case the
|
|
* controller has failed to re-detect a card interrupt. Read GPE8 and
|
|
* see if it is low and if so, signal a SDIO interrupt.
|
|
*
|
|
* This is currently called if a request is finished (we assume that the
|
|
* bus is now idle) and when the SDIO IRQ is enabled in case the IRQ is
|
|
* already being indicated.
|
|
*/
|
|
static void s3cmci_check_sdio_irq(struct s3cmci_host *host)
|
|
{
|
|
if (host->sdio_irqen) {
|
|
if (gpio_get_value(S3C2410_GPE(8)) == 0) {
|
|
pr_debug("%s: signalling irq\n", __func__);
|
|
mmc_signal_sdio_irq(host->mmc);
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline int get_data_buffer(struct s3cmci_host *host,
|
|
u32 *bytes, u32 **pointer)
|
|
{
|
|
struct scatterlist *sg;
|
|
|
|
if (host->pio_active == XFER_NONE)
|
|
return -EINVAL;
|
|
|
|
if ((!host->mrq) || (!host->mrq->data))
|
|
return -EINVAL;
|
|
|
|
if (host->pio_sgptr >= host->mrq->data->sg_len) {
|
|
dbg(host, dbg_debug, "no more buffers (%i/%i)\n",
|
|
host->pio_sgptr, host->mrq->data->sg_len);
|
|
return -EBUSY;
|
|
}
|
|
sg = &host->mrq->data->sg[host->pio_sgptr];
|
|
|
|
*bytes = sg->length;
|
|
*pointer = sg_virt(sg);
|
|
|
|
host->pio_sgptr++;
|
|
|
|
dbg(host, dbg_sg, "new buffer (%i/%i)\n",
|
|
host->pio_sgptr, host->mrq->data->sg_len);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline u32 fifo_count(struct s3cmci_host *host)
|
|
{
|
|
u32 fifostat = readl(host->base + S3C2410_SDIFSTA);
|
|
|
|
fifostat &= S3C2410_SDIFSTA_COUNTMASK;
|
|
return fifostat;
|
|
}
|
|
|
|
static inline u32 fifo_free(struct s3cmci_host *host)
|
|
{
|
|
u32 fifostat = readl(host->base + S3C2410_SDIFSTA);
|
|
|
|
fifostat &= S3C2410_SDIFSTA_COUNTMASK;
|
|
return 63 - fifostat;
|
|
}
|
|
|
|
/**
|
|
* s3cmci_enable_irq - enable IRQ, after having disabled it.
|
|
* @host: The device state.
|
|
* @more: True if more IRQs are expected from transfer.
|
|
*
|
|
* Enable the main IRQ if needed after it has been disabled.
|
|
*
|
|
* The IRQ can be one of the following states:
|
|
* - disabled during IDLE
|
|
* - disabled whilst processing data
|
|
* - enabled during transfer
|
|
* - enabled whilst awaiting SDIO interrupt detection
|
|
*/
|
|
static void s3cmci_enable_irq(struct s3cmci_host *host, bool more)
|
|
{
|
|
unsigned long flags;
|
|
bool enable = false;
|
|
|
|
local_irq_save(flags);
|
|
|
|
host->irq_enabled = more;
|
|
host->irq_disabled = false;
|
|
|
|
enable = more | host->sdio_irqen;
|
|
|
|
if (host->irq_state != enable) {
|
|
host->irq_state = enable;
|
|
|
|
if (enable)
|
|
enable_irq(host->irq);
|
|
else
|
|
disable_irq(host->irq);
|
|
}
|
|
|
|
local_irq_restore(flags);
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static void s3cmci_disable_irq(struct s3cmci_host *host, bool transfer)
|
|
{
|
|
unsigned long flags;
|
|
|
|
local_irq_save(flags);
|
|
|
|
/* pr_debug("%s: transfer %d\n", __func__, transfer); */
|
|
|
|
host->irq_disabled = transfer;
|
|
|
|
if (transfer && host->irq_state) {
|
|
host->irq_state = false;
|
|
disable_irq(host->irq);
|
|
}
|
|
|
|
local_irq_restore(flags);
|
|
}
|
|
|
|
static void do_pio_read(struct s3cmci_host *host)
|
|
{
|
|
int res;
|
|
u32 fifo;
|
|
u32 *ptr;
|
|
u32 fifo_words;
|
|
void __iomem *from_ptr;
|
|
|
|
/* write real prescaler to host, it might be set slow to fix */
|
|
writel(host->prescaler, host->base + S3C2410_SDIPRE);
|
|
|
|
from_ptr = host->base + host->sdidata;
|
|
|
|
while ((fifo = fifo_count(host))) {
|
|
if (!host->pio_bytes) {
|
|
res = get_data_buffer(host, &host->pio_bytes,
|
|
&host->pio_ptr);
|
|
if (res) {
|
|
host->pio_active = XFER_NONE;
|
|
host->complete_what = COMPLETION_FINALIZE;
|
|
|
|
dbg(host, dbg_pio, "pio_read(): "
|
|
"complete (no more data).\n");
|
|
return;
|
|
}
|
|
|
|
dbg(host, dbg_pio,
|
|
"pio_read(): new target: [%i]@[%p]\n",
|
|
host->pio_bytes, host->pio_ptr);
|
|
}
|
|
|
|
dbg(host, dbg_pio,
|
|
"pio_read(): fifo:[%02i] buffer:[%03i] dcnt:[%08X]\n",
|
|
fifo, host->pio_bytes,
|
|
readl(host->base + S3C2410_SDIDCNT));
|
|
|
|
/* If we have reached the end of the block, we can
|
|
* read a word and get 1 to 3 bytes. If we in the
|
|
* middle of the block, we have to read full words,
|
|
* otherwise we will write garbage, so round down to
|
|
* an even multiple of 4. */
|
|
if (fifo >= host->pio_bytes)
|
|
fifo = host->pio_bytes;
|
|
else
|
|
fifo -= fifo & 3;
|
|
|
|
host->pio_bytes -= fifo;
|
|
host->pio_count += fifo;
|
|
|
|
fifo_words = fifo >> 2;
|
|
ptr = host->pio_ptr;
|
|
while (fifo_words--)
|
|
*ptr++ = readl(from_ptr);
|
|
host->pio_ptr = ptr;
|
|
|
|
if (fifo & 3) {
|
|
u32 n = fifo & 3;
|
|
u32 data = readl(from_ptr);
|
|
u8 *p = (u8 *)host->pio_ptr;
|
|
|
|
while (n--) {
|
|
*p++ = data;
|
|
data >>= 8;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!host->pio_bytes) {
|
|
res = get_data_buffer(host, &host->pio_bytes, &host->pio_ptr);
|
|
if (res) {
|
|
dbg(host, dbg_pio,
|
|
"pio_read(): complete (no more buffers).\n");
|
|
host->pio_active = XFER_NONE;
|
|
host->complete_what = COMPLETION_FINALIZE;
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
enable_imask(host,
|
|
S3C2410_SDIIMSK_RXFIFOHALF | S3C2410_SDIIMSK_RXFIFOLAST);
|
|
}
|
|
|
|
static void do_pio_write(struct s3cmci_host *host)
|
|
{
|
|
void __iomem *to_ptr;
|
|
int res;
|
|
u32 fifo;
|
|
u32 *ptr;
|
|
|
|
to_ptr = host->base + host->sdidata;
|
|
|
|
while ((fifo = fifo_free(host)) > 3) {
|
|
if (!host->pio_bytes) {
|
|
res = get_data_buffer(host, &host->pio_bytes,
|
|
&host->pio_ptr);
|
|
if (res) {
|
|
dbg(host, dbg_pio,
|
|
"pio_write(): complete (no more data).\n");
|
|
host->pio_active = XFER_NONE;
|
|
|
|
return;
|
|
}
|
|
|
|
dbg(host, dbg_pio,
|
|
"pio_write(): new source: [%i]@[%p]\n",
|
|
host->pio_bytes, host->pio_ptr);
|
|
|
|
}
|
|
|
|
/* If we have reached the end of the block, we have to
|
|
* write exactly the remaining number of bytes. If we
|
|
* in the middle of the block, we have to write full
|
|
* words, so round down to an even multiple of 4. */
|
|
if (fifo >= host->pio_bytes)
|
|
fifo = host->pio_bytes;
|
|
else
|
|
fifo -= fifo & 3;
|
|
|
|
host->pio_bytes -= fifo;
|
|
host->pio_count += fifo;
|
|
|
|
fifo = (fifo + 3) >> 2;
|
|
ptr = host->pio_ptr;
|
|
while (fifo--)
|
|
writel(*ptr++, to_ptr);
|
|
host->pio_ptr = ptr;
|
|
}
|
|
|
|
enable_imask(host, S3C2410_SDIIMSK_TXFIFOHALF);
|
|
}
|
|
|
|
static void pio_tasklet(unsigned long data)
|
|
{
|
|
struct s3cmci_host *host = (struct s3cmci_host *) data;
|
|
|
|
s3cmci_disable_irq(host, true);
|
|
|
|
if (host->pio_active == XFER_WRITE)
|
|
do_pio_write(host);
|
|
|
|
if (host->pio_active == XFER_READ)
|
|
do_pio_read(host);
|
|
|
|
if (host->complete_what == COMPLETION_FINALIZE) {
|
|
clear_imask(host);
|
|
if (host->pio_active != XFER_NONE) {
|
|
dbg(host, dbg_err, "unfinished %s "
|
|
"- pio_count:[%u] pio_bytes:[%u]\n",
|
|
(host->pio_active == XFER_READ) ? "read" : "write",
|
|
host->pio_count, host->pio_bytes);
|
|
|
|
if (host->mrq->data)
|
|
host->mrq->data->error = -EINVAL;
|
|
}
|
|
|
|
s3cmci_enable_irq(host, false);
|
|
finalize_request(host);
|
|
} else
|
|
s3cmci_enable_irq(host, true);
|
|
}
|
|
|
|
/*
|
|
* ISR for SDI Interface IRQ
|
|
* Communication between driver and ISR works as follows:
|
|
* host->mrq points to current request
|
|
* host->complete_what Indicates when the request is considered done
|
|
* COMPLETION_CMDSENT when the command was sent
|
|
* COMPLETION_RSPFIN when a response was received
|
|
* COMPLETION_XFERFINISH when the data transfer is finished
|
|
* COMPLETION_XFERFINISH_RSPFIN both of the above.
|
|
* host->complete_request is the completion-object the driver waits for
|
|
*
|
|
* 1) Driver sets up host->mrq and host->complete_what
|
|
* 2) Driver prepares the transfer
|
|
* 3) Driver enables interrupts
|
|
* 4) Driver starts transfer
|
|
* 5) Driver waits for host->complete_rquest
|
|
* 6) ISR checks for request status (errors and success)
|
|
* 6) ISR sets host->mrq->cmd->error and host->mrq->data->error
|
|
* 7) ISR completes host->complete_request
|
|
* 8) ISR disables interrupts
|
|
* 9) Driver wakes up and takes care of the request
|
|
*
|
|
* Note: "->error"-fields are expected to be set to 0 before the request
|
|
* was issued by mmc.c - therefore they are only set, when an error
|
|
* contition comes up
|
|
*/
|
|
|
|
static irqreturn_t s3cmci_irq(int irq, void *dev_id)
|
|
{
|
|
struct s3cmci_host *host = dev_id;
|
|
struct mmc_command *cmd;
|
|
u32 mci_csta, mci_dsta, mci_fsta, mci_dcnt, mci_imsk;
|
|
u32 mci_cclear = 0, mci_dclear;
|
|
unsigned long iflags;
|
|
|
|
mci_dsta = readl(host->base + S3C2410_SDIDSTA);
|
|
mci_imsk = readl(host->base + host->sdiimsk);
|
|
|
|
if (mci_dsta & S3C2410_SDIDSTA_SDIOIRQDETECT) {
|
|
if (mci_imsk & S3C2410_SDIIMSK_SDIOIRQ) {
|
|
mci_dclear = S3C2410_SDIDSTA_SDIOIRQDETECT;
|
|
writel(mci_dclear, host->base + S3C2410_SDIDSTA);
|
|
|
|
mmc_signal_sdio_irq(host->mmc);
|
|
return IRQ_HANDLED;
|
|
}
|
|
}
|
|
|
|
spin_lock_irqsave(&host->complete_lock, iflags);
|
|
|
|
mci_csta = readl(host->base + S3C2410_SDICMDSTAT);
|
|
mci_dcnt = readl(host->base + S3C2410_SDIDCNT);
|
|
mci_fsta = readl(host->base + S3C2410_SDIFSTA);
|
|
mci_dclear = 0;
|
|
|
|
if ((host->complete_what == COMPLETION_NONE) ||
|
|
(host->complete_what == COMPLETION_FINALIZE)) {
|
|
host->status = "nothing to complete";
|
|
clear_imask(host);
|
|
goto irq_out;
|
|
}
|
|
|
|
if (!host->mrq) {
|
|
host->status = "no active mrq";
|
|
clear_imask(host);
|
|
goto irq_out;
|
|
}
|
|
|
|
cmd = host->cmd_is_stop ? host->mrq->stop : host->mrq->cmd;
|
|
|
|
if (!cmd) {
|
|
host->status = "no active cmd";
|
|
clear_imask(host);
|
|
goto irq_out;
|
|
}
|
|
|
|
if (!s3cmci_host_usedma(host)) {
|
|
if ((host->pio_active == XFER_WRITE) &&
|
|
(mci_fsta & S3C2410_SDIFSTA_TFDET)) {
|
|
|
|
disable_imask(host, S3C2410_SDIIMSK_TXFIFOHALF);
|
|
tasklet_schedule(&host->pio_tasklet);
|
|
host->status = "pio tx";
|
|
}
|
|
|
|
if ((host->pio_active == XFER_READ) &&
|
|
(mci_fsta & S3C2410_SDIFSTA_RFDET)) {
|
|
|
|
disable_imask(host,
|
|
S3C2410_SDIIMSK_RXFIFOHALF |
|
|
S3C2410_SDIIMSK_RXFIFOLAST);
|
|
|
|
tasklet_schedule(&host->pio_tasklet);
|
|
host->status = "pio rx";
|
|
}
|
|
}
|
|
|
|
if (mci_csta & S3C2410_SDICMDSTAT_CMDTIMEOUT) {
|
|
dbg(host, dbg_err, "CMDSTAT: error CMDTIMEOUT\n");
|
|
cmd->error = -ETIMEDOUT;
|
|
host->status = "error: command timeout";
|
|
goto fail_transfer;
|
|
}
|
|
|
|
if (mci_csta & S3C2410_SDICMDSTAT_CMDSENT) {
|
|
if (host->complete_what == COMPLETION_CMDSENT) {
|
|
host->status = "ok: command sent";
|
|
goto close_transfer;
|
|
}
|
|
|
|
mci_cclear |= S3C2410_SDICMDSTAT_CMDSENT;
|
|
}
|
|
|
|
if (mci_csta & S3C2410_SDICMDSTAT_CRCFAIL) {
|
|
if (cmd->flags & MMC_RSP_CRC) {
|
|
if (host->mrq->cmd->flags & MMC_RSP_136) {
|
|
dbg(host, dbg_irq,
|
|
"fixup: ignore CRC fail with long rsp\n");
|
|
} else {
|
|
/* note, we used to fail the transfer
|
|
* here, but it seems that this is just
|
|
* the hardware getting it wrong.
|
|
*
|
|
* cmd->error = -EILSEQ;
|
|
* host->status = "error: bad command crc";
|
|
* goto fail_transfer;
|
|
*/
|
|
}
|
|
}
|
|
|
|
mci_cclear |= S3C2410_SDICMDSTAT_CRCFAIL;
|
|
}
|
|
|
|
if (mci_csta & S3C2410_SDICMDSTAT_RSPFIN) {
|
|
if (host->complete_what == COMPLETION_RSPFIN) {
|
|
host->status = "ok: command response received";
|
|
goto close_transfer;
|
|
}
|
|
|
|
if (host->complete_what == COMPLETION_XFERFINISH_RSPFIN)
|
|
host->complete_what = COMPLETION_XFERFINISH;
|
|
|
|
mci_cclear |= S3C2410_SDICMDSTAT_RSPFIN;
|
|
}
|
|
|
|
/* errors handled after this point are only relevant
|
|
when a data transfer is in progress */
|
|
|
|
if (!cmd->data)
|
|
goto clear_status_bits;
|
|
|
|
/* Check for FIFO failure */
|
|
if (host->is2440) {
|
|
if (mci_fsta & S3C2440_SDIFSTA_FIFOFAIL) {
|
|
dbg(host, dbg_err, "FIFO failure\n");
|
|
host->mrq->data->error = -EILSEQ;
|
|
host->status = "error: 2440 fifo failure";
|
|
goto fail_transfer;
|
|
}
|
|
} else {
|
|
if (mci_dsta & S3C2410_SDIDSTA_FIFOFAIL) {
|
|
dbg(host, dbg_err, "FIFO failure\n");
|
|
cmd->data->error = -EILSEQ;
|
|
host->status = "error: fifo failure";
|
|
goto fail_transfer;
|
|
}
|
|
}
|
|
|
|
if (mci_dsta & S3C2410_SDIDSTA_RXCRCFAIL) {
|
|
dbg(host, dbg_err, "bad data crc (outgoing)\n");
|
|
cmd->data->error = -EILSEQ;
|
|
host->status = "error: bad data crc (outgoing)";
|
|
goto fail_transfer;
|
|
}
|
|
|
|
if (mci_dsta & S3C2410_SDIDSTA_CRCFAIL) {
|
|
dbg(host, dbg_err, "bad data crc (incoming)\n");
|
|
cmd->data->error = -EILSEQ;
|
|
host->status = "error: bad data crc (incoming)";
|
|
goto fail_transfer;
|
|
}
|
|
|
|
if (mci_dsta & S3C2410_SDIDSTA_DATATIMEOUT) {
|
|
dbg(host, dbg_err, "data timeout\n");
|
|
cmd->data->error = -ETIMEDOUT;
|
|
host->status = "error: data timeout";
|
|
goto fail_transfer;
|
|
}
|
|
|
|
if (mci_dsta & S3C2410_SDIDSTA_XFERFINISH) {
|
|
if (host->complete_what == COMPLETION_XFERFINISH) {
|
|
host->status = "ok: data transfer completed";
|
|
goto close_transfer;
|
|
}
|
|
|
|
if (host->complete_what == COMPLETION_XFERFINISH_RSPFIN)
|
|
host->complete_what = COMPLETION_RSPFIN;
|
|
|
|
mci_dclear |= S3C2410_SDIDSTA_XFERFINISH;
|
|
}
|
|
|
|
clear_status_bits:
|
|
writel(mci_cclear, host->base + S3C2410_SDICMDSTAT);
|
|
writel(mci_dclear, host->base + S3C2410_SDIDSTA);
|
|
|
|
goto irq_out;
|
|
|
|
fail_transfer:
|
|
host->pio_active = XFER_NONE;
|
|
|
|
close_transfer:
|
|
host->complete_what = COMPLETION_FINALIZE;
|
|
|
|
clear_imask(host);
|
|
tasklet_schedule(&host->pio_tasklet);
|
|
|
|
goto irq_out;
|
|
|
|
irq_out:
|
|
dbg(host, dbg_irq,
|
|
"csta:0x%08x dsta:0x%08x fsta:0x%08x dcnt:0x%08x status:%s.\n",
|
|
mci_csta, mci_dsta, mci_fsta, mci_dcnt, host->status);
|
|
|
|
spin_unlock_irqrestore(&host->complete_lock, iflags);
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
static void s3cmci_dma_done_callback(void *arg)
|
|
{
|
|
struct s3cmci_host *host = arg;
|
|
unsigned long iflags;
|
|
|
|
BUG_ON(!host->mrq);
|
|
BUG_ON(!host->mrq->data);
|
|
|
|
spin_lock_irqsave(&host->complete_lock, iflags);
|
|
|
|
dbg(host, dbg_dma, "DMA FINISHED\n");
|
|
|
|
host->dma_complete = 1;
|
|
host->complete_what = COMPLETION_FINALIZE;
|
|
|
|
tasklet_schedule(&host->pio_tasklet);
|
|
spin_unlock_irqrestore(&host->complete_lock, iflags);
|
|
|
|
}
|
|
|
|
static void finalize_request(struct s3cmci_host *host)
|
|
{
|
|
struct mmc_request *mrq = host->mrq;
|
|
struct mmc_command *cmd;
|
|
int debug_as_failure = 0;
|
|
|
|
if (host->complete_what != COMPLETION_FINALIZE)
|
|
return;
|
|
|
|
if (!mrq)
|
|
return;
|
|
cmd = host->cmd_is_stop ? mrq->stop : mrq->cmd;
|
|
|
|
if (cmd->data && (cmd->error == 0) &&
|
|
(cmd->data->error == 0)) {
|
|
if (s3cmci_host_usedma(host) && (!host->dma_complete)) {
|
|
dbg(host, dbg_dma, "DMA Missing (%d)!\n",
|
|
host->dma_complete);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Read response from controller. */
|
|
cmd->resp[0] = readl(host->base + S3C2410_SDIRSP0);
|
|
cmd->resp[1] = readl(host->base + S3C2410_SDIRSP1);
|
|
cmd->resp[2] = readl(host->base + S3C2410_SDIRSP2);
|
|
cmd->resp[3] = readl(host->base + S3C2410_SDIRSP3);
|
|
|
|
writel(host->prescaler, host->base + S3C2410_SDIPRE);
|
|
|
|
if (cmd->error)
|
|
debug_as_failure = 1;
|
|
|
|
if (cmd->data && cmd->data->error)
|
|
debug_as_failure = 1;
|
|
|
|
dbg_dumpcmd(host, cmd, debug_as_failure);
|
|
|
|
/* Cleanup controller */
|
|
writel(0, host->base + S3C2410_SDICMDARG);
|
|
writel(S3C2410_SDIDCON_STOP, host->base + S3C2410_SDIDCON);
|
|
writel(0, host->base + S3C2410_SDICMDCON);
|
|
clear_imask(host);
|
|
|
|
if (cmd->data && cmd->error)
|
|
cmd->data->error = cmd->error;
|
|
|
|
if (cmd->data && cmd->data->stop && (!host->cmd_is_stop)) {
|
|
host->cmd_is_stop = 1;
|
|
s3cmci_send_request(host->mmc);
|
|
return;
|
|
}
|
|
|
|
/* If we have no data transfer we are finished here */
|
|
if (!mrq->data)
|
|
goto request_done;
|
|
|
|
/* Calculate the amout of bytes transfer if there was no error */
|
|
if (mrq->data->error == 0) {
|
|
mrq->data->bytes_xfered =
|
|
(mrq->data->blocks * mrq->data->blksz);
|
|
} else {
|
|
mrq->data->bytes_xfered = 0;
|
|
}
|
|
|
|
/* If we had an error while transferring data we flush the
|
|
* DMA channel and the fifo to clear out any garbage. */
|
|
if (mrq->data->error != 0) {
|
|
if (s3cmci_host_usedma(host))
|
|
dmaengine_terminate_all(host->dma);
|
|
|
|
if (host->is2440) {
|
|
/* Clear failure register and reset fifo. */
|
|
writel(S3C2440_SDIFSTA_FIFORESET |
|
|
S3C2440_SDIFSTA_FIFOFAIL,
|
|
host->base + S3C2410_SDIFSTA);
|
|
} else {
|
|
u32 mci_con;
|
|
|
|
/* reset fifo */
|
|
mci_con = readl(host->base + S3C2410_SDICON);
|
|
mci_con |= S3C2410_SDICON_FIFORESET;
|
|
|
|
writel(mci_con, host->base + S3C2410_SDICON);
|
|
}
|
|
}
|
|
|
|
request_done:
|
|
host->complete_what = COMPLETION_NONE;
|
|
host->mrq = NULL;
|
|
|
|
s3cmci_check_sdio_irq(host);
|
|
mmc_request_done(host->mmc, mrq);
|
|
}
|
|
|
|
static void s3cmci_send_command(struct s3cmci_host *host,
|
|
struct mmc_command *cmd)
|
|
{
|
|
u32 ccon, imsk;
|
|
|
|
imsk = S3C2410_SDIIMSK_CRCSTATUS | S3C2410_SDIIMSK_CMDTIMEOUT |
|
|
S3C2410_SDIIMSK_RESPONSEND | S3C2410_SDIIMSK_CMDSENT |
|
|
S3C2410_SDIIMSK_RESPONSECRC;
|
|
|
|
enable_imask(host, imsk);
|
|
|
|
if (cmd->data)
|
|
host->complete_what = COMPLETION_XFERFINISH_RSPFIN;
|
|
else if (cmd->flags & MMC_RSP_PRESENT)
|
|
host->complete_what = COMPLETION_RSPFIN;
|
|
else
|
|
host->complete_what = COMPLETION_CMDSENT;
|
|
|
|
writel(cmd->arg, host->base + S3C2410_SDICMDARG);
|
|
|
|
ccon = cmd->opcode & S3C2410_SDICMDCON_INDEX;
|
|
ccon |= S3C2410_SDICMDCON_SENDERHOST | S3C2410_SDICMDCON_CMDSTART;
|
|
|
|
if (cmd->flags & MMC_RSP_PRESENT)
|
|
ccon |= S3C2410_SDICMDCON_WAITRSP;
|
|
|
|
if (cmd->flags & MMC_RSP_136)
|
|
ccon |= S3C2410_SDICMDCON_LONGRSP;
|
|
|
|
writel(ccon, host->base + S3C2410_SDICMDCON);
|
|
}
|
|
|
|
static int s3cmci_setup_data(struct s3cmci_host *host, struct mmc_data *data)
|
|
{
|
|
u32 dcon, imsk, stoptries = 3;
|
|
|
|
/* write DCON register */
|
|
|
|
if (!data) {
|
|
writel(0, host->base + S3C2410_SDIDCON);
|
|
return 0;
|
|
}
|
|
|
|
if ((data->blksz & 3) != 0) {
|
|
/* We cannot deal with unaligned blocks with more than
|
|
* one block being transferred. */
|
|
|
|
if (data->blocks > 1) {
|
|
pr_warn("%s: can't do non-word sized block transfers (blksz %d)\n",
|
|
__func__, data->blksz);
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
while (readl(host->base + S3C2410_SDIDSTA) &
|
|
(S3C2410_SDIDSTA_TXDATAON | S3C2410_SDIDSTA_RXDATAON)) {
|
|
|
|
dbg(host, dbg_err,
|
|
"mci_setup_data() transfer stillin progress.\n");
|
|
|
|
writel(S3C2410_SDIDCON_STOP, host->base + S3C2410_SDIDCON);
|
|
s3cmci_reset(host);
|
|
|
|
if ((stoptries--) == 0) {
|
|
dbg_dumpregs(host, "DRF");
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
dcon = data->blocks & S3C2410_SDIDCON_BLKNUM_MASK;
|
|
|
|
if (s3cmci_host_usedma(host))
|
|
dcon |= S3C2410_SDIDCON_DMAEN;
|
|
|
|
if (host->bus_width == MMC_BUS_WIDTH_4)
|
|
dcon |= S3C2410_SDIDCON_WIDEBUS;
|
|
|
|
dcon |= S3C2410_SDIDCON_BLOCKMODE;
|
|
|
|
if (data->flags & MMC_DATA_WRITE) {
|
|
dcon |= S3C2410_SDIDCON_TXAFTERRESP;
|
|
dcon |= S3C2410_SDIDCON_XFER_TXSTART;
|
|
}
|
|
|
|
if (data->flags & MMC_DATA_READ) {
|
|
dcon |= S3C2410_SDIDCON_RXAFTERCMD;
|
|
dcon |= S3C2410_SDIDCON_XFER_RXSTART;
|
|
}
|
|
|
|
if (host->is2440) {
|
|
dcon |= S3C2440_SDIDCON_DS_WORD;
|
|
dcon |= S3C2440_SDIDCON_DATSTART;
|
|
}
|
|
|
|
writel(dcon, host->base + S3C2410_SDIDCON);
|
|
|
|
/* write BSIZE register */
|
|
|
|
writel(data->blksz, host->base + S3C2410_SDIBSIZE);
|
|
|
|
/* add to IMASK register */
|
|
imsk = S3C2410_SDIIMSK_FIFOFAIL | S3C2410_SDIIMSK_DATACRC |
|
|
S3C2410_SDIIMSK_DATATIMEOUT | S3C2410_SDIIMSK_DATAFINISH;
|
|
|
|
enable_imask(host, imsk);
|
|
|
|
/* write TIMER register */
|
|
|
|
if (host->is2440) {
|
|
writel(0x007FFFFF, host->base + S3C2410_SDITIMER);
|
|
} else {
|
|
writel(0x0000FFFF, host->base + S3C2410_SDITIMER);
|
|
|
|
/* FIX: set slow clock to prevent timeouts on read */
|
|
if (data->flags & MMC_DATA_READ)
|
|
writel(0xFF, host->base + S3C2410_SDIPRE);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define BOTH_DIR (MMC_DATA_WRITE | MMC_DATA_READ)
|
|
|
|
static int s3cmci_prepare_pio(struct s3cmci_host *host, struct mmc_data *data)
|
|
{
|
|
int rw = (data->flags & MMC_DATA_WRITE) ? 1 : 0;
|
|
|
|
BUG_ON((data->flags & BOTH_DIR) == BOTH_DIR);
|
|
|
|
host->pio_sgptr = 0;
|
|
host->pio_bytes = 0;
|
|
host->pio_count = 0;
|
|
host->pio_active = rw ? XFER_WRITE : XFER_READ;
|
|
|
|
if (rw) {
|
|
do_pio_write(host);
|
|
enable_imask(host, S3C2410_SDIIMSK_TXFIFOHALF);
|
|
} else {
|
|
enable_imask(host, S3C2410_SDIIMSK_RXFIFOHALF
|
|
| S3C2410_SDIIMSK_RXFIFOLAST);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int s3cmci_prepare_dma(struct s3cmci_host *host, struct mmc_data *data)
|
|
{
|
|
int rw = data->flags & MMC_DATA_WRITE;
|
|
struct dma_async_tx_descriptor *desc;
|
|
struct dma_slave_config conf = {
|
|
.src_addr = host->mem->start + host->sdidata,
|
|
.dst_addr = host->mem->start + host->sdidata,
|
|
.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES,
|
|
.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES,
|
|
};
|
|
|
|
BUG_ON((data->flags & BOTH_DIR) == BOTH_DIR);
|
|
|
|
/* Restore prescaler value */
|
|
writel(host->prescaler, host->base + S3C2410_SDIPRE);
|
|
|
|
if (!rw)
|
|
conf.direction = DMA_DEV_TO_MEM;
|
|
else
|
|
conf.direction = DMA_MEM_TO_DEV;
|
|
|
|
dma_map_sg(mmc_dev(host->mmc), data->sg, data->sg_len,
|
|
mmc_get_dma_dir(data));
|
|
|
|
dmaengine_slave_config(host->dma, &conf);
|
|
desc = dmaengine_prep_slave_sg(host->dma, data->sg, data->sg_len,
|
|
conf.direction,
|
|
DMA_CTRL_ACK | DMA_PREP_INTERRUPT);
|
|
if (!desc)
|
|
goto unmap_exit;
|
|
desc->callback = s3cmci_dma_done_callback;
|
|
desc->callback_param = host;
|
|
dmaengine_submit(desc);
|
|
dma_async_issue_pending(host->dma);
|
|
|
|
return 0;
|
|
|
|
unmap_exit:
|
|
dma_unmap_sg(mmc_dev(host->mmc), data->sg, data->sg_len,
|
|
mmc_get_dma_dir(data));
|
|
return -ENOMEM;
|
|
}
|
|
|
|
static void s3cmci_send_request(struct mmc_host *mmc)
|
|
{
|
|
struct s3cmci_host *host = mmc_priv(mmc);
|
|
struct mmc_request *mrq = host->mrq;
|
|
struct mmc_command *cmd = host->cmd_is_stop ? mrq->stop : mrq->cmd;
|
|
|
|
host->ccnt++;
|
|
prepare_dbgmsg(host, cmd, host->cmd_is_stop);
|
|
|
|
/* Clear command, data and fifo status registers
|
|
Fifo clear only necessary on 2440, but doesn't hurt on 2410
|
|
*/
|
|
writel(0xFFFFFFFF, host->base + S3C2410_SDICMDSTAT);
|
|
writel(0xFFFFFFFF, host->base + S3C2410_SDIDSTA);
|
|
writel(0xFFFFFFFF, host->base + S3C2410_SDIFSTA);
|
|
|
|
if (cmd->data) {
|
|
int res = s3cmci_setup_data(host, cmd->data);
|
|
|
|
host->dcnt++;
|
|
|
|
if (res) {
|
|
dbg(host, dbg_err, "setup data error %d\n", res);
|
|
cmd->error = res;
|
|
cmd->data->error = res;
|
|
|
|
mmc_request_done(mmc, mrq);
|
|
return;
|
|
}
|
|
|
|
if (s3cmci_host_usedma(host))
|
|
res = s3cmci_prepare_dma(host, cmd->data);
|
|
else
|
|
res = s3cmci_prepare_pio(host, cmd->data);
|
|
|
|
if (res) {
|
|
dbg(host, dbg_err, "data prepare error %d\n", res);
|
|
cmd->error = res;
|
|
cmd->data->error = res;
|
|
|
|
mmc_request_done(mmc, mrq);
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Send command */
|
|
s3cmci_send_command(host, cmd);
|
|
|
|
/* Enable Interrupt */
|
|
s3cmci_enable_irq(host, true);
|
|
}
|
|
|
|
static void s3cmci_request(struct mmc_host *mmc, struct mmc_request *mrq)
|
|
{
|
|
struct s3cmci_host *host = mmc_priv(mmc);
|
|
|
|
host->status = "mmc request";
|
|
host->cmd_is_stop = 0;
|
|
host->mrq = mrq;
|
|
|
|
if (mmc_gpio_get_cd(mmc) == 0) {
|
|
dbg(host, dbg_err, "%s: no medium present\n", __func__);
|
|
host->mrq->cmd->error = -ENOMEDIUM;
|
|
mmc_request_done(mmc, mrq);
|
|
} else
|
|
s3cmci_send_request(mmc);
|
|
}
|
|
|
|
static void s3cmci_set_clk(struct s3cmci_host *host, struct mmc_ios *ios)
|
|
{
|
|
u32 mci_psc;
|
|
|
|
/* Set clock */
|
|
for (mci_psc = 0; mci_psc < 255; mci_psc++) {
|
|
host->real_rate = host->clk_rate / (host->clk_div*(mci_psc+1));
|
|
|
|
if (host->real_rate <= ios->clock)
|
|
break;
|
|
}
|
|
|
|
if (mci_psc > 255)
|
|
mci_psc = 255;
|
|
|
|
host->prescaler = mci_psc;
|
|
writel(host->prescaler, host->base + S3C2410_SDIPRE);
|
|
|
|
/* If requested clock is 0, real_rate will be 0, too */
|
|
if (ios->clock == 0)
|
|
host->real_rate = 0;
|
|
}
|
|
|
|
static void s3cmci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
|
|
{
|
|
struct s3cmci_host *host = mmc_priv(mmc);
|
|
u32 mci_con;
|
|
|
|
/* Set the power state */
|
|
|
|
mci_con = readl(host->base + S3C2410_SDICON);
|
|
|
|
switch (ios->power_mode) {
|
|
case MMC_POWER_ON:
|
|
case MMC_POWER_UP:
|
|
/* Configure GPE5...GPE10 pins in SD mode */
|
|
if (!host->pdev->dev.of_node)
|
|
s3c_gpio_cfgall_range(S3C2410_GPE(5), 6, S3C_GPIO_SFN(2),
|
|
S3C_GPIO_PULL_NONE);
|
|
|
|
if (host->pdata->set_power)
|
|
host->pdata->set_power(ios->power_mode, ios->vdd);
|
|
|
|
if (!host->is2440)
|
|
mci_con |= S3C2410_SDICON_FIFORESET;
|
|
|
|
break;
|
|
|
|
case MMC_POWER_OFF:
|
|
default:
|
|
if (!host->pdev->dev.of_node)
|
|
gpio_direction_output(S3C2410_GPE(5), 0);
|
|
|
|
if (host->is2440)
|
|
mci_con |= S3C2440_SDICON_SDRESET;
|
|
|
|
if (host->pdata->set_power)
|
|
host->pdata->set_power(ios->power_mode, ios->vdd);
|
|
|
|
break;
|
|
}
|
|
|
|
s3cmci_set_clk(host, ios);
|
|
|
|
/* Set CLOCK_ENABLE */
|
|
if (ios->clock)
|
|
mci_con |= S3C2410_SDICON_CLOCKTYPE;
|
|
else
|
|
mci_con &= ~S3C2410_SDICON_CLOCKTYPE;
|
|
|
|
writel(mci_con, host->base + S3C2410_SDICON);
|
|
|
|
if ((ios->power_mode == MMC_POWER_ON) ||
|
|
(ios->power_mode == MMC_POWER_UP)) {
|
|
dbg(host, dbg_conf, "running at %lukHz (requested: %ukHz).\n",
|
|
host->real_rate/1000, ios->clock/1000);
|
|
} else {
|
|
dbg(host, dbg_conf, "powered down.\n");
|
|
}
|
|
|
|
host->bus_width = ios->bus_width;
|
|
}
|
|
|
|
static void s3cmci_reset(struct s3cmci_host *host)
|
|
{
|
|
u32 con = readl(host->base + S3C2410_SDICON);
|
|
|
|
con |= S3C2440_SDICON_SDRESET;
|
|
writel(con, host->base + S3C2410_SDICON);
|
|
}
|
|
|
|
static void s3cmci_enable_sdio_irq(struct mmc_host *mmc, int enable)
|
|
{
|
|
struct s3cmci_host *host = mmc_priv(mmc);
|
|
unsigned long flags;
|
|
u32 con;
|
|
|
|
local_irq_save(flags);
|
|
|
|
con = readl(host->base + S3C2410_SDICON);
|
|
host->sdio_irqen = enable;
|
|
|
|
if (enable == host->sdio_irqen)
|
|
goto same_state;
|
|
|
|
if (enable) {
|
|
con |= S3C2410_SDICON_SDIOIRQ;
|
|
enable_imask(host, S3C2410_SDIIMSK_SDIOIRQ);
|
|
|
|
if (!host->irq_state && !host->irq_disabled) {
|
|
host->irq_state = true;
|
|
enable_irq(host->irq);
|
|
}
|
|
} else {
|
|
disable_imask(host, S3C2410_SDIIMSK_SDIOIRQ);
|
|
con &= ~S3C2410_SDICON_SDIOIRQ;
|
|
|
|
if (!host->irq_enabled && host->irq_state) {
|
|
disable_irq_nosync(host->irq);
|
|
host->irq_state = false;
|
|
}
|
|
}
|
|
|
|
writel(con, host->base + S3C2410_SDICON);
|
|
|
|
same_state:
|
|
local_irq_restore(flags);
|
|
|
|
s3cmci_check_sdio_irq(host);
|
|
}
|
|
|
|
static const struct mmc_host_ops s3cmci_ops = {
|
|
.request = s3cmci_request,
|
|
.set_ios = s3cmci_set_ios,
|
|
.get_ro = mmc_gpio_get_ro,
|
|
.get_cd = mmc_gpio_get_cd,
|
|
.enable_sdio_irq = s3cmci_enable_sdio_irq,
|
|
};
|
|
|
|
static struct s3c24xx_mci_pdata s3cmci_def_pdata = {
|
|
/* This is currently here to avoid a number of if (host->pdata)
|
|
* checks. Any zero fields to ensure reasonable defaults are picked. */
|
|
.no_wprotect = 1,
|
|
.no_detect = 1,
|
|
};
|
|
|
|
#ifdef CONFIG_ARM_S3C24XX_CPUFREQ
|
|
|
|
static int s3cmci_cpufreq_transition(struct notifier_block *nb,
|
|
unsigned long val, void *data)
|
|
{
|
|
struct s3cmci_host *host;
|
|
struct mmc_host *mmc;
|
|
unsigned long newclk;
|
|
unsigned long flags;
|
|
|
|
host = container_of(nb, struct s3cmci_host, freq_transition);
|
|
newclk = clk_get_rate(host->clk);
|
|
mmc = host->mmc;
|
|
|
|
if ((val == CPUFREQ_PRECHANGE && newclk > host->clk_rate) ||
|
|
(val == CPUFREQ_POSTCHANGE && newclk < host->clk_rate)) {
|
|
spin_lock_irqsave(&mmc->lock, flags);
|
|
|
|
host->clk_rate = newclk;
|
|
|
|
if (mmc->ios.power_mode != MMC_POWER_OFF &&
|
|
mmc->ios.clock != 0)
|
|
s3cmci_set_clk(host, &mmc->ios);
|
|
|
|
spin_unlock_irqrestore(&mmc->lock, flags);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline int s3cmci_cpufreq_register(struct s3cmci_host *host)
|
|
{
|
|
host->freq_transition.notifier_call = s3cmci_cpufreq_transition;
|
|
|
|
return cpufreq_register_notifier(&host->freq_transition,
|
|
CPUFREQ_TRANSITION_NOTIFIER);
|
|
}
|
|
|
|
static inline void s3cmci_cpufreq_deregister(struct s3cmci_host *host)
|
|
{
|
|
cpufreq_unregister_notifier(&host->freq_transition,
|
|
CPUFREQ_TRANSITION_NOTIFIER);
|
|
}
|
|
|
|
#else
|
|
static inline int s3cmci_cpufreq_register(struct s3cmci_host *host)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static inline void s3cmci_cpufreq_deregister(struct s3cmci_host *host)
|
|
{
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
|
|
static int s3cmci_state_show(struct seq_file *seq, void *v)
|
|
{
|
|
struct s3cmci_host *host = seq->private;
|
|
|
|
seq_printf(seq, "Register base = 0x%08x\n", (u32)host->base);
|
|
seq_printf(seq, "Clock rate = %ld\n", host->clk_rate);
|
|
seq_printf(seq, "Prescale = %d\n", host->prescaler);
|
|
seq_printf(seq, "is2440 = %d\n", host->is2440);
|
|
seq_printf(seq, "IRQ = %d\n", host->irq);
|
|
seq_printf(seq, "IRQ enabled = %d\n", host->irq_enabled);
|
|
seq_printf(seq, "IRQ disabled = %d\n", host->irq_disabled);
|
|
seq_printf(seq, "IRQ state = %d\n", host->irq_state);
|
|
seq_printf(seq, "CD IRQ = %d\n", host->irq_cd);
|
|
seq_printf(seq, "Do DMA = %d\n", s3cmci_host_usedma(host));
|
|
seq_printf(seq, "SDIIMSK at %d\n", host->sdiimsk);
|
|
seq_printf(seq, "SDIDATA at %d\n", host->sdidata);
|
|
|
|
return 0;
|
|
}
|
|
|
|
DEFINE_SHOW_ATTRIBUTE(s3cmci_state);
|
|
|
|
#define DBG_REG(_r) { .addr = S3C2410_SDI##_r, .name = #_r }
|
|
|
|
struct s3cmci_reg {
|
|
unsigned short addr;
|
|
unsigned char *name;
|
|
};
|
|
|
|
static const struct s3cmci_reg debug_regs[] = {
|
|
DBG_REG(CON),
|
|
DBG_REG(PRE),
|
|
DBG_REG(CMDARG),
|
|
DBG_REG(CMDCON),
|
|
DBG_REG(CMDSTAT),
|
|
DBG_REG(RSP0),
|
|
DBG_REG(RSP1),
|
|
DBG_REG(RSP2),
|
|
DBG_REG(RSP3),
|
|
DBG_REG(TIMER),
|
|
DBG_REG(BSIZE),
|
|
DBG_REG(DCON),
|
|
DBG_REG(DCNT),
|
|
DBG_REG(DSTA),
|
|
DBG_REG(FSTA),
|
|
{}
|
|
};
|
|
|
|
static int s3cmci_regs_show(struct seq_file *seq, void *v)
|
|
{
|
|
struct s3cmci_host *host = seq->private;
|
|
const struct s3cmci_reg *rptr = debug_regs;
|
|
|
|
for (; rptr->name; rptr++)
|
|
seq_printf(seq, "SDI%s\t=0x%08x\n", rptr->name,
|
|
readl(host->base + rptr->addr));
|
|
|
|
seq_printf(seq, "SDIIMSK\t=0x%08x\n", readl(host->base + host->sdiimsk));
|
|
|
|
return 0;
|
|
}
|
|
|
|
DEFINE_SHOW_ATTRIBUTE(s3cmci_regs);
|
|
|
|
static void s3cmci_debugfs_attach(struct s3cmci_host *host)
|
|
{
|
|
struct device *dev = &host->pdev->dev;
|
|
|
|
host->debug_root = debugfs_create_dir(dev_name(dev), NULL);
|
|
if (IS_ERR(host->debug_root)) {
|
|
dev_err(dev, "failed to create debugfs root\n");
|
|
return;
|
|
}
|
|
|
|
host->debug_state = debugfs_create_file("state", 0444,
|
|
host->debug_root, host,
|
|
&s3cmci_state_fops);
|
|
|
|
if (IS_ERR(host->debug_state))
|
|
dev_err(dev, "failed to create debug state file\n");
|
|
|
|
host->debug_regs = debugfs_create_file("regs", 0444,
|
|
host->debug_root, host,
|
|
&s3cmci_regs_fops);
|
|
|
|
if (IS_ERR(host->debug_regs))
|
|
dev_err(dev, "failed to create debug regs file\n");
|
|
}
|
|
|
|
static void s3cmci_debugfs_remove(struct s3cmci_host *host)
|
|
{
|
|
debugfs_remove(host->debug_regs);
|
|
debugfs_remove(host->debug_state);
|
|
debugfs_remove(host->debug_root);
|
|
}
|
|
|
|
#else
|
|
static inline void s3cmci_debugfs_attach(struct s3cmci_host *host) { }
|
|
static inline void s3cmci_debugfs_remove(struct s3cmci_host *host) { }
|
|
|
|
#endif /* CONFIG_DEBUG_FS */
|
|
|
|
static int s3cmci_probe_pdata(struct s3cmci_host *host)
|
|
{
|
|
struct platform_device *pdev = host->pdev;
|
|
struct mmc_host *mmc = host->mmc;
|
|
struct s3c24xx_mci_pdata *pdata;
|
|
int i, ret;
|
|
|
|
host->is2440 = platform_get_device_id(pdev)->driver_data;
|
|
|
|
for (i = S3C2410_GPE(5); i <= S3C2410_GPE(10); i++) {
|
|
ret = gpio_request(i, dev_name(&pdev->dev));
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "failed to get gpio %d\n", i);
|
|
|
|
for (i--; i >= S3C2410_GPE(5); i--)
|
|
gpio_free(i);
|
|
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
if (!pdev->dev.platform_data)
|
|
pdev->dev.platform_data = &s3cmci_def_pdata;
|
|
|
|
pdata = pdev->dev.platform_data;
|
|
|
|
if (pdata->no_wprotect)
|
|
mmc->caps2 |= MMC_CAP2_NO_WRITE_PROTECT;
|
|
|
|
if (pdata->no_detect)
|
|
mmc->caps |= MMC_CAP_NEEDS_POLL;
|
|
|
|
if (pdata->wprotect_invert)
|
|
mmc->caps2 |= MMC_CAP2_RO_ACTIVE_HIGH;
|
|
|
|
/* If we get -ENOENT we have no card detect GPIO line */
|
|
ret = mmc_gpiod_request_cd(mmc, "cd", 0, false, 0, NULL);
|
|
if (ret != -ENOENT) {
|
|
dev_err(&pdev->dev, "error requesting GPIO for CD %d\n",
|
|
ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = mmc_gpiod_request_ro(host->mmc, "wp", 0, 0, NULL);
|
|
if (ret != -ENOENT) {
|
|
dev_err(&pdev->dev, "error requesting GPIO for WP %d\n",
|
|
ret);
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int s3cmci_probe_dt(struct s3cmci_host *host)
|
|
{
|
|
struct platform_device *pdev = host->pdev;
|
|
struct s3c24xx_mci_pdata *pdata;
|
|
struct mmc_host *mmc = host->mmc;
|
|
int ret;
|
|
|
|
host->is2440 = (int) of_device_get_match_data(&pdev->dev);
|
|
|
|
ret = mmc_of_parse(mmc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL);
|
|
if (!pdata)
|
|
return -ENOMEM;
|
|
|
|
pdev->dev.platform_data = pdata;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int s3cmci_probe(struct platform_device *pdev)
|
|
{
|
|
struct s3cmci_host *host;
|
|
struct mmc_host *mmc;
|
|
int ret;
|
|
int i;
|
|
|
|
mmc = mmc_alloc_host(sizeof(struct s3cmci_host), &pdev->dev);
|
|
if (!mmc) {
|
|
ret = -ENOMEM;
|
|
goto probe_out;
|
|
}
|
|
|
|
host = mmc_priv(mmc);
|
|
host->mmc = mmc;
|
|
host->pdev = pdev;
|
|
|
|
if (pdev->dev.of_node)
|
|
ret = s3cmci_probe_dt(host);
|
|
else
|
|
ret = s3cmci_probe_pdata(host);
|
|
|
|
if (ret)
|
|
goto probe_free_host;
|
|
|
|
host->pdata = pdev->dev.platform_data;
|
|
|
|
spin_lock_init(&host->complete_lock);
|
|
tasklet_init(&host->pio_tasklet, pio_tasklet, (unsigned long) host);
|
|
|
|
if (host->is2440) {
|
|
host->sdiimsk = S3C2440_SDIIMSK;
|
|
host->sdidata = S3C2440_SDIDATA;
|
|
host->clk_div = 1;
|
|
} else {
|
|
host->sdiimsk = S3C2410_SDIIMSK;
|
|
host->sdidata = S3C2410_SDIDATA;
|
|
host->clk_div = 2;
|
|
}
|
|
|
|
host->complete_what = COMPLETION_NONE;
|
|
host->pio_active = XFER_NONE;
|
|
|
|
host->mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
if (!host->mem) {
|
|
dev_err(&pdev->dev,
|
|
"failed to get io memory region resource.\n");
|
|
|
|
ret = -ENOENT;
|
|
goto probe_free_gpio;
|
|
}
|
|
|
|
host->mem = request_mem_region(host->mem->start,
|
|
resource_size(host->mem), pdev->name);
|
|
|
|
if (!host->mem) {
|
|
dev_err(&pdev->dev, "failed to request io memory region.\n");
|
|
ret = -ENOENT;
|
|
goto probe_free_gpio;
|
|
}
|
|
|
|
host->base = ioremap(host->mem->start, resource_size(host->mem));
|
|
if (!host->base) {
|
|
dev_err(&pdev->dev, "failed to ioremap() io memory region.\n");
|
|
ret = -EINVAL;
|
|
goto probe_free_mem_region;
|
|
}
|
|
|
|
host->irq = platform_get_irq(pdev, 0);
|
|
if (host->irq <= 0) {
|
|
dev_err(&pdev->dev, "failed to get interrupt resource.\n");
|
|
ret = -EINVAL;
|
|
goto probe_iounmap;
|
|
}
|
|
|
|
if (request_irq(host->irq, s3cmci_irq, 0, DRIVER_NAME, host)) {
|
|
dev_err(&pdev->dev, "failed to request mci interrupt.\n");
|
|
ret = -ENOENT;
|
|
goto probe_iounmap;
|
|
}
|
|
|
|
/* We get spurious interrupts even when we have set the IMSK
|
|
* register to ignore everything, so use disable_irq() to make
|
|
* ensure we don't lock the system with un-serviceable requests. */
|
|
|
|
disable_irq(host->irq);
|
|
host->irq_state = false;
|
|
|
|
/* Depending on the dma state, get a DMA channel to use. */
|
|
|
|
if (s3cmci_host_usedma(host)) {
|
|
host->dma = dma_request_chan(&pdev->dev, "rx-tx");
|
|
ret = PTR_ERR_OR_ZERO(host->dma);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "cannot get DMA channel.\n");
|
|
goto probe_free_irq;
|
|
}
|
|
}
|
|
|
|
host->clk = clk_get(&pdev->dev, "sdi");
|
|
if (IS_ERR(host->clk)) {
|
|
dev_err(&pdev->dev, "failed to find clock source.\n");
|
|
ret = PTR_ERR(host->clk);
|
|
host->clk = NULL;
|
|
goto probe_free_dma;
|
|
}
|
|
|
|
ret = clk_prepare_enable(host->clk);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "failed to enable clock source.\n");
|
|
goto clk_free;
|
|
}
|
|
|
|
host->clk_rate = clk_get_rate(host->clk);
|
|
|
|
mmc->ops = &s3cmci_ops;
|
|
mmc->ocr_avail = MMC_VDD_32_33 | MMC_VDD_33_34;
|
|
#ifdef CONFIG_MMC_S3C_HW_SDIO_IRQ
|
|
mmc->caps = MMC_CAP_4_BIT_DATA | MMC_CAP_SDIO_IRQ;
|
|
#else
|
|
mmc->caps = MMC_CAP_4_BIT_DATA;
|
|
#endif
|
|
mmc->f_min = host->clk_rate / (host->clk_div * 256);
|
|
mmc->f_max = host->clk_rate / host->clk_div;
|
|
|
|
if (host->pdata->ocr_avail)
|
|
mmc->ocr_avail = host->pdata->ocr_avail;
|
|
|
|
mmc->max_blk_count = 4095;
|
|
mmc->max_blk_size = 4095;
|
|
mmc->max_req_size = 4095 * 512;
|
|
mmc->max_seg_size = mmc->max_req_size;
|
|
|
|
mmc->max_segs = 128;
|
|
|
|
dbg(host, dbg_debug,
|
|
"probe: mode:%s mapped mci_base:%p irq:%u irq_cd:%u dma:%p.\n",
|
|
(host->is2440?"2440":""),
|
|
host->base, host->irq, host->irq_cd, host->dma);
|
|
|
|
ret = s3cmci_cpufreq_register(host);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "failed to register cpufreq\n");
|
|
goto free_dmabuf;
|
|
}
|
|
|
|
ret = mmc_add_host(mmc);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "failed to add mmc host.\n");
|
|
goto free_cpufreq;
|
|
}
|
|
|
|
s3cmci_debugfs_attach(host);
|
|
|
|
platform_set_drvdata(pdev, mmc);
|
|
dev_info(&pdev->dev, "%s - using %s, %s SDIO IRQ\n", mmc_hostname(mmc),
|
|
s3cmci_host_usedma(host) ? "dma" : "pio",
|
|
mmc->caps & MMC_CAP_SDIO_IRQ ? "hw" : "sw");
|
|
|
|
return 0;
|
|
|
|
free_cpufreq:
|
|
s3cmci_cpufreq_deregister(host);
|
|
|
|
free_dmabuf:
|
|
clk_disable_unprepare(host->clk);
|
|
|
|
clk_free:
|
|
clk_put(host->clk);
|
|
|
|
probe_free_dma:
|
|
if (s3cmci_host_usedma(host))
|
|
dma_release_channel(host->dma);
|
|
|
|
probe_free_irq:
|
|
free_irq(host->irq, host);
|
|
|
|
probe_iounmap:
|
|
iounmap(host->base);
|
|
|
|
probe_free_mem_region:
|
|
release_mem_region(host->mem->start, resource_size(host->mem));
|
|
|
|
probe_free_gpio:
|
|
if (!pdev->dev.of_node)
|
|
for (i = S3C2410_GPE(5); i <= S3C2410_GPE(10); i++)
|
|
gpio_free(i);
|
|
|
|
probe_free_host:
|
|
mmc_free_host(mmc);
|
|
|
|
probe_out:
|
|
return ret;
|
|
}
|
|
|
|
static void s3cmci_shutdown(struct platform_device *pdev)
|
|
{
|
|
struct mmc_host *mmc = platform_get_drvdata(pdev);
|
|
struct s3cmci_host *host = mmc_priv(mmc);
|
|
|
|
if (host->irq_cd >= 0)
|
|
free_irq(host->irq_cd, host);
|
|
|
|
s3cmci_debugfs_remove(host);
|
|
s3cmci_cpufreq_deregister(host);
|
|
mmc_remove_host(mmc);
|
|
clk_disable_unprepare(host->clk);
|
|
}
|
|
|
|
static int s3cmci_remove(struct platform_device *pdev)
|
|
{
|
|
struct mmc_host *mmc = platform_get_drvdata(pdev);
|
|
struct s3cmci_host *host = mmc_priv(mmc);
|
|
int i;
|
|
|
|
s3cmci_shutdown(pdev);
|
|
|
|
clk_put(host->clk);
|
|
|
|
tasklet_disable(&host->pio_tasklet);
|
|
|
|
if (s3cmci_host_usedma(host))
|
|
dma_release_channel(host->dma);
|
|
|
|
free_irq(host->irq, host);
|
|
|
|
if (!pdev->dev.of_node)
|
|
for (i = S3C2410_GPE(5); i <= S3C2410_GPE(10); i++)
|
|
gpio_free(i);
|
|
|
|
iounmap(host->base);
|
|
release_mem_region(host->mem->start, resource_size(host->mem));
|
|
|
|
mmc_free_host(mmc);
|
|
return 0;
|
|
}
|
|
|
|
static const struct of_device_id s3cmci_dt_match[] = {
|
|
{
|
|
.compatible = "samsung,s3c2410-sdi",
|
|
.data = (void *)0,
|
|
},
|
|
{
|
|
.compatible = "samsung,s3c2412-sdi",
|
|
.data = (void *)1,
|
|
},
|
|
{
|
|
.compatible = "samsung,s3c2440-sdi",
|
|
.data = (void *)1,
|
|
},
|
|
{ /* sentinel */ },
|
|
};
|
|
MODULE_DEVICE_TABLE(of, s3cmci_dt_match);
|
|
|
|
static const struct platform_device_id s3cmci_driver_ids[] = {
|
|
{
|
|
.name = "s3c2410-sdi",
|
|
.driver_data = 0,
|
|
}, {
|
|
.name = "s3c2412-sdi",
|
|
.driver_data = 1,
|
|
}, {
|
|
.name = "s3c2440-sdi",
|
|
.driver_data = 1,
|
|
},
|
|
{ }
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(platform, s3cmci_driver_ids);
|
|
|
|
static struct platform_driver s3cmci_driver = {
|
|
.driver = {
|
|
.name = "s3c-sdi",
|
|
.of_match_table = s3cmci_dt_match,
|
|
},
|
|
.id_table = s3cmci_driver_ids,
|
|
.probe = s3cmci_probe,
|
|
.remove = s3cmci_remove,
|
|
.shutdown = s3cmci_shutdown,
|
|
};
|
|
|
|
module_platform_driver(s3cmci_driver);
|
|
|
|
MODULE_DESCRIPTION("Samsung S3C MMC/SD Card Interface driver");
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_AUTHOR("Thomas Kleffel <tk@maintech.de>, Ben Dooks <ben-linux@fluff.org>");
|