2011-12-23 17:28:08 +09:00
/*
2012-07-11 21:06:55 +09:00
* sec - core . c
2011-12-23 17:28:08 +09:00
*
2012-07-11 21:06:55 +09:00
* Copyright ( c ) 2012 Samsung Electronics Co . , Ltd
2011-12-23 17:28:08 +09:00
* http : //www.samsung.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 .
*
*/
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/err.h>
# include <linux/slab.h>
# include <linux/i2c.h>
2013-10-16 14:26:52 +05:30
# include <linux/of.h>
2013-02-03 15:49:47 -08:00
# include <linux/of_irq.h>
2011-12-23 17:28:08 +09:00
# include <linux/interrupt.h>
# include <linux/pm_runtime.h>
# include <linux/mutex.h>
# include <linux/mfd/core.h>
2012-07-11 21:07:16 +09:00
# include <linux/mfd/samsung/core.h>
# include <linux/mfd/samsung/irq.h>
# include <linux/mfd/samsung/rtc.h>
2014-03-14 17:21:57 +05:30
# include <linux/mfd/samsung/s2mpa01.h>
2013-07-01 23:02:19 +01:00
# include <linux/mfd/samsung/s2mps11.h>
2014-02-28 11:41:44 +01:00
# include <linux/mfd/samsung/s2mps14.h>
2013-07-01 23:02:19 +01:00
# include <linux/mfd/samsung/s5m8763.h>
# include <linux/mfd/samsung/s5m8767.h>
2011-12-23 17:28:08 +09:00
# include <linux/regmap.h>
2013-11-18 14:33:06 +01:00
static const struct mfd_cell s5m8751_devs [ ] = {
2012-01-20 16:09:12 +09:00
{
. name = " s5m8751-pmic " ,
} , {
. name = " s5m-charger " ,
} , {
. name = " s5m8751-codec " ,
} ,
} ;
2013-11-18 14:33:06 +01:00
static const struct mfd_cell s5m8763_devs [ ] = {
2012-01-20 16:09:12 +09:00
{
. name = " s5m8763-pmic " ,
} , {
. name = " s5m-rtc " ,
} , {
. name = " s5m-charger " ,
} ,
} ;
2013-11-18 14:33:06 +01:00
static const struct mfd_cell s5m8767_devs [ ] = {
2011-12-23 17:28:08 +09:00
{
. name = " s5m8767-pmic " ,
} , {
. name = " s5m-rtc " ,
2013-12-26 15:49:00 +05:30
} , {
. name = " s5m8767-clk " ,
2014-03-17 10:19:17 +01:00
. of_compatible = " samsung,s5m8767-clk " ,
2013-12-26 15:49:00 +05:30
}
2011-12-23 17:28:08 +09:00
} ;
2013-11-18 14:33:06 +01:00
static const struct mfd_cell s2mps11_devs [ ] = {
2012-07-11 21:07:55 +09:00
{
. name = " s2mps11-pmic " ,
2013-07-07 17:14:21 +05:30
} , {
. name = " s2mps11-clk " ,
2014-03-17 10:19:17 +01:00
. of_compatible = " samsung,s2mps11-clk " ,
2013-07-07 17:14:21 +05:30
}
2012-07-11 21:07:55 +09:00
} ;
2014-02-28 11:41:44 +01:00
static const struct mfd_cell s2mps14_devs [ ] = {
{
. name = " s2mps14-pmic " ,
} , {
. name = " s2mps14-rtc " ,
} , {
. name = " s2mps14-clk " ,
2014-03-17 10:19:17 +01:00
. of_compatible = " samsung,s2mps14-clk " ,
2014-02-28 11:41:44 +01:00
}
} ;
2014-03-14 17:21:57 +05:30
static const struct mfd_cell s2mpa01_devs [ ] = {
{
. name = " s2mpa01-pmic " ,
} ,
} ;
2013-02-03 15:49:47 -08:00
# ifdef CONFIG_OF
static struct of_device_id sec_dt_match [ ] = {
{ . compatible = " samsung,s5m8767-pmic " ,
. data = ( void * ) S5M8767X ,
2014-03-14 17:21:57 +05:30
} , {
. compatible = " samsung,s2mps11-pmic " ,
2013-06-29 18:21:20 +05:30
. data = ( void * ) S2MPS11X ,
2014-03-14 17:21:57 +05:30
} , {
. compatible = " samsung,s2mps14-pmic " ,
2014-02-28 11:41:44 +01:00
. data = ( void * ) S2MPS14X ,
2014-03-14 17:21:57 +05:30
} , {
. compatible = " samsung,s2mpa01-pmic " ,
. data = ( void * ) S2MPA01 ,
} , {
/* Sentinel */
2014-02-28 11:41:44 +01:00
} ,
2013-02-03 15:49:47 -08:00
} ;
# endif
2014-03-14 17:21:57 +05:30
static bool s2mpa01_volatile ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case S2MPA01_REG_INT1M :
case S2MPA01_REG_INT2M :
case S2MPA01_REG_INT3M :
return false ;
default :
return true ;
}
}
2013-07-02 18:51:16 +01:00
static bool s2mps11_volatile ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case S2MPS11_REG_INT1M :
case S2MPS11_REG_INT2M :
case S2MPS11_REG_INT3M :
return false ;
default :
return true ;
}
}
static bool s5m8763_volatile ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case S5M8763_REG_IRQM1 :
case S5M8763_REG_IRQM2 :
case S5M8763_REG_IRQM3 :
case S5M8763_REG_IRQM4 :
return false ;
default :
return true ;
}
}
2013-11-28 09:09:43 +01:00
static const struct regmap_config sec_regmap_config = {
2011-12-23 17:28:08 +09:00
. reg_bits = 8 ,
. val_bits = 8 ,
} ;
2014-03-14 17:21:57 +05:30
static const struct regmap_config s2mpa01_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = S2MPA01_REG_LDO_OVCB4 ,
. volatile_reg = s2mpa01_volatile ,
. cache_type = REGCACHE_FLAT ,
} ;
2013-11-28 09:09:43 +01:00
static const struct regmap_config s2mps11_regmap_config = {
2013-07-01 23:02:19 +01:00
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = S2MPS11_REG_L38CTRL ,
2013-07-02 18:51:16 +01:00
. volatile_reg = s2mps11_volatile ,
. cache_type = REGCACHE_FLAT ,
2013-07-01 23:02:19 +01:00
} ;
2014-02-28 11:41:44 +01:00
static const struct regmap_config s2mps14_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = S2MPS14_REG_LDODSCH3 ,
. volatile_reg = s2mps11_volatile ,
. cache_type = REGCACHE_FLAT ,
} ;
2013-11-28 09:09:43 +01:00
static const struct regmap_config s5m8763_regmap_config = {
2013-07-01 23:02:19 +01:00
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = S5M8763_REG_LBCNFG2 ,
2013-07-02 18:51:16 +01:00
. volatile_reg = s5m8763_volatile ,
. cache_type = REGCACHE_FLAT ,
2013-07-01 23:02:19 +01:00
} ;
2013-11-28 09:09:43 +01:00
static const struct regmap_config s5m8767_regmap_config = {
2013-07-01 23:02:19 +01:00
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = S5M8767_REG_LDO28CTRL ,
2013-07-02 18:51:16 +01:00
. volatile_reg = s2mps11_volatile ,
. cache_type = REGCACHE_FLAT ,
2013-07-01 23:02:19 +01:00
} ;
2013-02-03 15:49:47 -08:00
2014-02-28 11:41:44 +01:00
static const struct regmap_config s5m_rtc_regmap_config = {
2013-12-12 17:12:31 -08:00
. reg_bits = 8 ,
. val_bits = 8 ,
2014-02-28 11:41:41 +01:00
. max_register = SEC_RTC_REG_MAX ,
2013-12-12 17:12:31 -08:00
} ;
2014-02-28 11:41:44 +01:00
static const struct regmap_config s2mps14_rtc_regmap_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. max_register = S2MPS_RTC_REG_MAX ,
} ;
2013-02-03 15:49:47 -08:00
# ifdef CONFIG_OF
/*
* Only the common platform data elements for s5m8767 are parsed here from the
* device tree . Other sub - modules of s5m8767 such as pmic , rtc , charger and
* others have to parse their own platform data elements from device tree .
*
* The s5m8767 platform data structure is instantiated here and the drivers for
* the sub - modules need not instantiate another instance while parsing their
* platform data .
*/
static struct sec_platform_data * sec_pmic_i2c_parse_dt_pdata (
struct device * dev )
{
struct sec_platform_data * pd ;
pd = devm_kzalloc ( dev , sizeof ( * pd ) , GFP_KERNEL ) ;
if ( ! pd ) {
dev_err ( dev , " could not allocate memory for pdata \n " ) ;
return ERR_PTR ( - ENOMEM ) ;
}
/*
* ToDo : the ' wakeup ' member in the platform data is more of a linux
* specfic information . Hence , there is no binding for that yet and
* not parsed here .
*/
return pd ;
}
# else
2013-02-04 18:31:25 +00:00
static struct sec_platform_data * sec_pmic_i2c_parse_dt_pdata (
2013-02-03 15:49:47 -08:00
struct device * dev )
{
2013-12-07 14:02:15 +08:00
return NULL ;
2013-02-03 15:49:47 -08:00
}
# endif
2014-03-13 11:14:07 +09:00
static inline unsigned long sec_i2c_get_driver_data ( struct i2c_client * i2c ,
2013-02-03 15:49:47 -08:00
const struct i2c_device_id * id )
{
# ifdef CONFIG_OF
if ( i2c - > dev . of_node ) {
const struct of_device_id * match ;
match = of_match_node ( sec_dt_match , i2c - > dev . of_node ) ;
2014-03-13 11:14:07 +09:00
return ( unsigned long ) match - > data ;
2013-02-03 15:49:47 -08:00
}
# endif
2014-03-13 11:14:07 +09:00
return id - > driver_data ;
2013-02-03 15:49:47 -08:00
}
2012-07-11 21:06:55 +09:00
static int sec_pmic_probe ( struct i2c_client * i2c ,
2011-12-23 17:28:08 +09:00
const struct i2c_device_id * id )
{
2013-07-30 17:10:05 +09:00
struct sec_platform_data * pdata = dev_get_platdata ( & i2c - > dev ) ;
2014-02-28 11:41:42 +01:00
const struct regmap_config * regmap , * regmap_rtc ;
2012-07-11 21:06:55 +09:00
struct sec_pmic_dev * sec_pmic ;
2012-02-28 14:23:55 +08:00
int ret ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
sec_pmic = devm_kzalloc ( & i2c - > dev , sizeof ( struct sec_pmic_dev ) ,
2012-01-20 16:09:11 +09:00
GFP_KERNEL ) ;
2012-07-11 21:06:55 +09:00
if ( sec_pmic = = NULL )
2011-12-23 17:28:08 +09:00
return - ENOMEM ;
2012-07-11 21:06:55 +09:00
i2c_set_clientdata ( i2c , sec_pmic ) ;
sec_pmic - > dev = & i2c - > dev ;
sec_pmic - > i2c = i2c ;
sec_pmic - > irq = i2c - > irq ;
2013-02-03 15:49:47 -08:00
sec_pmic - > type = sec_i2c_get_driver_data ( i2c , id ) ;
if ( sec_pmic - > dev - > of_node ) {
pdata = sec_pmic_i2c_parse_dt_pdata ( sec_pmic - > dev ) ;
if ( IS_ERR ( pdata ) ) {
ret = PTR_ERR ( pdata ) ;
return ret ;
}
pdata - > device_type = sec_pmic - > type ;
}
2011-12-23 17:28:08 +09:00
if ( pdata ) {
2012-07-11 21:06:55 +09:00
sec_pmic - > device_type = pdata - > device_type ;
sec_pmic - > ono = pdata - > ono ;
sec_pmic - > irq_base = pdata - > irq_base ;
sec_pmic - > wakeup = pdata - > wakeup ;
2013-02-03 15:49:47 -08:00
sec_pmic - > pdata = pdata ;
2011-12-23 17:28:08 +09:00
}
2013-07-01 23:02:19 +01:00
switch ( sec_pmic - > device_type ) {
2014-03-14 17:21:57 +05:30
case S2MPA01 :
regmap = & s2mpa01_regmap_config ;
2014-03-18 14:11:26 +01:00
/*
* The rtc - s5m driver does not support S2MPA01 and there
* is no mfd_cell for S2MPA01 RTC device .
* However we must pass something to devm_regmap_init_i2c ( )
* so use S5M - like regmap config even though it wouldn ' t work .
*/
regmap_rtc = & s5m_rtc_regmap_config ;
2014-03-14 17:21:57 +05:30
break ;
2013-07-01 23:02:19 +01:00
case S2MPS11X :
regmap = & s2mps11_regmap_config ;
2014-02-28 11:41:42 +01:00
/*
* The rtc - s5m driver does not support S2MPS11 and there
* is no mfd_cell for S2MPS11 RTC device .
* However we must pass something to devm_regmap_init_i2c ( )
* so use S5M - like regmap config even though it wouldn ' t work .
*/
2014-02-28 11:41:44 +01:00
regmap_rtc = & s5m_rtc_regmap_config ;
break ;
case S2MPS14X :
regmap = & s2mps14_regmap_config ;
regmap_rtc = & s2mps14_rtc_regmap_config ;
2013-07-01 23:02:19 +01:00
break ;
case S5M8763X :
regmap = & s5m8763_regmap_config ;
2014-02-28 11:41:44 +01:00
regmap_rtc = & s5m_rtc_regmap_config ;
2013-07-01 23:02:19 +01:00
break ;
case S5M8767X :
regmap = & s5m8767_regmap_config ;
2014-02-28 11:41:44 +01:00
regmap_rtc = & s5m_rtc_regmap_config ;
2013-07-01 23:02:19 +01:00
break ;
default :
regmap = & sec_regmap_config ;
2014-02-28 11:41:44 +01:00
regmap_rtc = & s5m_rtc_regmap_config ;
2013-07-01 23:02:19 +01:00
break ;
}
2013-12-12 17:12:31 -08:00
sec_pmic - > regmap_pmic = devm_regmap_init_i2c ( i2c , regmap ) ;
if ( IS_ERR ( sec_pmic - > regmap_pmic ) ) {
ret = PTR_ERR ( sec_pmic - > regmap_pmic ) ;
2011-12-23 17:28:08 +09:00
dev_err ( & i2c - > dev , " Failed to allocate register map: %d \n " ,
2012-02-28 14:23:55 +08:00
ret ) ;
2012-04-25 10:03:44 +08:00
return ret ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
sec_pmic - > rtc = i2c_new_dummy ( i2c - > adapter , RTC_I2C_ADDR ) ;
2014-02-11 10:12:25 +01:00
if ( ! sec_pmic - > rtc ) {
dev_err ( & i2c - > dev , " Failed to allocate I2C for RTC \n " ) ;
return - ENODEV ;
}
2012-07-11 21:06:55 +09:00
i2c_set_clientdata ( sec_pmic - > rtc , sec_pmic ) ;
2011-12-23 17:28:08 +09:00
2014-02-28 11:41:42 +01:00
sec_pmic - > regmap_rtc = devm_regmap_init_i2c ( sec_pmic - > rtc , regmap_rtc ) ;
2013-12-12 17:12:31 -08:00
if ( IS_ERR ( sec_pmic - > regmap_rtc ) ) {
ret = PTR_ERR ( sec_pmic - > regmap_rtc ) ;
dev_err ( & i2c - > dev , " Failed to allocate RTC register map: %d \n " ,
ret ) ;
2014-03-18 13:57:50 +01:00
goto err_regmap_rtc ;
2013-12-12 17:12:31 -08:00
}
2012-01-16 09:08:42 +09:00
if ( pdata & & pdata - > cfg_pmic_irq )
2011-12-23 17:28:08 +09:00
pdata - > cfg_pmic_irq ( ) ;
2012-07-11 21:06:55 +09:00
sec_irq_init ( sec_pmic ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
pm_runtime_set_active ( sec_pmic - > dev ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
switch ( sec_pmic - > device_type ) {
2012-01-20 16:09:12 +09:00
case S5M8751X :
2012-07-11 21:06:55 +09:00
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s5m8751_devs ,
2012-09-11 15:16:36 +08:00
ARRAY_SIZE ( s5m8751_devs ) , NULL , 0 , NULL ) ;
2012-01-20 16:09:12 +09:00
break ;
case S5M8763X :
2012-07-11 21:06:55 +09:00
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s5m8763_devs ,
2012-09-11 15:16:36 +08:00
ARRAY_SIZE ( s5m8763_devs ) , NULL , 0 , NULL ) ;
2012-01-20 16:09:12 +09:00
break ;
case S5M8767X :
2012-07-11 21:06:55 +09:00
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s5m8767_devs ,
2012-09-11 15:16:36 +08:00
ARRAY_SIZE ( s5m8767_devs ) , NULL , 0 , NULL ) ;
2012-01-20 16:09:12 +09:00
break ;
2014-03-14 17:21:57 +05:30
case S2MPA01 :
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s2mpa01_devs ,
ARRAY_SIZE ( s2mpa01_devs ) , NULL , 0 , NULL ) ;
break ;
2012-07-11 21:07:55 +09:00
case S2MPS11X :
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s2mps11_devs ,
2012-09-11 15:16:36 +08:00
ARRAY_SIZE ( s2mps11_devs ) , NULL , 0 , NULL ) ;
2012-07-11 21:07:55 +09:00
break ;
2014-02-28 11:41:44 +01:00
case S2MPS14X :
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s2mps14_devs ,
ARRAY_SIZE ( s2mps14_devs ) , NULL , 0 , NULL ) ;
break ;
2012-01-20 16:09:12 +09:00
default :
/* If this happens the probe function is problem */
BUG ( ) ;
}
2011-12-23 17:28:08 +09:00
2013-05-15 18:00:48 +03:00
if ( ret )
2014-03-18 13:57:50 +01:00
goto err_mfd ;
2011-12-23 17:28:08 +09:00
2013-11-26 14:50:27 +01:00
device_init_wakeup ( sec_pmic - > dev , sec_pmic - > wakeup ) ;
2011-12-23 17:28:08 +09:00
return ret ;
2014-03-18 13:57:50 +01:00
err_mfd :
2012-07-11 21:06:55 +09:00
sec_irq_exit ( sec_pmic ) ;
2014-03-18 13:57:50 +01:00
err_regmap_rtc :
2012-07-11 21:06:55 +09:00
i2c_unregister_device ( sec_pmic - > rtc ) ;
2011-12-23 17:28:08 +09:00
return ret ;
}
2012-07-11 21:06:55 +09:00
static int sec_pmic_remove ( struct i2c_client * i2c )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
struct sec_pmic_dev * sec_pmic = i2c_get_clientdata ( i2c ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
mfd_remove_devices ( sec_pmic - > dev ) ;
sec_irq_exit ( sec_pmic ) ;
i2c_unregister_device ( sec_pmic - > rtc ) ;
2011-12-23 17:28:08 +09:00
return 0 ;
}
2014-01-26 11:38:43 +01:00
# ifdef CONFIG_PM_SLEEP
2013-11-26 14:50:27 +01:00
static int sec_pmic_suspend ( struct device * dev )
{
struct i2c_client * i2c = container_of ( dev , struct i2c_client , dev ) ;
struct sec_pmic_dev * sec_pmic = i2c_get_clientdata ( i2c ) ;
if ( device_may_wakeup ( dev ) ) {
enable_irq_wake ( sec_pmic - > irq ) ;
/*
* PMIC IRQ must be disabled during suspend for RTC alarm
* to work properly .
* When device is woken up from suspend by RTC Alarm , an
* interrupt occurs before resuming I2C bus controller .
* The interrupt is handled by regmap_irq_thread which tries
* to read RTC registers . This read fails ( I2C is still
* suspended ) and RTC Alarm interrupt is disabled .
*/
disable_irq ( sec_pmic - > irq ) ;
}
return 0 ;
}
static int sec_pmic_resume ( struct device * dev )
{
struct i2c_client * i2c = container_of ( dev , struct i2c_client , dev ) ;
struct sec_pmic_dev * sec_pmic = i2c_get_clientdata ( i2c ) ;
if ( device_may_wakeup ( dev ) ) {
disable_irq_wake ( sec_pmic - > irq ) ;
enable_irq ( sec_pmic - > irq ) ;
}
return 0 ;
}
2014-01-26 11:38:43 +01:00
# endif /* CONFIG_PM_SLEEP */
2013-11-26 14:50:27 +01:00
static SIMPLE_DEV_PM_OPS ( sec_pmic_pm_ops , sec_pmic_suspend , sec_pmic_resume ) ;
2012-07-11 21:06:55 +09:00
static const struct i2c_device_id sec_pmic_id [ ] = {
{ " sec_pmic " , 0 } ,
2011-12-23 17:28:08 +09:00
{ }
} ;
2012-07-11 21:06:55 +09:00
MODULE_DEVICE_TABLE ( i2c , sec_pmic_id ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
static struct i2c_driver sec_pmic_driver = {
2011-12-23 17:28:08 +09:00
. driver = {
2012-07-11 21:06:55 +09:00
. name = " sec_pmic " ,
2011-12-23 17:28:08 +09:00
. owner = THIS_MODULE ,
2013-11-26 14:50:27 +01:00
. pm = & sec_pmic_pm_ops ,
2013-02-03 15:49:47 -08:00
. of_match_table = of_match_ptr ( sec_dt_match ) ,
2011-12-23 17:28:08 +09:00
} ,
2012-07-11 21:06:55 +09:00
. probe = sec_pmic_probe ,
. remove = sec_pmic_remove ,
. id_table = sec_pmic_id ,
2011-12-23 17:28:08 +09:00
} ;
2012-07-11 21:06:55 +09:00
static int __init sec_pmic_init ( void )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
return i2c_add_driver ( & sec_pmic_driver ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
subsys_initcall ( sec_pmic_init ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
static void __exit sec_pmic_exit ( void )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
i2c_del_driver ( & sec_pmic_driver ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
module_exit ( sec_pmic_exit ) ;
2011-12-23 17:28:08 +09:00
MODULE_AUTHOR ( " Sangbeom Kim <sbkim73@samsung.com> " ) ;
MODULE_DESCRIPTION ( " Core support for the S5M MFD " ) ;
MODULE_LICENSE ( " GPL " ) ;