2009-12-17 15:27:20 -08:00
/*
* Freescale eSDHC controller driver .
*
2012-02-14 14:05:37 +08:00
* Copyright ( c ) 2007 , 2010 , 2012 Freescale Semiconductor , Inc .
2009-12-17 15:27:20 -08:00
* Copyright ( c ) 2009 MontaVista Software , Inc .
*
* Authors : Xiaobo Xie < X . Xie @ freescale . com >
* Anton Vorontsov < avorontsov @ ru . mvista . com >
*
* 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 .
*/
2013-06-27 12:00:05 -04:00
# include <linux/err.h>
2009-12-17 15:27:20 -08:00
# include <linux/io.h>
2012-02-14 14:05:37 +08:00
# include <linux/of.h>
2009-12-17 15:27:20 -08:00
# include <linux/delay.h>
2011-07-03 15:15:51 -04:00
# include <linux/module.h>
2009-12-17 15:27:20 -08:00
# include <linux/mmc/host.h>
2011-05-27 23:48:14 +08:00
# include "sdhci-pltfm.h"
2010-10-15 12:21:03 +02:00
# include "sdhci-esdhc.h"
2009-12-17 15:27:20 -08:00
2012-03-08 11:25:02 +08:00
# define VENDOR_V_22 0x12
2012-12-04 10:41:28 +08:00
# define VENDOR_V_23 0x13
2012-03-08 11:25:02 +08:00
static u32 esdhc_readl ( struct sdhci_host * host , int reg )
{
u32 ret ;
ret = in_be32 ( host - > ioaddr + reg ) ;
/*
* The bit of ADMA flag in eSDHC is not compatible with standard
* SDHC register , so set fake flag SDHCI_CAN_DO_ADMA2 when ADMA is
* supported by eSDHC .
* And for many FSL eSDHC controller , the reset value of field
* SDHCI_CAN_DO_ADMA1 is one , but some of them can ' t support ADMA ,
* only these vendor version is greater than 2.2 / 0x12 support ADMA .
* For FSL eSDHC , must aligned 4 - byte , so use 0xFC to read the
* the verdor version number , oxFE is SDHCI_HOST_VERSION .
*/
if ( ( reg = = SDHCI_CAPABILITIES ) & & ( ret & SDHCI_CAN_DO_ADMA1 ) ) {
u32 tmp = in_be32 ( host - > ioaddr + SDHCI_SLOT_INT_STATUS ) ;
tmp = ( tmp & SDHCI_VENDOR_VER_MASK ) > > SDHCI_VENDOR_VER_SHIFT ;
if ( tmp > VENDOR_V_22 )
ret | = SDHCI_CAN_DO_ADMA2 ;
}
return ret ;
}
2009-12-17 15:27:20 -08:00
static u16 esdhc_readw ( struct sdhci_host * host , int reg )
{
u16 ret ;
2011-09-09 20:05:46 +08:00
int base = reg & ~ 0x3 ;
int shift = ( reg & 0x2 ) * 8 ;
2009-12-17 15:27:20 -08:00
if ( unlikely ( reg = = SDHCI_HOST_VERSION ) )
2011-09-09 20:05:46 +08:00
ret = in_be32 ( host - > ioaddr + base ) & 0xffff ;
2009-12-17 15:27:20 -08:00
else
2011-09-09 20:05:46 +08:00
ret = ( in_be32 ( host - > ioaddr + base ) > > shift ) & 0xffff ;
return ret ;
}
static u8 esdhc_readb ( struct sdhci_host * host , int reg )
{
int base = reg & ~ 0x3 ;
int shift = ( reg & 0x3 ) * 8 ;
u8 ret = ( in_be32 ( host - > ioaddr + base ) > > shift ) & 0xff ;
2012-01-13 15:02:01 +08:00
/*
* " DMA select " locates at offset 0x28 in SD specification , but on
* P5020 or P3041 , it locates at 0x29 .
*/
if ( reg = = SDHCI_HOST_CONTROL ) {
u32 dma_bits ;
dma_bits = in_be32 ( host - > ioaddr + reg ) ;
/* DMA select is 22,23 bits in Protocol Control Register */
dma_bits = ( dma_bits > > 5 ) & SDHCI_CTRL_DMA_MASK ;
/* fixup the result */
ret & = ~ SDHCI_CTRL_DMA_MASK ;
ret | = dma_bits ;
}
2009-12-17 15:27:20 -08:00
return ret ;
}
2012-12-04 10:41:28 +08:00
static void esdhc_writel ( struct sdhci_host * host , u32 val , int reg )
{
/*
* Enable IRQSTATEN [ BGESEN ] is just to set IRQSTAT [ BGE ]
* when SYSCTL [ RSTD ] ) is set for some special operations .
* No any impact other operation .
*/
if ( reg = = SDHCI_INT_ENABLE )
val | = SDHCI_INT_BLK_GAP ;
sdhci_be32bs_writel ( host , val , reg ) ;
}
2009-12-17 15:27:20 -08:00
static void esdhc_writew ( struct sdhci_host * host , u16 val , int reg )
{
if ( reg = = SDHCI_BLOCK_SIZE ) {
/*
* Two last DMA bits are reserved , and first one is used for
* non - standard blksz of 4096 bytes that we don ' t support
* yet . So clear the DMA boundary bits .
*/
val & = ~ SDHCI_MAKE_BLKSZ ( 0x7 , 0 ) ;
}
sdhci_be32bs_writew ( host , val , reg ) ;
}
static void esdhc_writeb ( struct sdhci_host * host , u8 val , int reg )
{
2012-01-13 15:02:01 +08:00
/*
* " DMA select " location is offset 0x28 in SD specification , but on
* P5020 or P3041 , it ' s located at 0x29 .
*/
if ( reg = = SDHCI_HOST_CONTROL ) {
u32 dma_bits ;
2013-07-05 12:48:35 -04:00
/*
* If host control register is not standard , exit
* this function
*/
if ( host - > quirks2 & SDHCI_QUIRK2_BROKEN_HOST_CONTROL )
return ;
2012-01-13 15:02:01 +08:00
/* DMA select is 22,23 bits in Protocol Control Register */
dma_bits = ( val & SDHCI_CTRL_DMA_MASK ) < < 5 ;
clrsetbits_be32 ( host - > ioaddr + reg , SDHCI_CTRL_DMA_MASK < < 5 ,
dma_bits ) ;
val & = ~ SDHCI_CTRL_DMA_MASK ;
val | = in_be32 ( host - > ioaddr + reg ) & SDHCI_CTRL_DMA_MASK ;
}
2009-12-17 15:27:20 -08:00
/* Prevent SDHCI core from writing reserved bits (e.g. HISPD). */
if ( reg = = SDHCI_HOST_CONTROL )
val & = ~ ESDHC_HOST_CONTROL_RES ;
sdhci_be32bs_writeb ( host , val , reg ) ;
}
2012-12-04 10:41:28 +08:00
/*
* For Abort or Suspend after Stop at Block Gap , ignore the ADMA
* error ( IRQSTAT [ ADMAE ] ) if both Transfer Complete ( IRQSTAT [ TC ] )
* and Block Gap Event ( IRQSTAT [ BGE ] ) are also set .
* For Continue , apply soft reset for data ( SYSCTL [ RSTD ] ) ;
* and re - issue the entire read transaction from beginning .
*/
static void esdhci_of_adma_workaround ( struct sdhci_host * host , u32 intmask )
{
u32 tmp ;
bool applicable ;
dma_addr_t dmastart ;
dma_addr_t dmanow ;
tmp = in_be32 ( host - > ioaddr + SDHCI_SLOT_INT_STATUS ) ;
tmp = ( tmp & SDHCI_VENDOR_VER_MASK ) > > SDHCI_VENDOR_VER_SHIFT ;
applicable = ( intmask & SDHCI_INT_DATA_END ) & &
( intmask & SDHCI_INT_BLK_GAP ) & &
( tmp = = VENDOR_V_23 ) ;
if ( ! applicable )
return ;
host - > data - > error = 0 ;
dmastart = sg_dma_address ( host - > data - > sg ) ;
dmanow = dmastart + host - > data - > bytes_xfered ;
/*
* Force update to the next DMA block boundary .
*/
dmanow = ( dmanow & ~ ( SDHCI_DEFAULT_BOUNDARY_SIZE - 1 ) ) +
SDHCI_DEFAULT_BOUNDARY_SIZE ;
host - > data - > bytes_xfered = dmanow - dmastart ;
sdhci_writel ( host , dmanow , SDHCI_DMA_ADDRESS ) ;
}
2010-10-15 12:21:03 +02:00
static int esdhc_of_enable_dma ( struct sdhci_host * host )
2009-12-17 15:27:20 -08:00
{
setbits32 ( host - > ioaddr + ESDHC_DMA_SYSCTL , ESDHC_DMA_SNOOP ) ;
return 0 ;
}
2010-10-15 12:21:03 +02:00
static unsigned int esdhc_of_get_max_clock ( struct sdhci_host * host )
2009-12-17 15:27:20 -08:00
{
2011-07-20 17:13:36 -04:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2009-12-17 15:27:20 -08:00
2011-07-20 17:13:36 -04:00
return pltfm_host - > clock ;
2009-12-17 15:27:20 -08:00
}
2010-10-15 12:21:03 +02:00
static unsigned int esdhc_of_get_min_clock ( struct sdhci_host * host )
2009-12-17 15:27:20 -08:00
{
2011-07-20 17:13:36 -04:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2009-12-17 15:27:20 -08:00
2011-07-20 17:13:36 -04:00
return pltfm_host - > clock / 256 / 16 ;
2009-12-17 15:27:20 -08:00
}
2012-02-14 14:05:37 +08:00
static void esdhc_of_set_clock ( struct sdhci_host * host , unsigned int clock )
{
2013-09-13 19:11:32 +08:00
int pre_div = 2 ;
int div = 1 ;
u32 temp ;
if ( clock = = 0 )
goto out ;
2012-02-14 14:05:37 +08:00
/* Workaround to reduce the clock frequency for p1010 esdhc */
if ( of_find_compatible_node ( NULL , NULL , " fsl,p1010-esdhc " ) ) {
if ( clock > 20000000 )
clock - = 5000000 ;
if ( clock > 40000000 )
clock - = 5000000 ;
}
2013-09-13 19:11:32 +08:00
temp = sdhci_readl ( host , ESDHC_SYSTEM_CONTROL ) ;
temp & = ~ ( ESDHC_CLOCK_IPGEN | ESDHC_CLOCK_HCKEN | ESDHC_CLOCK_PEREN
| ESDHC_CLOCK_MASK ) ;
sdhci_writel ( host , temp , ESDHC_SYSTEM_CONTROL ) ;
while ( host - > max_clk / pre_div / 16 > clock & & pre_div < 256 )
pre_div * = 2 ;
while ( host - > max_clk / pre_div / div > clock & & div < 16 )
div + + ;
dev_dbg ( mmc_dev ( host - > mmc ) , " desired SD clock: %d, actual: %d \n " ,
2013-09-13 19:11:37 +08:00
clock , host - > max_clk / pre_div / div ) ;
2013-09-13 19:11:32 +08:00
pre_div > > = 1 ;
div - - ;
temp = sdhci_readl ( host , ESDHC_SYSTEM_CONTROL ) ;
temp | = ( ESDHC_CLOCK_IPGEN | ESDHC_CLOCK_HCKEN | ESDHC_CLOCK_PEREN
| ( div < < ESDHC_DIVIDER_SHIFT )
| ( pre_div < < ESDHC_PREDIV_SHIFT ) ) ;
sdhci_writel ( host , temp , ESDHC_SYSTEM_CONTROL ) ;
mdelay ( 1 ) ;
out :
host - > clock = clock ;
2012-02-14 14:05:37 +08:00
}
2012-02-04 17:13:13 -05:00
# ifdef CONFIG_PM
static u32 esdhc_proctl ;
static void esdhc_of_suspend ( struct sdhci_host * host )
{
esdhc_proctl = sdhci_be32bs_readl ( host , SDHCI_HOST_CONTROL ) ;
}
static void esdhc_of_resume ( struct sdhci_host * host )
{
esdhc_of_enable_dma ( host ) ;
sdhci_be32bs_writel ( host , esdhc_proctl , SDHCI_HOST_CONTROL ) ;
}
# endif
2012-10-25 13:47:19 +08:00
static void esdhc_of_platform_init ( struct sdhci_host * host )
{
u32 vvn ;
vvn = in_be32 ( host - > ioaddr + SDHCI_SLOT_INT_STATUS ) ;
vvn = ( vvn & SDHCI_VENDOR_VER_MASK ) > > SDHCI_VENDOR_VER_SHIFT ;
if ( vvn = = VENDOR_V_22 )
host - > quirks2 | = SDHCI_QUIRK2_HOST_NO_CMD23 ;
2012-11-23 17:25:03 +08:00
if ( vvn > VENDOR_V_22 )
host - > quirks & = ~ SDHCI_QUIRK_NO_BUSY_IRQ ;
2012-10-25 13:47:19 +08:00
}
2013-06-27 12:00:05 -04:00
static int esdhc_pltfm_bus_width ( struct sdhci_host * host , int width )
{
u32 ctrl ;
switch ( width ) {
case MMC_BUS_WIDTH_8 :
ctrl = ESDHC_CTRL_8BITBUS ;
break ;
case MMC_BUS_WIDTH_4 :
ctrl = ESDHC_CTRL_4BITBUS ;
break ;
default :
ctrl = 0 ;
break ;
}
clrsetbits_be32 ( host - > ioaddr + SDHCI_HOST_CONTROL ,
ESDHC_CTRL_BUSWIDTH_MASK , ctrl ) ;
return 0 ;
}
2013-03-13 19:26:05 +01:00
static const struct sdhci_ops sdhci_esdhc_ops = {
2012-03-08 11:25:02 +08:00
. read_l = esdhc_readl ,
2011-07-20 17:13:36 -04:00
. read_w = esdhc_readw ,
2011-09-09 20:05:46 +08:00
. read_b = esdhc_readb ,
2012-12-04 10:41:28 +08:00
. write_l = esdhc_writel ,
2011-07-20 17:13:36 -04:00
. write_w = esdhc_writew ,
. write_b = esdhc_writeb ,
2012-02-14 14:05:37 +08:00
. set_clock = esdhc_of_set_clock ,
2011-07-20 17:13:36 -04:00
. enable_dma = esdhc_of_enable_dma ,
. get_max_clock = esdhc_of_get_max_clock ,
. get_min_clock = esdhc_of_get_min_clock ,
2012-10-25 13:47:19 +08:00
. platform_init = esdhc_of_platform_init ,
2012-02-04 17:13:13 -05:00
# ifdef CONFIG_PM
. platform_suspend = esdhc_of_suspend ,
. platform_resume = esdhc_of_resume ,
# endif
2012-12-04 10:41:28 +08:00
. adma_workaround = esdhci_of_adma_workaround ,
2013-06-27 12:00:05 -04:00
. platform_bus_width = esdhc_pltfm_bus_width ,
2011-07-20 17:13:36 -04:00
} ;
2013-03-13 19:26:03 +01:00
static const struct sdhci_pltfm_data sdhci_esdhc_pdata = {
2012-03-08 11:25:02 +08:00
/*
* card detection could be handled via GPIO
* eSDHC cannot support End Attribute in NOP ADMA descriptor
*/
2011-03-21 13:22:13 +08:00
. quirks = ESDHC_DEFAULT_QUIRKS | SDHCI_QUIRK_BROKEN_CARD_DETECTION
2012-03-08 11:25:02 +08:00
| SDHCI_QUIRK_NO_CARD_NO_RESET
| SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC ,
2011-07-20 17:13:36 -04:00
. ops = & sdhci_esdhc_ops ,
2009-12-17 15:27:20 -08:00
} ;
2011-05-27 23:48:14 +08:00
2012-11-19 13:23:06 -05:00
static int sdhci_esdhc_probe ( struct platform_device * pdev )
2011-05-27 23:48:14 +08:00
{
2013-06-27 12:00:05 -04:00
struct sdhci_host * host ;
2013-07-05 12:48:35 -04:00
struct device_node * np ;
2013-06-27 12:00:05 -04:00
int ret ;
host = sdhci_pltfm_init ( pdev , & sdhci_esdhc_pdata , 0 ) ;
if ( IS_ERR ( host ) )
return PTR_ERR ( host ) ;
sdhci_get_of_property ( pdev ) ;
2013-07-05 12:48:35 -04:00
np = pdev - > dev . of_node ;
if ( of_device_is_compatible ( np , " fsl,p2020-esdhc " ) ) {
/*
* Freescale messed up with P2020 as it has a non - standard
* host control register
*/
host - > quirks2 | = SDHCI_QUIRK2_BROKEN_HOST_CONTROL ;
}
2013-06-27 12:00:05 -04:00
/* call to generic mmc_of_parse to support additional capabilities */
mmc_of_parse ( host - > mmc ) ;
2013-08-26 09:19:24 +08:00
mmc_of_parse_voltage ( np , & host - > ocr_mask ) ;
2013-06-27 12:00:05 -04:00
ret = sdhci_add_host ( host ) ;
if ( ret )
sdhci_pltfm_free ( pdev ) ;
return ret ;
2011-05-27 23:48:14 +08:00
}
2012-11-19 13:26:03 -05:00
static int sdhci_esdhc_remove ( struct platform_device * pdev )
2011-05-27 23:48:14 +08:00
{
return sdhci_pltfm_unregister ( pdev ) ;
}
static const struct of_device_id sdhci_esdhc_of_match [ ] = {
{ . compatible = " fsl,mpc8379-esdhc " } ,
{ . compatible = " fsl,mpc8536-esdhc " } ,
{ . compatible = " fsl,esdhc " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , sdhci_esdhc_of_match ) ;
static struct platform_driver sdhci_esdhc_driver = {
. driver = {
. name = " sdhci-esdhc " ,
. owner = THIS_MODULE ,
. of_match_table = sdhci_esdhc_of_match ,
2011-11-03 11:09:45 +01:00
. pm = SDHCI_PLTFM_PMOPS ,
2011-05-27 23:48:14 +08:00
} ,
. probe = sdhci_esdhc_probe ,
2012-11-19 13:20:26 -05:00
. remove = sdhci_esdhc_remove ,
2011-05-27 23:48:14 +08:00
} ;
2011-11-26 12:55:43 +08:00
module_platform_driver ( sdhci_esdhc_driver ) ;
2011-05-27 23:48:14 +08:00
MODULE_DESCRIPTION ( " SDHCI OF driver for Freescale MPC eSDHC " ) ;
MODULE_AUTHOR ( " Xiaobo Xie <X.Xie@freescale.com>, "
" Anton Vorontsov <avorontsov@ru.mvista.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;