2019-05-27 09:55:01 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2011-12-12 18:36:56 +04:00
/*
* I2C access for DA9052 PMICs .
*
* Copyright ( c ) 2011 Dialog Semiconductor Ltd .
*
* Author : David Dajun Chen < dchen @ diasemi . com >
*/
# include <linux/device.h>
# include <linux/module.h>
# include <linux/input.h>
# include <linux/mfd/core.h>
# include <linux/i2c.h>
# include <linux/err.h>
# include <linux/mfd/da9052/da9052.h>
# include <linux/mfd/da9052/reg.h>
2012-04-15 20:01:50 +04:00
# ifdef CONFIG_OF
# include <linux/of.h>
# include <linux/of_device.h>
# endif
2013-01-25 12:33:49 +04:00
/* I2C safe register check */
static inline bool i2c_safe_reg ( unsigned char reg )
{
switch ( reg ) {
case DA9052_STATUS_A_REG :
case DA9052_STATUS_B_REG :
case DA9052_STATUS_C_REG :
case DA9052_STATUS_D_REG :
case DA9052_ADC_RES_L_REG :
case DA9052_ADC_RES_H_REG :
case DA9052_VDD_RES_REG :
case DA9052_ICHG_AV_REG :
case DA9052_TBAT_RES_REG :
case DA9052_ADCIN4_RES_REG :
case DA9052_ADCIN5_RES_REG :
case DA9052_ADCIN6_RES_REG :
case DA9052_TJUNC_RES_REG :
case DA9052_TSI_X_MSB_REG :
case DA9052_TSI_Y_MSB_REG :
case DA9052_TSI_LSB_REG :
case DA9052_TSI_Z_MSB_REG :
return true ;
default :
return false ;
}
}
/*
* There is an issue with DA9052 and DA9053_AA / BA / BB PMIC where the PMIC
* gets lockup up or fails to respond following a system reset .
* This fix is to follow any read or write with a dummy read to a safe
* register .
*/
2013-02-12 00:48:01 +04:00
static int da9052_i2c_fix ( struct da9052 * da9052 , unsigned char reg )
2013-01-25 12:33:49 +04:00
{
int val ;
switch ( da9052 - > chip_id ) {
case DA9052 :
case DA9053_AA :
case DA9053_BA :
case DA9053_BB :
/* A dummy read to a safe register address. */
2015-10-28 19:51:11 +03:00
if ( ! i2c_safe_reg ( reg ) )
2013-01-25 12:33:49 +04:00
return regmap_read ( da9052 - > regmap ,
DA9052_PARK_REGISTER ,
& val ) ;
break ;
2014-02-19 20:32:47 +04:00
case DA9053_BC :
2013-01-25 12:33:49 +04:00
default :
/*
* For other chips parking of I2C register
* to a safe place is not required .
*/
break ;
}
return 0 ;
}
2013-09-02 11:46:11 +04:00
/*
* According to errata item 24 , multiwrite mode should be avoided
* in order to prevent register data corruption after power - down .
*/
static int da9052_i2c_disable_multiwrite ( struct da9052 * da9052 )
2011-12-12 18:36:56 +04:00
{
int reg_val , ret ;
ret = regmap_read ( da9052 - > regmap , DA9052_CONTROL_B_REG , & reg_val ) ;
if ( ret < 0 )
return ret ;
2013-09-02 11:46:11 +04:00
if ( ! ( reg_val & DA9052_CONTROL_B_WRITEMODE ) ) {
reg_val | = DA9052_CONTROL_B_WRITEMODE ;
2011-12-12 18:36:56 +04:00
ret = regmap_write ( da9052 - > regmap , DA9052_CONTROL_B_REG ,
reg_val ) ;
if ( ret < 0 )
return ret ;
}
return 0 ;
}
2012-07-13 20:24:26 +04:00
static const struct i2c_device_id da9052_i2c_id [ ] = {
2012-04-15 20:01:50 +04:00
{ " da9052 " , DA9052 } ,
{ " da9053-aa " , DA9053_AA } ,
{ " da9053-ba " , DA9053_BA } ,
{ " da9053-bb " , DA9053_BB } ,
2014-02-19 20:32:47 +04:00
{ " da9053-bc " , DA9053_BC } ,
2012-04-15 20:01:50 +04:00
{ }
} ;
2021-05-12 09:33:46 +03:00
MODULE_DEVICE_TABLE ( i2c , da9052_i2c_id ) ;
2012-04-15 20:01:50 +04:00
# ifdef CONFIG_OF
static const struct of_device_id dialog_dt_ids [ ] = {
{ . compatible = " dlg,da9052 " , . data = & da9052_i2c_id [ 0 ] } ,
{ . compatible = " dlg,da9053-aa " , . data = & da9052_i2c_id [ 1 ] } ,
2014-02-19 20:32:47 +04:00
{ . compatible = " dlg,da9053-ba " , . data = & da9052_i2c_id [ 2 ] } ,
2012-04-15 20:01:50 +04:00
{ . compatible = " dlg,da9053-bb " , . data = & da9052_i2c_id [ 3 ] } ,
2014-02-19 20:32:47 +04:00
{ . compatible = " dlg,da9053-bc " , . data = & da9052_i2c_id [ 4 ] } ,
2012-04-15 20:01:50 +04:00
{ /* sentinel */ }
} ;
# endif
2022-11-19 01:42:40 +03:00
static int da9052_i2c_probe ( struct i2c_client * client )
2011-12-12 18:36:56 +04:00
{
2022-11-19 01:42:40 +03:00
const struct i2c_device_id * id = i2c_client_get_device_id ( client ) ;
2011-12-12 18:36:56 +04:00
struct da9052 * da9052 ;
int ret ;
2012-05-11 05:29:51 +04:00
da9052 = devm_kzalloc ( & client - > dev , sizeof ( struct da9052 ) , GFP_KERNEL ) ;
2011-12-12 18:36:56 +04:00
if ( ! da9052 )
return - ENOMEM ;
da9052 - > dev = & client - > dev ;
da9052 - > chip_irq = client - > irq ;
2013-01-25 12:33:49 +04:00
da9052 - > fix_io = da9052_i2c_fix ;
2011-12-12 18:36:56 +04:00
i2c_set_clientdata ( client , da9052 ) ;
2012-05-11 05:29:51 +04:00
da9052 - > regmap = devm_regmap_init_i2c ( client , & da9052_regmap_config ) ;
2011-12-12 18:36:56 +04:00
if ( IS_ERR ( da9052 - > regmap ) ) {
ret = PTR_ERR ( da9052 - > regmap ) ;
dev_err ( & client - > dev , " Failed to allocate register map: %d \n " ,
ret ) ;
2012-05-11 05:29:51 +04:00
return ret ;
2011-12-12 18:36:56 +04:00
}
2013-09-02 11:46:11 +04:00
ret = da9052_i2c_disable_multiwrite ( da9052 ) ;
2011-12-12 18:36:56 +04:00
if ( ret < 0 )
2012-05-11 05:29:51 +04:00
return ret ;
2011-12-12 18:36:56 +04:00
2012-04-15 20:01:50 +04:00
# ifdef CONFIG_OF
2021-05-26 15:47:01 +03:00
if ( ! id )
id = of_device_get_match_data ( & client - > dev ) ;
2012-04-15 20:01:50 +04:00
# endif
if ( ! id ) {
ret = - ENODEV ;
dev_err ( & client - > dev , " id is null. \n " ) ;
2012-05-11 05:29:51 +04:00
return ret ;
2012-04-15 20:01:50 +04:00
}
2015-09-29 14:26:07 +03:00
return da9052_device_init ( da9052 , id - > driver_data ) ;
2011-12-12 18:36:56 +04:00
}
2022-08-15 11:02:30 +03:00
static void da9052_i2c_remove ( struct i2c_client * client )
2011-12-12 18:36:56 +04:00
{
struct da9052 * da9052 = i2c_get_clientdata ( client ) ;
da9052_device_exit ( da9052 ) ;
}
static struct i2c_driver da9052_i2c_driver = {
2022-11-19 01:42:40 +03:00
. probe_new = da9052_i2c_probe ,
2012-11-19 22:20:24 +04:00
. remove = da9052_i2c_remove ,
2011-12-12 18:36:56 +04:00
. id_table = da9052_i2c_id ,
. driver = {
. name = " da9052 " ,
2012-04-15 20:01:50 +04:00
# ifdef CONFIG_OF
. of_match_table = dialog_dt_ids ,
# endif
2011-12-12 18:36:56 +04:00
} ,
} ;
static int __init da9052_i2c_init ( void )
{
int ret ;
ret = i2c_add_driver ( & da9052_i2c_driver ) ;
if ( ret ! = 0 ) {
pr_err ( " DA9052 I2C registration failed %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
subsys_initcall ( da9052_i2c_init ) ;
static void __exit da9052_i2c_exit ( void )
{
i2c_del_driver ( & da9052_i2c_driver ) ;
}
module_exit ( da9052_i2c_exit ) ;
MODULE_AUTHOR ( " David Dajun Chen <dchen@diasemi.com> " ) ;
MODULE_DESCRIPTION ( " I2C driver for Dialog DA9052 PMIC " ) ;