From: jbrunet@baylibre.com (Jerome Brunet)
To: linus-amlogic@lists.infradead.org
Subject: [PATCH v6 6/9] pinctrl: meson: add support for GPIO interrupts
Date: Fri, 09 Jun 2017 11:06:25 +0200 [thread overview]
Message-ID: <1496999185.3552.60.camel@baylibre.com> (raw)
In-Reply-To: <a86d38af-9103-7557-9986-2dc187815569@gmail.com>
On Thu, 2017-06-08 at 21:39 +0200, Heiner Kallweit wrote:
> Add support for GPIO interrupts and make use of the just introduced
> irqchip driver handling the GPIO interrupt-controller.
>
> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
> ---
> v5:
> - changed Kconfig entry based on Neil's suggestion
> - extended comments
> - fixed indentation
> v6:
> - no changes
> ---
> ?drivers/pinctrl/Kconfig???????????????|???1 +
> ?drivers/pinctrl/meson/pinctrl-meson.c | 170
> +++++++++++++++++++++++++++++++++-
> ?2 files changed, 170 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/pinctrl/Kconfig b/drivers/pinctrl/Kconfig
> index 7ae04a97..86834dea 100644
> --- a/drivers/pinctrl/Kconfig
> +++ b/drivers/pinctrl/Kconfig
> @@ -166,6 +166,7 @@ config PINCTRL_MESON
> ? select PINCONF
> ? select GENERIC_PINCONF
> ? select GPIOLIB
> + select GPIOLIB_IRQCHIP
> ? select OF_GPIO
> ? select REGMAP_MMIO
> ?
> diff --git a/drivers/pinctrl/meson/pinctrl-meson.c
> b/drivers/pinctrl/meson/pinctrl-meson.c
> index 66ed70c1..7bacd4e3 100644
> --- a/drivers/pinctrl/meson/pinctrl-meson.c
> +++ b/drivers/pinctrl/meson/pinctrl-meson.c
> @@ -62,6 +62,8 @@
> ?#include "../pinctrl-utils.h"
> ?#include "pinctrl-meson.h"
> ?
> +static struct irq_domain *meson_pinctrl_irq_domain;
> +
> ?/**
> ? * meson_get_bank() - find the bank containing a given pin
> ? *
> @@ -497,6 +499,154 @@ static int meson_gpio_get(struct gpio_chip *chip,
> unsigned gpio)
> ? return !!(val & BIT(bit));
> ?}
> ?
> +static struct meson_pinctrl *meson_gpio_data_to_pc(struct irq_data *data)
> +{
> +????????struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
> +
> +????????return gpiochip_get_data(chip);
> +}
> +
> +static int meson_gpio_bank_hwirq(struct meson_bank *bank, unsigned int
> offset)
> +{
> +????????int hwirq;
> +
> +????????if (bank->irq_first < 0)
> +????????????????/* this bank cannot generate irqs */
> +????????????????return 0;
> +
> +????????hwirq = offset - bank->first + bank->irq_first;
> +
> +????????if (hwirq > bank->irq_last)
> +????????????????/* this pin cannot generate irqs */
> +????????????????return 0;
> +
> +????????return hwirq;
> +}
> +
> +static int meson_gpio_to_hwirq(struct irq_data *data)
> +{
> + struct meson_pinctrl *pc = meson_gpio_data_to_pc(data);
> + unsigned int offset = data->hwirq;
> +????????struct meson_bank *bank;
> +????????int hwirq, ret;
> +
> +????????offset += pc->data->pin_base;
> +
> +????????ret = meson_get_bank(pc, offset, &bank);
> +????????if (ret)
> +????????????????return ret;
> +
> +????????hwirq = meson_gpio_bank_hwirq(bank, offset);
> +????????if (!hwirq)
> +????????????????dev_dbg(pc->dev, "no interrupt for pin %u\n", offset);
> +
> +????????return hwirq;
> +}
> +
> +static void meson_gpio_irq_handler(struct irq_desc *desc)
> +{
> +????????struct irq_chip *chip = irq_desc_get_chip(desc);
> +????????struct irq_data *gpio_irq_data = irq_desc_get_handler_data(desc);
> +
> +????????chained_irq_enter(chip, desc);
> +
> +????????if (gpio_irq_data)
> +????????????????generic_handle_irq(gpio_irq_data->irq);
> +
> +????????chained_irq_exit(chip, desc);
> +}
> +
> +static void meson_gpio_irq_unmask(struct irq_data *data) {}
> +static void meson_gpio_irq_mask(struct irq_data *data) {}
> +
> +static void meson_gpio_irq_shutdown(struct irq_data *data)
> +{
> + int hwirq = meson_gpio_to_hwirq(data);
> + int irq;
> +
> + if (hwirq <= 0)
> + return;
> +
> + /*
> + ?* In case of IRQ_TYPE_EDGE_BOTH we need two parent interrupts,
> + ?* one for each edge. That's due to HW constraints.
> + ?* We use format 2 * GPIO_HWIRQ +(0|1) for the hwirq, so we can
> + ?* have one GPIO_HWIRQ twice and derive the GPIO_HWIRQ from hwirq
> + ?* by shifting hwirq one bit to the right.
> + ?*/
> + irq = irq_find_mapping(meson_pinctrl_irq_domain, hwirq * 2);
> + if (irq) {
> + irq_set_chained_handler_and_data(irq, handle_bad_irq, NULL);
> + irq_domain_free_irqs(irq, 1);
> + }
> +
> + irq = irq_find_mapping(meson_pinctrl_irq_domain, hwirq * 2 + 1);
> + if (irq) {
> + irq_set_chained_handler_and_data(irq, handle_bad_irq, NULL);
> + irq_domain_free_irqs(irq, 1);
> + }
> +}
> +
> +static int meson_gpio_irq_set_type(struct irq_data *data, unsigned int type)
> +{
> + int hwirq = meson_gpio_to_hwirq(data);
> + struct irq_fwspec fwspec;
> +????????int irq, irq2, num_slots;
> +
> + if (irqd_is_activated(data))
> + return -EBUSY;
Again, your implementation assume that:
* irq_set_type should be called only once in the life of the irq.
* and/or irq_set_type cannot be called from irq context
I don't agree with those assumptions.
If irq_set_type could not called several times, to change the type at runtime,
why would the irq framework bother providing "irq_set_irq_type" in the public
API ? Why bother with irq_set_type callback at all ? It could all be done in the
startup callback...
> +
> + if (hwirq < 0)
> + return hwirq;
> +
> + if (!hwirq)
> + return -ENXIO;
> +
> + fwspec.fwnode = meson_pinctrl_irq_domain->fwnode;
> + fwspec.param_count = 2;
> +
> + /*
> + ?* The chip can create an interrupt for either rising or falling edge
> + ?* only. Therefore use two interrupts in case of IRQ_TYPE_EDGE_BOTH,
> + ?* first for falling edge and second one for rising edge.
> + ?*/
> + num_slots = (type == IRQ_TYPE_EDGE_BOTH) ? 2 : 1;
> +
> + /* see comment in meson_gpio_irq_shutdown why we shift one bit here
> */
> + fwspec.param[0] = hwirq << 1;
> + if (num_slots == 1)
> + fwspec.param[1] = type;
> + else
> + fwspec.param[1] = IRQ_TYPE_EDGE_FALLING;
> +
> + irq = irq_create_fwspec_mapping(&fwspec);
> + if (!irq)
> + return -EINVAL;
> +
> + irq_set_chained_handler_and_data(irq, meson_gpio_irq_handler, data);
> +
> + if (num_slots > 1) {
> + fwspec.param[0]++;
> + fwspec.param[1] = IRQ_TYPE_EDGE_RISING;
> + irq2 = irq_create_fwspec_mapping(&fwspec);
> + if (!irq2) {
> + irq_domain_free_irqs(irq, 1);
> + return -EINVAL;
> + }
> + irq_set_chained_handler_and_data(irq2,
> meson_gpio_irq_handler, data);
> + }
> +
> +????????return 0;
> +}
> +
> +static struct irq_chip meson_gpio_irq_chip = {
> +????????.name = "GPIO",
> +????????.irq_set_type = meson_gpio_irq_set_type,
> +????????.irq_mask = meson_gpio_irq_mask,
> +????????.irq_unmask = meson_gpio_irq_unmask,
> +????????.irq_shutdown = meson_gpio_irq_shutdown,
> +};
> +
> ?static const struct of_device_id meson_pinctrl_dt_match[] = {
> ? {
> ? .compatible = "amlogic,meson8-cbus-pinctrl",
> @@ -558,7 +708,8 @@ static int meson_gpiolib_register(struct meson_pinctrl
> *pc)
> ? return ret;
> ? }
> ?
> - return 0;
> + return gpiochip_irqchip_add(&pc->chip, &meson_gpio_irq_chip, 0,
> + ????handle_simple_irq, IRQ_TYPE_NONE);
> ?}
> ?
> ?static struct regmap_config meson_regmap_config = {
> @@ -637,6 +788,23 @@ static int meson_pinctrl_parse_dt(struct meson_pinctrl
> *pc,
> ? return PTR_ERR(pc->reg_gpio);
> ? }
> ?
> + if (!meson_pinctrl_irq_domain) {
This be in " struct meson_pinctrl *pc" not a global.
Each instance should parse its own bindings and not rely on the other one for
this !
> + np = of_find_compatible_node(NULL, NULL, "amlogic,meson-gpio-
> intc");
> + if (!np) {
> + dev_err(pc->dev, "interrupt controller DT node not
> found\n");
> + return -EINVAL;
> + }
> +
> + meson_pinctrl_irq_domain = irq_find_host(np);
> + if (!meson_pinctrl_irq_domain) {
> + dev_err(pc->dev, "interrupt controller not found\n");
> + of_node_put(np);
> + return -EINVAL;
> + }
> +
> + of_node_put(np);
> + }
> +
> ? return 0;
> ?}
> ?
WARNING: multiple messages have this Message-ID (diff)
From: Jerome Brunet <jbrunet@baylibre.com>
To: Heiner Kallweit <hkallweit1@gmail.com>,
Mark Rutland <mark.rutland@arm.com>,
Marc Zyngier <marc.zyngier@arm.com>,
Linus Walleij <linus.walleij@linaro.org>,
Kevin Hilman <khilman@baylibre.com>,
Thomas Gleixner <tglx@linutronix.de>,
Rob Herring <robh@kernel.org>,
Neil Armstrong <narmstrong@baylibre.com>
Cc: devicetree@vger.kernel.org, linux-amlogic@lists.infradead.org,
linux-gpio@vger.kernel.org,
"thierry.reding@gmail.com" <thierry.reding@gmail.com>,
Thierry Reding <treding@nvidia.com>
Subject: Re: [PATCH v6 6/9] pinctrl: meson: add support for GPIO interrupts
Date: Fri, 09 Jun 2017 11:06:25 +0200 [thread overview]
Message-ID: <1496999185.3552.60.camel@baylibre.com> (raw)
In-Reply-To: <a86d38af-9103-7557-9986-2dc187815569@gmail.com>
On Thu, 2017-06-08 at 21:39 +0200, Heiner Kallweit wrote:
> Add support for GPIO interrupts and make use of the just introduced
> irqchip driver handling the GPIO interrupt-controller.
>
> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
> ---
> v5:
> - changed Kconfig entry based on Neil's suggestion
> - extended comments
> - fixed indentation
> v6:
> - no changes
> ---
> drivers/pinctrl/Kconfig | 1 +
> drivers/pinctrl/meson/pinctrl-meson.c | 170
> +++++++++++++++++++++++++++++++++-
> 2 files changed, 170 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/pinctrl/Kconfig b/drivers/pinctrl/Kconfig
> index 7ae04a97..86834dea 100644
> --- a/drivers/pinctrl/Kconfig
> +++ b/drivers/pinctrl/Kconfig
> @@ -166,6 +166,7 @@ config PINCTRL_MESON
> select PINCONF
> select GENERIC_PINCONF
> select GPIOLIB
> + select GPIOLIB_IRQCHIP
> select OF_GPIO
> select REGMAP_MMIO
>
> diff --git a/drivers/pinctrl/meson/pinctrl-meson.c
> b/drivers/pinctrl/meson/pinctrl-meson.c
> index 66ed70c1..7bacd4e3 100644
> --- a/drivers/pinctrl/meson/pinctrl-meson.c
> +++ b/drivers/pinctrl/meson/pinctrl-meson.c
> @@ -62,6 +62,8 @@
> #include "../pinctrl-utils.h"
> #include "pinctrl-meson.h"
>
> +static struct irq_domain *meson_pinctrl_irq_domain;
> +
> /**
> * meson_get_bank() - find the bank containing a given pin
> *
> @@ -497,6 +499,154 @@ static int meson_gpio_get(struct gpio_chip *chip,
> unsigned gpio)
> return !!(val & BIT(bit));
> }
>
> +static struct meson_pinctrl *meson_gpio_data_to_pc(struct irq_data *data)
> +{
> + struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
> +
> + return gpiochip_get_data(chip);
> +}
> +
> +static int meson_gpio_bank_hwirq(struct meson_bank *bank, unsigned int
> offset)
> +{
> + int hwirq;
> +
> + if (bank->irq_first < 0)
> + /* this bank cannot generate irqs */
> + return 0;
> +
> + hwirq = offset - bank->first + bank->irq_first;
> +
> + if (hwirq > bank->irq_last)
> + /* this pin cannot generate irqs */
> + return 0;
> +
> + return hwirq;
> +}
> +
> +static int meson_gpio_to_hwirq(struct irq_data *data)
> +{
> + struct meson_pinctrl *pc = meson_gpio_data_to_pc(data);
> + unsigned int offset = data->hwirq;
> + struct meson_bank *bank;
> + int hwirq, ret;
> +
> + offset += pc->data->pin_base;
> +
> + ret = meson_get_bank(pc, offset, &bank);
> + if (ret)
> + return ret;
> +
> + hwirq = meson_gpio_bank_hwirq(bank, offset);
> + if (!hwirq)
> + dev_dbg(pc->dev, "no interrupt for pin %u\n", offset);
> +
> + return hwirq;
> +}
> +
> +static void meson_gpio_irq_handler(struct irq_desc *desc)
> +{
> + struct irq_chip *chip = irq_desc_get_chip(desc);
> + struct irq_data *gpio_irq_data = irq_desc_get_handler_data(desc);
> +
> + chained_irq_enter(chip, desc);
> +
> + if (gpio_irq_data)
> + generic_handle_irq(gpio_irq_data->irq);
> +
> + chained_irq_exit(chip, desc);
> +}
> +
> +static void meson_gpio_irq_unmask(struct irq_data *data) {}
> +static void meson_gpio_irq_mask(struct irq_data *data) {}
> +
> +static void meson_gpio_irq_shutdown(struct irq_data *data)
> +{
> + int hwirq = meson_gpio_to_hwirq(data);
> + int irq;
> +
> + if (hwirq <= 0)
> + return;
> +
> + /*
> + * In case of IRQ_TYPE_EDGE_BOTH we need two parent interrupts,
> + * one for each edge. That's due to HW constraints.
> + * We use format 2 * GPIO_HWIRQ +(0|1) for the hwirq, so we can
> + * have one GPIO_HWIRQ twice and derive the GPIO_HWIRQ from hwirq
> + * by shifting hwirq one bit to the right.
> + */
> + irq = irq_find_mapping(meson_pinctrl_irq_domain, hwirq * 2);
> + if (irq) {
> + irq_set_chained_handler_and_data(irq, handle_bad_irq, NULL);
> + irq_domain_free_irqs(irq, 1);
> + }
> +
> + irq = irq_find_mapping(meson_pinctrl_irq_domain, hwirq * 2 + 1);
> + if (irq) {
> + irq_set_chained_handler_and_data(irq, handle_bad_irq, NULL);
> + irq_domain_free_irqs(irq, 1);
> + }
> +}
> +
> +static int meson_gpio_irq_set_type(struct irq_data *data, unsigned int type)
> +{
> + int hwirq = meson_gpio_to_hwirq(data);
> + struct irq_fwspec fwspec;
> + int irq, irq2, num_slots;
> +
> + if (irqd_is_activated(data))
> + return -EBUSY;
Again, your implementation assume that:
* irq_set_type should be called only once in the life of the irq.
* and/or irq_set_type cannot be called from irq context
I don't agree with those assumptions.
If irq_set_type could not called several times, to change the type at runtime,
why would the irq framework bother providing "irq_set_irq_type" in the public
API ? Why bother with irq_set_type callback at all ? It could all be done in the
startup callback...
> +
> + if (hwirq < 0)
> + return hwirq;
> +
> + if (!hwirq)
> + return -ENXIO;
> +
> + fwspec.fwnode = meson_pinctrl_irq_domain->fwnode;
> + fwspec.param_count = 2;
> +
> + /*
> + * The chip can create an interrupt for either rising or falling edge
> + * only. Therefore use two interrupts in case of IRQ_TYPE_EDGE_BOTH,
> + * first for falling edge and second one for rising edge.
> + */
> + num_slots = (type == IRQ_TYPE_EDGE_BOTH) ? 2 : 1;
> +
> + /* see comment in meson_gpio_irq_shutdown why we shift one bit here
> */
> + fwspec.param[0] = hwirq << 1;
> + if (num_slots == 1)
> + fwspec.param[1] = type;
> + else
> + fwspec.param[1] = IRQ_TYPE_EDGE_FALLING;
> +
> + irq = irq_create_fwspec_mapping(&fwspec);
> + if (!irq)
> + return -EINVAL;
> +
> + irq_set_chained_handler_and_data(irq, meson_gpio_irq_handler, data);
> +
> + if (num_slots > 1) {
> + fwspec.param[0]++;
> + fwspec.param[1] = IRQ_TYPE_EDGE_RISING;
> + irq2 = irq_create_fwspec_mapping(&fwspec);
> + if (!irq2) {
> + irq_domain_free_irqs(irq, 1);
> + return -EINVAL;
> + }
> + irq_set_chained_handler_and_data(irq2,
> meson_gpio_irq_handler, data);
> + }
> +
> + return 0;
> +}
> +
> +static struct irq_chip meson_gpio_irq_chip = {
> + .name = "GPIO",
> + .irq_set_type = meson_gpio_irq_set_type,
> + .irq_mask = meson_gpio_irq_mask,
> + .irq_unmask = meson_gpio_irq_unmask,
> + .irq_shutdown = meson_gpio_irq_shutdown,
> +};
> +
> static const struct of_device_id meson_pinctrl_dt_match[] = {
> {
> .compatible = "amlogic,meson8-cbus-pinctrl",
> @@ -558,7 +708,8 @@ static int meson_gpiolib_register(struct meson_pinctrl
> *pc)
> return ret;
> }
>
> - return 0;
> + return gpiochip_irqchip_add(&pc->chip, &meson_gpio_irq_chip, 0,
> + handle_simple_irq, IRQ_TYPE_NONE);
> }
>
> static struct regmap_config meson_regmap_config = {
> @@ -637,6 +788,23 @@ static int meson_pinctrl_parse_dt(struct meson_pinctrl
> *pc,
> return PTR_ERR(pc->reg_gpio);
> }
>
> + if (!meson_pinctrl_irq_domain) {
This be in " struct meson_pinctrl *pc" not a global.
Each instance should parse its own bindings and not rely on the other one for
this !
> + np = of_find_compatible_node(NULL, NULL, "amlogic,meson-gpio-
> intc");
> + if (!np) {
> + dev_err(pc->dev, "interrupt controller DT node not
> found\n");
> + return -EINVAL;
> + }
> +
> + meson_pinctrl_irq_domain = irq_find_host(np);
> + if (!meson_pinctrl_irq_domain) {
> + dev_err(pc->dev, "interrupt controller not found\n");
> + of_node_put(np);
> + return -EINVAL;
> + }
> +
> + of_node_put(np);
> + }
> +
> return 0;
> }
>
next prev parent reply other threads:[~2017-06-09 9:06 UTC|newest]
Thread overview: 38+ messages / expand[flat|nested] mbox.gz Atom feed top
2017-06-08 19:14 [PATCH v6 0/9] pinctrl: meson: add support for GPIO IRQs Heiner Kallweit
2017-06-08 19:14 ` Heiner Kallweit
2017-06-08 19:37 ` [PATCH v6 1/9] pinctrl: meson: add interrupts to pinctrl data Heiner Kallweit
2017-06-08 19:37 ` Heiner Kallweit
2017-06-09 7:30 ` Linus Walleij
2017-06-09 7:30 ` Linus Walleij
2017-06-08 19:38 ` [PATCH v6 2/9] irqchip: add Amlogic Meson GPIO irqchip driver Heiner Kallweit
2017-06-08 19:38 ` Heiner Kallweit
2017-06-09 7:31 ` Linus Walleij
2017-06-09 7:31 ` Linus Walleij
2017-06-09 9:07 ` Jerome Brunet
2017-06-09 9:07 ` Jerome Brunet
2017-06-09 18:38 ` Heiner Kallweit
2017-06-09 18:38 ` Heiner Kallweit
2017-06-09 21:15 ` Kevin Hilman
2017-06-09 21:15 ` Kevin Hilman
2017-06-09 22:30 ` Heiner Kallweit
2017-06-09 22:30 ` Heiner Kallweit
2017-06-09 23:30 ` Kevin Hilman
2017-06-09 23:30 ` Kevin Hilman
2017-06-08 19:38 ` [PATCH v6 3/9] dt-bindings: add Amlogic Meson GPIO interrupt-controller DT binding documentation Heiner Kallweit
2017-06-08 19:38 ` Heiner Kallweit
2017-06-08 19:38 ` [PATCH v6 4/9] ARM: dts: meson: add GPIO interrupt-controller support Heiner Kallweit
2017-06-08 19:38 ` Heiner Kallweit
2017-06-08 19:39 ` [PATCH v6 5/9] ARM64: " Heiner Kallweit
2017-06-08 19:39 ` Heiner Kallweit
2017-06-08 19:39 ` [PATCH v6 6/9] pinctrl: meson: add support for GPIO interrupts Heiner Kallweit
2017-06-08 19:39 ` Heiner Kallweit
2017-06-09 9:06 ` Jerome Brunet [this message]
2017-06-09 9:06 ` Jerome Brunet
2017-06-09 18:09 ` Heiner Kallweit
2017-06-09 18:09 ` Heiner Kallweit
2017-06-08 19:39 ` [PATCH v6 7/9] pinctrl: meson: update DT binding documentation Heiner Kallweit
2017-06-08 19:39 ` Heiner Kallweit
2017-06-08 19:39 ` [PATCH v6 8/9] ARM: dts: meson: mark gpio controllers as interrupt controllers Heiner Kallweit
2017-06-08 19:39 ` Heiner Kallweit
2017-06-08 19:39 ` [PATCH v6 9/9] ARM64: " Heiner Kallweit
2017-06-08 19:39 ` Heiner Kallweit
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=1496999185.3552.60.camel@baylibre.com \
--to=jbrunet@baylibre.com \
--cc=linus-amlogic@lists.infradead.org \
/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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.