From: Lee Jones <lee.jones@linaro.org>
To: Marcin Niestroj <m.niestroj@grinn-global.com>
Cc: Tony Lindgren <tony@atomide.com>,
Sebastian Reichel <sre@kernel.org>,
Dmitry Eremin-Solenikov <dbaryshkov@gmail.com>,
David Woodhouse <dwmw2@infradead.org>,
Rob Herring <robh+dt@kernel.org>, Pawel Moll <pawel.moll@arm.com>,
linux-omap@vger.kernel.org, linux-pm@vger.kernel.org,
linux-input@vger.kernel.org, devicetree@vger.kernel.org
Subject: Re: [PATCH v2 1/5] mfd: tps65217: Add support for IRQs
Date: Thu, 16 Jun 2016 11:35:34 +0100 [thread overview]
Message-ID: <20160616103534.GR4948@dell> (raw)
In-Reply-To: <a0bf75c0-26cf-e351-0046-a5f4d3303c29@grinn-global.com>
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@grinn-global.com>
> >>---
> >> 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 linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
next prev parent reply other threads:[~2016-06-16 10:34 UTC|newest]
Thread overview: 13+ messages / expand[flat|nested] mbox.gz Atom feed top
2016-06-14 13:29 [PATCH v2 0/5] mfd: tps65217: Add power-button and IRQ support Marcin Niestroj
2016-06-14 13:29 ` [PATCH v2 1/5] mfd: tps65217: Add support for IRQs Marcin Niestroj
2016-06-15 15:13 ` Lee Jones
2016-06-16 7:55 ` Marcin Niestroj
2016-06-16 10:35 ` Lee Jones [this message]
2016-06-14 13:29 ` [PATCH v2 2/5] power_supply: tps65217-charger: Fix NULL deref during property export Marcin Niestroj
2016-06-14 13:29 ` [PATCH v2 3/5] power_supply: tps65217-charger: Add support for IRQs Marcin Niestroj
2016-06-14 13:29 ` [PATCH v2 4/5] mfd: tps65217: Add power button as subdevice Marcin Niestroj
2016-06-15 14:12 ` Lee Jones
2016-06-15 14:51 ` Marcin Niestroj
2016-06-16 10:36 ` Lee Jones
2016-06-16 11:43 ` Marcin Niestroj
2016-06-14 13:29 ` [PATCH v2 5/5] Input: Add tps65217 power button driver Marcin Niestroj
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20160616103534.GR4948@dell \
--to=lee.jones@linaro.org \
--cc=dbaryshkov@gmail.com \
--cc=devicetree@vger.kernel.org \
--cc=dwmw2@infradead.org \
--cc=linux-input@vger.kernel.org \
--cc=linux-omap@vger.kernel.org \
--cc=linux-pm@vger.kernel.org \
--cc=m.niestroj@grinn-global.com \
--cc=pawel.moll@arm.com \
--cc=robh+dt@kernel.org \
--cc=sre@kernel.org \
--cc=tony@atomide.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).