diff options
Diffstat (limited to 'drivers/mmc/host/mmc_atsmb.c')
-rwxr-xr-x | drivers/mmc/host/mmc_atsmb.c | 2837 |
1 files changed, 2837 insertions, 0 deletions
diff --git a/drivers/mmc/host/mmc_atsmb.c b/drivers/mmc/host/mmc_atsmb.c new file mode 100755 index 00000000..150b7b12 --- /dev/null +++ b/drivers/mmc/host/mmc_atsmb.c @@ -0,0 +1,2837 @@ +/*++ +linux/drivers/mmc/host/mmc_atsmb.c + +Copyright (c) 2008 WonderMedia Technologies, Inc. + +This program is free software: you can redistribute it and/or modify it under the +terms of the GNU General Public License as published by the Free Software Foundation, +either version 2 of the License, or (at your option) any later version. + +This program is distributed in the hope that it will be useful, but WITHOUT +ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +PARTICULAR PURPOSE. See the GNU General Public License for more details. +You should have received a copy of the GNU General Public License along with +this program. If not, see <http://www.gnu.org/licenses/>. + +WonderMedia Technologies, Inc. +10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C. +--*/ + +//#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/platform_device.h> +#include <linux/interrupt.h> +#include <linux/delay.h> +#include <linux/err.h> +#include <linux/interrupt.h> +#include <linux/blkdev.h> +#include <linux/mmc/host.h> +#include <linux/mmc/card.h> +#include <linux/mmc/sd.h> +#include <linux/mmc/mmc.h> +#include <linux/mmc/sdio.h> +#include <linux/completion.h> +#include <linux/pagemap.h> +#include <linux/dma-mapping.h> +#include <asm/dma.h> +#include <asm/io.h> +#include <asm/irq.h> +#include <asm/memory.h> +#include <mach/hardware.h> +#include <asm/scatterlist.h> +#include <asm/sizes.h> +#include "mmc_atsmb.h" +//#include <mach/multicard.h> +#include <mach/irqs.h> +#include <linux/regulator/consumer.h> + +#define mprintk + +#if 0 +#define DBG(host, fmt, args...) \ + printk("%s: %s: " fmt, mmc_hostname(host->mmc), __func__ , args) +#endif +#define ATSMB_TIMEOUT_TIME (HZ*2) + + +static struct delayed_work mmc_work; + +static int g_atsmb_regulator; +static struct regulator *atsmb_regulator; + + +//add by jay,for modules support +static u64 wmt_sdmmc_dma_mask = 0xffffffffUL; +static struct resource wmt_sdmmc_resources[] = { + [0] = { + .start = SD0_SDIO_MMC_BASE_ADDR, + .end = (SD0_SDIO_MMC_BASE_ADDR + 0x3FF), + .flags = IORESOURCE_MEM, + }, + [1] = { + .start = IRQ_SDC0, + .end = IRQ_SDC0, + .flags = IORESOURCE_IRQ, + }, + [2] = { + .start = IRQ_SDC0_DMA, + .end = IRQ_SDC0_DMA, + .flags = IORESOURCE_IRQ, + }, + /*2008/10/6 RichardHsu-s*/ + [3] = { + .start = IRQ_PMC_WAKEUP, + .end = IRQ_PMC_WAKEUP, + .flags = IORESOURCE_IRQ, + }, + /*2008/10/6 RichardHsu-e*/ +}; + +struct kobject *atsmb_kobj; +struct mmc_host *mmc_host_attr = NULL; +// +static void atsmb_release(struct device * dev) {} + +/*#ifdef CONFIG_MMC_DEBUG*/ +#define MORE_INFO +#if 0 +#define DBG(x...) printk(KERN_ALERT x) +#define DBGR(x...) do { } while (0) +#else +#define DBG(x...) do { } while (0) +#define DBGR(x...) do { } while (0) +#endif + +/* when Read CRC error occur, and clear read CRC error by software reset.*/ +void atsmb_copy_reg(int direct) +{ + static u8 CTL, CMD_IDX, RSP_TYPE, BUS_MODE, INT_MASK_0, INT_MASK_1, SD_STS_0, SD_STS_1, SD_STS_2, SD_STS_3; + static u8 EXT_BUS_MODE, EXT_CTL_1, EXT_CTL_2, MAN_TUNE_VAL, SD_WRI_TUNE; + static u8 RSP_0, RSP_1, RSP_2, RSP_3, RSP_4, RSP_5, RSP_6, RSP_7; + static u8 RSP_8, RSP_9, RSP_10, RSP_11, RSP_12, RSP_13, RSP_14, RSP_15; + static u8 RSP_TOUT, CLK_SEL, EXT_CTL; + static u16 BLK_LEN, BLK_CNT, SHDW_BLKLEN, TIMER_VAL, CTL2; + static u32 CMD_ARG, CURBLK_CNT; + + /*direct 0: copy register to memory; 1: copy memory to register.*/ + if (direct == 0) { + CTL = *ATSMB0_CTL; + CMD_IDX = *ATSMB0_CMD_IDX; + RSP_TYPE = *ATSMB0_RSP_TYPE; + CMD_ARG = *ATSMB0_CMD_ARG; + BUS_MODE = *ATSMB0_BUS_MODE; + EXT_BUS_MODE = *ATSMB0_EXT_BUS_MODE; + CTL2 = *ATSMB0_CTL2; + BLK_LEN = *ATSMB0_BLK_LEN; + BLK_CNT = *ATSMB0_BLK_CNT; + RSP_0 = *ATSMB0_RSP_0; + RSP_1 = *ATSMB0_RSP_1; + RSP_2 = *ATSMB0_RSP_2; + RSP_3 = *ATSMB0_RSP_3; + RSP_4 = *ATSMB0_RSP_4; + RSP_5 = *ATSMB0_RSP_5; + RSP_6 = *ATSMB0_RSP_6; + RSP_7 = *ATSMB0_RSP_7; + RSP_8 = *ATSMB0_RSP_8; + RSP_9 = *ATSMB0_RSP_9; + RSP_10 = *ATSMB0_RSP_10; + RSP_11 = *ATSMB0_RSP_11; + RSP_12 = *ATSMB0_RSP_12; + RSP_13 = *ATSMB0_RSP_13; + RSP_14 = *ATSMB0_RSP_14; + RSP_15 = *ATSMB0_RSP_15; + CURBLK_CNT = *ATSMB0_CURBLK_CNT; + INT_MASK_0 = *ATSMB0_INT_MASK_0; + INT_MASK_1 = *ATSMB0_INT_MASK_1; + SD_STS_0 = *ATSMB0_SD_STS_0; + SD_STS_1 = *ATSMB0_SD_STS_1; + SD_STS_2 = *ATSMB0_SD_STS_2; + SD_STS_3 = *ATSMB0_SD_STS_3; + RSP_TOUT = *ATSMB0_RSP_TOUT; + CLK_SEL = *ATSMB0_CLK_SEL; + EXT_CTL = *ATSMB0_EXT_CTL; + EXT_CTL_1 = *ATSMB0_EXT_CTL_1; + EXT_CTL_2 = *ATSMB0_EXT_CTL_2; + SHDW_BLKLEN = *ATSMB0_SHDW_BLKLEN; + MAN_TUNE_VAL = *ATSMB0_MAN_TUNE_VAL; + SD_WRI_TUNE = *ATSMB0_SD_WRI_TUNE; + TIMER_VAL = *ATSMB0_TIMER_VAL; + } else { + *ATSMB0_CTL = CTL; + *ATSMB0_CMD_IDX = CMD_IDX; + *ATSMB0_RSP_TYPE = RSP_TYPE; + *ATSMB0_CMD_ARG = CMD_ARG; + *ATSMB0_BUS_MODE = BUS_MODE; + *ATSMB0_EXT_BUS_MODE = EXT_BUS_MODE; + *ATSMB0_CTL2 = CTL2; + *ATSMB0_BLK_LEN = BLK_LEN; + *ATSMB0_BLK_CNT = BLK_CNT; + *ATSMB0_RSP_0 = RSP_0; + *ATSMB0_RSP_1 = RSP_1; + *ATSMB0_RSP_2 = RSP_2; + *ATSMB0_RSP_3 = RSP_3; + *ATSMB0_RSP_4 = RSP_4; + *ATSMB0_RSP_5 = RSP_5; + *ATSMB0_RSP_6 = RSP_6; + *ATSMB0_RSP_7 = RSP_7; + *ATSMB0_RSP_8 = RSP_8; + *ATSMB0_RSP_9 = RSP_9; + *ATSMB0_RSP_10 = RSP_10; + *ATSMB0_RSP_11 = RSP_11; + *ATSMB0_RSP_12 = RSP_12; + *ATSMB0_RSP_13 = RSP_13; + *ATSMB0_RSP_14 = RSP_14; + *ATSMB0_RSP_15 = RSP_15; + *ATSMB0_CURBLK_CNT = CURBLK_CNT; + *ATSMB0_INT_MASK_0 = INT_MASK_0; + *ATSMB0_INT_MASK_1 = INT_MASK_1; + *ATSMB0_SD_STS_0 = SD_STS_0; + *ATSMB0_SD_STS_1 = SD_STS_1; + *ATSMB0_SD_STS_2 = SD_STS_2; + *ATSMB0_SD_STS_3 = SD_STS_3; + *ATSMB0_RSP_TOUT = RSP_TOUT; + *ATSMB0_CLK_SEL = CLK_SEL; + *ATSMB0_EXT_CTL = EXT_CTL; + *ATSMB0_EXT_CTL_1 = EXT_CTL_1; + *ATSMB0_EXT_CTL_2 = EXT_CTL_2; + *ATSMB0_SHDW_BLKLEN = SHDW_BLKLEN; + *ATSMB0_MAN_TUNE_VAL = MAN_TUNE_VAL; + *ATSMB0_SD_WRI_TUNE = SD_WRI_TUNE; + *ATSMB0_TIMER_VAL = TIMER_VAL; + } +} + +#if 0 +void atsmb_dump_reg(void) +{ + u8 CTL, CMD_IDX, RSP_TYPE, BUS_MODE, INT_MASK_0, INT_MASK_1, SD_STS_0, SD_STS_1, SD_STS_2, SD_STS_3; + u8 EXT_BUS_MODE, EXT_CTL_1, EXT_CTL_2, MAN_TUNE_VAL, SD_WRI_TUNE; + u8 RSP_0, RSP_1, RSP_2, RSP_3, RSP_4, RSP_5, RSP_6, RSP_7; + u8 RSP_8, RSP_9, RSP_10, RSP_11, RSP_12, RSP_13, RSP_14, RSP_15; + u8 RSP_TOUT, CLK_SEL, EXT_CTL; + u16 BLK_LEN, BLK_CNT, SHDW_BLKLEN, TIMER_VAL, CTL2; + u32 CMD_ARG, PDMA_GCR, PDMA_IER, PDMA_ISR, PDMA_DESPR, PDMA_RBR, PDMA_DAR, PDMA_BAR, PDMA_CPR, PDMA_CCR; + u32 CURBLK_CNT; + + CTL = *ATSMB0_CTL; + CMD_IDX = *ATSMB0_CMD_IDX; + RSP_TYPE = *ATSMB0_RSP_TYPE; + CMD_ARG = *ATSMB0_CMD_ARG; + BUS_MODE = *ATSMB0_BUS_MODE; + EXT_BUS_MODE = *ATSMB0_EXT_BUS_MODE; + CTL2 = *ATSMB0_CTL2; + BLK_LEN = *ATSMB0_BLK_LEN; + BLK_CNT = *ATSMB0_BLK_CNT; + RSP_0 = *ATSMB0_RSP_0; + RSP_1 = *ATSMB0_RSP_1; + RSP_2 = *ATSMB0_RSP_2; + RSP_3 = *ATSMB0_RSP_3; + RSP_4 = *ATSMB0_RSP_4; + RSP_5 = *ATSMB0_RSP_5; + RSP_6 = *ATSMB0_RSP_6; + RSP_7 = *ATSMB0_RSP_7; + RSP_8 = *ATSMB0_RSP_8; + RSP_9 = *ATSMB0_RSP_9; + RSP_10 = *ATSMB0_RSP_10; + RSP_11 = *ATSMB0_RSP_11; + RSP_12 = *ATSMB0_RSP_12; + RSP_13 = *ATSMB0_RSP_13; + RSP_14 = *ATSMB0_RSP_14; + RSP_15 = *ATSMB0_RSP_15; + CURBLK_CNT = *ATSMB0_CURBLK_CNT; + INT_MASK_0 = *ATSMB0_INT_MASK_0; + INT_MASK_1 = *ATSMB0_INT_MASK_1; + SD_STS_0 = *ATSMB0_SD_STS_0; + SD_STS_1 = *ATSMB0_SD_STS_1; + SD_STS_2 = *ATSMB0_SD_STS_2; + SD_STS_3 = *ATSMB0_SD_STS_3; + RSP_TOUT = *ATSMB0_RSP_TOUT; + CLK_SEL = *ATSMB0_CLK_SEL; + EXT_CTL = *ATSMB0_EXT_CTL; + EXT_CTL_1 = *ATSMB0_EXT_CTL_1; + EXT_CTL_2 = *ATSMB0_EXT_CTL_2; + SHDW_BLKLEN = *ATSMB0_SHDW_BLKLEN; + MAN_TUNE_VAL = *ATSMB0_MAN_TUNE_VAL; + SD_WRI_TUNE = *ATSMB0_SD_WRI_TUNE; + TIMER_VAL = *ATSMB0_TIMER_VAL; + + PDMA_GCR = *ATSMB0_PDMA_GCR; + PDMA_IER = *ATSMB0_PDMA_IER; + PDMA_ISR = *ATSMB0_PDMA_ISR; + PDMA_DESPR = *ATSMB0_PDMA_DESPR; + PDMA_RBR = *ATSMB0_PDMA_RBR; + PDMA_DAR = *ATSMB0_PDMA_DAR; + PDMA_BAR = *ATSMB0_PDMA_BAR; + PDMA_CPR = *ATSMB0_PDMA_CPR; + PDMA_CCR = *ATSMB0_PDMA_CCR; + + DBGR("\n+---------------------------Registers----------------------------+\n"); + + DBGR("%16s = 0x%8x |", "CTL", CTL); + DBGR("%16s = 0x%8x\n", "CMD_IDX", CMD_IDX); + + DBGR("%16s = 0x%8x |", "RSP_TYPE", RSP_TYPE); + DBGR("%16s = 0x%8x\n", "CMD_ARG", CMD_ARG); + + DBGR("%16s = 0x%8x |", "BUS_MODE", BUS_MODE); + DBGR("%16s = 0x%8x\n", "EXT_BUS_MODE", EXT_BUS_MODE); + + DBGR("%16s = 0x%8x |", "CTL2", CTL2); + DBGR("%16s = 0x%8x\n", "BLK_LEN", BLK_LEN); + + DBGR("%16s = 0x%8x |", "BLK_CNT", BLK_CNT); + DBGR("%16s = 0x%8x\n", "RSP_0", RSP_0); + + DBGR("%16s = 0x%8x |", "RSP_1", RSP_1); + DBGR("%16s = 0x%8x\n", "RSP_2", RSP_2); + + DBGR("%16s = 0x%8x |", "RSP_3", RSP_3); + DBGR("%16s = 0x%8x\n", "RSP_4", RSP_4); + + DBGR("%16s = 0x%8x |", "RSP_5", RSP_5); + DBGR("%16s = 0x%8x\n", "RSP_6", RSP_6); + + DBGR("%16s = 0x%8x |", "RSP_7", RSP_7); + DBGR("%16s = 0x%8x\n", "RSP_8", RSP_8); + + DBGR("%16s = 0x%8x |", "RSP_9", RSP_9); + DBGR("%16s = 0x%8x\n", "RSP_10", RSP_10); + + DBGR("%16s = 0x%8x |", "RSP_11", RSP_11); + DBGR("%16s = 0x%8x\n", "RSP_12", RSP_12); + + DBGR("%16s = 0x%8x |", "RSP_13", RSP_13); + DBGR("%16s = 0x%8x\n", "RSP_14", RSP_14); + + DBGR("%16s = 0x%8x\n", "RSP_15", RSP_15); + + DBGR("%16s = 0x%8x |", "CURBLK_CNT", CURBLK_CNT); + DBGR("%16s = 0x%8x\n", "INT_MASK_0", INT_MASK_0); + + DBGR("%16s = 0x%8x |", "INT_MASK_1", INT_MASK_1); + DBGR("%16s = 0x%8x\n", "SD_STS_0", SD_STS_0); + + DBGR("%16s = 0x%8x |", "SD_STS_1", SD_STS_1); + DBGR("%16s = 0x%8x\n", "SD_STS_2", SD_STS_2); + + DBGR("%16s = 0x%8x |", "SD_STS_3", SD_STS_3); + DBGR("%16s = 0x%8x\n", "RSP_TOUT", RSP_TOUT); + + DBGR("%16s = 0x%8x |", "CLK_SEL", CLK_SEL); + DBGR("%16s = 0x%8x\n", "EXT_CTL", EXT_CTL); + + DBGR("%16s = 0x%8x |", "EXT_CTL_1", EXT_CTL_1); + DBGR("%16s = 0x%8x\n", "EXT_CTL_2", EXT_CTL_2); + + DBGR("%16s = 0x%8x |", "SHDW_BLKLEN", SHDW_BLKLEN); + DBGR("%16s = 0x%8x\n", "MAN_TUNE_VAL", MAN_TUNE_VAL); + + DBGR("%16s = 0x%8x |", "SD_WRI_TUNE", SD_WRI_TUNE); + DBGR("%16s = 0x%8x\n", "TIMER_VAL", TIMER_VAL); + + DBGR("%16s = 0x%8x |", "PDMA_GCR", PDMA_GCR); + DBGR("%16s = 0x%8x\n", "PDMA_IER", PDMA_IER); + + DBGR("%16s = 0x%8x |", "PDMA_ISR", PDMA_ISR); + DBGR("%16s = 0x%8x\n", "PDMA_DESPR", PDMA_DESPR); + + DBGR("%16s = 0x%8x |", "PDMA_RBR", PDMA_RBR); + DBGR("%16s = 0x%8x\n", "PDMA_DAR", PDMA_DAR); + + DBGR("%16s = 0x%8x |", "PDMA_BAR", PDMA_BAR); + DBGR("%16s = 0x%8x\n", "PDMA_CPR", PDMA_CPR); + + DBGR("%16s = 0x%8x |", "PDMA_CCR", PDMA_CCR); + DBGR("\n+----------------------------------------------------------------+\n"); +} +#else +void atsmb_dump_reg(void) {} +#endif + +unsigned int fmax0 = 515633; +unsigned int MMC0_DRIVER_VERSION; +int SD0_function = 0; /*0: normal SD/MMC card reader , 1: internal SDIO wifi*/ +int SDXC0_function; +static int SD0_detect_pol = 0; +static int SD0_detect_pulldown = 0; +static int SD0_speed = 0; +//add by kevin guan +int SD0_ro_disable = 0; + +int SCC_ID(void){ + unsigned short val; + + val = REG16_VAL(SYSTEM_CFG_CTRL_BASE_ADDR + 0x2); + return val; +} + +int get_chip_version(void) /*2008/05/01 janshiue modify for A1 chip*/ +{ + u32 tmp; + + tmp = REG32_VAL(SYSTEM_CFG_CTRL_BASE_ADDR); + tmp = ((tmp & 0xF00) >> 4) + 0x90 + ((tmp & 0xFF) - 1); + return tmp; +} + +void get_driver_version(void) +{ + if (SCC_ID() == 0x3426) { + if (get_chip_version() < 0xA1) + MMC0_DRIVER_VERSION = MMC_DRV_3426_A0; + else if (get_chip_version() == 0xA1) + MMC0_DRIVER_VERSION = MMC_DRV_3426_A1; + else + MMC0_DRIVER_VERSION = MMC_DRV_3426_A2; + } else if (SCC_ID() == 0x3437) { + if (get_chip_version() < 0xA1) + MMC0_DRIVER_VERSION = MMC_DRV_3437_A0; + else + MMC0_DRIVER_VERSION = MMC_DRV_3437_A1; + } else if (SCC_ID() == 0x3429) { + MMC0_DRIVER_VERSION = MMC_DRV_3429; + } else if (SCC_ID() == 0x3451) { + if (get_chip_version() < 0xA1) + MMC0_DRIVER_VERSION = MMC_DRV_3451_A0; + } else if (SCC_ID() == 0x3465) { + MMC0_DRIVER_VERSION = MMC_DRV_3465; + } else if (SCC_ID() == 0x3445) { + MMC0_DRIVER_VERSION = MMC_DRV_3445; + } else if (SCC_ID() == 0x3481) { + MMC0_DRIVER_VERSION = MMC_DRV_3481; + } else if (SCC_ID() == 0x3498) { + MMC0_DRIVER_VERSION = MMC_DRV_3498; + } +} +/*2008/10/6 RichardHsu-e*/ + +/********************************************************************** +Name : atsmb_alloc_desc +Function : To config PDMA descriptor setting. +Calls : +Called by : +Parameter : +Author : Janshiue Wu +History : +***********************************************************************/ +static inline int atsmb_alloc_desc(struct atsmb_host *host, + unsigned int bytes) +{ + void *DescPool = NULL; + DBG("[%s] s\n",__func__); + + DescPool = dma_alloc_coherent(host->mmc->parent, bytes, &(host->DescPhyAddr), GFP_KERNEL); + if (!DescPool) { + DBG("can't allocal desc pool=%8X %8X\n", DescPool, host->DescPhyAddr); + DBG("[%s] e1\n",__func__); + return -1; + } + DBG("allocal desc pool=%8X %8X\n", DescPool, host->DescPhyAddr); + host->DescVirAddr = (unsigned long *)DescPool; + host->DescSize = bytes; + DBG("[%s] e2\n",__func__); + return 0; +} + +/********************************************************************** +Name : atsmb_config_desc +Function : To config PDMA descriptor setting. +Calls : +Called by : +Parameter : +Author : Janshiue Wu +History : +***********************************************************************/ +static inline void atsmb_config_desc(unsigned long *DescAddr, + unsigned long *BufferAddr, + unsigned long Blk_Cnt) +{ + + int i = 0 ; + unsigned long *CurDes = DescAddr; + DBG("[%s] s\n",__func__); + + /* the first Descriptor store for 1 Block data + * (512 bytes) + */ + for (i = 0 ; i < 3 ; i++) { + atsmb_init_short_desc(CurDes, 0x80, BufferAddr, 0); + BufferAddr += 0x20; + CurDes += sizeof(struct SD_PDMA_DESC_S)/4; + } + if (Blk_Cnt > 1) { + atsmb_init_long_desc(CurDes, 0x80, BufferAddr, CurDes + (sizeof(struct SD_PDMA_DESC_L)/4), 0); + BufferAddr += 0x20; + CurDes += sizeof(struct SD_PDMA_DESC_L)/4; + /* the following Descriptor store the rest Blocks data + * (Blk_Cnt - 1) * 512 bytes + */ + atsmb_init_long_desc(CurDes, (Blk_Cnt - 1) * 512, + BufferAddr, CurDes + (sizeof(struct SD_PDMA_DESC_L)/4), 1); + } else { + atsmb_init_long_desc(CurDes, 0x80, BufferAddr, CurDes + (sizeof(struct SD_PDMA_DESC_L)/4), 1); + } + DBG("[%s] e\n",__func__); +} +/********************************************************************** +Name : atsmb_config_dma +Function : To set des/src address, byte count to transfer, and DMA channel settings, + and DMA ctrl. register. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static inline void atsmb_config_dma(unsigned long config_dir, + unsigned long dma_mask, + struct atsmb_host *host) +{ + + DBG("[%s] s\n",__func__); + /* Enable DMA */ + *ATSMB0_PDMA_GCR = SD_PDMA_GCR_DMA_EN; + *ATSMB0_PDMA_GCR = SD_PDMA_GCR_SOFTRESET; + *ATSMB0_PDMA_GCR = SD_PDMA_GCR_DMA_EN; + /*open interrupt*/ + *ATSMB0_PDMA_IER = SD_PDMA_IER_INT_EN; + /*Make sure host could co-work with DMA*/ + *ATSMB0_SD_STS_2 |= ATSMB_CLK_FREEZ_EN; + + /*Set timer timeout value*/ + /*If clock is 390KHz*/ + if (host->current_clock < 400000) + *ATSMB0_TIMER_VAL = 0x200; /*1024*512*(1/390K) second*/ + else + *ATSMB0_TIMER_VAL = 0x1fff; /*why not to be 0xffff?*/ + + /*clear all DMA INT status for safety*/ + *ATSMB0_PDMA_ISR |= SD_PDMA_IER_INT_STS; + + /* hook desc */ + *ATSMB0_PDMA_DESPR = host->DescPhyAddr; + if (config_dir == DMA_CFG_WRITE) + *ATSMB0_PDMA_CCR &= SD_PDMA_CCR_IF_to_peripheral; + else + *ATSMB0_PDMA_CCR |= SD_PDMA_CCR_peripheral_to_IF; + + host->DmaIntMask = dma_mask; /*save success event*/ + + *ATSMB0_PDMA_CCR |= SD_PDMA_CCR_RUN; + DBG("[%s] e\n",__func__); +} + +/********************************************************************** +Name : atsmb_set_dma_end +Function : To set all descriptor end bit for removing card when PDMA + don't transfer data done. let PDMA transfer done +Calls : +Called by : +Parameter : +Author : Eason Chien +History : 2013/05/16 +***********************************************************************/ +void atsmb_set_dma_end(unsigned long *DescAddr, unsigned int bytes) +{ + int i = 0; + struct SD_PDMA_DESC_S *CurDes_S; + CurDes_S = (struct SD_PDMA_DESC_S *) DescAddr; + DBG("[%s] s\n", __func__); + DBG("[%s] byte = %d, size of %d\n", __func__, bytes, sizeof(struct SD_PDMA_DESC_S)); + + CurDes_S += 255; + + for (i = 255; i >=0 ;i--) { + /* Set end bit of all descriptor and let PDMA finish as soon as possible */ + CurDes_S->end = 1; + DBG("[%s] %d CurDes 0x%x, 0x%x, e=%x\n", __func__, i, CurDes_S, *CurDes_S); + CurDes_S -= 1; + } + DBG("[%s] e\n", __func__); +} + +/********************************************************************** +Name : atsmb_prep_cmd +Function : +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static inline void atsmb_prep_cmd(struct atsmb_host *host, + u32 opcode, + u32 arg, + unsigned int flags, + u16 blk_len, + u16 blk_cnt, + unsigned char int_maks_0, + unsigned char int_mask_1, + unsigned char cmd_type, + unsigned char op) +{ + DBG("[%s] s\n",__func__); + DBG("[%s] s opcode = %u arg = 0x%x flags = 0x%x\n",__func__,opcode,arg,flags); + /*set cmd operation code and arguments.*/ + host->opcode = opcode; + *ATSMB0_CMD_IDX = opcode; /* host->opcode is set for further use in ISR.*/ + *ATSMB0_CMD_ARG = arg; + +#if 0 /* Fixme to support SPI mode, James Tian*/ + if ((flags && MMC_RSP_NONE) == MMC_RSP_NONE) + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R0; + else if ((flags && MMC_RSP_R1) == MMC_RSP_R1) + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R1; + else if ((flags && MMC_RSP_R1B) == MMC_RSP_R1B) + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R1b; + else if ((flags && MMC_RSP_R2) == MMC_RSP_R2) + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R2; + else if ((flags && MMC_RSP_R3) == MMC_RSP_R3) + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R3; + else if ((flags && MMC_RSP_R6) == MMC_RSP_R6) + *ATSMB0_RSP_TYPE = ((opcode != SD_SEND_IF_COND) ? ATMSB_TYPE_R6 : ATMSB_TYPE_R7); + else + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R0; +#endif + +#if 1 + /*set cmd response type*/ + switch (flags) { + case MMC_RSP_NONE | MMC_CMD_AC: + case MMC_RSP_NONE | MMC_CMD_BC: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R0; + break; + case MMC_RSP_R1 | MMC_CMD_ADTC: + case MMC_RSP_R1 | MMC_CMD_AC: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R1; + break; + case MMC_RSP_R1B | MMC_CMD_AC: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R1b; + break; + case MMC_RSP_R2 | MMC_CMD_BCR: + case MMC_RSP_R2 | MMC_CMD_AC: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R2; + break; + case MMC_RSP_R3 | MMC_CMD_BCR: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R3; + break; + case MMC_RSP_R6 | MMC_CMD_BCR: /*MMC_RSP_R6 = MMC_RSP_R7.*/ + *ATSMB0_RSP_TYPE = ((opcode != SD_SEND_IF_COND) ? + ATMSB_TYPE_R6 : ATMSB_TYPE_R7); + break; + case MMC_RSP_SPI_R5 | MMC_RSP_R5 | MMC_CMD_AC: + case MMC_RSP_SPI_R5 | MMC_RSP_R5 | MMC_CMD_ADTC: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R5; + break; + case MMC_RSP_SPI_R4 | MMC_RSP_R4 | MMC_CMD_BCR: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R4; + break; + default: + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R0; + break; + } +#endif + /*SDIO cmd 52 , 53*/ + if ((opcode == SD_IO_RW_DIRECT) || + (opcode == SD_IO_RW_EXTENDED)) { + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R5; + *ATSMB0_RSP_TYPE |= BIT6; + } + /*SDIO cmd 5*/ + if ((opcode == SD_IO_SEND_OP_COND) && + ((flags & (MMC_RSP_PRESENT| + MMC_RSP_136| + MMC_RSP_CRC| + MMC_RSP_BUSY| + MMC_RSP_OPCODE)) == MMC_RSP_R4)) { + *ATSMB0_RSP_TYPE = ATMSB_TYPE_R4; + *ATSMB0_RSP_TYPE |= BIT6; + } + + /*reset Response FIFO*/ + *ATSMB0_CTL |= 0x08; + + /* SD Host enable Clock */ + *ATSMB0_BUS_MODE |= ATSMB_CST; + + /*Set Cmd-Rsp Timeout to be maximum value.*/ + *ATSMB0_RSP_TOUT = 0xFE; + + /*clear all status registers for safety*/ + *ATSMB0_SD_STS_0 |= 0xff; + *ATSMB0_SD_STS_1 |= 0xff; + *ATSMB0_SD_STS_2 |= 0xff; + //*ATSMB0_SD_STS_2 |= 0x7f; + *ATSMB0_SD_STS_3 |= 0xff; + + //set block length and block count for cmd requesting data + *ATSMB0_BLK_LEN &=~(0x0fff); + *ATSMB0_BLK_LEN |= blk_len; + //*ATSMB0_SHDW_BLKLEN = blk_len; + *ATSMB0_BLK_CNT = blk_cnt; + + + *ATSMB0_INT_MASK_0 |= int_maks_0; + *ATSMB0_INT_MASK_1 |= int_mask_1; + + //Set Auto stop for Multi-block access + if(cmd_type == 3 || cmd_type == 4) { + //auto stop command set. + *ATSMB0_EXT_CTL |= 0x01; + +/* + * Enable transaction abort. + * When CRC error occurs, CMD 12 would be automatically issued. + * That is why we cannot enable R/W CRC error INTs. + * If we enable CRC error INT, we would handle this INT in ISR and then issue CMD 12 via software. + */ +/* Don't auto issue stop command when write command error, +* because auto stop command can freezs PDMA. msut let PDMA transfer done. +* 2013/05/15 by Eason +*/ + if (cmd_type == 4) + *ATSMB0_BLK_LEN |= 0x0800; + } + + /*Set read or write*/ + if (op == 1) + *ATSMB0_CTL &= ~(0x04); + else if (op == 2) + *ATSMB0_CTL |= 0x04; + + /*for Non data access command, command type is 0.*/ + *ATSMB0_CTL &= 0x0F; + *ATSMB0_CTL |= (cmd_type<<4); + DBG("[%s] e\n",__func__); +} + +static inline void atsmb_issue_cmd(void) +{ + *ATSMB0_CTL |= ATSMB_START; + wmb(); +} +/********************************************************************** +Name : atsmb_request_end +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static void +atsmb_request_end(struct atsmb_host *host, struct mmc_request *mrq) +{ + DBG("[%s] s\n",__func__); + /* + * Need to drop the host lock here; mmc_request_done may call + * back into the driver... + */ + //kevin delete spin lock + //spin_unlock(&host->lock); + /*DBG("100");*/ + mmc_request_done(host->mmc, mrq); + /*DBG("101\n");*/ + //kevin delete spin lock + //spin_lock(&host->lock); + DBG("[%s] e\n",__func__); +} + +/********************************************************************** +Name : atsmb_data_done +Function : +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +void atsmb_wait_done(void *data) +{ + struct atsmb_host *host = (struct atsmb_host *) data; + DBG("[%s] s\n",__func__); + + WARN_ON(host->done_data == NULL); + complete(host->done_data); + host->done_data = NULL; + host->done = NULL; + DBG("[%s] e\n",__func__); +} + +/********************************************************************** +Name : atsmb_start_data +Function : If we start data, there must be only four cases. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static void atsmb_start_data(struct atsmb_host *host) +{ + DECLARE_COMPLETION(complete); + unsigned char cmd_type = 0; + unsigned char op = 0; /*0: non-operation; 1:read; 2: write*/ + unsigned char mask_0 = 0; + unsigned char mask_1 = 0; + unsigned long dma_mask = 0; + + struct mmc_data *data = host->data; + struct mmc_command *cmd = host->cmd; + + struct scatterlist *sg = NULL; + unsigned int sg_len = 0; + + unsigned int total_blks = 0; /*total block number to transfer*/ + u32 card_addr = 0; + unsigned long dma_len = 0; + unsigned long total_dma_len = 0; + dma_addr_t dma_phy = 0; /* physical address used for DMA*/ + unsigned int dma_times = 0; /*times dma need to transfer*/ + unsigned int dma_loop = 0; + unsigned int sg_num = 0; + int loop_cnt = 10000; + unsigned int sg_transfer_len = 0; /*record each time dma transfer sg length */ + + DBG("[%s] s\n",__func__); + data->bytes_xfered = 0; + cmd->error = 0; + data->error = 0; + + /*for loop*/ + sg = data->sg; + sg_len = data->sg_len; + + dma_times = (sg_len / MAX_DESC_NUM); + if (sg_len % MAX_DESC_NUM) + dma_times++; + DBG("dma_times = %d sg_len = %d sg = %x\n", dma_times, sg_len, sg); + card_addr = cmd->arg; /*may be it is block-addressed, or byte-addressed.*/ + total_blks = data->blocks; + dma_map_sg(&(host->mmc->class_dev), sg, sg_len, + ((data->flags)&MMC_DATA_READ) ? DMA_FROM_DEVICE : DMA_TO_DEVICE); + + + for (dma_loop = 1 ; dma_loop <= dma_times; dma_loop++, sg_len -= sg_transfer_len) { + DBG("dma_loop = %d sg_len = %d sg_transfer_len = %d\n", dma_loop, sg_len, sg_transfer_len); + if (dma_loop < dma_times) + sg_transfer_len = MAX_DESC_NUM; + else + sg_transfer_len = sg_len; + DBG("sg_transfer_len = %d\n", sg_transfer_len); + /* + *Firstly, check and wait till card is in the transfer state. + *For our hardware, we can not consider + *the card has successfully tranfered its state from data/rcv to trans, + *when auto stop INT occurs. + */ + loop_cnt = 10000; + do { + if (host->cmd->opcode == SD_IO_RW_EXTENDED) + break; + loop_cnt--; + WARN_ON(loop_cnt == 0); + host->done_data = &complete; + host->done = &atsmb_wait_done; + host->soft_timeout = 1; + atsmb_prep_cmd(host, + MMC_SEND_STATUS, + (host->mmc->card->rca)<<16, + MMC_RSP_R1 | MMC_CMD_AC, + 0, + 0, + 0, /*mask_0*/ + ATSMB_RSP_DONE_EN + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN, + 0, /*cmd type*/ + 0); /*read or write*/ + atsmb_issue_cmd(); + DBG("%16s = 0x%8x |", "INT_MASK_1", *ATSMB0_INT_MASK_1); + DBG("%16s = 0x%8x \n", "SD_STS_1", *ATSMB0_SD_STS_1); + /*ISR would completes it.*/ + wait_for_completion_timeout(&complete, ATSMB_TIMEOUT_TIME); + + WARN_ON(host->soft_timeout == 1); + if (host->soft_timeout == 1) { + DBG("%s soft_timeout.\n", __func__); + atsmb_dump_reg(); + } + if (cmd->error != MMC_ERR_NONE) { + cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("Getting Status failed.\n"); + goto end; + } + } while ((cmd->resp[0] & 0x1f00) != 0x900 && loop_cnt > 0); /*wait for trans state.*/ + + /* + * Now, we can safely issue read/write command. + * We can not consider this request as multi-block acess or single one via opcode, + * as request is splitted into sgs. + * Some sgs may be single one, some sgs may be multi one. + */ + + dma_phy = sg_dma_address(sg); + dma_len = sg_dma_len(sg); + DBG("dma_len = %d data->blksz = %d sg_len = %d\n", dma_len, data->blksz, sg_len); + /*SDIO read/write*/ + if (host->cmd->opcode == SD_IO_RW_EXTENDED) { + /*Single Block read/write*/ + if ((dma_len / (data->blksz)) == 1 && (sg_len == 1)) { + /* read operation*/ + if (data->flags & MMC_DATA_READ) { + host->opcode = SD_IO_RW_EXTENDED; + cmd_type = 2; + op = 1; + mask_0 = 0; /*BLOCK_XFER_DONE INT skipped, we use DMA TC INT*/ + mask_1 = (//ATSMB_SDIO_EN + ATSMB_READ_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = SD_PDMA_CCR_Evt_success; + DBG("[%s]SR opcode = %d type = 0x%x op = 0x%x m_0 = 0x%x m_1 = 0x%x d_m = 0x%x\n", + __func__,host->opcode,cmd_type,op,mask_0,mask_1,dma_mask); + } else { + /* write operation*/ + host->opcode = SD_IO_RW_EXTENDED; + cmd_type = 1; + op = 2; + /*====That is what we want===== DMA TC INT skipped*/ + mask_0 = ATSMB_BLOCK_XFER_DONE_EN; + mask_1 = (//ATSMB_SDIO_EN + ATSMB_WRITE_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = 0; + DBG("[%s]SW opcode = %d type = 0x%x op = 0x%x m_0 = 0x%x m_1 = 0x%x d_m = 0x%x\n", + __func__,host->opcode,cmd_type,op,mask_0,mask_1,dma_mask); + } + } else { + /*Multiple Block read/write*/ + /* read operation*/ + if (data->flags & MMC_DATA_READ) { + host->opcode = SD_IO_RW_EXTENDED; + cmd_type = 6; + op = 1; + mask_0 = 0; /*MULTI_XFER_DONE_EN skipped*/ + mask_1 = (//ATSMB_SDIO_EN /*====That is what we want=====*/ + ATSMB_READ_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = SD_PDMA_CCR_Evt_success; + DBG("[%s]MR opcode = %d type = 0x%x op = 0x%x m_0 = 0x%x m_1 = 0x%x d_m = 0x%x\n", + __func__,host->opcode,cmd_type,op,mask_0,mask_1,dma_mask); + } else { + /* write operation*/ + host->opcode = SD_IO_RW_EXTENDED; + cmd_type = 5; + op = 2; + mask_0 = ATSMB_MULTI_XFER_DONE_EN;//ATSMB_BLOCK_XFER_DONE_EN; /*MULTI_XFER_DONE INT skipped*/ + mask_1 = (//ATSMB_SDIO_EN /*====That is what we want=====*/ + ATSMB_WRITE_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = 0; + DBG("[%s]MR opcode = %d type = 0x%x op = 0x%x m_0 = 0x%x m_1 = 0x%x d_m = 0x%x\n", + __func__,host->opcode,cmd_type,op,mask_0,mask_1,dma_mask); + } + } + + } else { + if ((dma_len / (data->blksz)) == 1 && (sg_len == 1)) { + if (data->flags & MMC_DATA_READ) {/* read operation*/ + host->opcode = MMC_READ_SINGLE_BLOCK; + cmd_type = 2; + op = 1; + mask_0 = 0; /*BLOCK_XFER_DONE INT skipped, we use DMA TC INT*/ + mask_1 = (ATSMB_READ_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = SD_PDMA_CCR_Evt_success; + } else {/*write operation*/ + host->opcode = MMC_WRITE_BLOCK; + cmd_type = 1; + op = 2; + /*====That is what we want===== DMA TC INT skipped*/ + mask_0 = ATSMB_BLOCK_XFER_DONE_EN; + mask_1 = (ATSMB_WRITE_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = 0; + } + } else { /*more than one*/ + if (data->flags&MMC_DATA_READ) {/* read operation*/ + host->opcode = MMC_READ_MULTIPLE_BLOCK; + cmd_type = 4; + op = 1; + mask_0 = 0; /*MULTI_XFER_DONE_EN skipped*/ + mask_1 = (ATSMB_AUTO_STOP_EN /*====That is what we want=====*/ + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = 0; + } else {/*write operation*/ + host->opcode = MMC_WRITE_MULTIPLE_BLOCK; + cmd_type = 3; + op = 2; + mask_0 = 0; /*MULTI_XFER_DONE INT skipped*/ + mask_1 = (ATSMB_AUTO_STOP_EN /*====That is what we want=====*/ + |ATSMB_DATA_TIMEOUT_EN + /*Data Timeout and CRC error */ + |ATSMB_RSP_CRC_ERR_EN + |ATSMB_RSP_TIMEOUT_EN); + /*Resp Timeout and CRC error */ + dma_mask = 0; + } + } + } + /*To controller every sg done*/ + host->done_data = &complete; + host->done = &atsmb_wait_done; + /*sleep till ISR wakes us*/ + host->soft_timeout = 1; /*If INT comes early than software timer, it would be cleared.*/ + + total_dma_len = 0; + DBG("host->DescVirAddr = %x host->DescSize=%x\n", host->DescVirAddr, host->DescSize); + memset(host->DescVirAddr, 0, host->DescSize); + for (sg_num = 0 ; sg_num < sg_transfer_len ; sg++, sg_num++) { + + /* + * Now, we can safely issue read/write command. + * We can not consider this request as multi-block acess or single one via opcode, + * as request is splitted into sgs. + * Some sgs may be single one, some sgs may be multi one. + */ + + dma_phy = sg_dma_address(sg); + dma_len = sg_dma_len(sg); + total_dma_len = total_dma_len + dma_len; + DBG("sg_num=%d sg_transfer_len=%d sg=%x sg_len=%d total_dma_len=%d dma_len=%d\n", + sg_num, sg_transfer_len, sg, sg_len, total_dma_len, dma_len); + /*2009/01/15 janshiue add*/ + if (sg_num < sg_transfer_len - 1) { + /* means the last descporitor */ + atsmb_init_short_desc( + host->DescVirAddr + (sg_num * sizeof(struct SD_PDMA_DESC_S)/4), + dma_len, (unsigned long *)dma_phy, 0); + } else { + atsmb_init_short_desc( + host->DescVirAddr + (sg_num * sizeof(struct SD_PDMA_DESC_S)/4), + dma_len, (unsigned long *)dma_phy, 1); + } + /*2009/01/15 janshiue add*/ + + } + /*operate our hardware*/ + atsmb_prep_cmd(host, + host->opcode, + /*arg, may be byte addressed, may be block addressed.*/ + card_addr, + cmd->flags, + data->blksz - 1, /* in fact, it is useless.*/ + /* for single one, it is useless. but for multi one, */ + /* it would be used to tell auto stop function whether it is done.*/ + total_dma_len/(data->blksz), + mask_0, + mask_1, + cmd_type, + op); + + atsmb_config_dma((op == 1) ? DMA_CFG_READ : DMA_CFG_WRITE, + dma_mask, + host); + + atsmb_issue_cmd(); + wait_for_completion_timeout(&complete, + ATSMB_TIMEOUT_TIME*sg_transfer_len); /*ISR would completes it.*/ + + /* When the address of request plus length equal card bound, + * force this stop command response as pass. Eason 2012/4/20 */ + if (cmd->resp[0] == 0x80000b00) { + /*This caes used for SD2.0 and after MMC4.1 version*/ + if (card_addr+(total_dma_len/data->blksz) == host->mmc->card->csd.capacity) { + cmd->resp[0] = 0x00000b00; + /*printk("card_addr = %08X, card_length = %08X,card capacity = %08X\n", + card_addr,(total_dma_len/data->blksz),host->mmc->card->csd.capacity); + printk("card_resp[0]=%08X, addr = %08X\n",cmd->resp[0],cmd->resp);*/ + } + + /* This caes used for SD1.0 and before MMC 4.1 */ + if ((card_addr/data->blksz)+(total_dma_len/data->blksz) == host->mmc->card->csd.capacity) { + cmd->resp[0] = 0x00000b00; + /*printk("Eason test: cmd->arg = %08X, data->blksz = %08X, length = %08X\n", + card_addr,data->blksz,(total_dma_len/data->blksz));*/ + } + } + + if (host->soft_timeout == 1) { + atsmb_dump_reg(); + //*ATSMB0_BUS_MODE |= ATSMB_SFTRST; + *ATSMB0_PDMA_GCR = SD_PDMA_GCR_SOFTRESET; + /*disable INT */ + *ATSMB0_INT_MASK_0 &= ~(ATSMB_BLOCK_XFER_DONE_EN | ATSMB_MULTI_XFER_DONE_EN); + *ATSMB0_INT_MASK_1 &= ~(ATSMB_WRITE_CRC_ERR_EN|ATSMB_READ_CRC_ERR_EN|ATSMB_RSP_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN|ATSMB_AUTO_STOP_EN|ATSMB_RSP_TIMEOUT_EN|ATSMB_RSP_DONE_EN); + + cmd->error = -ETIMEDOUT; + data->error = -ETIMEDOUT; + } + + WARN_ON(host->soft_timeout == 1); + + /*check everything goes okay or not*/ + if (cmd->error != MMC_ERR_NONE + || data->error != MMC_ERR_NONE) { + DBG("CMD or Data failed error=%X DescVirAddr=%8X dma_phy=%8X dma_mask = %x\n", + cmd->error, host->DescVirAddr, dma_phy, host->DmaIntMask); + goto end; + } +// card_addr += total_dma_len>>(mmc_card_blockaddr(host->mmc->card_selected) ? 9 : 0); + card_addr += total_dma_len>>(mmc_card_blockaddr(host->mmc->card) ? 9 : 0); //zhf: modified by James Tian + data->bytes_xfered += total_dma_len; + + + } +// dma_unmap_sg(&(host->mmc->class_dev), data->sg, data->sg_len, +// ((data->flags)&MMC_DATA_READ) ? DMA_FROM_DEVICE : DMA_TO_DEVICE); + + WARN_ON(total_blks != (data->bytes_xfered / data->blksz)); + host->opcode = 0; +end: + dma_unmap_sg(&(host->mmc->class_dev), data->sg, data->sg_len, + ((data->flags)&MMC_DATA_READ) ? DMA_FROM_DEVICE : DMA_TO_DEVICE); + spin_lock(&host->lock); + atsmb_request_end(host, host->mrq); + spin_unlock(&host->lock); + DBG("[%s] e\n",__func__); +} + + +/********************************************************************** +Name : atsmb_cmd_with_data_back +Function : +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static void atsmb_cmd_with_data_back(struct atsmb_host *host) +{ + DECLARE_COMPLETION(complete); + + struct scatterlist *sg = NULL; + unsigned int sg_len = 0; + DBG("[%s] s\n",__func__); + /*for loop*/ + sg = host->data->sg; + sg_len = host->data->sg_len; + /*To controller every sg done*/ + host->done_data = &complete; + host->done = &atsmb_wait_done; + dma_map_sg(&(host->mmc->class_dev), sg, sg_len, DMA_FROM_DEVICE); + + /*2009/01/15 janshiue add*/ + memset(host->DescVirAddr, 0, host->DescSize); + atsmb_init_long_desc(host->DescVirAddr, sg_dma_len(sg), (unsigned long *)sg_dma_address(sg), 0, 1); + /*2009/01/15 janshiue add*/ + /*prepare for cmd*/ + atsmb_prep_cmd(host, /*host*/ + host->cmd->opcode, /*opcode*/ + host->cmd->arg, /*arg*/ + host->cmd->flags, /*flags*/ + sg_dma_len(sg)-1, /*block length*/ + 0, /*block size: It looks like useless*/ + 0, /*int_mask_0*/ + (ATSMB_RSP_CRC_ERR_EN|ATSMB_RSP_TIMEOUT_EN + |ATSMB_READ_CRC_ERR_EN |ATSMB_DATA_TIMEOUT_EN), /*int_mask_1*/ + 2, /*cmd_type*/ + 1); /*op*/ + + atsmb_config_dma(DMA_CFG_READ, + SD_PDMA_CCR_Evt_success, + host); + atsmb_issue_cmd(); + /*ISR would completes it.*/ + wait_for_completion_timeout(&complete, ATSMB_TIMEOUT_TIME); + + dma_unmap_sg(&(host->mmc->class_dev), host->data->sg, host->data->sg_len, + ((host->data->flags)&MMC_DATA_READ) ? DMA_FROM_DEVICE : DMA_TO_DEVICE); + spin_lock(&host->lock); + atsmb_request_end(host, host->mrq); + spin_unlock(&host->lock); + DBG("[%s] e\n",__func__); +} +/********************************************************************** +Name : atsmb_start_cmd +Function : +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static void atsmb_start_cmd(struct atsmb_host *host) +{ + unsigned char int_mask_0,int_mask_1; + int_mask_0 = 0; + int_mask_1 = ATSMB_RSP_DONE_EN|ATSMB_RSP_CRC_ERR_EN|ATSMB_RSP_TIMEOUT_EN; + + DBG("[%s] s\n",__func__); + + atsmb_prep_cmd(host, + host->cmd->opcode, + host->cmd->arg, + host->cmd->flags, + 0, /*useless*/ + 0, /*useless*/ + int_mask_0, /*mask_0*/ + int_mask_1, /*mask_1*/ + 0, /*cmd type*/ + 0); /*read or write*/ + atsmb_issue_cmd(); + DBG("[%s] e\n",__func__); +} +/********************************************************************** +Name : atsmb_fmt_check_rsp +Function : +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static inline void atsmb_fmt_check_rsp(struct atsmb_host *host) +{ + + u8 tmp_resp[4] = {0}; + int i, j, k; + DBG("[%s] s\n",__func__); + if (host->cmd->flags != (MMC_RSP_R2 | MMC_CMD_AC) + && host->cmd->flags != (MMC_RSP_R2 | MMC_CMD_BCR)) { + for (j = 0, k = 1; j < 4; j++, k++) + tmp_resp[j] = *(REG8_PTR(_RSP_0 + MMC_ATSMB0_BASE+k)); + + host->cmd->resp[0] = (tmp_resp[0] << 24) | + (tmp_resp[1]<<16) | (tmp_resp[2]<<8) | (tmp_resp[3]); + } else { + for (i = 0, k = 1; i < 4; i++) { /*R2 has 4 u32 response.*/ + for (j = 0; j < 4; j++) { + if (k < 16) + tmp_resp[j] = *(REG8_PTR(_RSP_0 + MMC_ATSMB0_BASE+k)); + else /* k =16*/ + tmp_resp[j] = *(ATSMB0_RSP_0); + k++; + } + host->cmd->resp[i] = (tmp_resp[0]<<24) | (tmp_resp[1]<<16) | + (tmp_resp[2]<<8) | (tmp_resp[3]); + } + } + + /* + * For the situation that we need response, + * but response registers give us all zeros, we consider this operation timeout. + */ + if (host->cmd->flags != (MMC_RSP_NONE | MMC_CMD_AC) + && host->cmd->flags != (MMC_RSP_NONE | MMC_CMD_BC)) { + if (host->cmd->resp[0] == 0 && host->cmd->resp[1] == 0 + && host->cmd->resp[2] == 0 && host->cmd->resp[3] == 0) { + host->cmd->error = -ETIMEDOUT; //zhf: modified by James Tian + } + } + DBG("[%s] e\n",__func__); +} +/********************************************************************** +Name : atsmb_get_slot_status +Function : Our host only supports one slot. +Calls : +Called by : +Parameter : +returns : 1: in slot; 0: not in slot. +Author : Leo Lee +History : +***********************************************************************/ +static int atsmb_get_slot_status(struct mmc_host *mmc) +{ +// struct atsmb_host *host = mmc_priv(mmc); + unsigned char status_0 = 0; +// unsigned long flags; + unsigned long ret = 0; + DBG("[%s] s\n",__func__); +// spin_lock_irqsave(&host->lock, flags); // Vincent Li mark out for CONFIG_PREEMPT_RT + status_0 = *ATSMB0_SD_STS_0; +// spin_unlock_irqrestore(&host->lock, flags); // Vincent Li mark out for CONFIG_PREEMPT_RT + /* after WM3400 A1 ATSMB_CARD_IN_SLOT_GPI = 1 means not in slot*/ + if (MMC0_DRIVER_VERSION >= MMC_DRV_3426_A0) { + if(SD0_detect_pol) + ret = ((status_0 & ATSMB_CARD_NOT_IN_SLOT_GPI) ? 1 : 0); + else + ret = ((status_0 & ATSMB_CARD_NOT_IN_SLOT_GPI) ? 0 : 1); + + DBG("[%s] e1\n",__func__); + return ret; + } else { + ret = ((status_0 & ATSMB_CARD_NOT_IN_SLOT_GPI) ? 1 : 0); + DBG("[%s] e2\n",__func__); + return ret; + } + DBG("[%s] e3\n",__func__); + return 0; +} + +/********************************************************************** +Name : atsmb_init_short_desc +Function : +Calls : +Called by : +Parameter : +Author : Janshiue Wu +History : +***********************************************************************/ +int atsmb_init_short_desc(unsigned long *DescAddr, unsigned int ReqCount, unsigned long *BufferAddr, int End) +{ + struct SD_PDMA_DESC_S *CurDes_S; + DBG("[%s] s\n",__func__); + CurDes_S = (struct SD_PDMA_DESC_S *) DescAddr; + CurDes_S->ReqCount = ReqCount; + CurDes_S->i = 0; + CurDes_S->format = 0; + CurDes_S->DataBufferAddr = BufferAddr; + if (End) { + CurDes_S->end = 1; + CurDes_S->i = 1; + } + DBG("[%s] e\n",__func__); + return 0; +} + +/********************************************************************** +Name : atsmb_init_long_desc +Function : +Calls : +Called by : +Parameter : +Author : Janshiue Wu +History : +***********************************************************************/ +int atsmb_init_long_desc(unsigned long *DescAddr, unsigned int ReqCount, + unsigned long *BufferAddr, unsigned long *BranchAddr, int End) +{ + struct SD_PDMA_DESC_L *CurDes_L; + DBG("[%s] s\n",__func__); + CurDes_L = (struct SD_PDMA_DESC_L *) DescAddr; + CurDes_L->ReqCount = ReqCount; + CurDes_L->i = 0; + CurDes_L->format = 1; + CurDes_L->DataBufferAddr = BufferAddr; + CurDes_L->BranchAddr = BranchAddr; + if (End) { + CurDes_L->end = 1; + CurDes_L->i = 1; + } + DBG("[%s] e\n",__func__); + return 0; +} + +/********************************************************************** +Name : atsmb_dma_isr +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static irqreturn_t atsmb_dma_isr(int irq, void *dev_id) +{ + + struct atsmb_host *host = dev_id; + u8 status_0, status_1, status_2, status_3; + u32 pdma_event_code = 0; + DBG("[%s] s\n",__func__); + + disable_irq_nosync(irq); + spin_lock(&host->lock); + /*Get INT status*/ + status_0 = *ATSMB0_SD_STS_0; + status_1 = *ATSMB0_SD_STS_1; + status_2 = *ATSMB0_SD_STS_2; + status_3 = *ATSMB0_SD_STS_3; + /* after WM3426 A0 using PDMA */ + if (MMC0_DRIVER_VERSION >= MMC_DRV_3426_A0) { + + pdma_event_code = *ATSMB0_PDMA_CCR & SD_PDMA_CCR_EvtCode; + + /* clear INT status to notify HW clear EventCode*/ + *ATSMB0_PDMA_ISR |= SD_PDMA_IER_INT_STS; + + /*printk("dma_isr event code = %X\n", *ATSMB0_PDMA_CCR);*/ + /*We expect cmd with data sending back or read single block cmd run here.*/ + if (pdma_event_code == SD_PDMA_CCR_Evt_success) { + /*means need to update the data->error and cmd->error*/ + if (host->DmaIntMask == SD_PDMA_CCR_Evt_success) { + if (host->data != NULL) { + host->data->error = MMC_ERR_NONE; + host->cmd->error = MMC_ERR_NONE; + } else { + DBG("dma_isr host->data is NULL\n"); + /*disable INT*/ + *ATSMB0_PDMA_IER &= ~SD_PDMA_IER_INT_EN; + goto out; + } + } + /*else do nothing*/ + DBG("dma_isr PDMA OK\n"); + /*atsmb_dump_reg();*/ + } + /*But unluckily, we should also be prepare for all kinds of error situation.*/ + else { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("** dma_isr PDMA fail** event code = %X\n", *ATSMB0_PDMA_CCR); + atsmb_dump_reg(); + } + + if (host->DmaIntMask == SD_PDMA_CCR_Evt_success) + atsmb_fmt_check_rsp(host); + + /*disable INT*/ + *ATSMB0_PDMA_IER &= ~SD_PDMA_IER_INT_EN; + } + + /*wake up some one who is sleeping.*/ + if ((pdma_event_code != SD_PDMA_CCR_Evt_success) || (host->DmaIntMask == SD_PDMA_CCR_Evt_success)) { + if (host->done_data) {/* We only use done_data when requesting data.*/ + host->soft_timeout = 0; + host->done(host); + } else + atsmb_request_end(host, host->mrq); /*for cmd with data sending back.*/ + } + +out: + spin_unlock(&host->lock); + enable_irq(irq); + DBG("[%s] e\n",__func__); + return IRQ_HANDLED; +} + +/********************************************************************** +Name : atsmb_regular_isr +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +//static irqreturn_t atsmb_regular_isr(int irq, void *dev_id, struct pt_regs *regs) +irqreturn_t atsmb_regular_isr(int irq, void *dev_id) +{ + + struct atsmb_host *host = dev_id; + u8 status_0, status_1, status_2, status_3,mask_0,mask_1; + u32 pdma_sts; + + DBG("[%s] s\n",__func__); + WARN_ON(host == NULL); + + disable_irq_nosync(irq); + spin_lock(&host->lock); + + /*Get INT status*/ + status_0 = *ATSMB0_SD_STS_0; + status_1 = *ATSMB0_SD_STS_1; + status_2 = *ATSMB0_SD_STS_2; + status_3 = *ATSMB0_SD_STS_3; + + mask_0 = *ATSMB0_INT_MASK_0; + mask_1 = *ATSMB0_INT_MASK_1; + /******************************************************* + card insert interrupt + ********************************************************/ + if ((status_0 & ATSMB_DEVICE_INSERT) /*Status Set and IRQ enabled*/ + /*To aviod the situation that we intentionally disable IRQ to do rescan.*/ + && (*ATSMB0_INT_MASK_0 & 0x80)) { + + if (host->mmc->ops->get_slot_status(host->mmc)) { + host->mmc->scan_retry = 3; + host->mmc->card_scan_status = false; + } else { + host->mmc->scan_retry = 0; + host->mmc->card_scan_status = false; + } + + mmc_detect_change(host->mmc, HZ/2); + /*Taipei Side Request: Disable INSERT IRQ when doing rescan.*/ + //*ATSMB0_INT_MASK_0 &= (~0x80);/* or 40?*/ //zhf: marked by James Tian + /*Then we clear the INT status*/ + //iowrite32(ATSMB_DEVICE_INSERT, host->base+_SD_STS_0); + *ATSMB0_SD_STS_0 |= ATSMB_DEVICE_INSERT; + spin_unlock(&host->lock); + enable_irq(irq); + DBG("[%s] e1\n",__func__); + return IRQ_HANDLED; + } + + if ((status_1 & mask_1)& ATSMB_SDIO_INT) { + spin_unlock(&host->lock); + mmc_signal_sdio_irq(host->mmc); + + if (((status_1 & mask_1) == ATSMB_SDIO_INT) && ((status_0 & mask_0) == 0)) { + + enable_irq(irq); + DBG("[%s] e2\n",__func__); + + return IRQ_HANDLED; + } + spin_lock(&host->lock); + } + pdma_sts = *ATSMB0_PDMA_CCR; + + if (((status_0 & mask_0) | (status_1 & mask_1)) == 0) { + spin_unlock(&host->lock); + enable_irq(irq); + DBG("[%s] e3\n",__func__); + return IRQ_HANDLED; + } + /******************************************************* + command interrupt. + *******************************************************/ + /*for write single block*/ + if (host->opcode == MMC_WRITE_BLOCK) { + if ((status_0 & ATSMB_BLOCK_XFER_DONE) + && (status_1 & ATSMB_RSP_DONE)) { + host->data->error = MMC_ERR_NONE; + host->cmd->error = MMC_ERR_NONE; + /* okay, what we want.*/ + } else { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err1\n",__func__); + atsmb_dump_reg(); + } + } else if (host->opcode == MMC_WRITE_MULTIPLE_BLOCK + || host->opcode == MMC_READ_MULTIPLE_BLOCK) { + if ((status_1 & (ATSMB_AUTO_STOP|ATSMB_RSP_DONE)) + && (status_0 & ATSMB_MULTI_XFER_DONE)) { + /*If CRC error occurs, I think this INT would not occrs.*/ + /*okay, that is what we want.*/ + host->data->error = MMC_ERR_NONE; + host->cmd->error = MMC_ERR_NONE; + } else { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err2\n",__func__); + atsmb_dump_reg(); + + } + } else if (host->opcode == MMC_READ_SINGLE_BLOCK) {/* we want DMA TC, run here, must be error.*/ + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err3\n",__func__); + atsmb_dump_reg(); + } else if (host->opcode == SD_IO_RW_EXTENDED){ + /*Write operation*/ + if (*ATSMB0_CTL & BIT2) { + if ((*ATSMB0_CTL & 0xf0) == 0x10) { /*single block write*/ + if ((status_0 & ATSMB_BLOCK_XFER_DONE) + && (status_1 & ATSMB_RSP_DONE)) { + host->data->error = MMC_ERR_NONE; + host->cmd->error = MMC_ERR_NONE; + /* okay, what we want.*/ + } else { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err4 status_0 = %x status_1 = %x\n",__func__,status_0,status_1); + } + + } else if ((*ATSMB0_CTL & 0xf0) == 0x50) { + if ((status_0 & ATSMB_MULTI_XFER_DONE) + && (status_1 & ATSMB_RSP_DONE)) { + host->data->error = MMC_ERR_NONE; + host->cmd->error = MMC_ERR_NONE; + /* okay, what we want.*/ + } else { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err4-2 status_0 = %x status_1 = %x\n",__func__,status_0,status_1); + } + } else { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err4-3 status_0 = %x status_1 = %x\n",__func__,status_0,status_1); + } + } + else { + /*Read operation*/ + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err5\n",__func__); + } + + + } else { +/* command, not request data*/ +/* the command which need data sending back,*/ +/* like switch_function, send_ext_csd, send_scr, send_num_wr_blocks.*/ +/* NOTICE: we also send status before reading or writing data, so SEND_STATUS should be excluded.*/ + if (host->data && host->opcode != MMC_SEND_STATUS) { + host->data->error = -EIO; //MMC_ERR_FAILED; + host->cmd->error = -EIO; //MMC_ERR_FAILED; + DBG("[%s] err6\n",__func__); + atsmb_dump_reg(); + } else { /* Just command, no need data sending back.*/ + if (status_1 & ATSMB_RSP_DONE) { + /*Firstly, check data-response is busy or not.*/ + if (host->cmd->flags == (MMC_RSP_R1B | MMC_CMD_AC)) { + int i = 10000; + + while (status_2 & ATSMB_RSP_BUSY) { + status_2 = *ATSMB0_SD_STS_2; + if (--i == 0) + break; + DBG(" IRQ:Status_2 = %d, busy!\n", status_2); + } + if (i == 0) + printk("[MMC driver] Error :SD data-response always busy!"); + } +#if 1 +/*for our host, even no card in slot, for SEND_STATUS also returns no error.*/ +/*The protocol layer depends on SEND_STATUS to check whether card is in slot or not.*/ +/*In fact, we can also avoid this situation by checking the response whether they are all zeros.*/ + if (!atsmb_get_slot_status(host->mmc) && host->opcode == MMC_SEND_STATUS) { + host->cmd->retries = 0; /* No retry.*/ +// host->cmd->error = MMC_ERR_INVALID; + host->cmd->error = -EINVAL; + } else +#endif + host->cmd->error = MMC_ERR_NONE; + } else { + if (status_1 & ATSMB_RSP_TIMEOUT) {/* RSP_Timeout .*/ +// host->cmd->error = MMC_ERR_TIMEOUT; + host->cmd->error = -ETIMEDOUT; + DBG("[%s] err7\n",__func__); + } else {/*or RSP CRC error*/ +// host->cmd->error = MMC_ERR_BADCRC; + host->cmd->error = -EILSEQ; + DBG("[%s] err8\n",__func__); + } + atsmb_dump_reg(); + } + } + } + atsmb_fmt_check_rsp(host); + + /*disable INT */ + *ATSMB0_INT_MASK_0 &= ~(ATSMB_BLOCK_XFER_DONE_EN | ATSMB_MULTI_XFER_DONE_EN); + *ATSMB0_INT_MASK_1 &= ~(ATSMB_WRITE_CRC_ERR_EN|ATSMB_READ_CRC_ERR_EN|ATSMB_RSP_CRC_ERR_EN + |ATSMB_DATA_TIMEOUT_EN|ATSMB_AUTO_STOP_EN|ATSMB_RSP_TIMEOUT_EN|ATSMB_RSP_DONE_EN); + + + /*clear INT status. In fact, we will clear again before issuing new command.*/ + *ATSMB0_SD_STS_0 |= status_0; + *ATSMB0_SD_STS_1 |= status_1; + + /* when read CRC error occur, and the status can't write one to clear. + * To clear read CRC error status , can do software reset. This is HW bug. 2013/3/21*/ + if ((*ATSMB0_SD_STS_1 & BIT6) == 0x40) { + DBG("[%s] host0 CMD%d Read CRC error occur\n",__func__,host->cmd->opcode); + /* Save SD card register */ + atsmb_copy_reg(0); + /* Software reset */ + *ATSMB0_BUS_MODE |= BIT7; + /* restore SD card register */ + atsmb_copy_reg(1); + } + + if (*ATSMB0_PDMA_ISR & SD_PDMA_IER_INT_STS) + *ATSMB0_PDMA_ISR |= SD_PDMA_IER_INT_STS; + + /*wake up some one who is sleeping.*/ + if (host->done_data) { /* We only use done_data when requesting data.*/ + host->soft_timeout = 0; + host->done(host); + } else + atsmb_request_end(host, host->mrq); /*for cmd without data.*/ + + spin_unlock(&host->lock); + enable_irq(irq); + DBG("[%s] e4\n",__func__); + return IRQ_HANDLED; +} +EXPORT_SYMBOL(atsmb_regular_isr); + +/********************************************************************** +Name : atsmb_get_ro +Function :. +Calls : +Called by : +Parameter : +returns : 0 : write protection is disabled. 1: write protection is enabled. +Author : Leo Lee +History : +***********************************************************************/ +int atsmb_get_ro(struct mmc_host *mmc) +{ + struct atsmb_host *host = mmc_priv(mmc); + unsigned char status_0 = 0; + unsigned long flags; + unsigned long ret = 0; + DBG("[%s] s\n",__func__); + //add by kevin guan + if(SD0_ro_disable) + return 0; + spin_lock_irqsave(&host->lock, flags); + status_0 = *ATSMB0_SD_STS_0; + spin_unlock_irqrestore(&host->lock, flags); + DBG("[%s]\nstatus_0 = 0x%x\n", __func__,status_0); + ret = ((status_0 & ATSMB_WRITE_PROTECT) ? 0 : 1); + DBG("[%s] e\n",__func__); + return ret; +} +/********************************************************************** +Name : atsmb_dump_host_regs +Function : +Calls : +Called by : +Parameter : +returns : +Author : Leo Lee +History : +***********************************************************************/ +void atsmb_dump_host_regs(struct mmc_host *mmc) +{ + DBG("[%s] s\n",__func__); + atsmb_dump_reg(); + DBG("[%s] e\n",__func__); +} +EXPORT_SYMBOL(atsmb_dump_host_regs); + +/********************************************************************** +Name : atsmb_enable_sdio_irq +Function : +Calls : +Called by : +Parameter : +returns : +Author : Tommy Huang +History : +***********************************************************************/ +static void atsmb_enable_sdio_irq(struct mmc_host *mmc, int enable) +{ + struct atsmb_host *host = mmc_priv(mmc); + unsigned long flags; + + DBG("[%s] s enable = %d *ATSMB0_INT_MASK_1 = %x\n",__func__,enable,*ATSMB0_INT_MASK_1); + spin_lock_irqsave(&host->lock, flags); + + if (enable) { + *ATSMB0_INT_MASK_1 |= ATSMB_SDIO_EN; + } else { + *ATSMB0_INT_MASK_1 &= ~ATSMB_SDIO_EN; + } + + spin_unlock_irqrestore(&host->lock, flags); + DBG("[%s] e\n",__func__); + +} +/********************************************************************** +Name : atsmb_request +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static void atsmb_request(struct mmc_host *mmc, struct mmc_request *mrq) +{ + + struct atsmb_host *host = mmc_priv(mmc); + DBG("[%s] s\n",__func__); + + /* May retry process comes here.*/ + host->mrq = mrq; + host->data = mrq->data; + host->cmd = mrq->cmd; + host->done_data = NULL; + host->done = NULL; + + /*for data request*/ + if (host->data) { + if (host->cmd->opcode == MMC_WRITE_BLOCK + || host->cmd->opcode == MMC_WRITE_MULTIPLE_BLOCK + || host->cmd->opcode == MMC_READ_SINGLE_BLOCK + || host->cmd->opcode == MMC_READ_MULTIPLE_BLOCK + || host->cmd->opcode == SD_IO_RW_EXTENDED) { + atsmb_start_data(host); + } else { + atsmb_cmd_with_data_back(host); + } + } else { + atsmb_start_cmd(host); + } + DBG("[%s] e\n",__func__); +} +/********************************************************************** +Name : atsmb_set_clock +Function :. +Calls : +Called by : +Parameter : +Author : Eason Chien +History :2012/7/19 +***********************************************************************/ +static int atsmb_set_clock(struct mmc_host *mmc, unsigned int clock) +{ + int clock_multiplier = 1; + DBG("clock = %u\n",clock); + + if (*ATSMB0_EXT_BUS_MODE & BIT4) /*Enable DDR50*/ + clock_multiplier = 2; + + if(SD0_speed){ + if (clock == mmc->f_min) { + DBG("[%s]ATSMB Host 390KHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 1, 390); + } + else if (clock >= 208000000) { + DBG("[%s]ATSMB Host 208MHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + + + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + DBG("[%s] *DPCTL = 0x%x DPCTL = 0x%x\n", + __func__, SD0_DPCTL_4BYTE_ADDR, SD0_DPCTL_4BYTE_VAL); + DBG("[%s] *DNCTL = 0x%x DNCTL = 0x%x\n", + __func__, SD0_DNCTL_4BYTE_ADDR, SD0_DNCTL_4BYTE_VAL); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 80); + } else if (clock >= 100000000) { + DBG("[%s]ATSMB Host 100MHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 80); + } else if (clock >= 50000000) { + DBG("[%s]ATSMB Host 50MHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 45 * clock_multiplier); + } else if ((clock >= 25000000) && (clock < 50000000)) { + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + DBG("[%s]ATSMB Host 25MHz\n", __func__); + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 24 * clock_multiplier); + } else if ((clock >= 20000000) && (clock < 25000000)) { + DBG("[%s]ATSMB Host 20MHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 20 * clock_multiplier); + } else { + DBG("[%s]ATSMB Host 390KHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 1, 390 * clock_multiplier); + } + }else{ + if (clock == mmc->f_min) { + DBG("[%s]ATSMB Host 390KHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 1, 390); + }else if (clock >= 25000000) { + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + DBG("[%s]ATSMB Host 25MHz\n", __func__); + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 24 * clock_multiplier); + } else if ((clock >= 20000000) && (clock < 25000000)) { + DBG("[%s]ATSMB Host 20MHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 2, 20 * clock_multiplier); + } else { + DBG("[%s]ATSMB Host 390KHz\n", __func__); + if ((*ATSMB0_CTL2 & BIT3) == 0) { + DBG("[%s] 1.8V clock = %u\n",__func__,clock); + + /*Set DPCTL 010 DNCTL 001*/ + SD0_DPCTL_4BYTE_VAL = (BIT1 | BIT4 | BIT7 | BIT10 | BIT13 | BIT16 | BIT19); + SD0_DNCTL_4BYTE_VAL = (BIT0 | BIT3 | BIT6 | BIT9 | BIT12 | BIT15 | BIT18); + DBG("[%s] *ATSMB0_CTL2 = 0x%x \n", __func__, *ATSMB0_CTL2); + } + return auto_pll_divisor(DEV_SDMMC0, SET_DIV, 1, 390 * clock_multiplier); + } + } +} + +static void atsmb_enable_power(struct atsmb_host *host, unsigned long flags) +{ + int result; + spin_unlock_irqrestore(&host->lock, flags); + if (g_atsmb_regulator) { + if (regulator_is_enabled(atsmb_regulator) == 0) { + DBG("[%s] Turn on power\n", __FUNCTION__); + result = regulator_enable(atsmb_regulator); + if (result != 0) + printk(KERN_ALERT "[%s] regulator_enable FAIL\n", __FUNCTION__); + } + } else { + GPIO_OD_GP13_SD0_BYTE_VAL &= ~GPIO_SD0_POWER; + } + spin_lock_irqsave(&host->lock, flags); +} + +static void atsmb_disable_power(struct atsmb_host *host, unsigned long flags) +{ + int result; + spin_unlock_irqrestore(&host->lock, flags); + if (g_atsmb_regulator) { + if (regulator_is_enabled(atsmb_regulator) == 1) { + DBG("[%s] Turn off power\n", __FUNCTION__); + result = regulator_disable(atsmb_regulator); + if (result != 0) + printk(KERN_ALERT "[%s] regulator_disable FAIL\n", __FUNCTION__); + } + } else { + GPIO_CTRL_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + GPIO_OC_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + /*set internal pull up*/ + PULL_CTRL_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + /*set internal pull enable*/ + PULL_EN_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + /*disable SD0 power*/ + GPIO_OD_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + } + spin_lock_irqsave(&host->lock, flags); +} + +/********************************************************************** +Name : atsmb_set_ios +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static void atsmb_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) +{ + + struct atsmb_host *host = mmc_priv(mmc); + unsigned long flags; + DBG("[%s] s\n",__func__); + spin_lock_irqsave(&host->lock, flags); + + if (ios->power_mode == MMC_POWER_OFF) { + if (MMC0_DRIVER_VERSION == MMC_DRV_3498) { + /* set all descriptor end bit for removing card when + * PDMA don't transfer data done. let PDMA transfer done + */ + atsmb_set_dma_end(host->DescVirAddr, host->DescSize); + + /* stop SD output clock */ + *ATSMB0_BUS_MODE &= ~(ATSMB_CST); + + /* disable SD Card power */ + /*set SD0 power pin as GPO pin*/ + atsmb_disable_power(host, flags); + + /* Disable Pull up/down resister of SD Bus */ + /*GPIO_PULL_CTRL_GP13_XDIO_BYTE_VAL &= ~SD0_PIN; marked by eason 2012/3/29*/ + + /* Config SD0 to GPIO */ + GPIO_CTRL_GP13_SD0_BYTE_VAL |= SD0_PIN; + + /* SD0 all pins output low */ + GPIO_OD_GP13_SD0_BYTE_VAL &= ~SD0_PIN; + + /* Config SD0 to GPO */ + GPIO_OC_GP13_SD0_BYTE_VAL |= SD0_PIN; + + } + + } else if (ios->power_mode == MMC_POWER_UP) { + if (MMC0_DRIVER_VERSION == MMC_DRV_3498) { + /* disable SD Card power */ + /*set SD0 power pin as GPO pin*/ + GPIO_CTRL_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + GPIO_OC_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + + /*set internal pull up*/ + PULL_CTRL_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + /*set internal pull enable*/ + PULL_EN_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + /*disable SD0 power*/ + GPIO_OD_GP13_SD0_BYTE_VAL |= GPIO_SD0_POWER; + + /* Config SD PIN share, WM3498 SD0 no share pin */ + /*PIN_SHARING_SEL_4BYTE_VAL &= ~GPIO_SD0_PinShare; */ + + /* do not config GPIO_SD0_CD because ISR has already run, + * config card detect will issue ISR storm. + */ + /* + GPIO_OD_GP12_SPI_BYTE_VAL |= GPIO_SD0_18SEL; + GPIO_OC_GP12_SPI_BYTE_VAL |= GPIO_SD0_18SEL; + GPIO_CTRL_GP12_SPI_BYTE_VAL |= GPIO_SD0_18SEL; + */ + *ATSMB0_CTL2 |= BIT3; /*Set 3.3V*/ + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + *ATSMB0_CTL2 &= ~(BIT0 | BIT1 | BIT2); + *ATSMB0_EXT_BUS_MODE &= ~BIT4; /*Disable DDR50*/ + + /*Set 18SEL as function pin*/ + GPIO_CTRL_GP12_SPI_BYTE_VAL &= ~GPIO_SD0_18SEL; + + + /* Config SD to GPIO */ + GPIO_CTRL_GP13_SD0_BYTE_VAL |= SD0_PIN; + + /* SD all pins output low */ + GPIO_OD_GP13_SD0_BYTE_VAL &= ~SD0_PIN; + + /* Config SD to GPO */ + GPIO_OC_GP13_SD0_BYTE_VAL |= SD0_PIN; + + /* Pull up/down resister of SD Bus */ + /*Disable Clock & CMD Pull enable*/ + PULL_EN_GP13_SD0_BYTE_VAL &= ~(GPIO_SD0_Clock | GPIO_SD0_Command); + + /*Set CD ,WP ,DATA pin pull up*/ + if (SD0_detect_pulldown) + PULL_CTRL_GP63_SD02_BYTE_VAL &= ~GPIO_SD0_CD; + else + PULL_CTRL_GP63_SD02_BYTE_VAL |= GPIO_SD0_CD; + PULL_CTRL_GP13_SD0_BYTE_VAL |= (GPIO_SD0_Data | GPIO_SD0_WriteProtect); + + /*Enable CD ,WP ,DATA internal pull*/ + PULL_EN_GP63_SD02_BYTE_VAL |= GPIO_SD0_CD; + PULL_EN_GP13_SD0_BYTE_VAL |= (GPIO_SD0_Data | GPIO_SD0_WriteProtect); + + /*Set SD0 PAD signal drive strength as 0*/ + SD0_DPCTL_4BYTE_VAL = 0; + SD0_DNCTL_4BYTE_VAL = 0; + + spin_unlock_irqrestore(&host->lock, flags); + msleep(100); + spin_lock_irqsave(&host->lock, flags); + + /* enable SD Card Power */ + atsmb_enable_power(host, flags); + + /* enable SD output clock */ + *ATSMB0_BUS_MODE |= ATSMB_CST; + + spin_unlock_irqrestore(&host->lock, flags); + msleep(10); + spin_lock_irqsave(&host->lock, flags); + + /* Config SD0 back to function */ + GPIO_CTRL_GP13_SD0_BYTE_VAL &= ~SD0_PIN; + GPIO_CTRL_GP63_SD02CD_BYTE_VAL &= ~GPIO_SD0_CD; + } + } else { + /*nothing to do when powering on.*/ + } + + host->current_clock = atsmb_set_clock(mmc,ios->clock); + + if (ios->bus_width == MMC_BUS_WIDTH_8) { + *ATSMB0_EXT_CTL |= BIT2; + } else if (ios->bus_width == MMC_BUS_WIDTH_4) { + *ATSMB0_BUS_MODE |= ATSMB_BUS_WIDTH_4; + *ATSMB0_EXT_CTL &= ~BIT2; + } else { + *ATSMB0_BUS_MODE &= ~(ATSMB_BUS_WIDTH_4); + *ATSMB0_EXT_CTL &= ~BIT2; + } + +/* +#define MMC_TIMING_LEGACY 0 +#define MMC_TIMING_MMC_HS 1 +#define MMC_TIMING_SD_HS 2 +#define MMC_TIMING_UHS_SDR12 MMC_TIMING_LEGACY +#define MMC_TIMING_UHS_SDR25 MMC_TIMING_SD_HS +#define MMC_TIMING_UHS_SDR50 3 +#define MMC_TIMING_UHS_SDR104 4 +#define MMC_TIMING_UHS_DDR50 5 +#define MMC_TIMING_MMC_HS200 6 +*/ + DBG("[%s]timing = 0x%x clock = %u\n", __func__, ios->timing, ios->clock); + + if (ios->timing == MMC_TIMING_LEGACY) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + *ATSMB0_CTL2 &= ~(BIT0 | BIT1 | BIT2); + } else if (ios->timing == MMC_TIMING_MMC_HS) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + *ATSMB0_CTL2 &= ~(BIT0 | BIT1 | BIT2); + } else if (ios->timing == MMC_TIMING_SD_HS) { + *ATSMB0_EXT_CTL |= BIT7; /*HIGH SPEED MODE*/ + *ATSMB0_CTL2 &= ~(BIT0 | BIT1 | BIT2); + } else if (ios->timing == MMC_TIMING_UHS_SDR12) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + /*UHS Mode Select = 000 SDR12*/ + *ATSMB0_CTL2 &= ~(BIT0 | BIT1 | BIT2); + } else if (ios->timing == MMC_TIMING_UHS_SDR25) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + /*UHS Mode Select = 001 SDR25*/ + *ATSMB0_CTL2 |= BIT0; + *ATSMB0_CTL2 &= ~(BIT1 | BIT2); + } else if (ios->timing == MMC_TIMING_UHS_SDR50) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + /*UHS Mode Select = 010 SDR50*/ + *ATSMB0_CTL2 |= BIT1; + *ATSMB0_CTL2 &= ~(BIT0 | BIT2); + } else if (ios->timing == MMC_TIMING_UHS_SDR104) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + /*UHS Mode Select = 011 SDR104*/ + *ATSMB0_CTL2 |= (BIT0 | BIT1); + *ATSMB0_CTL2 &= ~BIT2; + } else if (ios->timing == MMC_TIMING_UHS_DDR50) { + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + /*UHS Mode Select = 100 DDR50*/ + *ATSMB0_CTL2 |= BIT2; + *ATSMB0_CTL2 &= ~(BIT0 | BIT1); + *ATSMB0_EXT_BUS_MODE |= BIT4; /*Enable DDR50*/ + /* enable SD output clock */ + *ATSMB0_BUS_MODE &= ~ATSMB_CST; + spin_unlock_irqrestore(&host->lock, flags); + msleep(1); + spin_lock_irqsave(&host->lock, flags); + + *ATSMB0_EXT_BUS_MODE |= BIT4; /*Enable DDR50*/ + /* enable SD output clock */ + *ATSMB0_BUS_MODE |= ATSMB_CST; + } else + *ATSMB0_EXT_CTL &= ~BIT7; /*Disable High Speed Mode*/ + +/* spin_unlock_irqrestore(&host->lock, flags); + msleep(100); + spin_lock_irqsave(&host->lock, flags); */ + + DBG("[%s] *ATSMB0_CTL2 = 0x%x *ATSMB0_EXT_CTL = 0x%x *ATSMB0_EXT_BUS_MODE = 0x%x\n", + __func__, *ATSMB0_CTL2, *ATSMB0_EXT_CTL, *ATSMB0_EXT_BUS_MODE); + spin_unlock_irqrestore(&host->lock, flags); + DBG("[%s] e\n",__func__); +} + +int atsmb_start_signal_voltage_switch(struct mmc_host *host, struct mmc_ios *ios) +{ + DBG("[%s] s\n",__func__); + + if(ios->signal_voltage == MMC_SIGNAL_VOLTAGE_180) { + /*Set CMD ,DATA0~3 pin as GPO low*/ + /* SD0 all pins output low */ + GPIO_OD_GP13_SD0_BYTE_VAL &= ~(GPIO_SD0_Command | GPIO_SD0_Data); + /* Config SD0 to GPO */ + GPIO_OC_GP13_SD0_BYTE_VAL |= (GPIO_SD0_Command | GPIO_SD0_Data);; + /* Config SD0 to GPIO */ + GPIO_CTRL_GP13_SD0_BYTE_VAL |= (GPIO_SD0_Command | GPIO_SD0_Data); + + + /*Wait for 1ms*/ + msleep(1); + + /*Stop Clock*/ + *ATSMB0_BUS_MODE &= ~(ATSMB_CST); + + + /*Enable clock & Set 1.8V*/ + *ATSMB0_CTL2 &= ~BIT3; + + /*Wait for 10ms*/ + msleep(100); + + /*Enable Clock*/ + *ATSMB0_BUS_MODE |= ATSMB_CST; + + + /*Set CMD DATA0~3 pin as function pin*/ + GPIO_CTRL_GP13_SD0_BYTE_VAL &= ~GPIO_SD0_Command; + GPIO_CTRL_GP13_SD0_BYTE_VAL &= ~GPIO_SD0_Data; + + + DBG("Set SD0 1.8v = 0x%x\n",ios->signal_voltage); + + return 0; + } else if (ios->signal_voltage == MMC_SIGNAL_VOLTAGE_330){ + *ATSMB0_CTL2 |= BIT3; + DBG("Set SD0 3.3v = 0x%x\n",ios->signal_voltage); + return 0; + } + + printk("Set voltage error = 0x%x\n",ios->signal_voltage); + return -EIO; + DBG("[%s] e\n",__func__); +} + +static const struct mmc_host_ops atsmb_ops = { + .request = atsmb_request, + .set_ios = atsmb_set_ios, + .get_ro = atsmb_get_ro, + .get_slot_status = atsmb_get_slot_status, + .dump_host_regs = atsmb_dump_host_regs, + .enable_sdio_irq = atsmb_enable_sdio_irq, + .start_signal_voltage_switch = atsmb_start_signal_voltage_switch, +}; + +extern int wmt_getsyspara(char *varname, unsigned char *varval, int *varlen); + +/********************************************************************** +Name :atsmb_probe +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static int __init atsmb_probe(struct platform_device *pdev) +{ + struct device *dev = &pdev->dev; + struct mmc_host *mmc_host = NULL; + struct atsmb_host *atsmb_host = NULL; + struct resource *resource = NULL; + int irq[2] = {0}; + int ret = 0; + + DBG("[%s] s\n",__func__); + if (!pdev) { + ret = -EINVAL; /* Invalid argument */ + goto the_end; + } + + /*Enable SD host clock*/ + auto_pll_divisor(DEV_SDMMC0, CLK_ENABLE, 0, 0); + + if (MMC0_DRIVER_VERSION == MMC_DRV_3498) { + /* Pull up/down resister of SD CD */ + if(SD0_detect_pulldown) + PULL_CTRL_GP63_SD02_BYTE_VAL &= ~GPIO_SD0_CD; /*pull down CD*/ + else + PULL_CTRL_GP63_SD02_BYTE_VAL |= GPIO_SD0_CD; /*pull up CD*/ + PULL_EN_GP63_SD02_BYTE_VAL |= GPIO_SD0_CD; + + /* config CardDetect pin to SD function */ + GPIO_CTRL_GP63_SD02CD_BYTE_VAL &= ~GPIO_SD0_CD; + } + + + resource = platform_get_resource(pdev, IORESOURCE_MEM, 0); + if (!resource) { + ret = -ENXIO; /* No such device or address */ + printk(KERN_ALERT "[MMC/SD driver] Getting platform resources failed!\n"); + goto the_end; + } +#if 0 + if (!request_mem_region(resource->start, SZ_1K, MMC0_DRIVER_NAME)) { + ret = -EBUSY; + printk(KERN_ALERT "[MMC/SD driver] Request memory region failed!\n"); + goto the_end ; + } +#endif + irq[0] = platform_get_irq(pdev, 0); /*get IRQ for device*/; + irq[1] = platform_get_irq(pdev, 1); /*get IRQ for dma*/; + + if (irq[0] == NO_IRQ || irq[1] == NO_IRQ) { + ret = -ENXIO;/* No such device or address */ + printk(KERN_ALERT "[MMC/SD driver] Get platform IRQ failed!\n"); + goto rls_region; + } + + /*allocate a standard msp_host structure attached with a atsmb structure*/ + mmc_host = mmc_alloc_host(sizeof(struct atsmb_host), dev); + if (!mmc_host) { + ret = -ENOMEM; + printk(KERN_ALERT "[MMC/SD driver] Allocating driver's data failed!\n"); + goto rls_region; + } + mmc_host->wmt_host_index = 0; /*to identify host number*/ + + dev_set_drvdata(dev, (void *)mmc_host); /* mmc_host is driver data for the atsmb dev.*/ + atsmb_host = mmc_priv(mmc_host); + + mmc_host->ops = &atsmb_ops; + + mmc_host->ocr_avail = MMC_VDD_32_33|MMC_VDD_33_34; + + mmc_host->f_min = 390425; /*390.425Hz = 400MHz/64/16*/ + mmc_host->f_max = 100000000; /* Set max frequency 100MHz*/ + + if (SDXC0_function == 1) { + mmc_host->caps = MMC_CAP_4_BIT_DATA | MMC_CAP_SD_HIGHSPEED | MMC_CAP_MMC_HIGHSPEED | MMC_CAP_SDIO_IRQ + | MMC_CAP_UHS_SDR12 | MMC_CAP_UHS_SDR25 | MMC_CAP_UHS_DDR50 | MMC_CAP_1_8V_DDR; + /* |MMC_CAP_8_BIT_DATA;*/ //zhf: marked by James Tian + } else { + mmc_host->caps = MMC_CAP_4_BIT_DATA | MMC_CAP_SD_HIGHSPEED | MMC_CAP_MMC_HIGHSPEED | MMC_CAP_SDIO_IRQ; + /* |MMC_CAP_8_BIT_DATA;*/ //zhf: marked by James Tian + } + + mmc_host->max_segs = 128; /*we use software sg. so we could manage even larger number.*/ + + /*1MB per each request */ + /*we have a 16 bit block number register, and block length is 512 bytes.*/ + mmc_host->max_req_size = 16*512*(mmc_host->max_segs); + mmc_host->max_seg_size = 65024; /* 0x7F*512 PDMA one descriptor can transfer 64K-1 byte*/ + mmc_host->max_blk_size = 2048; /* our block length register is 11 bits.*/ + mmc_host->max_blk_count = (mmc_host->max_req_size)/512; + + /*set the specified host -- ATSMB*/ +#ifdef CONFIG_MMC_UNSAFE_RESUME + mmc_host->card_attath_status = card_attach_status_unchange; +#endif + sema_init(&mmc_host->req_sema,1); /*initial request semaphore*/ +#if 0 + atsmb_host->base = ioremap(resource->start, SZ_1K); + if (!atsmb_host->base) { + printk(KERN_ALERT "[MMC/SD driver] IO remap failed!\n"); + ret = -ENOMEM; + goto fr_host; + } +#endif + atsmb_host->base = (void *)resource->start; + atsmb_host->mmc = mmc_host; + spin_lock_init(&atsmb_host->lock); + atsmb_host->res = resource;/* for atsmb_remove*/ + + /*disable all interrupt and clear status by resetting controller.*/ + *ATSMB0_BUS_MODE |= ATSMB_SFTRST; + *ATSMB0_BLK_LEN &= ~(0xa000); + *ATSMB0_SD_STS_0 |= 0xff; + *ATSMB0_SD_STS_1 |= 0xff; + + /* WM3437 A0 default not output clock, after SFTRST need to enable SD clock */ + //if (MMC0_DRIVER_VERSION >= MMC_DRV_3437_A0) /* including 3429 */ + *ATSMB0_BUS_MODE |= ATSMB_CST; + + atsmb_host->regular_irq = irq[0]; + atsmb_host->dma_irq = irq[1]; + + ret = request_irq(atsmb_host->regular_irq, + atsmb_regular_isr, + IRQF_SHARED, //SA_SHIRQ, /*SA_INTERRUPT, * that is okay?*/ //zhf: modified by James Tian, should be IRQF_SHARED? + MMC0_DRIVER_NAME, + (void *)atsmb_host); + if (ret) { + printk(KERN_ALERT "[MMC/SD driver] Failed to register regular ISR!\n"); + goto unmap; + } + + ret = request_irq(atsmb_host->dma_irq, + atsmb_dma_isr, + IRQF_DISABLED, // SA_INTERRUPT, //zhf: modified by James Tian + MMC0_DRIVER_NAME, + (void *)atsmb_host); + if (ret) { + printk(KERN_ALERT "[MMC/SD driver] Failed to register DMA ISR!\n"); + goto fr_regular_isr; + } + + /* config CardDetect pin to SD function */ + /*GPIO_CTRL_GP63_SD02CD_BYTE_VAL &= ~GPIO_SD0_CD; add by eason 2012/3/29*/ + + /*wait card detect status change*/ + //msleep(10); + + /*enable card insertion interrupt and enable DMA and its Global INT*/ + *ATSMB0_BLK_LEN |= (0xa000); /* also, we enable GPIO to detect card.*/ + *ATSMB0_SD_STS_0 |= 0xff; + *ATSMB0_INT_MASK_0 |= 0x80; /*or 0x40?*/ + + /*allocation dma descriptor*/ + ret = atsmb_alloc_desc(atsmb_host, sizeof(struct SD_PDMA_DESC_S) * MAX_DESC_NUM); + if (ret == -1) { + printk(KERN_ALERT "[MMC/SD driver] Failed to allocate DMA descriptor!\n"); + goto fr_dma_isr; + } + printk(KERN_INFO "WMT ATSMB (AHB To SD/MMC Bus) controller registered!\n"); + + if (SD0_function == SDIO_WIFI) + mmc_add_host(mmc_host,false); + else + mmc_add_host(mmc_host,true); + + mmc_host_attr = mmc_host; + + DBG("[%s] e1\n",__func__); + return 0; + +fr_dma_isr: + free_irq(atsmb_host->dma_irq, atsmb_host); +fr_regular_isr: + free_irq(atsmb_host->regular_irq, atsmb_host); +unmap: + //iounmap(atsmb_host->base); +//fr_host: + dev_set_drvdata(dev, NULL); + mmc_free_host(mmc_host); +rls_region: + //release_mem_region(resource->start, SZ_1K); +the_end: + printk(KERN_ALERT "[MMC/SD driver] ATSMB0 probe Failed!\n") ; + DBG("[%s] e2\n",__func__); + return ret; +} +/********************************************************************** +Name : atsmb_remove +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static int atsmb_remove(struct platform_device *pdev) +{ + struct device *dev = &pdev->dev; + struct mmc_host *mmc_host = (struct mmc_host *)dev_get_drvdata(dev); + struct atsmb_host *atsmb_host; + + DBG("[%s] s\n",__func__); + atsmb_host = mmc_priv(mmc_host); + if (!mmc_host || !atsmb_host) { + printk(KERN_ALERT "[MMC/SD driver] ATSMB0 remove method failed!\n"); + DBG("[%s] e1\n",__func__); + return -ENXIO; + } + mmc_remove_host(mmc_host); + + /*disable interrupt by resetting controller -- for safey*/ + *ATSMB0_BUS_MODE |= ATSMB_SFTRST; + *ATSMB0_BLK_LEN &= ~(0xa000); + *ATSMB0_SD_STS_0 |= 0xff; + *ATSMB0_SD_STS_1 |= 0xff; + + (void)free_irq(atsmb_host->regular_irq, atsmb_host); + (void)free_irq(atsmb_host->dma_irq, atsmb_host); + (void)iounmap(atsmb_host->base); + (void)release_mem_region(atsmb_host->res->start, SZ_1K); + dev_set_drvdata(dev, NULL); + /*free dma descriptor*/ + dma_free_coherent(atsmb_host->mmc->parent, atsmb_host->DescSize, + atsmb_host->DescVirAddr, atsmb_host->DescPhyAddr); + (void)mmc_free_host(mmc_host);/* also free atsmb_host.*/ + DBG("[%s] e2\n",__func__); + return 0; +} + +/********************************************************************** +Name : atsmb_shutdown +Function :. +Calls : +Called by : +Parameter : +Author : Tommy Huang +History : +***********************************************************************/ +static void atsmb_shutdown(struct platform_device *pdev) +{ + /*atsmb_shutdown don't be used now.*/ + /*struct device *dev = &pdev->dev; + struct mmc_host *mmc_host = (struct mmc_host *)dev_get_drvdata(dev);*/ + + DBG("[%s] s\n",__func__); + if (SD0_function != SDIO_WIFI) { + /*Disable card detect interrupt*/ + *ATSMB0_INT_MASK_0 &= ~0x80; + } + DBG("[%s] e\n",__func__); + +} + +/********************************************************************** +Name : atsmb_suspend +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +#ifdef CONFIG_PM +static int atsmb_suspend(struct platform_device *pdev, pm_message_t state) +{ + struct device *dev = &pdev->dev; + struct mmc_host *mmc = (struct mmc_host *)dev_get_drvdata(dev); + int ret = 0; + DBG("[%s] s\n",__func__); + + if (mmc) { + /*struct atsmb_host *host = mmc_priv(mmc);*/ + ret = mmc_suspend_host(mmc); + if (ret == 0) { + /*disable all interrupt and clear status by resetting controller. */ + *ATSMB0_BUS_MODE |= ATSMB_SFTRST; + *ATSMB0_BLK_LEN &= ~(0xa000); + *ATSMB0_SD_STS_0 |= 0xff; + *ATSMB0_SD_STS_1 |= 0xff; + + } + /*disable source clock*/ + auto_pll_divisor(DEV_SDMMC0, CLK_DISABLE, 0, 0); +#ifdef CONFIG_MMC_UNSAFE_RESUME + /*clean SD card attatch status change*/ + PMCWS_VAL |= BIT19; + mmc->card_attath_status = card_attach_status_unchange; +#endif + } + + DBG("[%s] e\n",__func__); + return ret; +} +/********************************************************************** +Name : atsmb_resume +Function :. +Calls : +Called by : +Parameter : +Author : Leo Lee +History : +***********************************************************************/ +static int atsmb_resume(struct platform_device *pdev) +{ + struct device *dev = &pdev->dev; + struct mmc_host *mmc = (struct mmc_host *)dev_get_drvdata(dev); + int ret = 0; + DBG("[%s] s\n",__func__); + + /* + * enable interrupt, DMA, etc. + * Supply power to slot. + */ + if (mmc) { + /*enable source clock*/ + auto_pll_divisor(DEV_SDMMC0, CLK_ENABLE, 0, 0); + + udelay(1); + /*enable card insertion interrupt and enable DMA and its Global INT*/ + *ATSMB0_BUS_MODE |= ATSMB_SFTRST; + *ATSMB0_BLK_LEN |= (0xa000); + *ATSMB0_INT_MASK_0 |= 0x80; /* or 40?*/ +#ifdef CONFIG_MMC_UNSAFE_RESUME + /*modify SD card attatch status change*/ + if ((PMCWS_VAL & BIT19) && !mmc->bus_dead) { + /*card change when suspend mode*/ + mmc->card_attath_status = card_attach_status_change; + /*clean SD card attatch status change*/ + PMCWS_VAL |= BIT19; + } +#endif + ret = mmc_resume_host(mmc); + } + + DBG("[%s] e\n",__func__); + return ret; +} +#else +#define atsmb_suspend NULL +#define atsmb_resume NULL +#endif + +static struct platform_driver atsmb_driver = { + .driver.name = "sdmmc", + //.probe = atsmb_probe, + .remove = atsmb_remove, + .shutdown = atsmb_shutdown, + .suspend = atsmb_suspend, + .resume = atsmb_resume, +}; + +static struct platform_device wmt_sdmmc_device = { + .name = "sdmmc", + .id = 0, + .dev = { + .dma_mask = &wmt_sdmmc_dma_mask, + .coherent_dma_mask = ~0, + .release = atsmb_release, + }, + .num_resources = ARRAY_SIZE(wmt_sdmmc_resources), + .resource = wmt_sdmmc_resources, +}; + + +static ssize_t atsmb_state_show(struct kobject *kobj, struct kobj_attribute *attr, + char *buf) +{ + int ret = 0; + int card_state; + + DBG("[%s] s\n",__func__); + if (mmc_host_attr->card != NULL) + card_state = 1; + else + card_state = 0; + DBG("[%s]card_state = %d\n",__func__,card_state); + ret = sprintf(buf, "%d\n", card_state); + DBG("[%s] e\n",__func__); + return ret; +} + +static ssize_t atsmb_state_store(struct kobject *kobj, struct kobj_attribute *attr, + const char *buf, size_t n) +{ + int val; + DBG("[%s] s\n",__func__); + if (sscanf(buf, "%d", &val) == 1) { + DBG("[%s] val = %d\n",__func__,val); + if ((val == 1) && (mmc_host_attr->card == NULL)) { + DBG("[%s]add card\n",__func__); + mmc_detect_change(mmc_host_attr, 0); + msleep(500); + } else if ((val == 0) && (mmc_host_attr->card != NULL)) { + DBG("[%s]remove card\n",__func__); + mmc_force_remove_card(mmc_host_attr); + } + DBG("[%s] e1\n",__func__); + return n; + } + DBG("[%s] e2\n",__func__); + return -EINVAL; +} + +static struct kobj_attribute atsmb_state_attr = { \ + .attr = { \ + .name = __stringify(state), \ + .mode = 0755, \ + }, \ + .show = atsmb_state_show, \ + .store = atsmb_state_store, \ +}; + +static struct attribute * g[] = { + &atsmb_state_attr.attr, + NULL, +}; + +static struct attribute_group attr_group = { + .attrs = g, +}; + //kevin add to check tf card stats every 1 sec + +static void wmt_mmc_work(struct work_struct *work) +{ + + static int card_state_save = -1; + if(mmc_host_attr!=NULL){ + struct atsmb_host *host = mmc_priv(mmc_host_attr); + int card_state; + if (mmc_host_attr->card != NULL) + card_state = 1; + else + card_state = 0; + + + //printk("check %d %d\n",atsmb_get_slot_status(host->mmc),card_state); + if(atsmb_get_slot_status(host->mmc)!=card_state){ + //the second time error,goto detect or remove card + if(card_state_save == card_state){ + + //printk("xxxxxxxxxxxxxxxxxxxxxxxxcatch error %dxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n",card_state); + mmc_detect_change(mmc_host_attr, 0); + + } + //mark status error + card_state_save =card_state; + }else + card_state_save = -1; + + } + schedule_delayed_work(&mmc_work, 1*HZ); +} +static int __init atsmb_init(void) +{ + int ret; + + int retval; + unsigned char buf[80]; + int varlen = 80; + char *varname = "wmt.sd0.param"; + int temp = 0, sd_enable = 1; /*0 :disable 1:enable*/ + unsigned char colon; + DBG("[%s] s\n",__func__); + +#ifdef CONFIG_MTD_WMT_SF + /*Read system param to identify host function 0: SD/MMC 1:SDIO wifi*/ + retval = wmt_getsyspara(varname, buf, &varlen); + if (retval == 0) { + sscanf(buf,"%x%c%d%c%d%c%d%c%d", &temp, &colon, &SD0_function,&colon,&SD0_detect_pol,&colon,&SD0_detect_pulldown,&colon,&SD0_speed); + printk(KERN_ALERT "wmt.sd0.param = %x%c%d%c%d%c%d%c%d\n", temp, colon, SD0_function,colon,SD0_detect_pol,colon,SD0_detect_pulldown,colon,SD0_speed); + sd_enable = temp & 0xf; + SDXC0_function = (temp >> 4) & 0xf; + printk(KERN_ALERT "SD0 ebable = %x, SDXC = %x, function = %x\n", + sd_enable, SDXC0_function, SD0_function); + + if (SD0_function < 0 || SD0_function >= SD_MAX_FUN) { + sd_enable = 1; + SD0_function = 0; + printk(KERN_ALERT "wmt.sd0.param func err\n"); + } + if (colon != ':') { + sd_enable = 1; + printk(KERN_ALERT "wmt.sd0.param colon err\n"); + } + + } else { + printk(KERN_ALERT "Default wmt.sd0.param = %x:%d\n", temp, SD0_function); + } + + //add by kevin guan + retval = wmt_getsyspara("wmt.sd0.ro.disable", buf, &varlen); + if (retval == 0) { + sscanf(buf,"%d", &SD0_ro_disable); + printk(KERN_ALERT "SD0_ro_disable %d\n",SD0_ro_disable); + } + +#endif + /*SD function disable*/ + if (sd_enable != 1) { + return -ENODEV; + } + + + get_driver_version(); + + if (platform_device_register(&wmt_sdmmc_device))//add by jay,for modules support + return -1; + //ret = platform_driver_register(&atsmb_driver); + + /* Register regulator for SD host power switch */ + atsmb_regulator = regulator_get(NULL, "ldo4"); + if (IS_ERR(atsmb_regulator)) + g_atsmb_regulator = 0; + else + g_atsmb_regulator = 1; + printk("[SD/MMC]use_regulator = %d\n", g_atsmb_regulator); + + /* Force to disable ldo4 regulator, because Power switch is turned on, default */ + if (g_atsmb_regulator) { + ret = regulator_force_disable(atsmb_regulator); + if (ret != 0) + printk(KERN_ALERT "[%s] get regulator Fail, ret = %d\n", __FUNCTION__, ret); + } + + ret = platform_driver_probe(&atsmb_driver, atsmb_probe); + + atsmb_kobj = kobject_create_and_add("mmc0", NULL); + if (!atsmb_kobj) + return -ENOMEM; + + + { + //add by kevin guan + int detect_disable = 0; + retval = wmt_getsyspara("wmt.sd0.detect.disable", buf, &varlen); + if (retval == 0) { + sscanf(buf,"%d", &detect_disable); + printk(KERN_ALERT "detect_disable %d\n",detect_disable); + } + + if(detect_disable){ + //tf slot without detect pin. + //do nothing; + }else{ + //kevin add to check tf card stats every 1 sec + INIT_DELAYED_WORK(&mmc_work, wmt_mmc_work); + schedule_delayed_work(&mmc_work, 1*HZ); + } + } + + return sysfs_create_group(atsmb_kobj, &attr_group); + + DBG("[%s] e\n",__func__); + return ret; +} + +static void __exit atsmb_exit(void) +{ + DBG("[%s] s\n",__func__); + (void)platform_driver_unregister(&atsmb_driver); + (void)platform_device_unregister(&wmt_sdmmc_device);//add by jay,for modules support + DBG("[%s] e\n",__func__); +} + +module_init(atsmb_init); +module_exit(atsmb_exit); +module_param(fmax0, uint, 0444); + +MODULE_AUTHOR("WonderMedia Technologies, Inc."); +MODULE_DESCRIPTION("WMT [AHB to SD/MMC Bridge] driver"); +MODULE_LICENSE("GPL"); |