2018-08-22 00:02:17 +02:00
// SPDX-License-Identifier: GPL-2.0
2008-07-15 16:02:21 +01:00
/*
2017-05-10 11:25:25 +02:00
* Driver for the MMC / SD / SDIO cell found in :
*
* TC6393XB TC6391XB TC6387XB T7L66XB ASIC3
2008-07-15 16:02:21 +01:00
*
2017-05-30 14:50:52 +02:00
* Copyright ( C ) 2017 Renesas Electronics Corporation
* Copyright ( C ) 2017 Horms Solutions , Simon Horman
2011-03-23 12:42:44 +01:00
* Copyright ( C ) 2007 Ian Molton
* Copyright ( C ) 2004 Ian Molton
2008-07-15 16:02:21 +01:00
*/
2010-11-23 17:24:11 +01:00
2018-08-23 13:44:16 +09:00
# include <linux/delay.h>
2010-11-23 17:24:11 +01:00
# include <linux/device.h>
2008-07-15 16:02:21 +01:00
# include <linux/mfd/core.h>
# include <linux/mfd/tmio.h>
2010-11-23 17:24:11 +01:00
# include <linux/mmc/host.h>
# include <linux/module.h>
# include <linux/pagemap.h>
# include <linux/scatterlist.h>
2011-01-05 17:36:14 -05:00
2011-03-23 12:42:44 +01:00
# include "tmio_mmc.h"
2008-07-15 16:02:21 +01:00
2018-10-10 12:51:32 +09:00
/* Registers specific to this variant */
# define CTL_SDIO_REGS 0x100
# define CTL_CLK_AND_WAIT_CTL 0x138
# define CTL_RESET_SDIO 0x1e0
2018-08-23 13:44:16 +09:00
static void tmio_mmc_clk_start ( struct tmio_mmc_host * host )
{
sd_ctrl_write16 ( host , CTL_SD_CARD_CLK_CTL , CLK_CTL_SCLKEN |
sd_ctrl_read16 ( host , CTL_SD_CARD_CLK_CTL ) ) ;
usleep_range ( 10000 , 11000 ) ;
sd_ctrl_write16 ( host , CTL_CLK_AND_WAIT_CTL , 0x0100 ) ;
usleep_range ( 10000 , 11000 ) ;
}
static void tmio_mmc_clk_stop ( struct tmio_mmc_host * host )
{
sd_ctrl_write16 ( host , CTL_CLK_AND_WAIT_CTL , 0x0000 ) ;
usleep_range ( 10000 , 11000 ) ;
sd_ctrl_write16 ( host , CTL_SD_CARD_CLK_CTL , ~ CLK_CTL_SCLKEN &
sd_ctrl_read16 ( host , CTL_SD_CARD_CLK_CTL ) ) ;
usleep_range ( 10000 , 11000 ) ;
}
static void tmio_mmc_set_clock ( struct tmio_mmc_host * host ,
unsigned int new_clock )
{
2018-08-30 14:16:03 +02:00
unsigned int divisor ;
2018-08-23 13:44:20 +09:00
u32 clk = 0 ;
int clk_sel ;
2018-08-23 13:44:16 +09:00
if ( new_clock = = 0 ) {
tmio_mmc_clk_stop ( host ) ;
return ;
}
2018-08-23 13:44:20 +09:00
divisor = host - > pdata - > hclk / new_clock ;
2018-08-23 13:44:16 +09:00
2018-08-30 14:14:38 +02:00
/* bit7 set: 1/512, ... bit0 set: 1/4, all bits clear: 1/2 */
clk_sel = ( divisor < = 1 ) ;
clk = clk_sel ? 0 : ( roundup_pow_of_two ( divisor ) > > 2 ) ;
2018-08-23 13:44:16 +09:00
2018-08-23 13:44:20 +09:00
host - > pdata - > set_clk_div ( host - > pdev , clk_sel ) ;
2018-08-23 13:44:16 +09:00
sd_ctrl_write16 ( host , CTL_SD_CARD_CLK_CTL , ~ CLK_CTL_SCLKEN &
sd_ctrl_read16 ( host , CTL_SD_CARD_CLK_CTL ) ) ;
sd_ctrl_write16 ( host , CTL_SD_CARD_CLK_CTL , clk & CLK_CTL_DIV_MASK ) ;
usleep_range ( 10000 , 11000 ) ;
tmio_mmc_clk_start ( host ) ;
}
2018-10-10 12:51:31 +09:00
static void tmio_mmc_reset ( struct tmio_mmc_host * host )
{
sd_ctrl_write16 ( host , CTL_RESET_SDIO , 0x0000 ) ;
usleep_range ( 10000 , 11000 ) ;
sd_ctrl_write16 ( host , CTL_RESET_SDIO , 0x0001 ) ;
usleep_range ( 10000 , 11000 ) ;
}
2013-10-23 14:57:50 +02:00
# ifdef CONFIG_PM_SLEEP
static int tmio_mmc_suspend ( struct device * dev )
2008-07-15 16:02:21 +01:00
{
2013-10-23 14:57:50 +02:00
struct platform_device * pdev = to_platform_device ( dev ) ;
const struct mfd_cell * cell = mfd_get_cell ( pdev ) ;
2008-07-15 16:02:21 +01:00
int ret ;
2014-08-25 12:28:20 +02:00
ret = pm_runtime_force_suspend ( dev ) ;
2008-07-15 16:02:21 +01:00
/* Tell MFD core it can disable us now.*/
if ( ! ret & & cell - > disable )
2013-10-23 14:57:50 +02:00
cell - > disable ( pdev ) ;
2008-07-15 16:02:21 +01:00
return ret ;
}
2013-10-23 14:57:50 +02:00
static int tmio_mmc_resume ( struct device * dev )
2008-07-15 16:02:21 +01:00
{
2013-10-23 14:57:50 +02:00
struct platform_device * pdev = to_platform_device ( dev ) ;
const struct mfd_cell * cell = mfd_get_cell ( pdev ) ;
2008-07-15 16:02:21 +01:00
int ret = 0 ;
/* Tell the MFD core we are ready to be enabled */
2011-05-05 16:13:12 +00:00
if ( cell - > resume )
2013-10-23 14:57:50 +02:00
ret = cell - > resume ( pdev ) ;
2008-07-15 16:02:21 +01:00
2011-05-05 16:13:12 +00:00
if ( ! ret )
2014-08-25 12:28:20 +02:00
ret = pm_runtime_force_resume ( dev ) ;
2008-07-15 16:02:21 +01:00
return ret ;
}
# endif
2012-11-19 13:23:06 -05:00
static int tmio_mmc_probe ( struct platform_device * pdev )
2008-07-15 16:02:21 +01:00
{
2011-03-23 12:42:44 +01:00
const struct mfd_cell * cell = mfd_get_cell ( pdev ) ;
2009-06-04 20:12:31 +02:00
struct tmio_mmc_data * pdata ;
2008-07-15 16:02:21 +01:00
struct tmio_mmc_host * host ;
2013-11-20 00:30:55 -08:00
struct resource * res ;
2011-05-06 11:02:33 +00:00
int ret = - EINVAL , irq ;
2008-07-15 16:02:21 +01:00
2011-03-23 12:42:44 +01:00
if ( pdev - > num_resources ! = 2 )
2008-07-15 16:02:21 +01:00
goto out ;
2011-04-06 11:38:14 +02:00
pdata = pdev - > dev . platform_data ;
2009-06-04 20:12:34 +02:00
if ( ! pdata | | ! pdata - > hclk )
2009-06-04 20:12:31 +02:00
goto out ;
2009-06-04 20:12:34 +02:00
2011-05-06 11:02:33 +00:00
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 ) {
ret = irq ;
goto out ;
}
2008-07-15 16:02:21 +01:00
/* Tell the MFD core we are ready to be enabled */
if ( cell - > enable ) {
2011-03-23 12:42:44 +01:00
ret = cell - > enable ( pdev ) ;
2008-07-15 16:02:21 +01:00
if ( ret )
2011-03-23 12:42:44 +01:00
goto out ;
2008-07-15 16:02:21 +01:00
}
2013-11-20 00:30:55 -08:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2015-04-27 00:01:06 +01:00
if ( ! res ) {
ret = - EINVAL ;
goto cell_disable ;
}
2013-11-20 00:30:55 -08:00
2018-01-18 01:28:02 +09:00
host = tmio_mmc_host_alloc ( pdev , pdata ) ;
2018-01-18 01:28:01 +09:00
if ( IS_ERR ( host ) ) {
ret = PTR_ERR ( host ) ;
2010-02-17 16:38:23 +09:00
goto cell_disable ;
2018-01-18 01:28:01 +09:00
}
2008-07-15 16:02:21 +01:00
2015-01-13 04:58:20 +00:00
/* SD control register space size is 0x200, 0x400 for bus_shift=1 */
host - > bus_shift = resource_size ( res ) > > 10 ;
2018-08-23 13:44:16 +09:00
host - > set_clock = tmio_mmc_set_clock ;
2018-10-10 12:51:31 +09:00
host - > reset = tmio_mmc_reset ;
2015-01-13 04:58:20 +00:00
2018-01-18 01:28:02 +09:00
host - > mmc - > f_max = pdata - > hclk ;
host - > mmc - > f_min = pdata - > hclk / 512 ;
2018-01-18 01:28:04 +09:00
ret = tmio_mmc_host_probe ( host ) ;
2015-01-13 04:57:22 +00:00
if ( ret )
goto host_free ;
2015-04-27 00:01:36 +01:00
ret = devm_request_irq ( & pdev - > dev , irq , tmio_mmc_irq ,
2017-06-16 18:11:03 +02:00
IRQF_TRIGGER_FALLING ,
dev_name ( & pdev - > dev ) , host ) ;
2011-05-06 11:02:33 +00:00
if ( ret )
goto host_remove ;
2020-11-16 17:42:52 +01:00
pr_info ( " %s at 0x%p irq %d \n " , mmc_hostname ( host - > mmc ) , host - > ctl , irq ) ;
2008-07-15 16:02:21 +01:00
return 0 ;
2011-05-06 11:02:33 +00:00
host_remove :
tmio_mmc_host_remove ( host ) ;
2015-01-13 04:57:22 +00:00
host_free :
tmio_mmc_host_free ( host ) ;
2010-02-17 16:38:23 +09:00
cell_disable :
if ( cell - > disable )
2011-03-23 12:42:44 +01:00
cell - > disable ( pdev ) ;
2008-07-15 16:02:21 +01:00
out :
return ret ;
}
2012-11-19 13:26:03 -05:00
static int tmio_mmc_remove ( struct platform_device * pdev )
2008-07-15 16:02:21 +01:00
{
2011-03-23 12:42:44 +01:00
const struct mfd_cell * cell = mfd_get_cell ( pdev ) ;
2017-11-25 01:24:39 +09:00
struct tmio_mmc_host * host = platform_get_drvdata ( pdev ) ;
2008-07-15 16:02:21 +01:00
2017-11-25 01:24:39 +09:00
tmio_mmc_host_remove ( host ) ;
if ( cell - > disable )
cell - > disable ( pdev ) ;
2008-07-15 16:02:21 +01:00
return 0 ;
}
/* ------------------- device registration ----------------------- */
2013-10-23 14:57:50 +02:00
static const struct dev_pm_ops tmio_mmc_dev_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS ( tmio_mmc_suspend , tmio_mmc_resume )
2014-12-04 00:34:11 +01:00
SET_RUNTIME_PM_OPS ( tmio_mmc_host_runtime_suspend ,
2017-06-16 18:11:03 +02:00
tmio_mmc_host_runtime_resume , NULL )
2013-10-23 14:57:50 +02:00
} ;
2008-07-15 16:02:21 +01:00
static struct platform_driver tmio_mmc_driver = {
. driver = {
. name = " tmio-mmc " ,
2020-09-03 16:24:36 -07:00
. probe_type = PROBE_PREFER_ASYNCHRONOUS ,
2013-10-23 14:57:50 +02:00
. pm = & tmio_mmc_dev_pm_ops ,
2008-07-15 16:02:21 +01:00
} ,
. probe = tmio_mmc_probe ,
2012-11-19 13:20:26 -05:00
. remove = tmio_mmc_remove ,
2008-07-15 16:02:21 +01:00
} ;
2011-11-26 12:55:43 +08:00
module_platform_driver ( tmio_mmc_driver ) ;
2008-07-15 16:02:21 +01:00
MODULE_DESCRIPTION ( " Toshiba TMIO SD/MMC driver " ) ;
MODULE_AUTHOR ( " Ian Molton <spyro@f2s.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS ( " platform:tmio-mmc " ) ;