2020-04-07 20:38:49 +03:00
// SPDX-License-Identifier: GPL-2.0-only
/* MCP23S08 I2C GPIO driver */
# include <linux/i2c.h>
# include <linux/mod_devicetable.h>
# include <linux/module.h>
# include <linux/regmap.h>
# include "pinctrl-mcp23s08.h"
2022-11-18 23:44:04 +01:00
static int mcp230xx_probe ( struct i2c_client * client )
2020-04-07 20:38:49 +03:00
{
2023-10-01 16:01:13 +01:00
const struct mcp23s08_info * info ;
2020-04-07 20:38:49 +03:00
struct device * dev = & client - > dev ;
struct mcp23s08 * mcp ;
int ret ;
mcp = devm_kzalloc ( dev , sizeof ( * mcp ) , GFP_KERNEL ) ;
if ( ! mcp )
return - ENOMEM ;
2023-10-01 16:01:13 +01:00
info = i2c_get_match_data ( client ) ;
if ( ! info )
return dev_err_probe ( dev , - EINVAL , " invalid device type \n " ) ;
2020-04-07 20:38:49 +03:00
2023-10-01 16:01:13 +01:00
mcp - > reg_shift = info - > reg_shift ;
mcp - > chip . ngpio = info - > ngpio ;
mcp - > chip . label = info - > label ;
mcp - > regmap = devm_regmap_init_i2c ( client , info - > regmap ) ;
2020-04-07 20:38:49 +03:00
if ( IS_ERR ( mcp - > regmap ) )
return PTR_ERR ( mcp - > regmap ) ;
mcp - > irq = client - > irq ;
mcp - > pinctrl_desc . name = " mcp23xxx-pinctrl " ;
2023-10-01 16:01:13 +01:00
ret = mcp23s08_probe_one ( mcp , dev , client - > addr , info - > type , - 1 ) ;
2020-04-07 20:38:49 +03:00
if ( ret )
return ret ;
i2c_set_clientdata ( client , mcp ) ;
return 0 ;
}
2023-10-01 16:01:13 +01:00
static const struct mcp23s08_info mcp23008_i2c = {
. regmap = & mcp23x08_regmap ,
. label = " mcp23008 " ,
. type = MCP_TYPE_008 ,
. ngpio = 8 ,
. reg_shift = 0 ,
} ;
static const struct mcp23s08_info mcp23017_i2c = {
. regmap = & mcp23x17_regmap ,
. label = " mcp23017 " ,
. type = MCP_TYPE_017 ,
. ngpio = 16 ,
. reg_shift = 1 ,
} ;
static const struct mcp23s08_info mcp23018_i2c = {
. regmap = & mcp23x17_regmap ,
. label = " mcp23018 " ,
. type = MCP_TYPE_018 ,
. ngpio = 16 ,
. reg_shift = 1 ,
} ;
2020-04-07 20:38:49 +03:00
static const struct i2c_device_id mcp230xx_id [ ] = {
2023-10-01 16:01:13 +01:00
{ " mcp23008 " , ( kernel_ulong_t ) & mcp23008_i2c } ,
{ " mcp23017 " , ( kernel_ulong_t ) & mcp23017_i2c } ,
{ " mcp23018 " , ( kernel_ulong_t ) & mcp23018_i2c } ,
2020-04-07 20:38:49 +03:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , mcp230xx_id ) ;
static const struct of_device_id mcp23s08_i2c_of_match [ ] = {
2023-10-01 16:01:13 +01:00
{ . compatible = " microchip,mcp23008 " , . data = & mcp23008_i2c } ,
{ . compatible = " microchip,mcp23017 " , . data = & mcp23017_i2c } ,
{ . compatible = " microchip,mcp23018 " , . data = & mcp23018_i2c } ,
2020-04-07 20:38:49 +03:00
/* NOTE: The use of the mcp prefix is deprecated and will be removed. */
2023-10-01 16:01:13 +01:00
{ . compatible = " mcp,mcp23008 " , . data = & mcp23008_i2c } ,
{ . compatible = " mcp,mcp23017 " , . data = & mcp23017_i2c } ,
2020-04-07 20:38:49 +03:00
{ }
} ;
MODULE_DEVICE_TABLE ( of , mcp23s08_i2c_of_match ) ;
static struct i2c_driver mcp230xx_driver = {
. driver = {
. name = " mcp230xx " ,
. of_match_table = mcp23s08_i2c_of_match ,
} ,
2023-05-25 22:42:58 +02:00
. probe = mcp230xx_probe ,
2020-04-07 20:38:49 +03:00
. id_table = mcp230xx_id ,
} ;
static int __init mcp23s08_i2c_init ( void )
{
return i2c_add_driver ( & mcp230xx_driver ) ;
}
/*
* Register after I ² C postcore initcall and before
* subsys initcalls that may rely on these GPIOs .
*/
subsys_initcall ( mcp23s08_i2c_init ) ;
static void mcp23s08_i2c_exit ( void )
{
i2c_del_driver ( & mcp230xx_driver ) ;
}
module_exit ( mcp23s08_i2c_exit ) ;
MODULE_LICENSE ( " GPL " ) ;