2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2012-06-19 19:35:49 +04:00
/*
* Arizona - i2c . c - - Arizona I2C bus interface
*
* Copyright 2012 Wolfson Microelectronics plc
*
* Author : Mark Brown < broonie @ opensource . wolfsonmicro . com >
*/
# include <linux/err.h>
# include <linux/i2c.h>
# include <linux/module.h>
# include <linux/pm_runtime.h>
# include <linux/regmap.h>
# include <linux/regulator/consumer.h>
# include <linux/slab.h>
2013-10-16 12:56:56 +04:00
# include <linux/of.h>
2012-06-19 19:35:49 +04:00
# include <linux/mfd/arizona/core.h>
# include "arizona.h"
2012-11-19 22:23:04 +04:00
static int arizona_i2c_probe ( struct i2c_client * i2c ,
2014-07-02 17:28:46 +04:00
const struct i2c_device_id * id )
2012-06-19 19:35:49 +04:00
{
2021-01-21 00:49:54 +03:00
const void * match_data ;
2012-06-19 19:35:49 +04:00
struct arizona * arizona ;
2015-10-02 15:29:14 +03:00
const struct regmap_config * regmap_config = NULL ;
2021-01-21 00:49:54 +03:00
unsigned long type = 0 ;
2014-07-02 17:28:46 +04:00
int ret ;
2012-06-19 19:35:49 +04:00
2021-01-21 00:49:54 +03:00
match_data = device_get_match_data ( & i2c - > dev ) ;
if ( match_data )
type = ( unsigned long ) match_data ;
else if ( id )
2013-03-25 04:11:27 +04:00
type = id - > driver_data ;
switch ( type ) {
2012-06-19 19:35:49 +04:00
case WM5102 :
2015-10-02 15:29:14 +03:00
if ( IS_ENABLED ( CONFIG_MFD_WM5102 ) )
regmap_config = & wm5102_i2c_regmap ;
2012-06-19 19:35:49 +04:00
break ;
2012-07-10 15:37:58 +04:00
case WM5110 :
2015-01-17 18:21:22 +03:00
case WM8280 :
2015-10-02 15:29:14 +03:00
if ( IS_ENABLED ( CONFIG_MFD_WM5110 ) )
regmap_config = & wm5110_i2c_regmap ;
2012-07-10 15:37:58 +04:00
break ;
2013-06-13 12:43:29 +04:00
case WM8997 :
2015-10-02 15:29:14 +03:00
if ( IS_ENABLED ( CONFIG_MFD_WM8997 ) )
regmap_config = & wm8997_i2c_regmap ;
2013-06-13 12:43:29 +04:00
break ;
2015-07-03 18:16:35 +03:00
case WM8998 :
case WM1814 :
2015-10-02 15:29:14 +03:00
if ( IS_ENABLED ( CONFIG_MFD_WM8998 ) )
regmap_config = & wm8998_i2c_regmap ;
2015-07-03 18:16:35 +03:00
break ;
2012-06-19 19:35:49 +04:00
default :
2015-10-02 15:29:15 +03:00
dev_err ( & i2c - > dev , " Unknown device type %ld \n " , type ) ;
2012-06-19 19:35:49 +04:00
return - EINVAL ;
}
2015-10-02 15:29:14 +03:00
if ( ! regmap_config ) {
dev_err ( & i2c - > dev ,
" No kernel support for device type %ld \n " , type ) ;
return - EINVAL ;
}
2012-06-19 19:35:49 +04:00
arizona = devm_kzalloc ( & i2c - > dev , sizeof ( * arizona ) , GFP_KERNEL ) ;
if ( arizona = = NULL )
return - ENOMEM ;
arizona - > regmap = devm_regmap_init_i2c ( i2c , regmap_config ) ;
if ( IS_ERR ( arizona - > regmap ) ) {
ret = PTR_ERR ( arizona - > regmap ) ;
dev_err ( & i2c - > dev , " Failed to allocate register map: %d \n " ,
ret ) ;
return ret ;
}
2015-10-02 15:29:15 +03:00
arizona - > type = type ;
2012-06-19 19:35:49 +04:00
arizona - > dev = & i2c - > dev ;
arizona - > irq = i2c - > irq ;
return arizona_dev_init ( arizona ) ;
}
2022-08-15 11:02:30 +03:00
static void arizona_i2c_remove ( struct i2c_client * i2c )
2012-06-19 19:35:49 +04:00
{
struct arizona * arizona = dev_get_drvdata ( & i2c - > dev ) ;
2015-10-28 16:54:07 +03:00
2012-06-19 19:35:49 +04:00
arizona_dev_exit ( arizona ) ;
}
static const struct i2c_device_id arizona_i2c_id [ ] = {
{ " wm5102 " , WM5102 } ,
2012-07-10 15:37:58 +04:00
{ " wm5110 " , WM5110 } ,
2015-01-17 18:21:22 +03:00
{ " wm8280 " , WM8280 } ,
2013-06-13 12:43:29 +04:00
{ " wm8997 " , WM8997 } ,
2015-07-03 18:16:35 +03:00
{ " wm8998 " , WM8998 } ,
{ " wm1814 " , WM1814 } ,
2012-06-19 19:35:49 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , arizona_i2c_id ) ;
2021-09-28 19:30:35 +03:00
# ifdef CONFIG_OF
2022-01-04 19:54:35 +03:00
static const struct of_device_id arizona_i2c_of_match [ ] = {
2021-09-28 19:30:35 +03:00
{ . compatible = " wlf,wm5102 " , . data = ( void * ) WM5102 } ,
{ . compatible = " wlf,wm5110 " , . data = ( void * ) WM5110 } ,
{ . compatible = " wlf,wm8280 " , . data = ( void * ) WM8280 } ,
{ . compatible = " wlf,wm8997 " , . data = ( void * ) WM8997 } ,
{ . compatible = " wlf,wm8998 " , . data = ( void * ) WM8998 } ,
{ . compatible = " wlf,wm1814 " , . data = ( void * ) WM1814 } ,
{ } ,
} ;
# endif
2012-06-19 19:35:49 +04:00
static struct i2c_driver arizona_i2c_driver = {
. driver = {
. name = " arizona " ,
. pm = & arizona_pm_ops ,
2021-09-28 19:30:35 +03:00
. of_match_table = of_match_ptr ( arizona_i2c_of_match ) ,
2012-06-19 19:35:49 +04:00
} ,
. probe = arizona_i2c_probe ,
2012-11-19 22:20:24 +04:00
. remove = arizona_i2c_remove ,
2012-06-19 19:35:49 +04:00
. id_table = arizona_i2c_id ,
} ;
module_i2c_driver ( arizona_i2c_driver ) ;
2021-01-21 00:49:53 +03:00
MODULE_SOFTDEP ( " pre: arizona_ldo1 " ) ;
2012-06-19 19:35:49 +04:00
MODULE_DESCRIPTION ( " Arizona I2C bus interface " ) ;
MODULE_AUTHOR ( " Mark Brown <broonie@opensource.wolfsonmicro.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;