mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-27 08:14:35 +08:00
gpio: ixp4xx: Convert to hierarchical GPIOLIB_IRQCHIP
This modifies the IXP4xx driver to use the new helpers to handle the remapping of parent to child hardware irqs in the gpiolib core. This pulls the majority of the code out of the driver and use the generic code in gpiolib. Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Marc Zyngier <marc.zyngier@arm.com> Cc: Lina Iyer <ilina@codeaurora.org> Cc: Jon Hunter <jonathanh@nvidia.com> Cc: Sowjanya Komatineni <skomatineni@nvidia.com> Cc: Bitan Biswas <bbiswas@nvidia.com> Cc: linux-tegra@vger.kernel.org Cc: Thierry Reding <treding@nvidia.com> Cc: Brian Masney <masneyb@onstation.org> Signed-off-by: Linus Walleij <linus.walleij@linaro.org> Link: https://lore.kernel.org/r/20190808123242.5359-2-linus.walleij@linaro.org
This commit is contained in:
parent
fdd61a013a
commit
aa7d618ac6
@ -288,7 +288,7 @@ config GPIO_IXP4XX
|
||||
depends on ARM # For <asm/mach-types.h>
|
||||
depends on ARCH_IXP4XX
|
||||
select GPIO_GENERIC
|
||||
select IRQ_DOMAIN
|
||||
select GPIOLIB_IRQCHIP
|
||||
select IRQ_DOMAIN_HIERARCHY
|
||||
help
|
||||
Say yes here to support the GPIO functionality of a number of Intel
|
||||
|
@ -47,7 +47,6 @@
|
||||
* @dev: containing device for this instance
|
||||
* @fwnode: the fwnode for this GPIO chip
|
||||
* @gc: gpiochip for this instance
|
||||
* @domain: irqdomain for this chip instance
|
||||
* @base: remapped I/O-memory base
|
||||
* @irq_edge: Each bit represents an IRQ: 1: edge-triggered,
|
||||
* 0: level triggered
|
||||
@ -56,48 +55,22 @@ struct ixp4xx_gpio {
|
||||
struct device *dev;
|
||||
struct fwnode_handle *fwnode;
|
||||
struct gpio_chip gc;
|
||||
struct irq_domain *domain;
|
||||
void __iomem *base;
|
||||
unsigned long long irq_edge;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct ixp4xx_gpio_map - IXP4 GPIO to parent IRQ map
|
||||
* @gpio_offset: offset of the IXP4 GPIO line
|
||||
* @parent_hwirq: hwirq on the parent IRQ controller
|
||||
*/
|
||||
struct ixp4xx_gpio_map {
|
||||
int gpio_offset;
|
||||
int parent_hwirq;
|
||||
};
|
||||
|
||||
/* GPIO lines 0..12 have corresponding IRQs, GPIOs 13..15 have no IRQs */
|
||||
const struct ixp4xx_gpio_map ixp4xx_gpiomap[] = {
|
||||
{ .gpio_offset = 0, .parent_hwirq = 6 },
|
||||
{ .gpio_offset = 1, .parent_hwirq = 7 },
|
||||
{ .gpio_offset = 2, .parent_hwirq = 19 },
|
||||
{ .gpio_offset = 3, .parent_hwirq = 20 },
|
||||
{ .gpio_offset = 4, .parent_hwirq = 21 },
|
||||
{ .gpio_offset = 5, .parent_hwirq = 22 },
|
||||
{ .gpio_offset = 6, .parent_hwirq = 23 },
|
||||
{ .gpio_offset = 7, .parent_hwirq = 24 },
|
||||
{ .gpio_offset = 8, .parent_hwirq = 25 },
|
||||
{ .gpio_offset = 9, .parent_hwirq = 26 },
|
||||
{ .gpio_offset = 10, .parent_hwirq = 27 },
|
||||
{ .gpio_offset = 11, .parent_hwirq = 28 },
|
||||
{ .gpio_offset = 12, .parent_hwirq = 29 },
|
||||
};
|
||||
|
||||
static void ixp4xx_gpio_irq_ack(struct irq_data *d)
|
||||
{
|
||||
struct ixp4xx_gpio *g = irq_data_get_irq_chip_data(d);
|
||||
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
||||
struct ixp4xx_gpio *g = gpiochip_get_data(gc);
|
||||
|
||||
__raw_writel(BIT(d->hwirq), g->base + IXP4XX_REG_GPIS);
|
||||
}
|
||||
|
||||
static void ixp4xx_gpio_irq_unmask(struct irq_data *d)
|
||||
{
|
||||
struct ixp4xx_gpio *g = irq_data_get_irq_chip_data(d);
|
||||
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
||||
struct ixp4xx_gpio *g = gpiochip_get_data(gc);
|
||||
|
||||
/* ACK when unmasking if not edge-triggered */
|
||||
if (!(g->irq_edge & BIT(d->hwirq)))
|
||||
@ -108,7 +81,8 @@ static void ixp4xx_gpio_irq_unmask(struct irq_data *d)
|
||||
|
||||
static int ixp4xx_gpio_irq_set_type(struct irq_data *d, unsigned int type)
|
||||
{
|
||||
struct ixp4xx_gpio *g = irq_data_get_irq_chip_data(d);
|
||||
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
||||
struct ixp4xx_gpio *g = gpiochip_get_data(gc);
|
||||
int line = d->hwirq;
|
||||
unsigned long flags;
|
||||
u32 int_style;
|
||||
@ -187,122 +161,31 @@ static struct irq_chip ixp4xx_gpio_irqchip = {
|
||||
.irq_set_type = ixp4xx_gpio_irq_set_type,
|
||||
};
|
||||
|
||||
static int ixp4xx_gpio_to_irq(struct gpio_chip *gc, unsigned int offset)
|
||||
static int ixp4xx_gpio_child_to_parent_hwirq(struct gpio_chip *gc,
|
||||
unsigned int child,
|
||||
unsigned int child_type,
|
||||
unsigned int *parent,
|
||||
unsigned int *parent_type)
|
||||
{
|
||||
struct ixp4xx_gpio *g = gpiochip_get_data(gc);
|
||||
struct irq_fwspec fwspec;
|
||||
/* All these interrupts are level high in the CPU */
|
||||
*parent_type = IRQ_TYPE_LEVEL_HIGH;
|
||||
|
||||
fwspec.fwnode = g->fwnode;
|
||||
fwspec.param_count = 2;
|
||||
fwspec.param[0] = offset;
|
||||
fwspec.param[1] = IRQ_TYPE_NONE;
|
||||
|
||||
return irq_create_fwspec_mapping(&fwspec);
|
||||
}
|
||||
|
||||
static int ixp4xx_gpio_irq_domain_translate(struct irq_domain *domain,
|
||||
struct irq_fwspec *fwspec,
|
||||
unsigned long *hwirq,
|
||||
unsigned int *type)
|
||||
{
|
||||
int ret;
|
||||
|
||||
/* We support standard DT translation */
|
||||
if (is_of_node(fwspec->fwnode) && fwspec->param_count == 2) {
|
||||
return irq_domain_translate_twocell(domain, fwspec,
|
||||
hwirq, type);
|
||||
/* GPIO lines 0..12 have dedicated IRQs */
|
||||
if (child == 0) {
|
||||
*parent = 6;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* This goes away when we transition to DT */
|
||||
if (is_fwnode_irqchip(fwspec->fwnode)) {
|
||||
ret = irq_domain_translate_twocell(domain, fwspec,
|
||||
hwirq, type);
|
||||
if (ret)
|
||||
return ret;
|
||||
WARN_ON(*type == IRQ_TYPE_NONE);
|
||||
if (child == 1) {
|
||||
*parent = 7;
|
||||
return 0;
|
||||
}
|
||||
if (child >= 2 && child <= 12) {
|
||||
*parent = child + 17;
|
||||
return 0;
|
||||
}
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
static int ixp4xx_gpio_irq_domain_alloc(struct irq_domain *d,
|
||||
unsigned int irq, unsigned int nr_irqs,
|
||||
void *data)
|
||||
{
|
||||
struct ixp4xx_gpio *g = d->host_data;
|
||||
irq_hw_number_t hwirq;
|
||||
unsigned int type = IRQ_TYPE_NONE;
|
||||
struct irq_fwspec *fwspec = data;
|
||||
int ret;
|
||||
int i;
|
||||
|
||||
ret = ixp4xx_gpio_irq_domain_translate(d, fwspec, &hwirq, &type);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
dev_dbg(g->dev, "allocate IRQ %d..%d, hwirq %lu..%lu\n",
|
||||
irq, irq + nr_irqs - 1,
|
||||
hwirq, hwirq + nr_irqs - 1);
|
||||
|
||||
for (i = 0; i < nr_irqs; i++) {
|
||||
struct irq_fwspec parent_fwspec;
|
||||
const struct ixp4xx_gpio_map *map;
|
||||
int j;
|
||||
|
||||
/* Not all lines support IRQs */
|
||||
for (j = 0; j < ARRAY_SIZE(ixp4xx_gpiomap); j++) {
|
||||
map = &ixp4xx_gpiomap[j];
|
||||
if (map->gpio_offset == hwirq)
|
||||
break;
|
||||
}
|
||||
if (j == ARRAY_SIZE(ixp4xx_gpiomap)) {
|
||||
dev_err(g->dev, "can't look up hwirq %lu\n", hwirq);
|
||||
return -EINVAL;
|
||||
}
|
||||
dev_dbg(g->dev, "found parent hwirq %u\n", map->parent_hwirq);
|
||||
|
||||
/*
|
||||
* We set handle_bad_irq because the .set_type() should
|
||||
* always be invoked and set the right type of handler.
|
||||
*/
|
||||
irq_domain_set_info(d,
|
||||
irq + i,
|
||||
hwirq + i,
|
||||
&ixp4xx_gpio_irqchip,
|
||||
g,
|
||||
handle_bad_irq,
|
||||
NULL, NULL);
|
||||
irq_set_probe(irq + i);
|
||||
|
||||
/*
|
||||
* Create a IRQ fwspec to send up to the parent irqdomain:
|
||||
* specify the hwirq we address on the parent and tie it
|
||||
* all together up the chain.
|
||||
*/
|
||||
parent_fwspec.fwnode = d->parent->fwnode;
|
||||
parent_fwspec.param_count = 2;
|
||||
parent_fwspec.param[0] = map->parent_hwirq;
|
||||
/* This parent only handles asserted level IRQs */
|
||||
parent_fwspec.param[1] = IRQ_TYPE_LEVEL_HIGH;
|
||||
dev_dbg(g->dev, "alloc_irqs_parent for %d parent hwirq %d\n",
|
||||
irq + i, map->parent_hwirq);
|
||||
ret = irq_domain_alloc_irqs_parent(d, irq + i, 1,
|
||||
&parent_fwspec);
|
||||
if (ret)
|
||||
dev_err(g->dev,
|
||||
"failed to allocate parent hwirq %d for hwirq %lu\n",
|
||||
map->parent_hwirq, hwirq);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct irq_domain_ops ixp4xx_gpio_irqdomain_ops = {
|
||||
.translate = ixp4xx_gpio_irq_domain_translate,
|
||||
.alloc = ixp4xx_gpio_irq_domain_alloc,
|
||||
.free = irq_domain_free_irqs_common,
|
||||
};
|
||||
|
||||
static int ixp4xx_gpio_probe(struct platform_device *pdev)
|
||||
{
|
||||
unsigned long flags;
|
||||
@ -311,8 +194,8 @@ static int ixp4xx_gpio_probe(struct platform_device *pdev)
|
||||
struct irq_domain *parent;
|
||||
struct resource *res;
|
||||
struct ixp4xx_gpio *g;
|
||||
struct gpio_irq_chip *girq;
|
||||
int ret;
|
||||
int i;
|
||||
|
||||
g = devm_kzalloc(dev, sizeof(*g), GFP_KERNEL);
|
||||
if (!g)
|
||||
@ -326,6 +209,35 @@ static int ixp4xx_gpio_probe(struct platform_device *pdev)
|
||||
return PTR_ERR(g->base);
|
||||
}
|
||||
|
||||
/*
|
||||
* When we convert to device tree we will simply look up the
|
||||
* parent irqdomain using irq_find_host(parent) as parent comes
|
||||
* from IRQCHIP_DECLARE(), then use of_node_to_fwnode() to get
|
||||
* the fwnode. For now we need this boardfile style code.
|
||||
*/
|
||||
if (np) {
|
||||
struct device_node *irq_parent;
|
||||
|
||||
irq_parent = of_irq_find_parent(np);
|
||||
if (!irq_parent) {
|
||||
dev_err(dev, "no IRQ parent node\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
parent = irq_find_host(irq_parent);
|
||||
if (!parent) {
|
||||
dev_err(dev, "no IRQ parent domain\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
g->fwnode = of_node_to_fwnode(np);
|
||||
} else {
|
||||
parent = ixp4xx_get_irq_domain();
|
||||
g->fwnode = irq_domain_alloc_fwnode(g->base);
|
||||
if (!g->fwnode) {
|
||||
dev_err(dev, "no domain base\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Make sure GPIO 14 and 15 are NOT used as clocks but GPIO on
|
||||
* specific machines.
|
||||
@ -360,7 +272,6 @@ static int ixp4xx_gpio_probe(struct platform_device *pdev)
|
||||
dev_err(dev, "unable to init generic GPIO\n");
|
||||
return ret;
|
||||
}
|
||||
g->gc.to_irq = ixp4xx_gpio_to_irq;
|
||||
g->gc.ngpio = 16;
|
||||
g->gc.label = "IXP4XX_GPIO_CHIP";
|
||||
/*
|
||||
@ -372,86 +283,22 @@ static int ixp4xx_gpio_probe(struct platform_device *pdev)
|
||||
g->gc.parent = &pdev->dev;
|
||||
g->gc.owner = THIS_MODULE;
|
||||
|
||||
girq = &g->gc.irq;
|
||||
girq->chip = &ixp4xx_gpio_irqchip;
|
||||
girq->fwnode = g->fwnode;
|
||||
girq->parent_domain = parent;
|
||||
girq->child_to_parent_hwirq = ixp4xx_gpio_child_to_parent_hwirq;
|
||||
girq->handler = handle_bad_irq;
|
||||
girq->default_type = IRQ_TYPE_NONE;
|
||||
|
||||
ret = devm_gpiochip_add_data(dev, &g->gc, g);
|
||||
if (ret) {
|
||||
dev_err(dev, "failed to add SoC gpiochip\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* When we convert to device tree we will simply look up the
|
||||
* parent irqdomain using irq_find_host(parent) as parent comes
|
||||
* from IRQCHIP_DECLARE(), then use of_node_to_fwnode() to get
|
||||
* the fwnode. For now we need this boardfile style code.
|
||||
*/
|
||||
if (np) {
|
||||
struct device_node *irq_parent;
|
||||
|
||||
irq_parent = of_irq_find_parent(np);
|
||||
if (!irq_parent) {
|
||||
dev_err(dev, "no IRQ parent node\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
parent = irq_find_host(irq_parent);
|
||||
if (!parent) {
|
||||
dev_err(dev, "no IRQ parent domain\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
g->fwnode = of_node_to_fwnode(np);
|
||||
} else {
|
||||
parent = ixp4xx_get_irq_domain();
|
||||
g->fwnode = irq_domain_alloc_fwnode(g->base);
|
||||
if (!g->fwnode) {
|
||||
dev_err(dev, "no domain base\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
}
|
||||
g->domain = irq_domain_create_hierarchy(parent,
|
||||
IRQ_DOMAIN_FLAG_HIERARCHY,
|
||||
ARRAY_SIZE(ixp4xx_gpiomap),
|
||||
g->fwnode,
|
||||
&ixp4xx_gpio_irqdomain_ops,
|
||||
g);
|
||||
if (!g->domain) {
|
||||
irq_domain_free_fwnode(g->fwnode);
|
||||
dev_err(dev, "no hierarchical irq domain\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* After adding OF support, this is no longer needed: irqs
|
||||
* will be allocated for the respective fwnodes.
|
||||
*/
|
||||
if (!np) {
|
||||
for (i = 0; i < ARRAY_SIZE(ixp4xx_gpiomap); i++) {
|
||||
const struct ixp4xx_gpio_map *map = &ixp4xx_gpiomap[i];
|
||||
struct irq_fwspec fwspec;
|
||||
|
||||
fwspec.fwnode = g->fwnode;
|
||||
/* This is the hwirq for the GPIO line side of things */
|
||||
fwspec.param[0] = map->gpio_offset;
|
||||
fwspec.param[1] = IRQ_TYPE_EDGE_RISING;
|
||||
fwspec.param_count = 2;
|
||||
ret = __irq_domain_alloc_irqs(g->domain,
|
||||
-1, /* just pick something */
|
||||
1,
|
||||
NUMA_NO_NODE,
|
||||
&fwspec,
|
||||
false,
|
||||
NULL);
|
||||
if (ret < 0) {
|
||||
irq_domain_free_fwnode(g->fwnode);
|
||||
dev_err(dev,
|
||||
"can not allocate irq for GPIO line %d parent hwirq %d in hierarchy domain: %d\n",
|
||||
map->gpio_offset, map->parent_hwirq,
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
platform_set_drvdata(pdev, g);
|
||||
dev_info(dev, "IXP4 GPIO @%p registered\n", g->base);
|
||||
dev_info(dev, "IXP4 GPIO registered\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user