Re: [PATCH v2 1/5] mfd: tps65217: Add support for IRQs

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




On Thu, 16 Jun 2016, Marcin Niestroj wrote:

> Hi,
> 
> On 15.06.2016 17:13, Lee Jones wrote:
> >On Tue, 14 Jun 2016, Marcin Niestroj wrote:
> >
> >>Add support for handling IRQs: power button, AC and USB power state
> >>changes. Mask and interrupt bits are shared within one register, which
> >>prevents us to use regmap_irq implementation. New irq_domain is created in
> >>order to add interrupt handling for each tps65217's subsystem. IRQ
> >>resources have been added for charger subsystem to be able to notify about
> >>AC and USB state changes.
> >>
> >>Signed-off-by: Marcin Niestroj <m.niestroj@xxxxxxxxxxxxxxxx>
> >>---
> >> drivers/mfd/Kconfig          |   1 +
> >> drivers/mfd/tps65217.c       | 182 ++++++++++++++++++++++++++++++++++++++++++-
> >> include/linux/mfd/tps65217.h |  11 +++
> >> 3 files changed, 192 insertions(+), 2 deletions(-)
> >>
> >>diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig
> >>index 1bcf601..f8c9580 100644
> >>--- a/drivers/mfd/Kconfig
> >>+++ b/drivers/mfd/Kconfig
> >>@@ -1200,6 +1200,7 @@ config MFD_TPS65217
> >> 	depends on I2C
> >> 	select MFD_CORE
> >> 	select REGMAP_I2C
> >>+	select IRQ_DOMAIN
> >> 	help
> >> 	  If you say yes here you get support for the TPS65217 series of
> >> 	  Power Management / White LED chips.
> >>diff --git a/drivers/mfd/tps65217.c b/drivers/mfd/tps65217.c
> >>index 049a6fc..d49f94e 100644
> >>--- a/drivers/mfd/tps65217.c
> >>+++ b/drivers/mfd/tps65217.c
> >>@@ -23,6 +23,9 @@
> >> #include <linux/i2c.h>
> >> #include <linux/slab.h>
> >> #include <linux/regmap.h>
> >>+#include <linux/interrupt.h>
> >>+#include <linux/irq.h>
> >>+#include <linux/irqdomain.h>
> >
> >Header includes should be alphabetical.
> >
> >Please take the opportunity to recorder them.
> 
> Ok, will do that.
> 
> >
> >> #include <linux/err.h>
> >> #include <linux/of.h>
> >> #include <linux/of_device.h>
> >>@@ -30,7 +33,81 @@
> >> #include <linux/mfd/core.h>
> >> #include <linux/mfd/tps65217.h>
> >>
> >>-static const struct mfd_cell tps65217s[] = {
> >>+static struct resource charger_resources[] = {
> >>+	DEFINE_RES_IRQ_NAMED(TPS65217_IRQ_AC, "AC"),
> >>+	DEFINE_RES_IRQ_NAMED(TPS65217_IRQ_USB, "USB"),
> >>+};
> >>+
> >>+struct tps65217_irq {
> >>+	int mask;
> >>+	int interrupt;
> >>+};
> >>+
> >>+static const struct tps65217_irq tps65217_irqs[] = {
> >
> >Not sure how this compiles, since you have this struct and the enums
> >in the header named the same.  Better to differentiate between them.
> 
> I think that in C these two names are in different namespaces. The name
> in the header is just a enum tag name, but tps65217_irqs object here is
> an ordinary identifier. Similarly we use "struct regmap *regmap;"
> declaration in other places in the kernel.
> I also tried to be consistent with other drivers: tps65218, tps65086,
> tps65912.

Yes, I think the 'static' helps you there.

> Please confirm if I should change these names.

It would save a little confusion.

> >>+	[TPS65217_IRQ_PB] = {
> >>+		.mask = TPS65217_INT_PBM,
> >>+		.interrupt = TPS65217_INT_PBI,
> >>+	},
> >>+	[TPS65217_IRQ_AC] = {
> >>+		.mask = TPS65217_INT_ACM,
> >>+		.interrupt = TPS65217_INT_ACI,
> >>+	},
> >>+	[TPS65217_IRQ_USB] = {
> >>+		.mask = TPS65217_INT_USBM,
> >>+		.interrupt = TPS65217_INT_USBI,
> >>+	},
> >>+};
> >>+
> >>+static void tps65217_irq_lock(struct irq_data *data)
> >>+{
> >>+	struct tps65217 *tps = irq_data_get_irq_chip_data(data);
> >>+
> >>+	mutex_lock(&tps->irq_lock);
> >>+}
> >>+
> >>+static void tps65217_irq_sync_unlock(struct irq_data *data)
> >>+{
> >>+	struct tps65217 *tps = irq_data_get_irq_chip_data(data);
> >>+	int ret;
> >>+
> >>+	ret = tps65217_reg_write(tps, TPS65217_REG_INT, tps->irq_mask,
> >>+				TPS65217_PROTECT_NONE);
> >>+	if (ret != 0)
> >>+		dev_err(tps->dev, "Failed to sync IRQ masks\n");
> >>+
> >>+	mutex_unlock(&tps->irq_lock);
> >>+}
> >>+
> >>+static const inline struct tps65217_irq *
> >>+irq_to_tps65217_irq(struct tps65217 *tps, struct irq_data *data)
> >>+{
> >>+	return &tps65217_irqs[data->hwirq];
> >>+}
> >>+
> >>+static void tps65217_irq_enable(struct irq_data *data)
> >>+{
> >>+	struct tps65217 *tps = irq_data_get_irq_chip_data(data);
> >>+	const struct tps65217_irq *irq_data = irq_to_tps65217_irq(tps, data);
> >>+
> >>+	tps->irq_mask &= ~irq_data->mask;
> >>+}
> >>+
> >>+static void tps65217_irq_disable(struct irq_data *data)
> >>+{
> >>+	struct tps65217 *tps = irq_data_get_irq_chip_data(data);
> >>+	const struct tps65217_irq *irq_data = irq_to_tps65217_irq(tps, data);
> >>+
> >>+	tps->irq_mask |= irq_data->mask;
> >>+}
> >
> >Nit: Swap this with enable, either here or in the struct below.
> 
> Sure
> 
> >
> >>+static struct irq_chip tps65217_irq_chip = {
> >>+	.irq_bus_lock		= tps65217_irq_lock,
> >>+	.irq_bus_sync_unlock	= tps65217_irq_sync_unlock,
> >>+	.irq_disable		= tps65217_irq_disable,
> >>+	.irq_enable		= tps65217_irq_enable,
> >>+};
> >>+
> >>+static struct mfd_cell tps65217s[] = {
> >> 	{
> >> 		.name = "tps65217-pmic",
> >> 		.of_compatible = "ti,tps65217-pmic",
> >>@@ -41,10 +118,89 @@ static const struct mfd_cell tps65217s[] = {
> >> 	},
> >> 	{
> >> 		.name = "tps65217-charger",
> >>+		.num_resources = ARRAY_SIZE(charger_resources),
> >>+		.resources = charger_resources,
> >> 		.of_compatible = "ti,tps65217-charger",
> >> 	},
> >> };
> >>
> >>+static irqreturn_t tps65217_irq_thread(int irq, void *data)
> >>+{
> >>+	struct tps65217 *tps = data;
> >>+	unsigned int status;
> >>+	bool handled = false;
> >>+	int i;
> >>+	int ret;
> >>+
> >>+	ret = tps65217_reg_read(tps, TPS65217_REG_INT, &status);
> >>+	if (ret < 0) {
> >>+		dev_err(tps->dev, "Failed to read IRQ status: %d\n",
> >>+			ret);
> >>+		return IRQ_NONE;
> >>+	}
> >>+
> >>+	for (i = 0; i < ARRAY_SIZE(tps65217_irqs); i++) {
> >>+		if (status & tps65217_irqs[i].interrupt) {
> >>+			handle_nested_irq(irq_find_mapping(tps->irq_domain, i));
> >>+			handled = true;
> >>+		}
> >>+	}
> >>+
> >>+	if (handled)
> >>+		return IRQ_HANDLED;
> >>+	else
> >>+		return IRQ_NONE;
> >
> >The else is kinda redundant here.
> >
> >... and I think it's possible to get a warning saying something about
> >coming to the end of a non-void function.
> 
> Will change that in new patch version.
> 
> >
> >>+}
> >>+
> >>+static int tps65217_irq_map(struct irq_domain *h, unsigned int virq,
> >>+			irq_hw_number_t hw)
> >>+{
> >>+	struct tps65217 *tps = h->host_data;
> >>+
> >>+	irq_set_chip_data(virq, tps);
> >>+	irq_set_chip_and_handler(virq, &tps65217_irq_chip, handle_edge_irq);
> >>+	irq_set_nested_thread(virq, 1);
> >>+	irq_set_noprobe(virq);
> >>+
> >>+	return 0;
> >>+}
> >>+
> >>+static const struct irq_domain_ops tps65217_irq_domain_ops = {
> >>+	.map = tps65217_irq_map,
> >>+};
> >>+
> >>+static int tps65217_irq_init(struct tps65217 *tps, int irq)
> >>+{
> >>+	int ret;
> >>+
> >>+	mutex_init(&tps->irq_lock);
> >>+
> >>+	/* Mask all interrupt sources */
> >>+	tps->irq_mask = (TPS65217_INT_RESERVEDM | TPS65217_INT_PBM
> >>+			| TPS65217_INT_ACM | TPS65217_INT_USBM);
> >>+	tps65217_reg_write(tps, TPS65217_REG_INT, tps->irq_mask,
> >>+			TPS65217_PROTECT_NONE);
> >>+
> >>+	tps->irq_domain = irq_domain_add_linear(tps->dev->of_node,
> >
> >I think it's suggested that you use *add_simple() these days.
> 
> In Documentation/IRQ-domain.txt they are both described.
> *add_simple() is under 'Legacy' section and "It is used when the
> driver cannot be immediately converted to use the linear mapping."

So that's how they're doing it now.  Very well.

> >>+		TPS65217_NUM_IRQ, &tps65217_irq_domain_ops, tps);
> >>+	if (!tps->irq_domain) {
> >>+		dev_err(tps->dev, "Could not create IRQ domain\n");
> >>+		return -ENOMEM;
> >>+	}
> >>+
> >>+	ret = devm_request_threaded_irq(tps->dev, irq, NULL,
> >>+					tps65217_irq_thread,
> >>+					IRQF_TRIGGER_RISING | IRQF_ONESHOT,
> >>+					"tps65217-irq", tps);
> >>+	if (ret) {
> >>+		dev_err(tps->dev, "Failed to request IRQ %d: %d\n",
> >>+			irq, ret);
> >>+		return ret;
> >>+	}
> >>+
> >>+	return 0;
> >>+}
> >>+
> >> /**
> >>  * tps65217_reg_read: Read a single tps65217 register.
> >>  *
> >>@@ -149,11 +305,22 @@ int tps65217_clear_bits(struct tps65217 *tps, unsigned int reg,
> >> }
> >> EXPORT_SYMBOL_GPL(tps65217_clear_bits);
> >>
> >>+static bool tps65217_volatile_reg(struct device *dev, unsigned int reg)
> >>+{
> >>+	switch (reg) {
> >>+	case TPS65217_REG_INT:
> >>+		return true;
> >>+	default:
> >>+		return false;
> >>+	}
> >>+}
> >>+
> >> static const struct regmap_config tps65217_regmap_config = {
> >> 	.reg_bits = 8,
> >> 	.val_bits = 8,
> >>
> >> 	.max_register = TPS65217_REG_MAX,
> >>+	.volatile_reg = tps65217_volatile_reg,
> >> };
> >>
> >> static const struct of_device_id tps65217_of_match[] = {
> >>@@ -205,8 +372,19 @@ static int tps65217_probe(struct i2c_client *client,
> >> 		return ret;
> >> 	}
> >>
> >>+	if (client->irq) {
> >>+		tps65217_irq_init(tps, client->irq);
> >>+	} else {
> >>+		int i;
> >>+
> >>+		/* Don't tell children about IRQ resources which won't fire */
> >>+		for (i = 0; i < ARRAY_SIZE(tps65217s); i++)
> >>+			tps65217s[i].num_resources = 0;
> >>+	}
> >>+
> >> 	ret = devm_mfd_add_devices(tps->dev, -1, tps65217s,
> >>-				   ARRAY_SIZE(tps65217s), NULL, 0, NULL);
> >>+				   ARRAY_SIZE(tps65217s), NULL, 0,
> >>+				   tps->irq_domain);
> >> 	if (ret < 0) {
> >> 		dev_err(tps->dev, "mfd_add_devices failed: %d\n", ret);
> >> 		return ret;
> >>diff --git a/include/linux/mfd/tps65217.h b/include/linux/mfd/tps65217.h
> >>index ac7fba4..f80a5d3 100644
> >>--- a/include/linux/mfd/tps65217.h
> >>+++ b/include/linux/mfd/tps65217.h
> >>@@ -73,6 +73,7 @@
> >> #define TPS65217_PPATH_AC_CURRENT_MASK	0x0C
> >> #define TPS65217_PPATH_USB_CURRENT_MASK	0x03
> >>
> >>+#define TPS65217_INT_RESERVEDM		BIT(7)
> >> #define TPS65217_INT_PBM		BIT(6)
> >> #define TPS65217_INT_ACM		BIT(5)
> >> #define TPS65217_INT_USBM		BIT(4)
> >>@@ -233,6 +234,13 @@ struct tps65217_bl_pdata {
> >> 	int dft_brightness;
> >> };
> >>
> >>+enum tps65217_irqs {
> >>+	TPS65217_IRQ_PB,
> >>+	TPS65217_IRQ_AC,
> >>+	TPS65217_IRQ_USB,
> >>+	TPS65217_NUM_IRQ
> >>+};
> >>+
> >> /**
> >>  * struct tps65217_board - packages regulator init data
> >>  * @tps65217_regulator_data: regulator initialization values
> >>@@ -257,6 +265,9 @@ struct tps65217 {
> >> 	unsigned long id;
> >> 	struct regulator_desc desc[TPS65217_NUM_REGULATOR];
> >> 	struct regmap *regmap;
> >>+	struct irq_domain *irq_domain;
> >>+	struct mutex irq_lock;
> >>+	u8 irq_mask;
> >> };
> >>
> >> static inline struct tps65217 *dev_to_tps65217(struct device *dev)
> >
> 

-- 
Lee Jones
Linaro STMicroelectronics Landing Team Lead
Linaro.org │ Open source software for ARM SoCs
Follow Linaro: Facebook | Twitter | Blog
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Device Tree Compilter]     [Device Tree Spec]     [Linux Driver Backports]     [Video for Linux]     [Linux USB Devel]     [Linux PCI Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [XFree86]     [Yosemite Backpacking]
  Powered by Linux