2009-06-04 13:57:29 +02:00
/*
* sdhci - pltfm . c Support for SDHCI platform devices
* Copyright ( c ) 2009 Intel Corporation
*
2011-05-27 23:48:14 +08:00
* Copyright ( c ) 2007 Freescale Semiconductor , Inc .
* 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
static struct sdhci_ops sdhci_pltfm_ops = {
} ;
2011-05-27 23:48:14 +08:00
# ifdef CONFIG_OF
static bool sdhci_of_wp_inverted ( struct device_node * np )
{
if ( of_get_property ( np , " sdhci,wp-inverted " , NULL ) )
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 ) ;
const __be32 * clk ;
int size ;
if ( of_device_is_available ( np ) ) {
if ( of_get_property ( np , " sdhci,auto-cmd12 " , NULL ) )
host - > quirks | = SDHCI_QUIRK_MULTIBLOCK_READ_ACMD12 ;
if ( of_get_property ( np , " sdhci,1-bit-only " , NULL ) )
host - > quirks | = SDHCI_QUIRK_FORCE_1_BIT_DATA ;
if ( sdhci_of_wp_inverted ( np ) )
host - > quirks | = SDHCI_QUIRK_INVERTED_WRITE_PROTECT ;
clk = of_get_property ( np , " clock-frequency " , & size ) ;
if ( clk & & size = = sizeof ( * clk ) & & * clk )
pltfm_host - > clock = be32_to_cpup ( clk ) ;
}
}
# 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 ,
struct sdhci_pltfm_data * pdata )
2009-06-04 13:57:29 +02:00
{
struct sdhci_host * host ;
2010-10-15 12:20:59 +02:00
struct sdhci_pltfm_host * pltfm_host ;
2011-07-01 09:35:28 +08:00
struct device_node * np = pdev - > dev . of_node ;
2009-06-04 13:57:29 +02:00
struct resource * iomem ;
int ret ;
iomem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! iomem ) {
ret = - ENOMEM ;
goto err ;
}
2010-05-26 14:41:56 -07:00
if ( resource_size ( iomem ) < 0x100 )
2011-05-27 23:48:12 +08:00
dev_err ( & pdev - > dev , " Invalid iomem size! \n " ) ;
2009-06-04 13:57:29 +02:00
2010-10-15 12:20:59 +02:00
/* Some PCI-based MFD need the parent here */
2011-07-01 09:35:28 +08:00
if ( pdev - > dev . parent ! = & platform_bus & & ! np )
2010-10-15 12:20:59 +02:00
host = sdhci_alloc_host ( pdev - > dev . parent , sizeof ( * pltfm_host ) ) ;
2009-06-04 13:57:29 +02:00
else
2010-10-15 12:20:59 +02:00
host = sdhci_alloc_host ( & pdev - > dev , sizeof ( * pltfm_host ) ) ;
2009-06-04 13:57:29 +02:00
if ( IS_ERR ( host ) ) {
ret = PTR_ERR ( host ) ;
goto err ;
}
2010-10-15 12:20:59 +02:00
pltfm_host = sdhci_priv ( host ) ;
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 ;
if ( pdata )
host - > quirks = pdata - > quirks ;
2009-06-04 13:57:29 +02:00
host - > irq = platform_get_irq ( pdev , 0 ) ;
if ( ! request_mem_region ( iomem - > start , resource_size ( iomem ) ,
mmc_hostname ( host - > mmc ) ) ) {
dev_err ( & pdev - > dev , " cannot request region \n " ) ;
ret = - EBUSY ;
goto err_request ;
}
host - > ioaddr = ioremap ( iomem - > start , resource_size ( iomem ) ) ;
if ( ! host - > ioaddr ) {
dev_err ( & pdev - > dev , " failed to remap registers \n " ) ;
ret = - ENOMEM ;
goto err_remap ;
}
platform_set_drvdata ( pdev , host ) ;
2011-05-27 23:48:12 +08:00
return host ;
2009-06-04 13:57:29 +02:00
err_remap :
release_mem_region ( iomem - > start , resource_size ( iomem ) ) ;
err_request :
sdhci_free_host ( host ) ;
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 ) ;
struct resource * iomem = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
iounmap ( host - > ioaddr ) ;
release_mem_region ( iomem - > start , resource_size ( iomem ) ) ;
sdhci_free_host ( host ) ;
platform_set_drvdata ( pdev , NULL ) ;
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 ,
struct sdhci_pltfm_data * pdata )
{
struct sdhci_host * host ;
int ret = 0 ;
host = sdhci_pltfm_init ( pdev , pdata ) ;
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 ) ;
int dead = ( readl ( host - > ioaddr + SDHCI_INT_STATUS ) = = 0xffffffff ) ;
sdhci_remove_host ( host , dead ) ;
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
2010-09-28 10:41:27 +02:00
# ifdef CONFIG_PM
2011-11-03 11:09:45 +01: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
2011-11-03 11:09:45 +01:00
return sdhci_suspend_host ( host ) ;
2010-09-28 10:41:27 +02:00
}
2011-11-03 11:09:45 +01: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 ) ;
}
2011-11-03 11:09:45 +01:00
const struct dev_pm_ops sdhci_pltfm_pmops = {
. suspend = sdhci_pltfm_suspend ,
. resume = sdhci_pltfm_resume ,
} ;
EXPORT_SYMBOL_GPL ( sdhci_pltfm_pmops ) ;
2010-09-28 10:41:27 +02:00
# endif /* CONFIG_PM */
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 " ) ;