[PATCH RESEND v2 1/4] mfd: intel_soc_pmic: Core driver
From: Zhu, Lejun
Date: Thu May 22 2014 - 20:41:45 EST
This patch provides the common code for the intel_soc_pmic MFD driver,
such as read/write register and set up IRQ.
v2:
- Use regmap instead of our own callbacks for read/write.
- Add one missing EXPORT_SYMBOL.
- Remove some duplicate code and put them into pmic_regmap_load_from_hw.
Signed-off-by: Yang, Bin <bin.yang@xxxxxxxxx>
Signed-off-by: Zhu, Lejun <lejun.zhu@xxxxxxxxxxxxxxx>
---
drivers/mfd/intel_soc_pmic_core.c | 521 +++++++++++++++++++++++++++++++++++++
drivers/mfd/intel_soc_pmic_core.h | 83 ++++++
include/linux/mfd/intel_soc_pmic.h | 29 +++
3 files changed, 633 insertions(+)
create mode 100644 drivers/mfd/intel_soc_pmic_core.c
create mode 100644 drivers/mfd/intel_soc_pmic_core.h
create mode 100644 include/linux/mfd/intel_soc_pmic.h
diff --git a/drivers/mfd/intel_soc_pmic_core.c b/drivers/mfd/intel_soc_pmic_core.c
new file mode 100644
index 0000000..76d366e
--- /dev/null
+++ b/drivers/mfd/intel_soc_pmic_core.c
@@ -0,0 +1,521 @@
+/*
+ * intel_soc_pmic_core.c - Intel SoC PMIC Core Functions
+ *
+ * Copyright (C) 2013, 2014 Intel Corporation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License version
+ * 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * Author: Yang, Bin <bin.yang@xxxxxxxxx>
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/delay.h>
+#include <linux/mutex.h>
+#include <linux/mfd/core.h>
+#include <linux/err.h>
+#include <linux/irq.h>
+#include <linux/interrupt.h>
+#include <linux/workqueue.h>
+#include <linux/acpi.h>
+#include <linux/version.h>
+#include <linux/gpio.h>
+#include <linux/regmap.h>
+#include <linux/mfd/intel_soc_pmic.h>
+#include "intel_soc_pmic_core.h"
+
+struct cell_dev_pdata {
+ struct list_head list;
+ const char *name;
+ void *data;
+ int len;
+};
+static LIST_HEAD(pdata_list);
+
+static DEFINE_MUTEX(pmic_lock); /* protect the following variables */
+static struct intel_soc_pmic *pmic;
+static int cache_offset = -1;
+static unsigned int cache_read_val;
+static unsigned int cache_write_val;
+static int cache_write_pending;
+static int cache_flags;
+
+struct device *intel_soc_pmic_dev(void)
+{
+ return pmic->dev;
+}
+EXPORT_SYMBOL(intel_soc_pmic_dev);
+
+/*
+ * Read from a PMIC register
+ */
+int intel_soc_pmic_readb(int reg)
+{
+ int ret;
+ unsigned int val;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic) {
+ ret = -EIO;
+ } else {
+ ret = regmap_read(pmic->regmap, reg, &val);
+ if (!ret)
+ ret = val;
+ }
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+EXPORT_SYMBOL(intel_soc_pmic_readb);
+
+/*
+ * Write to a PMIC register
+ */
+int intel_soc_pmic_writeb(int reg, u8 val)
+{
+ int ret;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic)
+ ret = -EIO;
+ else
+ ret = regmap_write(pmic->regmap, reg, val);
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+EXPORT_SYMBOL(intel_soc_pmic_writeb);
+
+/*
+ * Set 1 bit in a PMIC register
+ */
+int intel_soc_pmic_setb(int reg, u8 mask)
+{
+ int ret;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic)
+ ret = -EIO;
+ else
+ ret = regmap_update_bits(pmic->regmap, reg, mask, mask);
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+EXPORT_SYMBOL(intel_soc_pmic_setb);
+
+/*
+ * Clear 1 bit in a PMIC register
+ */
+int intel_soc_pmic_clearb(int reg, u8 mask)
+{
+ int ret;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic)
+ ret = -EIO;
+ else
+ ret = regmap_update_bits(pmic->regmap, reg, mask, 0);
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+EXPORT_SYMBOL(intel_soc_pmic_clearb);
+
+/*
+ * Set and clear multiple bits of a PMIC register
+ */
+int intel_soc_pmic_update(int reg, u8 val, u8 mask)
+{
+ int ret;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic)
+ ret = -EIO;
+ else
+ ret = regmap_update_bits(pmic->regmap, reg, mask, val);
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+EXPORT_SYMBOL(intel_soc_pmic_update);
+
+/*
+ * Set platform_data of the child devices. Needs to be called before
+ * the MFD device is added.
+ */
+int intel_soc_pmic_set_pdata(const char *name, void *data, int len)
+{
+ struct cell_dev_pdata *pdata;
+
+ pdata = kzalloc(sizeof(*pdata), GFP_KERNEL);
+ if (!pdata) {
+ pr_err("intel_pmic: can't set pdata!\n");
+ return -ENOMEM;
+ }
+
+ pdata->name = name;
+ pdata->data = data;
+ pdata->len = len;
+
+ list_add_tail(&pdata->list, &pdata_list);
+
+ return 0;
+}
+EXPORT_SYMBOL(intel_soc_pmic_set_pdata);
+
+static void __pmic_regmap_flush(void)
+{
+ if (cache_write_pending)
+ WARN_ON(regmap_write(pmic->regmap, cache_offset,
+ cache_write_val));
+ cache_offset = -1;
+ cache_write_pending = 0;
+}
+
+static void pmic_regmap_flush(void)
+{
+ mutex_lock(&pmic_lock);
+
+ if (pmic)
+ __pmic_regmap_flush();
+
+ mutex_unlock(&pmic_lock);
+}
+
+static int pmic_regmap_load_from_hw(struct intel_pmic_regmap *map)
+{
+ int ret;
+
+ if (cache_offset == map->offset) {
+ if (cache_flags != map->flags) {
+ dev_warn(pmic->dev, "Same reg with diff flags\n");
+ __pmic_regmap_flush();
+ }
+ }
+
+ if (cache_offset != map->offset) {
+ __pmic_regmap_flush();
+ if (IS_PMIC_REG_WO(map) || IS_PMIC_REG_W1C(map)) {
+ cache_write_val = 0;
+ ret = regmap_read(pmic->regmap, map->offset,
+ &cache_read_val);
+ } else {
+ ret = regmap_read(pmic->regmap, map->offset,
+ &cache_read_val);
+ cache_write_val = cache_read_val;
+ }
+ if (ret) {
+ dev_err(pmic->dev, "Register access error\n");
+ return ret;
+ }
+ cache_offset = map->offset;
+ cache_flags = map->flags;
+ }
+
+ return 0;
+}
+
+static int pmic_regmap_write(struct intel_pmic_regmap *map, int val)
+{
+ int ret = 0;
+
+ if (!IS_PMIC_REG_VALID(map))
+ return -ENXIO;
+
+ if (IS_PMIC_REG_INV(map))
+ val = ~val;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic) {
+ ret = -ENODEV;
+ goto err;
+ }
+
+ ret = pmic_regmap_load_from_hw(map);
+ if (ret)
+ goto err;
+
+ val = ((val & map->mask) << map->shift);
+ cache_write_val &= ~(map->mask << map->shift);
+ cache_write_val |= val;
+ cache_write_pending = 1;
+
+ if (!IS_PMIC_REG_WO(map) && !IS_PMIC_REG_W1C(map))
+ cache_read_val = cache_write_val;
+
+err:
+ dev_dbg(pmic->dev, "offset=%x, shift=%x, mask=%x, flags=%x\n",
+ map->offset, map->shift, map->mask, map->flags);
+ dev_dbg(pmic->dev, "cache_read=%x, cache_write=%x, ret=%x\n",
+ cache_read_val, cache_write_val, ret);
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+
+static int pmic_regmap_read(struct intel_pmic_regmap *map)
+{
+ int ret = 0;
+
+ if (!IS_PMIC_REG_VALID(map))
+ return -ENXIO;
+
+ mutex_lock(&pmic_lock);
+
+ if (!pmic) {
+ ret = -ENODEV;
+ goto err;
+ }
+
+ ret = pmic_regmap_load_from_hw(map);
+ if (ret)
+ goto err;
+
+ if (IS_PMIC_REG_INV(map))
+ ret = ~cache_read_val;
+ else
+ ret = cache_read_val;
+
+ ret = (ret >> map->shift) & map->mask;
+ if (!IS_PMIC_REG_WO(map) && !IS_PMIC_REG_W1C(map))
+ cache_write_val = cache_read_val;
+
+err:
+ dev_dbg(pmic->dev, "offset=%x, shift=%x, mask=%x, flags=%x\n",
+ map->offset, map->shift, map->mask, map->flags);
+ dev_dbg(pmic->dev, "cache_read=%x, cache_write=%x, ret=%x\n",
+ cache_read_val, cache_write_val, ret);
+
+ mutex_unlock(&pmic_lock);
+
+ return ret;
+}
+
+static void pmic_irq_enable(struct irq_data *data)
+{
+ clear_bit(PMIC_IRQ_MASK_BIT(data->irq - pmic->irq_base),
+ &PMIC_IRQ_MASK(pmic, data->irq - pmic->irq_base));
+ pmic->irq_need_update = 1;
+}
+
+static void pmic_irq_disable(struct irq_data *data)
+{
+ set_bit(PMIC_IRQ_MASK_BIT(data->irq - pmic->irq_base),
+ &PMIC_IRQ_MASK(pmic, data->irq - pmic->irq_base));
+ pmic->irq_need_update = 1;
+}
+
+static void pmic_irq_sync_unlock(struct irq_data *data)
+{
+ int val, irq_offset;
+ if (pmic->irq_need_update) {
+ irq_offset = data->irq - pmic->irq_base;
+ val = !!test_bit(PMIC_IRQ_MASK_BIT(irq_offset),
+ &PMIC_IRQ_MASK(pmic, irq_offset));
+ pmic_regmap_write(&pmic->irq_regmap[irq_offset].mask, val);
+ pmic->irq_need_update = 0;
+ pmic_regmap_flush();
+ }
+ mutex_unlock(&pmic->irq_lock);
+}
+
+static void pmic_irq_lock(struct irq_data *data)
+{
+ mutex_lock(&pmic->irq_lock);
+}
+
+static irqreturn_t pmic_irq_isr(int irq, void *data)
+{
+ return IRQ_WAKE_THREAD;
+}
+
+static irqreturn_t pmic_irq_thread(int irq, void *data)
+{
+ int i;
+
+ mutex_lock(&pmic->irq_lock);
+
+ for (i = 0; i < pmic->irq_num; i++) {
+ if (test_bit(PMIC_IRQ_MASK_BIT(i), &PMIC_IRQ_MASK(pmic, i)))
+ continue;
+
+ if (pmic_regmap_read(&pmic->irq_regmap[i].status)) {
+ pmic_regmap_write(&pmic->irq_regmap[i].ack, 1);
+ handle_nested_irq(pmic->irq_base + i);
+ }
+ }
+
+ pmic_regmap_flush();
+
+ mutex_unlock(&pmic->irq_lock);
+
+ return IRQ_HANDLED;
+}
+
+static struct irq_chip pmic_irq_chip = {
+ .name = "intel_soc_pmic",
+ .irq_bus_lock = pmic_irq_lock,
+ .irq_bus_sync_unlock = pmic_irq_sync_unlock,
+ .irq_disable = pmic_irq_disable,
+ .irq_enable = pmic_irq_enable,
+};
+
+static int pmic_irq_init(void)
+{
+ int cur_irq;
+ int ret;
+ int i;
+
+ for (i = 0; i < pmic->irq_num; i++) {
+ pmic_regmap_write(&pmic->irq_regmap[i].mask, 1);
+ set_bit(PMIC_IRQ_MASK_BIT(i), &PMIC_IRQ_MASK(pmic, i));
+ }
+
+ for (i = 0; i < pmic->irq_num; i++)
+ pmic_regmap_write(&pmic->irq_regmap[i].ack, 1);
+
+ pmic_regmap_flush();
+
+ pmic->irq_base = irq_alloc_descs(-1, 0, pmic->irq_num, 0);
+ if (pmic->irq_base < 0) {
+ dev_warn(pmic->dev, "Failed to allocate IRQs: %d\n",
+ pmic->irq_base);
+ pmic->irq_base = 0;
+ ret = -EINVAL;
+ goto out;
+ } else {
+ dev_dbg(pmic->dev, "PMIC IRQ Base:%d\n", pmic->irq_base);
+ }
+
+ for (cur_irq = pmic->irq_base;
+ cur_irq < pmic->irq_num + pmic->irq_base;
+ cur_irq++) {
+ irq_set_chip_data(cur_irq, pmic);
+ irq_set_chip_and_handler(cur_irq, &pmic_irq_chip,
+ handle_edge_irq);
+ irq_set_nested_thread(cur_irq, 1);
+ irq_set_noprobe(cur_irq);
+ }
+
+ if (gpio_is_valid(pmic->pmic_int_gpio)) {
+ ret = gpio_request_one(pmic->pmic_int_gpio,
+ GPIOF_DIR_IN, "PMIC Interupt");
+ if (ret) {
+ dev_err(pmic->dev, "Request PMIC_INT gpio error\n");
+ goto out_free_desc;
+ }
+
+ pmic->irq = gpio_to_irq(pmic->pmic_int_gpio);
+ }
+
+ ret = request_threaded_irq(pmic->irq, pmic_irq_isr, pmic_irq_thread,
+ pmic->irq_flags, "intel_soc_pmic", pmic);
+ if (ret != 0) {
+ dev_err(pmic->dev, "Failed to request IRQ %d: %d\n",
+ pmic->irq, ret);
+ goto out_free_gpio;
+ }
+
+ ret = enable_irq_wake(pmic->irq);
+ if (ret != 0)
+ dev_warn(pmic->dev, "Can't enable IRQ as wake source: %d\n",
+ ret);
+
+ return 0;
+
+out_free_gpio:
+ if (gpio_is_valid(pmic->pmic_int_gpio)) {
+ gpio_free(pmic->pmic_int_gpio);
+ pmic->irq = 0;
+ }
+out_free_desc:
+ irq_free_descs(pmic->irq_base, pmic->irq_num);
+out:
+ return ret;
+}
+
+int intel_pmic_add(struct intel_soc_pmic *chip)
+{
+ int i, ret;
+ struct cell_dev_pdata *pdata;
+
+ mutex_lock(&pmic_lock);
+
+ if (pmic != NULL) {
+ mutex_unlock(&pmic_lock);
+ return -EBUSY;
+ }
+
+ pmic = chip;
+
+ mutex_unlock(&pmic_lock);
+
+ mutex_init(&chip->irq_lock);
+
+ if (chip->init) {
+ ret = chip->init();
+ if (ret != 0)
+ goto err;
+ }
+
+ ret = pmic_irq_init();
+ if (ret != 0)
+ goto err;
+
+ for (i = 0; chip->cell_dev[i].name != NULL; i++) {
+ list_for_each_entry(pdata, &pdata_list, list) {
+ if (!strcmp(pdata->name, chip->cell_dev[i].name)) {
+ chip->cell_dev[i].platform_data = pdata->data;
+ chip->cell_dev[i].pdata_size = pdata->len;
+ }
+ }
+ }
+
+ return mfd_add_devices(chip->dev, -1, chip->cell_dev, i,
+ NULL, chip->irq_base, NULL);
+
+err:
+ mutex_lock(&pmic_lock);
+ if (pmic == chip)
+ pmic = NULL;
+ mutex_unlock(&pmic_lock);
+ return ret;
+}
+
+int intel_pmic_remove(struct intel_soc_pmic *chip)
+{
+ mutex_lock(&pmic_lock);
+
+ if (pmic != chip) {
+ mutex_unlock(&pmic_lock);
+ return -ENODEV;
+ }
+
+ pmic = NULL;
+
+ mutex_unlock(&pmic_lock);
+
+ mfd_remove_devices(chip->dev);
+
+ return 0;
+}
diff --git a/drivers/mfd/intel_soc_pmic_core.h b/drivers/mfd/intel_soc_pmic_core.h
new file mode 100644
index 0000000..6b54962
--- /dev/null
+++ b/drivers/mfd/intel_soc_pmic_core.h
@@ -0,0 +1,83 @@
+/*
+ * intel_soc_pmic_core.h - Intel SoC PMIC MFD Driver
+ *
+ * Copyright (C) 2012-2014 Intel Corporation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License version
+ * 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * Author: Yang, Bin <bin.yang@xxxxxxxxx>
+ */
+
+#ifndef __INTEL_SOC_PMIC_CORE_H__
+#define __INTEL_SOC_PMIC_CORE_H__
+
+#define INTEL_PMIC_IRQ_MAX 128
+#define INTEL_PMIC_REG_NULL {-1,}
+
+#define INTEL_PMIC_REG_INV BIT(0) /*value revert*/
+#define INTEL_PMIC_REG_WO BIT(1) /*write only*/
+#define INTEL_PMIC_REG_RO BIT(2) /*read only*/
+#define INTEL_PMIC_REG_W1C BIT(3) /*write 1 clear*/
+#define INTEL_PMIC_REG_RC BIT(4) /*read clear*/
+#define IS_PMIC_REG_INV(_map) (_map->flags & INTEL_PMIC_REG_INV)
+#define IS_PMIC_REG_WO(_map) (_map->flags & INTEL_PMIC_REG_WO)
+#define IS_PMIC_REG_RO(_map) (_map->flags & INTEL_PMIC_REG_RO)
+#define IS_PMIC_REG_W1C(_map) (_map->flags & INTEL_PMIC_REG_W1C)
+#define IS_PMIC_REG_RC(_map) (_map->flags & INTEL_PMIC_REG_RC)
+#define IS_PMIC_REG_VALID(_map) \
+ ((_map->mask != 0) && (_map->offset >= 0))
+
+#define PMIC_IRQREG_MASK 0
+#define PMIC_IRQREG_STATUS 1
+#define PMIC_IRQREG_ACK 2
+
+#define PMIC_IRQ_MASK_NBITS 32
+#define PMIC_IRQ_MASK_LEN (INTEL_PMIC_IRQ_MAX / PMIC_IRQ_MASK_NBITS)
+#define PMIC_IRQ_MASK(pmic, offset) \
+ (pmic->irq_mask[(offset) / PMIC_IRQ_MASK_NBITS])
+#define PMIC_IRQ_MASK_BIT(offset) ((offset) % PMIC_IRQ_MASK_NBITS)
+
+struct intel_pmic_regmap {
+ int offset;
+ int shift;
+ int mask;
+ int flags;
+};
+
+struct intel_pmic_irqregmap {
+ struct intel_pmic_regmap mask;
+ struct intel_pmic_regmap status;
+ struct intel_pmic_regmap ack;
+};
+
+struct intel_soc_pmic {
+ const char *label;
+ unsigned long irq_flags;
+ int irq_num;
+ struct mfd_cell *cell_dev;
+ struct intel_pmic_irqregmap *irq_regmap;
+ struct regmap_config *regmap_cfg;
+ int (*init)(void);
+ struct device *dev;
+ struct mutex irq_lock; /* irq_bus_lock */
+ int irq_need_update;
+ int irq;
+ int irq_base;
+ unsigned long irq_mask[PMIC_IRQ_MASK_LEN];
+ int pmic_int_gpio;
+ struct regmap *regmap;
+};
+
+int intel_pmic_add(struct intel_soc_pmic *chip);
+int intel_pmic_remove(struct intel_soc_pmic *chip);
+
+extern struct intel_soc_pmic crystal_cove_pmic;
+
+#endif /* __INTEL_SOC_PMIC_CORE_H__ */
diff --git a/include/linux/mfd/intel_soc_pmic.h b/include/linux/mfd/intel_soc_pmic.h
new file mode 100644
index 0000000..88c28d9
--- /dev/null
+++ b/include/linux/mfd/intel_soc_pmic.h
@@ -0,0 +1,29 @@
+/*
+ * intel_soc_pmic.h - Intel SoC PMIC Driver
+ *
+ * Copyright (C) 2012-2014 Intel Corporation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License version
+ * 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * Author: Yang, Bin <bin.yang@xxxxxxxxx>
+ */
+
+#ifndef __INTEL_SOC_PMIC_H__
+#define __INTEL_SOC_PMIC_H__
+
+int intel_soc_pmic_readb(int reg);
+int intel_soc_pmic_writeb(int reg, u8 val);
+int intel_soc_pmic_setb(int reg, u8 mask);
+int intel_soc_pmic_clearb(int reg, u8 mask);
+int intel_soc_pmic_update(int reg, u8 val, u8 mask);
+int intel_soc_pmic_set_pdata(const char *name, void *data, int len);
+struct device *intel_soc_pmic_dev(void);
+
+#endif /* __INTEL_SOC_PMIC_H__ */
--
1.8.3.2
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/