From: Thierry Reding <thierry.reding-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
To: Linus Walleij <linus.walleij-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org>
Cc: Suresh Mangipudi
<smangipudi-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>,
Laxman Dewangan
<ldewangan-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>,
Jon Hunter <jonathanh-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>,
linux-gpio-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
linux-tegra-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
Subject: Re: [PATCH v2] gpio: Add Tegra186 support
Date: Fri, 31 Mar 2017 15:10:06 +0200 [thread overview]
Message-ID: <20170331131006.GC29779@ulmo.ba.sec> (raw)
In-Reply-To: <20170310162629.31455-1-thierry.reding-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
[-- Attachment #1: Type: text/plain, Size: 21831 bytes --]
On Fri, Mar 10, 2017 at 05:26:29PM +0100, Thierry Reding wrote:
> From: Thierry Reding <treding-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
>
> Tegra186 has two GPIO controllers that are largely register compatible
> between one another but are completely different from the controller
> found on earlier generations.
>
> Signed-off-by: Thierry Reding <treding-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
> ---
> Changes in v2:
> - add pin names to allow easy lookup using the chardev interface
> - distinguish AON and main GPIO controllers by label
> - use gpiochip_get_data() instead of container_of()
> - use C99 initializers
>
> Hi Linus,
>
> This addresses all comments I got on the last revision, except for your
> request to use gpiolib's IRQ chip helpers. I investigated using it with
> the driver but still can't figure out how it's supposed to work given
> our requirement to service multiple parent interrupts.
>
> Thierry
>
> drivers/gpio/Kconfig | 8 +
> drivers/gpio/Makefile | 1 +
> drivers/gpio/gpio-tegra186.c | 636 +++++++++++++++++++++++++++++++++++++++++++
> 3 files changed, 645 insertions(+)
> create mode 100644 drivers/gpio/gpio-tegra186.c
Hi Linus,
any objections to merging this for v4.12? I know Laxman has some
reservations about the internal numbering, but I hope my explanation in
the other subthread shows that this is the simplest way to achieve what
we want with the least amount of surprises.
Thierry
> diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig
> index 05043071fc98..f74b61cfd113 100644
> --- a/drivers/gpio/Kconfig
> +++ b/drivers/gpio/Kconfig
> @@ -431,6 +431,14 @@ config GPIO_TEGRA
> help
> Say yes here to support GPIO pins on NVIDIA Tegra SoCs.
>
> +config GPIO_TEGRA186
> + tristate "NVIDIA Tegra186 GPIO support"
> + default ARCH_TEGRA_186_SOC
> + depends on ARCH_TEGRA_186_SOC || COMPILE_TEST
> + depends on OF_GPIO
> + help
> + Say yes here to support GPIO pins on NVIDIA Tegra186 SoCs.
> +
> config GPIO_TS4800
> tristate "TS-4800 DIO blocks and compatibles"
> depends on OF_GPIO
> diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile
> index becb96c724fe..cad0eb2c3531 100644
> --- a/drivers/gpio/Makefile
> +++ b/drivers/gpio/Makefile
> @@ -111,6 +111,7 @@ obj-$(CONFIG_GPIO_SYSCON) += gpio-syscon.o
> obj-$(CONFIG_GPIO_TB10X) += gpio-tb10x.o
> obj-$(CONFIG_GPIO_TC3589X) += gpio-tc3589x.o
> obj-$(CONFIG_GPIO_TEGRA) += gpio-tegra.o
> +obj-$(CONFIG_GPIO_TEGRA186) += gpio-tegra186.o
> obj-$(CONFIG_GPIO_TIMBERDALE) += gpio-timberdale.o
> obj-$(CONFIG_GPIO_PALMAS) += gpio-palmas.o
> obj-$(CONFIG_GPIO_TPIC2810) += gpio-tpic2810.o
> diff --git a/drivers/gpio/gpio-tegra186.c b/drivers/gpio/gpio-tegra186.c
> new file mode 100644
> index 000000000000..87e02d9165d8
> --- /dev/null
> +++ b/drivers/gpio/gpio-tegra186.c
> @@ -0,0 +1,636 @@
> +/*
> + * Copyright (c) 2016-2017 NVIDIA Corporation
> + *
> + * Author: Thierry Reding <treding-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
> + *
> + * This software is licensed under the terms of the GNU General Public
> + * License version 2, as published by the Free Software Foundation, and
> + * may be copied, distributed, and modified under those terms.
> + */
> +
> +#include <linux/gpio/driver.h>
> +#include <linux/gpio.h>
> +#include <linux/interrupt.h>
> +#include <linux/irq.h>
> +#include <linux/module.h>
> +#include <linux/of_device.h>
> +#include <linux/of_irq.h>
> +#include <linux/platform_device.h>
> +
> +#include <dt-bindings/gpio/tegra186-gpio.h>
> +
> +#define TEGRA186_GPIO_ENABLE_CONFIG 0x00
> +#define TEGRA186_GPIO_ENABLE_CONFIG_ENABLE BIT(0)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_OUT BIT(1)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_NONE (0x0 << 2)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_LEVEL (0x1 << 2)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_SINGLE_EDGE (0x2 << 2)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_DOUBLE_EDGE (0x3 << 2)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_MASK (0x3 << 2)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_LEVEL BIT(4)
> +#define TEGRA186_GPIO_ENABLE_CONFIG_INTERRUPT BIT(6)
> +
> +#define TEGRA186_GPIO_DEBOUNCE_CONTROL 0x04
> +#define TEGRA186_GPIO_DEBOUNCE_CONTROL_THRESHOLD(x) ((x) & 0xff)
> +
> +#define TEGRA186_GPIO_INPUT 0x08
> +#define TEGRA186_GPIO_INPUT_HIGH BIT(0)
> +
> +#define TEGRA186_GPIO_OUTPUT_CONTROL 0x0c
> +#define TEGRA186_GPIO_OUTPUT_CONTROL_FLOATED BIT(0)
> +
> +#define TEGRA186_GPIO_OUTPUT_VALUE 0x10
> +#define TEGRA186_GPIO_OUTPUT_VALUE_HIGH BIT(0)
> +
> +#define TEGRA186_GPIO_INTERRUPT_CLEAR 0x14
> +
> +#define TEGRA186_GPIO_INTERRUPT_STATUS(x) (0x100 + (x) * 4)
> +
> +struct tegra_gpio_port {
> + const char *name;
> + unsigned int offset;
> + unsigned int pins;
> +};
> +
> +struct tegra_gpio_soc {
> + const struct tegra_gpio_port *ports;
> + unsigned int num_ports;
> + const char *name;
> +};
> +
> +struct tegra_gpio {
> + struct gpio_chip gpio;
> + struct irq_chip intc;
> + unsigned int num_irq;
> + unsigned int *irq;
> +
> + const struct tegra_gpio_soc *soc;
> +
> + void __iomem *base;
> +
> + struct irq_domain *domain;
> +};
> +
> +static const struct tegra_gpio_port *
> +tegra186_gpio_get_port(struct tegra_gpio *gpio, unsigned int *pin)
> +{
> + unsigned int start = 0, i;
> +
> + for (i = 0; i < gpio->soc->num_ports; i++) {
> + const struct tegra_gpio_port *port = &gpio->soc->ports[i];
> +
> + if (*pin >= start && *pin < start + port->pins) {
> + *pin -= start;
> + return port;
> + }
> +
> + start += port->pins;
> + }
> +
> + return NULL;
> +}
> +
> +static void __iomem *tegra186_gpio_get_base(struct tegra_gpio *gpio,
> + unsigned int pin)
> +{
> + const struct tegra_gpio_port *port;
> +
> + port = tegra186_gpio_get_port(gpio, &pin);
> + if (!port)
> + return NULL;
> +
> + return gpio->base + port->offset + pin * 0x20;
> +}
> +
> +static int tegra186_gpio_get_direction(struct gpio_chip *chip,
> + unsigned int offset)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, offset);
> + if (WARN_ON(base == NULL))
> + return -ENODEV;
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + if (value & TEGRA186_GPIO_ENABLE_CONFIG_OUT)
> + return GPIOF_DIR_OUT;
> +
> + return GPIOF_DIR_IN;
> +}
> +
> +static int tegra186_gpio_direction_input(struct gpio_chip *chip,
> + unsigned int offset)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, offset);
> + if (WARN_ON(base == NULL))
> + return -ENODEV;
> +
> + value = readl(base + TEGRA186_GPIO_OUTPUT_CONTROL);
> + value |= TEGRA186_GPIO_OUTPUT_CONTROL_FLOATED;
> + writel(value, base + TEGRA186_GPIO_OUTPUT_CONTROL);
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_ENABLE;
> + value &= ~TEGRA186_GPIO_ENABLE_CONFIG_OUT;
> + writel(value, base + TEGRA186_GPIO_ENABLE_CONFIG);
> +
> + return 0;
> +}
> +
> +static int tegra186_gpio_direction_output(struct gpio_chip *chip,
> + unsigned int offset, int level)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> + void __iomem *base;
> + u32 value;
> +
> + /* configure output level first */
> + chip->set(chip, offset, level);
> +
> + base = tegra186_gpio_get_base(gpio, offset);
> + if (WARN_ON(base == NULL))
> + return -EINVAL;
> +
> + /* set the direction */
> + value = readl(base + TEGRA186_GPIO_OUTPUT_CONTROL);
> + value &= ~TEGRA186_GPIO_OUTPUT_CONTROL_FLOATED;
> + writel(value, base + TEGRA186_GPIO_OUTPUT_CONTROL);
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_ENABLE;
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_OUT;
> + writel(value, base + TEGRA186_GPIO_ENABLE_CONFIG);
> +
> + return 0;
> +}
> +
> +static int tegra186_gpio_get(struct gpio_chip *chip, unsigned int offset)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, offset);
> + if (WARN_ON(base == NULL))
> + return -ENODEV;
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + if (value & TEGRA186_GPIO_ENABLE_CONFIG_OUT)
> + value = readl(base + TEGRA186_GPIO_OUTPUT_VALUE);
> + else
> + value = readl(base + TEGRA186_GPIO_INPUT);
> +
> + return value & BIT(0);
> +}
> +
> +static void tegra186_gpio_set(struct gpio_chip *chip, unsigned int offset,
> + int level)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, offset);
> + if (WARN_ON(base == NULL))
> + return;
> +
> + value = readl(base + TEGRA186_GPIO_OUTPUT_VALUE);
> + if (level == 0)
> + value &= ~TEGRA186_GPIO_OUTPUT_VALUE_HIGH;
> + else
> + value |= TEGRA186_GPIO_OUTPUT_VALUE_HIGH;
> +
> + writel(value, base + TEGRA186_GPIO_OUTPUT_VALUE);
> +}
> +
> +static int tegra186_gpio_to_irq(struct gpio_chip *chip, unsigned int offset)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> +
> + return irq_find_mapping(gpio->domain, offset);
> +}
> +
> +static int tegra186_gpio_of_xlate(struct gpio_chip *chip,
> + const struct of_phandle_args *spec,
> + u32 *flags)
> +{
> + struct tegra_gpio *gpio = gpiochip_get_data(chip);
> + unsigned int port, pin, i, offset = 0;
> +
> + if (WARN_ON(chip->of_gpio_n_cells < 2))
> + return -EINVAL;
> +
> + if (WARN_ON(spec->args_count < chip->of_gpio_n_cells))
> + return -EINVAL;
> +
> + port = spec->args[0] / 8;
> + pin = spec->args[0] % 8;
> +
> + if (port >= gpio->soc->num_ports) {
> + dev_err(chip->parent, "invalid port number: %u\n", port);
> + return -EINVAL;
> + }
> +
> + for (i = 0; i < port; i++)
> + offset += gpio->soc->ports[i].pins;
> +
> + if (flags)
> + *flags = spec->args[1];
> +
> + return offset + pin;
> +}
> +
> +static void tegra186_irq_ack(struct irq_data *data)
> +{
> + struct tegra_gpio *gpio = irq_data_get_irq_chip_data(data);
> + void __iomem *base;
> +
> + base = tegra186_gpio_get_base(gpio, data->hwirq);
> + if (WARN_ON(base == NULL))
> + return;
> +
> + writel(1, base + TEGRA186_GPIO_INTERRUPT_CLEAR);
> +}
> +
> +static void tegra186_irq_mask(struct irq_data *data)
> +{
> + struct tegra_gpio *gpio = irq_data_get_irq_chip_data(data);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, data->hwirq);
> + if (WARN_ON(base == NULL))
> + return;
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + value &= ~TEGRA186_GPIO_ENABLE_CONFIG_INTERRUPT;
> + writel(value, base + TEGRA186_GPIO_ENABLE_CONFIG);
> +}
> +
> +static void tegra186_irq_unmask(struct irq_data *data)
> +{
> + struct tegra_gpio *gpio = irq_data_get_irq_chip_data(data);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, data->hwirq);
> + if (WARN_ON(base == NULL))
> + return;
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_INTERRUPT;
> + writel(value, base + TEGRA186_GPIO_ENABLE_CONFIG);
> +}
> +
> +static int tegra186_irq_set_type(struct irq_data *data, unsigned int flow)
> +{
> + struct tegra_gpio *gpio = irq_data_get_irq_chip_data(data);
> + void __iomem *base;
> + u32 value;
> +
> + base = tegra186_gpio_get_base(gpio, data->hwirq);
> + if (WARN_ON(base == NULL))
> + return -ENODEV;
> +
> + value = readl(base + TEGRA186_GPIO_ENABLE_CONFIG);
> + value &= ~TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_MASK;
> + value &= ~TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_LEVEL;
> +
> + switch (flow & IRQ_TYPE_SENSE_MASK) {
> + case IRQ_TYPE_NONE:
> + break;
> +
> + case IRQ_TYPE_EDGE_RISING:
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_SINGLE_EDGE;
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_LEVEL;
> + break;
> +
> + case IRQ_TYPE_EDGE_FALLING:
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_SINGLE_EDGE;
> + break;
> +
> + case IRQ_TYPE_EDGE_BOTH:
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_DOUBLE_EDGE;
> + break;
> +
> + case IRQ_TYPE_LEVEL_HIGH:
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_LEVEL;
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_LEVEL;
> + break;
> +
> + case IRQ_TYPE_LEVEL_LOW:
> + value |= TEGRA186_GPIO_ENABLE_CONFIG_TRIGGER_TYPE_LEVEL;
> + break;
> +
> + default:
> + return -EINVAL;
> + }
> +
> + writel(value, base + TEGRA186_GPIO_ENABLE_CONFIG);
> +
> + if ((flow & IRQ_TYPE_EDGE_BOTH) == 0)
> + irq_set_handler_locked(data, handle_level_irq);
> + else
> + irq_set_handler_locked(data, handle_edge_irq);
> +
> + return 0;
> +}
> +
> +static void tegra186_gpio_irq(struct irq_desc *desc)
> +{
> + struct tegra_gpio *gpio = irq_desc_get_handler_data(desc);
> + struct irq_chip *chip = irq_desc_get_chip(desc);
> + unsigned int i, offset = 0;
> +
> + chained_irq_enter(chip, desc);
> +
> + for (i = 0; i < gpio->soc->num_ports; i++) {
> + const struct tegra_gpio_port *port = &gpio->soc->ports[i];
> + void __iomem *base = gpio->base + port->offset;
> + unsigned int pin, irq;
> + unsigned long value;
> +
> + value = readl(base + TEGRA186_GPIO_INTERRUPT_STATUS(1));
> +
> + for_each_set_bit(pin, &value, port->pins) {
> + irq = irq_find_mapping(gpio->domain, offset + pin);
> + if (WARN_ON(irq == 0))
> + continue;
> +
> + generic_handle_irq(irq);
> + }
> +
> + offset += port->pins;
> + }
> +
> + chained_irq_exit(chip, desc);
> +}
> +
> +static int tegra186_gpio_irq_domain_xlate(struct irq_domain *domain,
> + struct device_node *np,
> + const u32 *spec, unsigned int size,
> + unsigned long *hwirq,
> + unsigned int *type)
> +{
> + struct tegra_gpio *gpio = domain->host_data;
> + unsigned int port, pin, i, offset = 0;
> +
> + if (size < 2)
> + return -EINVAL;
> +
> + port = spec[0] / 8;
> + pin = spec[0] % 8;
> +
> + if (port >= gpio->soc->num_ports) {
> + dev_err(gpio->gpio.parent, "invalid port number: %u\n", port);
> + return -EINVAL;
> + }
> +
> + for (i = 0; i < port; i++)
> + offset += gpio->soc->ports[i].pins;
> +
> + *type = spec[1] & IRQ_TYPE_SENSE_MASK;
> + *hwirq = offset + pin;
> +
> + return 0;
> +}
> +
> +static const struct irq_domain_ops tegra186_gpio_irq_domain_ops = {
> + .xlate = tegra186_gpio_irq_domain_xlate,
> +};
> +
> +static struct lock_class_key tegra186_gpio_lock_class;
> +
> +static int tegra186_gpio_probe(struct platform_device *pdev)
> +{
> + unsigned int i, j, irq, offset = 0;
> + struct tegra_gpio *gpio;
> + struct resource *res;
> + char **names;
> + int err;
> +
> + gpio = devm_kzalloc(&pdev->dev, sizeof(*gpio), GFP_KERNEL);
> + if (!gpio)
> + return -ENOMEM;
> +
> + gpio->soc = of_device_get_match_data(&pdev->dev);
> +
> + res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "gpio");
> + gpio->base = devm_ioremap_resource(&pdev->dev, res);
> + if (IS_ERR(gpio->base))
> + return PTR_ERR(gpio->base);
> +
> + err = of_irq_count(pdev->dev.of_node);
> + if (err < 0)
> + return err;
> +
> + gpio->num_irq = err;
> +
> + gpio->irq = devm_kcalloc(&pdev->dev, gpio->num_irq, sizeof(*gpio->irq),
> + GFP_KERNEL);
> + if (!gpio->irq)
> + return -ENOMEM;
> +
> + for (i = 0; i < gpio->num_irq; i++) {
> + err = platform_get_irq(pdev, i);
> + if (err < 0)
> + return err;
> +
> + gpio->irq[i] = err;
> + }
> +
> + gpio->gpio.label = gpio->soc->name;
> + gpio->gpio.parent = &pdev->dev;
> +
> + gpio->gpio.get_direction = tegra186_gpio_get_direction;
> + gpio->gpio.direction_input = tegra186_gpio_direction_input;
> + gpio->gpio.direction_output = tegra186_gpio_direction_output;
> + gpio->gpio.get = tegra186_gpio_get,
> + gpio->gpio.set = tegra186_gpio_set;
> + gpio->gpio.to_irq = tegra186_gpio_to_irq;
> +
> + gpio->gpio.base = -1;
> +
> + for (i = 0; i < gpio->soc->num_ports; i++)
> + gpio->gpio.ngpio += gpio->soc->ports[i].pins;
> +
> + names = devm_kcalloc(gpio->gpio.parent, gpio->gpio.ngpio,
> + sizeof(*names), GFP_KERNEL);
> + if (!names)
> + return -ENOMEM;
> +
> + for (i = 0; i < gpio->soc->num_ports; i++) {
> + const struct tegra_gpio_port *port = &gpio->soc->ports[i];
> + char *name;
> +
> + for (j = 0; j < port->pins; j++) {
> + name = devm_kasprintf(gpio->gpio.parent, GFP_KERNEL,
> + "P%s.%02x", port->name, j);
> + if (!name)
> + return -ENOMEM;
> +
> + names[offset + j] = name;
> + }
> +
> + offset += port->pins;
> + }
> +
> + gpio->gpio.names = (const char * const *)names;
> +
> + gpio->gpio.of_node = pdev->dev.of_node;
> + gpio->gpio.of_gpio_n_cells = 2;
> + gpio->gpio.of_xlate = tegra186_gpio_of_xlate;
> +
> + gpio->intc.name = pdev->dev.of_node->name;
> + gpio->intc.irq_ack = tegra186_irq_ack;
> + gpio->intc.irq_mask = tegra186_irq_mask;
> + gpio->intc.irq_unmask = tegra186_irq_unmask;
> + gpio->intc.irq_set_type = tegra186_irq_set_type;
> +
> + platform_set_drvdata(pdev, gpio);
> +
> + err = devm_gpiochip_add_data(&pdev->dev, &gpio->gpio, gpio);
> + if (err < 0)
> + return err;
> +
> + gpio->domain = irq_domain_add_linear(pdev->dev.of_node,
> + gpio->gpio.ngpio,
> + &tegra186_gpio_irq_domain_ops,
> + gpio);
> + if (!gpio->domain)
> + return -ENODEV;
> +
> + for (i = 0; i < gpio->gpio.ngpio; i++) {
> + irq = irq_create_mapping(gpio->domain, i);
> + if (irq == 0) {
> + dev_err(&pdev->dev,
> + "failed to create IRQ mapping for GPIO#%u\n",
> + i);
> + continue;
> + }
> +
> + irq_set_lockdep_class(irq, &tegra186_gpio_lock_class);
> + irq_set_chip_data(irq, gpio);
> + irq_set_chip_and_handler(irq, &gpio->intc, handle_simple_irq);
> + }
> +
> + for (i = 0; i < gpio->num_irq; i++)
> + irq_set_chained_handler_and_data(gpio->irq[i],
> + tegra186_gpio_irq,
> + gpio);
> +
> + return 0;
> +}
> +
> +static int tegra186_gpio_remove(struct platform_device *pdev)
> +{
> + struct tegra_gpio *gpio = platform_get_drvdata(pdev);
> + unsigned int i, irq;
> +
> + for (i = 0; i < gpio->num_irq; i++)
> + irq_set_chained_handler_and_data(gpio->irq[i], NULL, NULL);
> +
> + for (i = 0; i < gpio->gpio.ngpio; i++) {
> + irq = irq_find_mapping(gpio->domain, i);
> + irq_dispose_mapping(irq);
> + }
> +
> + irq_domain_remove(gpio->domain);
> +
> + return 0;
> +}
> +
> +#define TEGRA_MAIN_GPIO_PORT(port, base, count) \
> + [TEGRA_MAIN_GPIO_PORT_##port] = { \
> + .name = #port, \
> + .offset = base, \
> + .pins = count, \
> + }
> +
> +static const struct tegra_gpio_port tegra186_main_ports[] = {
> + TEGRA_MAIN_GPIO_PORT( A, 0x2000, 7),
> + TEGRA_MAIN_GPIO_PORT( B, 0x3000, 7),
> + TEGRA_MAIN_GPIO_PORT( C, 0x3200, 7),
> + TEGRA_MAIN_GPIO_PORT( D, 0x3400, 6),
> + TEGRA_MAIN_GPIO_PORT( E, 0x2200, 8),
> + TEGRA_MAIN_GPIO_PORT( F, 0x2400, 6),
> + TEGRA_MAIN_GPIO_PORT( G, 0x4200, 6),
> + TEGRA_MAIN_GPIO_PORT( H, 0x1000, 7),
> + TEGRA_MAIN_GPIO_PORT( I, 0x0800, 8),
> + TEGRA_MAIN_GPIO_PORT( J, 0x5000, 8),
> + TEGRA_MAIN_GPIO_PORT( K, 0x5200, 1),
> + TEGRA_MAIN_GPIO_PORT( L, 0x1200, 8),
> + TEGRA_MAIN_GPIO_PORT( M, 0x5600, 6),
> + TEGRA_MAIN_GPIO_PORT( N, 0x0000, 7),
> + TEGRA_MAIN_GPIO_PORT( O, 0x0200, 4),
> + TEGRA_MAIN_GPIO_PORT( P, 0x4000, 7),
> + TEGRA_MAIN_GPIO_PORT( Q, 0x0400, 6),
> + TEGRA_MAIN_GPIO_PORT( R, 0x0a00, 6),
> + TEGRA_MAIN_GPIO_PORT( T, 0x0600, 4),
> + TEGRA_MAIN_GPIO_PORT( X, 0x1400, 8),
> + TEGRA_MAIN_GPIO_PORT( Y, 0x1600, 7),
> + TEGRA_MAIN_GPIO_PORT(BB, 0x2600, 2),
> + TEGRA_MAIN_GPIO_PORT(CC, 0x5400, 4),
> +};
> +
> +static const struct tegra_gpio_soc tegra186_main_soc = {
> + .num_ports = ARRAY_SIZE(tegra186_main_ports),
> + .ports = tegra186_main_ports,
> + .name = "tegra186-gpio",
> +};
> +
> +#define TEGRA_AON_GPIO_PORT(port, base, count) \
> + [TEGRA_AON_GPIO_PORT_##port] = { \
> + .name = #port, \
> + .offset = base, \
> + .pins = count, \
> + }
> +
> +static const struct tegra_gpio_port tegra186_aon_ports[] = {
> + TEGRA_AON_GPIO_PORT( S, 0x0200, 5),
> + TEGRA_AON_GPIO_PORT( U, 0x0400, 6),
> + TEGRA_AON_GPIO_PORT( V, 0x0800, 8),
> + TEGRA_AON_GPIO_PORT( W, 0x0a00, 8),
> + TEGRA_AON_GPIO_PORT( Z, 0x0e00, 4),
> + TEGRA_AON_GPIO_PORT(AA, 0x0c00, 8),
> + TEGRA_AON_GPIO_PORT(EE, 0x0600, 3),
> + TEGRA_AON_GPIO_PORT(FF, 0x0000, 5),
> +};
> +
> +static const struct tegra_gpio_soc tegra186_aon_soc = {
> + .num_ports = ARRAY_SIZE(tegra186_aon_ports),
> + .ports = tegra186_aon_ports,
> + .name = "tegra186-gpio-aon",
> +};
> +
> +static const struct of_device_id tegra186_gpio_of_match[] = {
> + {
> + .compatible = "nvidia,tegra186-gpio",
> + .data = &tegra186_main_soc
> + }, {
> + .compatible = "nvidia,tegra186-gpio-aon",
> + .data = &tegra186_aon_soc
> + }, {
> + /* sentinel */
> + }
> +};
> +
> +static struct platform_driver tegra186_gpio_driver = {
> + .driver = {
> + .name = "tegra186-gpio",
> + .of_match_table = tegra186_gpio_of_match,
> + },
> + .probe = tegra186_gpio_probe,
> + .remove = tegra186_gpio_remove,
> +};
> +module_platform_driver(tegra186_gpio_driver);
> +
> +MODULE_DESCRIPTION("NVIDIA Tegra186 GPIO controller driver");
> +MODULE_AUTHOR("Thierry Reding <treding-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>");
> +MODULE_LICENSE("GPL v2");
> --
> 2.12.0
>
[-- Attachment #2: signature.asc --]
[-- Type: application/pgp-signature, Size: 833 bytes --]
next prev parent reply other threads:[~2017-03-31 13:10 UTC|newest]
Thread overview: 12+ messages / expand[flat|nested] mbox.gz Atom feed top
2017-03-10 16:26 [PATCH v2] gpio: Add Tegra186 support Thierry Reding
2017-03-13 7:06 ` Thierry Reding
[not found] ` <20170313070623.GA15513-EkSeR96xj6Pcmrwk2tT4+A@public.gmane.org>
2017-03-14 15:20 ` Linus Walleij
[not found] ` <20170310162629.31455-1-thierry.reding-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2017-03-13 16:22 ` Laxman Dewangan
[not found] ` <58C6C72C.4080408-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org>
2017-03-13 17:44 ` Thierry Reding
2017-03-31 13:10 ` Thierry Reding [this message]
[not found] ` <20170331131006.GC29779-EkSeR96xj6Pcmrwk2tT4+A@public.gmane.org>
2017-03-31 13:36 ` Linus Walleij
[not found] ` <CACRpkdZwb37kSgKDo=6v-G102KGs0NVB1CZg+MEOKDaA8dKuLg-JsoAwUIsXosN+BqQ9rBEUg@public.gmane.org>
2017-04-01 7:45 ` Laxman Dewangan
2017-03-31 13:59 ` Linus Walleij
[not found] ` <CACRpkdZpev7op=4e7DwYfnRpNBPN6U_EXd6G3PqM-XQudLHqow-JsoAwUIsXosN+BqQ9rBEUg@public.gmane.org>
2017-03-31 14:54 ` Thierry Reding
[not found] ` <20170331145437.GA16964-EkSeR96xj6Pcmrwk2tT4+A@public.gmane.org>
2017-04-01 17:46 ` Linus Walleij
2017-04-03 5:28 ` Thierry Reding
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=20170331131006.GC29779@ulmo.ba.sec \
--to=thierry.reding-re5jqeeqqe8avxtiumwx3w@public.gmane.org \
--cc=jonathanh-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org \
--cc=ldewangan-DDmLM1+adcrQT0dZR+AlfA@public.gmane.org \
--cc=linus.walleij-QSEj5FYQhm4dnm+yROfE0A@public.gmane.org \
--cc=linux-gpio-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=linux-tegra-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=smangipudi-DDmLM1+adcrQT0dZR+AlfA@public.gmane.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 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).