提交 9e1e7263 编写于 作者: M Matt Porter 提交者: Lee Jones

mfd: bcm590xx: Add support for secondary I2C slave address

BCM590xx utilizes a secondary I2C slave address to access additional
register space. Add support for the secondary address space by
instantiating a dummy I2C device with the appropriate secondary
I2C slave address. Also expose a secondary regmap register space so
that MFD drivers can access this secondary i2c slave address space.
Signed-off-by: NMatt Porter <mporter@linaro.org>
Signed-off-by: NLee Jones <lee.jones@linaro.org>
上级 bb7f32fe
...@@ -28,39 +28,71 @@ static const struct mfd_cell bcm590xx_devs[] = { ...@@ -28,39 +28,71 @@ static const struct mfd_cell bcm590xx_devs[] = {
}, },
}; };
static const struct regmap_config bcm590xx_regmap_config = { static const struct regmap_config bcm590xx_regmap_config_pri = {
.reg_bits = 8, .reg_bits = 8,
.val_bits = 8, .val_bits = 8,
.max_register = BCM590XX_MAX_REGISTER, .max_register = BCM590XX_MAX_REGISTER_PRI,
.cache_type = REGCACHE_RBTREE, .cache_type = REGCACHE_RBTREE,
}; };
static int bcm590xx_i2c_probe(struct i2c_client *i2c, static const struct regmap_config bcm590xx_regmap_config_sec = {
.reg_bits = 8,
.val_bits = 8,
.max_register = BCM590XX_MAX_REGISTER_SEC,
.cache_type = REGCACHE_RBTREE,
};
static int bcm590xx_i2c_probe(struct i2c_client *i2c_pri,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
struct bcm590xx *bcm590xx; struct bcm590xx *bcm590xx;
int ret; int ret;
bcm590xx = devm_kzalloc(&i2c->dev, sizeof(*bcm590xx), GFP_KERNEL); bcm590xx = devm_kzalloc(&i2c_pri->dev, sizeof(*bcm590xx), GFP_KERNEL);
if (!bcm590xx) if (!bcm590xx)
return -ENOMEM; return -ENOMEM;
i2c_set_clientdata(i2c, bcm590xx); i2c_set_clientdata(i2c_pri, bcm590xx);
bcm590xx->dev = &i2c->dev; bcm590xx->dev = &i2c_pri->dev;
bcm590xx->i2c_client = i2c; bcm590xx->i2c_pri = i2c_pri;
bcm590xx->regmap = devm_regmap_init_i2c(i2c, &bcm590xx_regmap_config); bcm590xx->regmap_pri = devm_regmap_init_i2c(i2c_pri,
if (IS_ERR(bcm590xx->regmap)) { &bcm590xx_regmap_config_pri);
ret = PTR_ERR(bcm590xx->regmap); if (IS_ERR(bcm590xx->regmap_pri)) {
dev_err(&i2c->dev, "regmap initialization failed: %d\n", ret); ret = PTR_ERR(bcm590xx->regmap_pri);
dev_err(&i2c_pri->dev, "primary regmap init failed: %d\n", ret);
return ret; return ret;
} }
ret = mfd_add_devices(&i2c->dev, -1, bcm590xx_devs, /* Secondary I2C slave address is the base address with A(2) asserted */
bcm590xx->i2c_sec = i2c_new_dummy(i2c_pri->adapter,
i2c_pri->addr | BIT(2));
if (IS_ERR_OR_NULL(bcm590xx->i2c_sec)) {
dev_err(&i2c_pri->dev, "failed to add secondary I2C device\n");
return -ENODEV;
}
i2c_set_clientdata(bcm590xx->i2c_sec, bcm590xx);
bcm590xx->regmap_sec = devm_regmap_init_i2c(bcm590xx->i2c_sec,
&bcm590xx_regmap_config_sec);
if (IS_ERR(bcm590xx->regmap_sec)) {
ret = PTR_ERR(bcm590xx->regmap_sec);
dev_err(&bcm590xx->i2c_sec->dev,
"secondary regmap init failed: %d\n", ret);
goto err;
}
ret = mfd_add_devices(&i2c_pri->dev, -1, bcm590xx_devs,
ARRAY_SIZE(bcm590xx_devs), NULL, 0, NULL); ARRAY_SIZE(bcm590xx_devs), NULL, 0, NULL);
if (ret < 0) if (ret < 0) {
dev_err(&i2c->dev, "failed to add sub-devices: %d\n", ret); dev_err(&i2c_pri->dev, "failed to add sub-devices: %d\n", ret);
goto err;
}
return 0;
err:
i2c_unregister_device(bcm590xx->i2c_sec);
return ret; return ret;
} }
......
...@@ -19,12 +19,15 @@ ...@@ -19,12 +19,15 @@
#include <linux/regmap.h> #include <linux/regmap.h>
/* max register address */ /* max register address */
#define BCM590XX_MAX_REGISTER 0xe7 #define BCM590XX_MAX_REGISTER_PRI 0xe7
#define BCM590XX_MAX_REGISTER_SEC 0xf0
struct bcm590xx { struct bcm590xx {
struct device *dev; struct device *dev;
struct i2c_client *i2c_client; struct i2c_client *i2c_pri;
struct regmap *regmap; struct i2c_client *i2c_sec;
struct regmap *regmap_pri;
struct regmap *regmap_sec;
unsigned int id; unsigned int id;
}; };
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册