|
@@ -23,6 +23,77 @@
|
|
|
#include <linux/mfd/da9150/core.h>
|
|
|
#include <linux/mfd/da9150/registers.h>
|
|
|
|
|
|
+/* Raw device access, used for QIF */
|
|
|
+static int da9150_i2c_read_device(struct i2c_client *client, u8 addr, int count,
|
|
|
+ u8 *buf)
|
|
|
+{
|
|
|
+ struct i2c_msg xfer;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Read is split into two transfers as device expects STOP/START rather
|
|
|
+ * than repeated start to carry out this kind of access.
|
|
|
+ */
|
|
|
+
|
|
|
+ /* Write address */
|
|
|
+ xfer.addr = client->addr;
|
|
|
+ xfer.flags = 0;
|
|
|
+ xfer.len = 1;
|
|
|
+ xfer.buf = &addr;
|
|
|
+
|
|
|
+ ret = i2c_transfer(client->adapter, &xfer, 1);
|
|
|
+ if (ret != 1) {
|
|
|
+ if (ret < 0)
|
|
|
+ return ret;
|
|
|
+ else
|
|
|
+ return -EIO;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Read data */
|
|
|
+ xfer.addr = client->addr;
|
|
|
+ xfer.flags = I2C_M_RD;
|
|
|
+ xfer.len = count;
|
|
|
+ xfer.buf = buf;
|
|
|
+
|
|
|
+ ret = i2c_transfer(client->adapter, &xfer, 1);
|
|
|
+ if (ret == 1)
|
|
|
+ return 0;
|
|
|
+ else if (ret < 0)
|
|
|
+ return ret;
|
|
|
+ else
|
|
|
+ return -EIO;
|
|
|
+}
|
|
|
+
|
|
|
+static int da9150_i2c_write_device(struct i2c_client *client, u8 addr,
|
|
|
+ int count, const u8 *buf)
|
|
|
+{
|
|
|
+ struct i2c_msg xfer;
|
|
|
+ u8 *reg_data;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ reg_data = kzalloc(1 + count, GFP_KERNEL);
|
|
|
+ if (!reg_data)
|
|
|
+ return -ENOMEM;
|
|
|
+
|
|
|
+ reg_data[0] = addr;
|
|
|
+ memcpy(®_data[1], buf, count);
|
|
|
+
|
|
|
+ /* Write address & data */
|
|
|
+ xfer.addr = client->addr;
|
|
|
+ xfer.flags = 0;
|
|
|
+ xfer.len = 1 + count;
|
|
|
+ xfer.buf = reg_data;
|
|
|
+
|
|
|
+ ret = i2c_transfer(client->adapter, &xfer, 1);
|
|
|
+ kfree(reg_data);
|
|
|
+ if (ret == 1)
|
|
|
+ return 0;
|
|
|
+ else if (ret < 0)
|
|
|
+ return ret;
|
|
|
+ else
|
|
|
+ return -EIO;
|
|
|
+}
|
|
|
+
|
|
|
static bool da9150_volatile_reg(struct device *dev, unsigned int reg)
|
|
|
{
|
|
|
switch (reg) {
|
|
@@ -107,6 +178,28 @@ static const struct regmap_config da9150_regmap_config = {
|
|
|
.volatile_reg = da9150_volatile_reg,
|
|
|
};
|
|
|
|
|
|
+void da9150_read_qif(struct da9150 *da9150, u8 addr, int count, u8 *buf)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ ret = da9150_i2c_read_device(da9150->core_qif, addr, count, buf);
|
|
|
+ if (ret < 0)
|
|
|
+ dev_err(da9150->dev, "Failed to read from QIF 0x%x: %d\n",
|
|
|
+ addr, ret);
|
|
|
+}
|
|
|
+EXPORT_SYMBOL_GPL(da9150_read_qif);
|
|
|
+
|
|
|
+void da9150_write_qif(struct da9150 *da9150, u8 addr, int count, const u8 *buf)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ ret = da9150_i2c_write_device(da9150->core_qif, addr, count, buf);
|
|
|
+ if (ret < 0)
|
|
|
+ dev_err(da9150->dev, "Failed to write to QIF 0x%x: %d\n",
|
|
|
+ addr, ret);
|
|
|
+}
|
|
|
+EXPORT_SYMBOL_GPL(da9150_write_qif);
|
|
|
+
|
|
|
u8 da9150_reg_read(struct da9150 *da9150, u16 reg)
|
|
|
{
|
|
|
int val, ret;
|
|
@@ -297,19 +390,35 @@ static struct resource da9150_charger_resources[] = {
|
|
|
},
|
|
|
};
|
|
|
|
|
|
+static struct resource da9150_fg_resources[] = {
|
|
|
+ DEFINE_RES_IRQ_NAMED(DA9150_IRQ_FG, "FG"),
|
|
|
+};
|
|
|
+
|
|
|
+enum da9150_dev_idx {
|
|
|
+ DA9150_GPADC_IDX = 0,
|
|
|
+ DA9150_CHARGER_IDX,
|
|
|
+ DA9150_FG_IDX,
|
|
|
+};
|
|
|
+
|
|
|
static struct mfd_cell da9150_devs[] = {
|
|
|
- {
|
|
|
+ [DA9150_GPADC_IDX] = {
|
|
|
.name = "da9150-gpadc",
|
|
|
.of_compatible = "dlg,da9150-gpadc",
|
|
|
.resources = da9150_gpadc_resources,
|
|
|
.num_resources = ARRAY_SIZE(da9150_gpadc_resources),
|
|
|
},
|
|
|
- {
|
|
|
+ [DA9150_CHARGER_IDX] = {
|
|
|
.name = "da9150-charger",
|
|
|
.of_compatible = "dlg,da9150-charger",
|
|
|
.resources = da9150_charger_resources,
|
|
|
.num_resources = ARRAY_SIZE(da9150_charger_resources),
|
|
|
},
|
|
|
+ [DA9150_FG_IDX] = {
|
|
|
+ .name = "da9150-fuel-gauge",
|
|
|
+ .of_compatible = "dlg,da9150-fuel-gauge",
|
|
|
+ .resources = da9150_fg_resources,
|
|
|
+ .num_resources = ARRAY_SIZE(da9150_fg_resources),
|
|
|
+ },
|
|
|
};
|
|
|
|
|
|
static int da9150_probe(struct i2c_client *client,
|
|
@@ -317,6 +426,7 @@ static int da9150_probe(struct i2c_client *client,
|
|
|
{
|
|
|
struct da9150 *da9150;
|
|
|
struct da9150_pdata *pdata = dev_get_platdata(&client->dev);
|
|
|
+ int qif_addr;
|
|
|
int ret;
|
|
|
|
|
|
da9150 = devm_kzalloc(&client->dev, sizeof(*da9150), GFP_KERNEL);
|
|
@@ -335,16 +445,41 @@ static int da9150_probe(struct i2c_client *client,
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
- da9150->irq_base = pdata ? pdata->irq_base : -1;
|
|
|
+ /* Setup secondary I2C interface for QIF access */
|
|
|
+ qif_addr = da9150_reg_read(da9150, DA9150_CORE2WIRE_CTRL_A);
|
|
|
+ qif_addr = (qif_addr & DA9150_CORE_BASE_ADDR_MASK) >> 1;
|
|
|
+ qif_addr |= DA9150_QIF_I2C_ADDR_LSB;
|
|
|
+ da9150->core_qif = i2c_new_dummy(client->adapter, qif_addr);
|
|
|
+ if (!da9150->core_qif) {
|
|
|
+ dev_err(da9150->dev, "Failed to attach QIF client\n");
|
|
|
+ return -ENODEV;
|
|
|
+ }
|
|
|
+
|
|
|
+ i2c_set_clientdata(da9150->core_qif, da9150);
|
|
|
+
|
|
|
+ if (pdata) {
|
|
|
+ da9150->irq_base = pdata->irq_base;
|
|
|
+
|
|
|
+ da9150_devs[DA9150_FG_IDX].platform_data = pdata->fg_pdata;
|
|
|
+ da9150_devs[DA9150_FG_IDX].pdata_size =
|
|
|
+ sizeof(struct da9150_fg_pdata);
|
|
|
+ } else {
|
|
|
+ da9150->irq_base = -1;
|
|
|
+ }
|
|
|
|
|
|
ret = regmap_add_irq_chip(da9150->regmap, da9150->irq,
|
|
|
IRQF_TRIGGER_LOW | IRQF_ONESHOT,
|
|
|
da9150->irq_base, &da9150_regmap_irq_chip,
|
|
|
&da9150->regmap_irq_data);
|
|
|
- if (ret)
|
|
|
- return ret;
|
|
|
+ if (ret) {
|
|
|
+ dev_err(da9150->dev, "Failed to add regmap irq chip: %d\n",
|
|
|
+ ret);
|
|
|
+ goto regmap_irq_fail;
|
|
|
+ }
|
|
|
+
|
|
|
|
|
|
da9150->irq_base = regmap_irq_chip_get_base(da9150->regmap_irq_data);
|
|
|
+
|
|
|
enable_irq_wake(da9150->irq);
|
|
|
|
|
|
ret = mfd_add_devices(da9150->dev, -1, da9150_devs,
|
|
@@ -352,11 +487,17 @@ static int da9150_probe(struct i2c_client *client,
|
|
|
da9150->irq_base, NULL);
|
|
|
if (ret) {
|
|
|
dev_err(da9150->dev, "Failed to add child devices: %d\n", ret);
|
|
|
- regmap_del_irq_chip(da9150->irq, da9150->regmap_irq_data);
|
|
|
- return ret;
|
|
|
+ goto mfd_fail;
|
|
|
}
|
|
|
|
|
|
return 0;
|
|
|
+
|
|
|
+mfd_fail:
|
|
|
+ regmap_del_irq_chip(da9150->irq, da9150->regmap_irq_data);
|
|
|
+regmap_irq_fail:
|
|
|
+ i2c_unregister_device(da9150->core_qif);
|
|
|
+
|
|
|
+ return ret;
|
|
|
}
|
|
|
|
|
|
static int da9150_remove(struct i2c_client *client)
|
|
@@ -365,6 +506,7 @@ static int da9150_remove(struct i2c_client *client)
|
|
|
|
|
|
regmap_del_irq_chip(da9150->irq, da9150->regmap_irq_data);
|
|
|
mfd_remove_devices(da9150->dev);
|
|
|
+ i2c_unregister_device(da9150->core_qif);
|
|
|
|
|
|
return 0;
|
|
|
}
|