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>
# include <linux/interrupt.h>
# include <linux/pm_runtime.h>
# include <linux/mutex.h>
# include <linux/mfd/core.h>
2012-07-11 21:06:40 +09:00
# include <linux/mfd/samsung/s5m-core.h>
# include <linux/mfd/samsung/s5m-pmic.h>
# include <linux/mfd/samsung/s5m-rtc.h>
2011-12-23 17:28:08 +09:00
# include <linux/regmap.h>
2012-01-20 16:09:12 +09:00
static struct mfd_cell s5m8751_devs [ ] = {
{
. name = " s5m8751-pmic " ,
} , {
. name = " s5m-charger " ,
} , {
. name = " s5m8751-codec " ,
} ,
} ;
static struct mfd_cell s5m8763_devs [ ] = {
{
. name = " s5m8763-pmic " ,
} , {
. name = " s5m-rtc " ,
} , {
. name = " s5m-charger " ,
} ,
} ;
static struct mfd_cell s5m8767_devs [ ] = {
2011-12-23 17:28:08 +09:00
{
. name = " s5m8767-pmic " ,
} , {
. name = " s5m-rtc " ,
} ,
} ;
2012-07-11 21:06:55 +09:00
int sec_reg_read ( struct sec_pmic_dev * sec_pmic , u8 reg , void * dest )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
return regmap_read ( sec_pmic - > regmap , reg , dest ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
EXPORT_SYMBOL_GPL ( sec_reg_read ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
int sec_bulk_read ( struct sec_pmic_dev * sec_pmic , u8 reg , int count , u8 * buf )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
return regmap_bulk_read ( sec_pmic - > regmap , reg , buf , count ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
EXPORT_SYMBOL_GPL ( sec_bulk_read ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
int sec_reg_write ( struct sec_pmic_dev * sec_pmic , u8 reg , u8 value )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
return regmap_write ( sec_pmic - > regmap , reg , value ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
EXPORT_SYMBOL_GPL ( sec_reg_write ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
int sec_bulk_write ( struct sec_pmic_dev * sec_pmic , u8 reg , int count , u8 * buf )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
return regmap_raw_write ( sec_pmic - > regmap , reg , buf , count ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
EXPORT_SYMBOL_GPL ( sec_bulk_write ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
int sec_reg_update ( struct sec_pmic_dev * sec_pmic , u8 reg , u8 val , u8 mask )
2011-12-23 17:28:08 +09:00
{
2012-07-11 21:06:55 +09:00
return regmap_update_bits ( sec_pmic - > regmap , reg , mask , val ) ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
EXPORT_SYMBOL_GPL ( sec_reg_update ) ;
2011-12-23 17:28:08 +09:00
2012-07-11 21:06:55 +09:00
static struct regmap_config sec_regmap_config = {
2011-12-23 17:28:08 +09:00
. reg_bits = 8 ,
. val_bits = 8 ,
} ;
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 )
{
2012-07-11 21:06:55 +09:00
struct sec_platform_data * pdata = i2c - > dev . platform_data ;
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 ;
sec_pmic - > type = id - > driver_data ;
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 ;
2011-12-23 17:28:08 +09:00
}
2012-07-11 21:06:55 +09:00
sec_pmic - > regmap = devm_regmap_init_i2c ( i2c , & sec_regmap_config ) ;
if ( IS_ERR ( sec_pmic - > regmap ) ) {
ret = PTR_ERR ( sec_pmic - > regmap ) ;
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 ) ;
i2c_set_clientdata ( sec_pmic - > rtc , sec_pmic ) ;
2011-12-23 17:28:08 +09: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-01-20 16:09:12 +09:00
ARRAY_SIZE ( s5m8751_devs ) , NULL , 0 ) ;
break ;
case S5M8763X :
2012-07-11 21:06:55 +09:00
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s5m8763_devs ,
2012-01-20 16:09:12 +09:00
ARRAY_SIZE ( s5m8763_devs ) , NULL , 0 ) ;
break ;
case S5M8767X :
2012-07-11 21:06:55 +09:00
ret = mfd_add_devices ( sec_pmic - > dev , - 1 , s5m8767_devs ,
2012-01-20 16:09:12 +09:00
ARRAY_SIZE ( s5m8767_devs ) , NULL , 0 ) ;
break ;
default :
/* If this happens the probe function is problem */
BUG ( ) ;
}
2011-12-23 17:28:08 +09:00
if ( ret < 0 )
goto err ;
return ret ;
err :
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 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 ;
}
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 ,
} ,
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 " ) ;