2009-06-04 13:57:29 +02:00
/*
* sdhci - pltfm . c Support for SDHCI platform devices
* Copyright ( c ) 2009 Intel Corporation
*
2012-01-16 14:13:03 +08:00
* Copyright ( c ) 2007 , 2011 Freescale Semiconductor , Inc .
2011-05-27 23:48:14 +08:00
* Copyright ( c ) 2009 MontaVista Software , Inc .
*
* Authors : Xiaobo Xie < X . Xie @ freescale . com >
* Anton Vorontsov < avorontsov @ ru . mvista . com >
*
2009-06-04 13:57:29 +02:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* 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 , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/* Supports:
* SDHCI platform devices
*
* Inspired by sdhci - pci . c , by Pierre Ossman
*/
2011-05-27 23:48:12 +08:00
# include <linux/err.h>
2011-07-03 15:15:51 -04:00
# include <linux/module.h>
2011-05-27 23:48:14 +08:00
# include <linux/of.h>
# ifdef CONFIG_PPC
# include <asm/machdep.h>
# endif
2010-08-10 18:01:47 -07:00
# include "sdhci-pltfm.h"
2009-06-04 13:57:29 +02:00
2013-01-28 19:27:12 +01:00
unsigned int sdhci_pltfm_clk_get_max_clock ( struct sdhci_host * host )
{
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
return clk_get_rate ( pltfm_host - > clk ) ;
}
EXPORT_SYMBOL_GPL ( sdhci_pltfm_clk_get_max_clock ) ;
2013-03-13 19:26:05 +01:00
static const struct sdhci_ops sdhci_pltfm_ops = {
2014-04-25 12:58:55 +01:00
. set_clock = sdhci_set_clock ,
2014-04-25 12:57:07 +01:00
. set_bus_width = sdhci_set_bus_width ,
2014-04-25 12:57:12 +01:00
. reset = sdhci_reset ,
2014-04-25 12:59:26 +01:00
. set_uhs_signaling = sdhci_set_uhs_signaling ,
2009-06-04 13:57:29 +02:00
} ;
2011-05-27 23:48:14 +08:00
# ifdef CONFIG_OF
static bool sdhci_of_wp_inverted ( struct device_node * np )
{
2012-05-13 00:14:24 -04:00
if ( of_get_property ( np , " sdhci,wp-inverted " , NULL ) | |
of_get_property ( np , " wp-inverted " , NULL ) )
2011-05-27 23:48:14 +08:00
return true ;
/* Old device trees don't have the wp-inverted property. */
# ifdef CONFIG_PPC
return machine_is ( mpc837x_rdb ) | | machine_is ( mpc837x_mds ) ;
# else
return false ;
# endif /* CONFIG_PPC */
}
void sdhci_get_of_property ( struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
struct sdhci_host * host = platform_get_drvdata ( pdev ) ;
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2012-05-13 00:14:24 -04:00
u32 bus_width ;
2011-05-27 23:48:14 +08:00
2015-02-04 14:33:53 +08:00
if ( of_get_property ( np , " sdhci,auto-cmd12 " , NULL ) )
host - > quirks | = SDHCI_QUIRK_MULTIBLOCK_READ_ACMD12 ;
2011-05-27 23:48:14 +08:00
2015-02-04 14:33:53 +08:00
if ( of_get_property ( np , " sdhci,1-bit-only " , NULL ) | |
( of_property_read_u32 ( np , " bus-width " , & bus_width ) = = 0 & &
bus_width = = 1 ) )
host - > quirks | = SDHCI_QUIRK_FORCE_1_BIT_DATA ;
2011-05-27 23:48:14 +08:00
2015-02-04 14:33:53 +08:00
if ( sdhci_of_wp_inverted ( np ) )
host - > quirks | = SDHCI_QUIRK_INVERTED_WRITE_PROTECT ;
2011-05-27 23:48:14 +08:00
2015-02-04 14:33:53 +08:00
if ( of_get_property ( np , " broken-cd " , NULL ) )
host - > quirks | = SDHCI_QUIRK_BROKEN_CARD_DETECTION ;
2012-09-15 12:20:00 +08:00
2015-02-04 14:33:53 +08:00
if ( of_get_property ( np , " no-1-8-v " , NULL ) )
host - > quirks2 | = SDHCI_QUIRK2_NO_1_8_V ;
2012-11-25 13:02:54 -05:00
2015-02-04 14:33:53 +08:00
if ( of_device_is_compatible ( np , " fsl,p2020-rev1-esdhc " ) )
host - > quirks | = SDHCI_QUIRK_BROKEN_DMA ;
2012-01-16 14:13:03 +08:00
2015-02-04 14:33:53 +08:00
if ( of_device_is_compatible ( np , " fsl,p2020-esdhc " ) | |
of_device_is_compatible ( np , " fsl,p1010-esdhc " ) | |
of_device_is_compatible ( np , " fsl,t4240-esdhc " ) | |
of_device_is_compatible ( np , " fsl,mpc8536-esdhc " ) )
host - > quirks | = SDHCI_QUIRK_BROKEN_TIMEOUT_VAL ;
2012-01-16 14:13:04 +08:00
2015-09-17 15:09:54 +02:00
of_property_read_u32 ( np , " clock-frequency " , & pltfm_host - > clock ) ;
2012-11-19 10:26:20 +05:30
2015-02-04 14:33:53 +08:00
if ( of_find_property ( np , " keep-power-in-suspend " , NULL ) )
host - > mmc - > pm_caps | = MMC_PM_KEEP_POWER ;
2012-11-19 10:26:20 +05:30
2015-10-21 11:10:02 +01:00
if ( of_property_read_bool ( np , " wakeup-source " ) | |
of_property_read_bool ( np , " enable-sdio-wakeup " ) ) /* legacy */
2015-02-04 14:33:53 +08:00
host - > mmc - > pm_caps | = MMC_PM_WAKE_SDIO_IRQ ;
2011-05-27 23:48:14 +08:00
}
# else
void sdhci_get_of_property ( struct platform_device * pdev ) { }
# endif /* CONFIG_OF */
2011-06-02 10:57:50 +08:00
EXPORT_SYMBOL_GPL ( sdhci_get_of_property ) ;
2011-05-27 23:48:14 +08:00
2011-05-27 23:48:12 +08:00
struct sdhci_host * sdhci_pltfm_init ( struct platform_device * pdev ,
2013-05-29 13:50:05 -07:00
const struct sdhci_pltfm_data * pdata ,
size_t priv_size )
2009-06-04 13:57:29 +02:00
{
struct sdhci_host * host ;
struct resource * iomem ;
2016-04-20 11:16:30 +09:00
void __iomem * ioaddr ;
2016-04-20 11:16:31 +09:00
int irq , ret ;
2009-06-04 13:57:29 +02:00
iomem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2016-04-20 11:16:30 +09:00
ioaddr = devm_ioremap_resource ( & pdev - > dev , iomem ) ;
if ( IS_ERR ( ioaddr ) ) {
ret = PTR_ERR ( ioaddr ) ;
2009-06-04 13:57:29 +02:00
goto err ;
}
2016-04-20 11:16:31 +09:00
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 ) {
dev_err ( & pdev - > dev , " failed to get IRQ number \n " ) ;
ret = irq ;
goto err ;
}
2014-08-12 11:37:52 +01:00
host = sdhci_alloc_host ( & pdev - > dev ,
sizeof ( struct sdhci_pltfm_host ) + priv_size ) ;
2009-06-04 13:57:29 +02:00
if ( IS_ERR ( host ) ) {
ret = PTR_ERR ( host ) ;
goto err ;
}
2016-04-20 11:16:30 +09:00
host - > ioaddr = ioaddr ;
2016-04-20 11:16:31 +09:00
host - > irq = irq ;
2011-05-27 23:48:12 +08:00
host - > hw_name = dev_name ( & pdev - > dev ) ;
2010-05-26 14:41:55 -07:00
if ( pdata & & pdata - > ops )
host - > ops = pdata - > ops ;
else
host - > ops = & sdhci_pltfm_ops ;
2013-05-26 13:59:26 -04:00
if ( pdata ) {
2010-05-26 14:41:55 -07:00
host - > quirks = pdata - > quirks ;
2013-05-26 13:59:26 -04:00
host - > quirks2 = pdata - > quirks2 ;
}
2009-06-04 13:57:29 +02:00
platform_set_drvdata ( pdev , host ) ;
2011-05-27 23:48:12 +08:00
return host ;
2009-06-04 13:57:29 +02:00
err :
2011-05-27 23:48:12 +08:00
dev_err ( & pdev - > dev , " %s failed %d \n " , __func__ , ret ) ;
return ERR_PTR ( ret ) ;
2009-06-04 13:57:29 +02:00
}
2011-06-02 10:57:50 +08:00
EXPORT_SYMBOL_GPL ( sdhci_pltfm_init ) ;
2009-06-04 13:57:29 +02:00
2011-05-27 23:48:12 +08:00
void sdhci_pltfm_free ( struct platform_device * pdev )
2009-06-04 13:57:29 +02:00
{
struct sdhci_host * host = platform_get_drvdata ( pdev ) ;
sdhci_free_host ( host ) ;
2011-05-27 23:48:12 +08:00
}
2011-06-02 10:57:50 +08:00
EXPORT_SYMBOL_GPL ( sdhci_pltfm_free ) ;
2009-06-04 13:57:29 +02:00
2011-05-27 23:48:12 +08:00
int sdhci_pltfm_register ( struct platform_device * pdev ,
2013-05-29 13:50:05 -07:00
const struct sdhci_pltfm_data * pdata ,
size_t priv_size )
2011-05-27 23:48:12 +08:00
{
struct sdhci_host * host ;
int ret = 0 ;
2013-05-29 13:50:05 -07:00
host = sdhci_pltfm_init ( pdev , pdata , priv_size ) ;
2011-05-27 23:48:12 +08:00
if ( IS_ERR ( host ) )
return PTR_ERR ( host ) ;
2011-05-27 23:48:14 +08:00
sdhci_get_of_property ( pdev ) ;
2011-05-27 23:48:12 +08:00
ret = sdhci_add_host ( host ) ;
if ( ret )
sdhci_pltfm_free ( pdev ) ;
return ret ;
2009-06-04 13:57:29 +02:00
}
2011-06-02 10:57:50 +08:00
EXPORT_SYMBOL_GPL ( sdhci_pltfm_register ) ;
2009-06-04 13:57:29 +02:00
2011-05-27 23:48:12 +08:00
int sdhci_pltfm_unregister ( struct platform_device * pdev )
{
struct sdhci_host * host = platform_get_drvdata ( pdev ) ;
2015-02-27 15:47:30 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2011-05-27 23:48:12 +08:00
int dead = ( readl ( host - > ioaddr + SDHCI_INT_STATUS ) = = 0xffffffff ) ;
sdhci_remove_host ( host , dead ) ;
2015-02-27 15:47:30 +08:00
clk_disable_unprepare ( pltfm_host - > clk ) ;
2011-05-27 23:48:12 +08:00
sdhci_pltfm_free ( pdev ) ;
return 0 ;
}
2011-06-02 10:57:50 +08:00
EXPORT_SYMBOL_GPL ( sdhci_pltfm_unregister ) ;
2010-08-10 18:01:47 -07:00
2016-07-27 12:48:04 +02:00
# ifdef CONFIG_PM_SLEEP
2016-07-27 12:25:38 +02:00
static int sdhci_pltfm_suspend ( struct device * dev )
2010-09-28 10:41:27 +02:00
{
2011-11-03 11:09:45 +01:00
struct sdhci_host * host = dev_get_drvdata ( dev ) ;
2010-09-28 10:41:27 +02:00
2017-03-20 19:50:32 +02:00
if ( host - > tuning_mode ! = SDHCI_TUNING_MODE_3 )
mmc_retune_needed ( host - > mmc ) ;
2011-11-03 11:09:45 +01:00
return sdhci_suspend_host ( host ) ;
2010-09-28 10:41:27 +02:00
}
2016-07-27 12:25:38 +02:00
static int sdhci_pltfm_resume ( struct device * dev )
2010-09-28 10:41:27 +02:00
{
2011-11-03 11:09:45 +01:00
struct sdhci_host * host = dev_get_drvdata ( dev ) ;
2010-09-28 10:41:27 +02:00
return sdhci_resume_host ( host ) ;
}
2016-07-27 12:48:04 +02:00
# endif
2011-11-03 11:09:45 +01:00
const struct dev_pm_ops sdhci_pltfm_pmops = {
2016-07-27 12:48:04 +02:00
SET_SYSTEM_SLEEP_PM_OPS ( sdhci_pltfm_suspend , sdhci_pltfm_resume )
2011-11-03 11:09:45 +01:00
} ;
EXPORT_SYMBOL_GPL ( sdhci_pltfm_pmops ) ;
2011-06-02 10:57:50 +08:00
static int __init sdhci_pltfm_drv_init ( void )
{
pr_info ( " sdhci-pltfm: SDHCI platform and OF driver helper \n " ) ;
return 0 ;
}
module_init ( sdhci_pltfm_drv_init ) ;
static void __exit sdhci_pltfm_drv_exit ( void )
{
}
module_exit ( sdhci_pltfm_drv_exit ) ;
MODULE_DESCRIPTION ( " SDHCI platform and OF driver helper " ) ;
MODULE_AUTHOR ( " Intel Corporation " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;