2012-08-15 10:28:46 -06:00
/*
* max8907 . c - mfd driver for MAX8907
*
* Copyright ( C ) 2010 Gyungoh Yoo < jack . yoo @ maxim - ic . com >
* Copyright ( C ) 2010 - 2012 , NVIDIA CORPORATION . All rights reserved .
*
* 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 .
*/
# include <linux/err.h>
# include <linux/i2c.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/mfd/core.h>
# include <linux/mfd/max8907.h>
# include <linux/module.h>
# include <linux/of_device.h>
# include <linux/regmap.h>
# include <linux/slab.h>
static struct mfd_cell max8907_cells [ ] = {
{ . name = " max8907-regulator " , } ,
{ . name = " max8907-rtc " , } ,
} ;
static bool max8907_gen_is_volatile_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MAX8907_REG_ON_OFF_IRQ1 :
case MAX8907_REG_ON_OFF_STAT :
case MAX8907_REG_ON_OFF_IRQ2 :
case MAX8907_REG_CHG_IRQ1 :
case MAX8907_REG_CHG_IRQ2 :
case MAX8907_REG_CHG_STAT :
return true ;
default :
return false ;
}
}
static bool max8907_gen_is_precious_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MAX8907_REG_ON_OFF_IRQ1 :
case MAX8907_REG_ON_OFF_IRQ2 :
case MAX8907_REG_CHG_IRQ1 :
case MAX8907_REG_CHG_IRQ2 :
return true ;
default :
return false ;
}
}
static bool max8907_gen_is_writeable_reg ( struct device * dev , unsigned int reg )
{
return ! max8907_gen_is_volatile_reg ( dev , reg ) ;
}
static const struct regmap_config max8907_regmap_gen_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. volatile_reg = max8907_gen_is_volatile_reg ,
. precious_reg = max8907_gen_is_precious_reg ,
. writeable_reg = max8907_gen_is_writeable_reg ,
. max_register = MAX8907_REG_LDO20VOUT ,
. cache_type = REGCACHE_RBTREE ,
} ;
static bool max8907_rtc_is_volatile_reg ( struct device * dev , unsigned int reg )
{
if ( reg < = MAX8907_REG_RTC_YEAR2 )
return true ;
switch ( reg ) {
case MAX8907_REG_RTC_STATUS :
case MAX8907_REG_RTC_IRQ :
return true ;
default :
return false ;
}
}
static bool max8907_rtc_is_precious_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MAX8907_REG_RTC_IRQ :
return true ;
default :
return false ;
}
}
static bool max8907_rtc_is_writeable_reg ( struct device * dev , unsigned int reg )
{
switch ( reg ) {
case MAX8907_REG_RTC_STATUS :
case MAX8907_REG_RTC_IRQ :
return false ;
default :
return true ;
}
}
static const struct regmap_config max8907_regmap_rtc_config = {
. reg_bits = 8 ,
. val_bits = 8 ,
. volatile_reg = max8907_rtc_is_volatile_reg ,
. precious_reg = max8907_rtc_is_precious_reg ,
. writeable_reg = max8907_rtc_is_writeable_reg ,
. max_register = MAX8907_REG_MPL_CNTL ,
. cache_type = REGCACHE_RBTREE ,
} ;
static const struct regmap_irq max8907_chg_irqs [ ] = {
{ . reg_offset = 0 , . mask = 1 < < 0 , } ,
{ . reg_offset = 0 , . mask = 1 < < 1 , } ,
{ . reg_offset = 0 , . mask = 1 < < 2 , } ,
{ . reg_offset = 1 , . mask = 1 < < 0 , } ,
{ . reg_offset = 1 , . mask = 1 < < 1 , } ,
{ . reg_offset = 1 , . mask = 1 < < 2 , } ,
{ . reg_offset = 1 , . mask = 1 < < 3 , } ,
{ . reg_offset = 1 , . mask = 1 < < 4 , } ,
{ . reg_offset = 1 , . mask = 1 < < 5 , } ,
{ . reg_offset = 1 , . mask = 1 < < 6 , } ,
{ . reg_offset = 1 , . mask = 1 < < 7 , } ,
} ;
static const struct regmap_irq_chip max8907_chg_irq_chip = {
. name = " max8907 chg " ,
. status_base = MAX8907_REG_CHG_IRQ1 ,
. mask_base = MAX8907_REG_CHG_IRQ1_MASK ,
. wake_base = MAX8907_REG_CHG_IRQ1_MASK ,
. irq_reg_stride = MAX8907_REG_CHG_IRQ2 - MAX8907_REG_CHG_IRQ1 ,
. num_regs = 2 ,
. irqs = max8907_chg_irqs ,
. num_irqs = ARRAY_SIZE ( max8907_chg_irqs ) ,
} ;
static const struct regmap_irq max8907_on_off_irqs [ ] = {
{ . reg_offset = 0 , . mask = 1 < < 0 , } ,
{ . reg_offset = 0 , . mask = 1 < < 1 , } ,
{ . reg_offset = 0 , . mask = 1 < < 2 , } ,
{ . reg_offset = 0 , . mask = 1 < < 3 , } ,
{ . reg_offset = 0 , . mask = 1 < < 4 , } ,
{ . reg_offset = 0 , . mask = 1 < < 5 , } ,
{ . reg_offset = 0 , . mask = 1 < < 6 , } ,
{ . reg_offset = 0 , . mask = 1 < < 7 , } ,
{ . reg_offset = 1 , . mask = 1 < < 0 , } ,
{ . reg_offset = 1 , . mask = 1 < < 1 , } ,
} ;
static const struct regmap_irq_chip max8907_on_off_irq_chip = {
. name = " max8907 on_off " ,
. status_base = MAX8907_REG_ON_OFF_IRQ1 ,
. mask_base = MAX8907_REG_ON_OFF_IRQ1_MASK ,
. irq_reg_stride = MAX8907_REG_ON_OFF_IRQ2 - MAX8907_REG_ON_OFF_IRQ1 ,
. num_regs = 2 ,
. irqs = max8907_on_off_irqs ,
. num_irqs = ARRAY_SIZE ( max8907_on_off_irqs ) ,
} ;
static const struct regmap_irq max8907_rtc_irqs [ ] = {
{ . reg_offset = 0 , . mask = 1 < < 2 , } ,
{ . reg_offset = 0 , . mask = 1 < < 3 , } ,
} ;
static const struct regmap_irq_chip max8907_rtc_irq_chip = {
. name = " max8907 rtc " ,
. status_base = MAX8907_REG_RTC_IRQ ,
. mask_base = MAX8907_REG_RTC_IRQ_MASK ,
. num_regs = 1 ,
. irqs = max8907_rtc_irqs ,
. num_irqs = ARRAY_SIZE ( max8907_rtc_irqs ) ,
} ;
2012-09-20 16:10:46 +02:00
static struct max8907 * max8907_pm_off ;
2012-09-18 16:51:19 -06:00
static void max8907_power_off ( void )
{
regmap_update_bits ( max8907_pm_off - > regmap_gen , MAX8907_REG_RESET_CNFG ,
MAX8907_MASK_POWER_OFF , MAX8907_MASK_POWER_OFF ) ;
}
2012-11-19 13:23:04 -05:00
static int max8907_i2c_probe ( struct i2c_client * i2c ,
2012-08-15 10:28:46 -06:00
const struct i2c_device_id * id )
{
struct max8907 * max8907 ;
int ret ;
2012-09-18 16:51:19 -06:00
struct max8907_platform_data * pdata = dev_get_platdata ( & i2c - > dev ) ;
bool pm_off = false ;
if ( pdata )
pm_off = pdata - > pm_off ;
else if ( i2c - > dev . of_node )
pm_off = of_property_read_bool ( i2c - > dev . of_node ,
" maxim,system-power-controller " ) ;
2012-08-15 10:28:46 -06:00
max8907 = devm_kzalloc ( & i2c - > dev , sizeof ( struct max8907 ) , GFP_KERNEL ) ;
if ( ! max8907 ) {
ret = - ENOMEM ;
goto err_alloc_drvdata ;
}
max8907 - > dev = & i2c - > dev ;
dev_set_drvdata ( max8907 - > dev , max8907 ) ;
max8907 - > i2c_gen = i2c ;
i2c_set_clientdata ( i2c , max8907 ) ;
max8907 - > regmap_gen = devm_regmap_init_i2c ( i2c ,
& max8907_regmap_gen_config ) ;
if ( IS_ERR ( max8907 - > regmap_gen ) ) {
ret = PTR_ERR ( max8907 - > regmap_gen ) ;
dev_err ( & i2c - > dev , " gen regmap init failed: %d \n " , ret ) ;
goto err_regmap_gen ;
}
max8907 - > i2c_rtc = i2c_new_dummy ( i2c - > adapter , MAX8907_RTC_I2C_ADDR ) ;
if ( ! max8907 - > i2c_rtc ) {
ret = - ENOMEM ;
goto err_dummy_rtc ;
}
i2c_set_clientdata ( max8907 - > i2c_rtc , max8907 ) ;
max8907 - > regmap_rtc = devm_regmap_init_i2c ( max8907 - > i2c_rtc ,
& max8907_regmap_rtc_config ) ;
if ( IS_ERR ( max8907 - > regmap_rtc ) ) {
ret = PTR_ERR ( max8907 - > regmap_rtc ) ;
dev_err ( & i2c - > dev , " rtc regmap init failed: %d \n " , ret ) ;
goto err_regmap_rtc ;
}
irq_set_status_flags ( max8907 - > i2c_gen - > irq , IRQ_NOAUTOEN ) ;
ret = regmap_add_irq_chip ( max8907 - > regmap_gen , max8907 - > i2c_gen - > irq ,
IRQF_ONESHOT | IRQF_SHARED , - 1 ,
& max8907_chg_irq_chip ,
& max8907 - > irqc_chg ) ;
if ( ret ! = 0 ) {
dev_err ( & i2c - > dev , " failed to add chg irq chip: %d \n " , ret ) ;
goto err_irqc_chg ;
}
ret = regmap_add_irq_chip ( max8907 - > regmap_gen , max8907 - > i2c_gen - > irq ,
IRQF_ONESHOT | IRQF_SHARED , - 1 ,
& max8907_on_off_irq_chip ,
& max8907 - > irqc_on_off ) ;
if ( ret ! = 0 ) {
dev_err ( & i2c - > dev , " failed to add on off irq chip: %d \n " , ret ) ;
goto err_irqc_on_off ;
}
ret = regmap_add_irq_chip ( max8907 - > regmap_rtc , max8907 - > i2c_gen - > irq ,
IRQF_ONESHOT | IRQF_SHARED , - 1 ,
& max8907_rtc_irq_chip ,
& max8907 - > irqc_rtc ) ;
if ( ret ! = 0 ) {
dev_err ( & i2c - > dev , " failed to add rtc irq chip: %d \n " , ret ) ;
goto err_irqc_rtc ;
}
enable_irq ( max8907 - > i2c_gen - > irq ) ;
ret = mfd_add_devices ( max8907 - > dev , - 1 , max8907_cells ,
ARRAY_SIZE ( max8907_cells ) , NULL , 0 , NULL ) ;
if ( ret ! = 0 ) {
dev_err ( & i2c - > dev , " failed to add MFD devices %d \n " , ret ) ;
goto err_add_devices ;
}
2012-09-18 16:51:19 -06:00
if ( pm_off & & ! pm_power_off ) {
max8907_pm_off = max8907 ;
pm_power_off = max8907_power_off ;
}
2012-08-15 10:28:46 -06:00
return 0 ;
err_add_devices :
regmap_del_irq_chip ( max8907 - > i2c_gen - > irq , max8907 - > irqc_rtc ) ;
err_irqc_rtc :
regmap_del_irq_chip ( max8907 - > i2c_gen - > irq , max8907 - > irqc_on_off ) ;
err_irqc_on_off :
regmap_del_irq_chip ( max8907 - > i2c_gen - > irq , max8907 - > irqc_chg ) ;
err_irqc_chg :
err_regmap_rtc :
i2c_unregister_device ( max8907 - > i2c_rtc ) ;
err_dummy_rtc :
err_regmap_gen :
err_alloc_drvdata :
return ret ;
}
2012-11-19 13:26:01 -05:00
static int max8907_i2c_remove ( struct i2c_client * i2c )
2012-08-15 10:28:46 -06:00
{
struct max8907 * max8907 = i2c_get_clientdata ( i2c ) ;
mfd_remove_devices ( max8907 - > dev ) ;
regmap_del_irq_chip ( max8907 - > i2c_gen - > irq , max8907 - > irqc_rtc ) ;
regmap_del_irq_chip ( max8907 - > i2c_gen - > irq , max8907 - > irqc_on_off ) ;
regmap_del_irq_chip ( max8907 - > i2c_gen - > irq , max8907 - > irqc_chg ) ;
i2c_unregister_device ( max8907 - > i2c_rtc ) ;
return 0 ;
}
# ifdef CONFIG_OF
static struct of_device_id max8907_of_match [ ] = {
{ . compatible = " maxim,max8907 " } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , max8907_of_match ) ;
# endif
static const struct i2c_device_id max8907_i2c_id [ ] = {
{ " max8907 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , max8907_i2c_id ) ;
static struct i2c_driver max8907_i2c_driver = {
. driver = {
. name = " max8907 " ,
. owner = THIS_MODULE ,
. of_match_table = of_match_ptr ( max8907_of_match ) ,
} ,
. probe = max8907_i2c_probe ,
. remove = max8907_i2c_remove ,
. id_table = max8907_i2c_id ,
} ;
static int __init max8907_i2c_init ( void )
{
int ret = - ENODEV ;
ret = i2c_add_driver ( & max8907_i2c_driver ) ;
if ( ret ! = 0 )
pr_err ( " Failed to register I2C driver: %d \n " , ret ) ;
return ret ;
}
subsys_initcall ( max8907_i2c_init ) ;
static void __exit max8907_i2c_exit ( void )
{
i2c_del_driver ( & max8907_i2c_driver ) ;
}
module_exit ( max8907_i2c_exit ) ;
MODULE_DESCRIPTION ( " MAX8907 multi-function core driver " ) ;
MODULE_AUTHOR ( " Gyungoh Yoo <jack.yoo@maxim-ic.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;