[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20140521012716.GC8664@titan.lakedaemon.net>
Date: Tue, 20 May 2014 21:27:16 -0400
From: Jason Cooper <jason@...edaemon.net>
To: Florian Fainelli <f.fainelli@...il.com>
Cc: arm@...nel.org, Brian Norris <computersforpeace@...il.com>,
Rob Herring <robh+dt@...nel.org>,
Pawel Moll <pawel.moll@....com>,
Mark Rutland <mark.rutland@....com>,
Ian Campbell <ijc+devicetree@...lion.org.uk>,
Kumar Gala <galak@...eaurora.org>,
Randy Dunlap <rdunlap@...radead.org>,
Thomas Gleixner <tglx@...utronix.de>,
"open list:OPEN FIRMWARE AND..." <devicetree@...r.kernel.org>,
"open list:DOCUMENTATION" <linux-doc@...r.kernel.org>,
open list <linux-kernel@...r.kernel.org>,
linux-arm-kernel@...r.kernel.org
Subject: Re: [PATCH RESEND 1/2] irqchip: add Broadcom Set Top Box Level-2
interrupt controller
Florian,
On Tue, May 20, 2014 at 12:13:48PM -0700, Florian Fainelli wrote:
> This patch adds support for the Level-2 interrupt controller hardware
> found in Broadcom Set Top Box System-on-a-Chip devices. This interrupt
> controller is implemented using the generic IRQ chip driver with
> separate enable and disable registers.
>
> Signed-off-by: Brian Norris <computersforpeace@...il.com>
> Signed-off-by: Florian Fainelli <f.fainelli@...il.com>
> ---
> drivers/irqchip/Kconfig | 6 ++
> drivers/irqchip/Makefile | 1 +
> drivers/irqchip/irq-brcmstb-l2.c | 200 +++++++++++++++++++++++++++++++++++++++
> 3 files changed, 207 insertions(+)
> create mode 100644 drivers/irqchip/irq-brcmstb-l2.c
>
> diff --git a/drivers/irqchip/Kconfig b/drivers/irqchip/Kconfig
> index d770f7406631..bbb746e35500 100644
> --- a/drivers/irqchip/Kconfig
> +++ b/drivers/irqchip/Kconfig
> @@ -30,6 +30,12 @@ config ARM_VIC_NR
> The maximum number of VICs available in the system, for
> power management.
>
> +config BRCMSTB_L2_IRQ
> + bool
> + depends on ARM
> + select GENERIC_IRQ_CHIP
> + select IRQ_DOMAIN
> +
> config DW_APB_ICTL
> bool
> select IRQ_DOMAIN
> diff --git a/drivers/irqchip/Makefile b/drivers/irqchip/Makefile
> index f180f8d5fb7b..62a13e5ef98f 100644
> --- a/drivers/irqchip/Makefile
> +++ b/drivers/irqchip/Makefile
> @@ -29,3 +29,4 @@ obj-$(CONFIG_TB10X_IRQC) += irq-tb10x.o
> obj-$(CONFIG_XTENSA) += irq-xtensa-pic.o
> obj-$(CONFIG_XTENSA_MX) += irq-xtensa-mx.o
> obj-$(CONFIG_IRQ_CROSSBAR) += irq-crossbar.o
> +obj-$(CONFIG_BRCMSTB_L2_IRQ) += irq-brcmstb-l2.o
> diff --git a/drivers/irqchip/irq-brcmstb-l2.c b/drivers/irqchip/irq-brcmstb-l2.c
> new file mode 100644
> index 000000000000..336f0ae5591a
> --- /dev/null
> +++ b/drivers/irqchip/irq-brcmstb-l2.c
> @@ -0,0 +1,200 @@
> +/*
> + * Generic Broadcom Set Top Box Level 2 Interrupt controller driver
> + *
> + * Copyright (C) 2014 Broadcom Corporation
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope that it will be useful,
> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> + * GNU General Public License for more details.
> + */
> +
> +#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
> +
> +#include <linux/init.h>
> +#include <linux/slab.h>
> +#include <linux/module.h>
> +#include <linux/platform_device.h>
> +#include <linux/of.h>
> +#include <linux/of_irq.h>
> +#include <linux/of_address.h>
> +#include <linux/of_platform.h>
> +#include <linux/interrupt.h>
> +#include <linux/irq.h>
> +#include <linux/io.h>
> +#include <linux/irqdomain.h>
> +#include <linux/irqchip.h>
> +#include <linux/irqchip/chained_irq.h>
> +
> +#include <asm/mach/irq.h>
> +
> +#include "irqchip.h"
> +
> +/* Register offsets in the L2 interrupt controller */
> +#define CPU_STATUS 0x00
> +#define CPU_SET 0x04
> +#define CPU_CLEAR 0x08
> +#define CPU_MASK_STATUS 0x0c
> +#define CPU_MASK_SET 0x10
> +#define CPU_MASK_CLEAR 0x14
> +
> +/* L2 intc private data structure */
> +struct brcmstb_l2_intc_data {
> + int parent_irq;
> + void __iomem *base;
> + struct irq_domain *domain;
> + bool can_wake;
> + u32 saved_mask; /* for suspend/resume */
> +};
> +
> +static void brcmstb_l2_intc_irq_handle(unsigned int irq, struct irq_desc *desc)
> +{
> + struct brcmstb_l2_intc_data *b = irq_desc_get_handler_data(desc);
> + struct irq_chip *chip = irq_get_chip(irq);
> + u32 status;
> +
> + chained_irq_enter(chip, desc);
> +
> + status = __raw_readl(b->base + CPU_STATUS) &
> + ~(__raw_readl(b->base + CPU_MASK_STATUS));
> +
> + if (status == 0) {
> + do_bad_IRQ(irq, desc);
> + goto out;
> + }
> +
> + do {
> + irq = ffs(status) - 1;
> + /* ack at our level */
> + __raw_writel(1 << irq, b->base + CPU_CLEAR);
> + status &= ~(1 << irq);
> + generic_handle_irq(irq_find_mapping(b->domain, irq));
> + } while (status);
> +out:
> + chained_irq_exit(chip, desc);
> +}
> +
> +static void brcmstb_l2_intc_suspend(struct irq_data *d)
> +{
> + struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d);
> + struct brcmstb_l2_intc_data *b = gc->private;
> +
> + irq_gc_lock(gc);
> + /* Save the current mask */
> + b->saved_mask = __raw_readl(b->base + CPU_MASK_STATUS);
> +
> + if (b->can_wake) {
> + /* Program the wakeup mask */
> + __raw_writel(~gc->wake_active, b->base + CPU_MASK_SET);
> + __raw_writel(gc->wake_active, b->base + CPU_MASK_CLEAR);
> + }
> + irq_gc_unlock(gc);
> +}
> +
> +static void brcmstb_l2_intc_resume(struct irq_data *d)
> +{
> + struct irq_chip_generic *gc = irq_data_get_irq_chip_data(d);
> + struct brcmstb_l2_intc_data *b = gc->private;
> +
> + irq_gc_lock(gc);
> + /* Clear unmasked non-wakeup interrupts */
> + __raw_writel(~b->saved_mask & ~gc->wake_active, b->base + CPU_CLEAR);
> +
> + /* Restore the saved mask */
> + __raw_writel(b->saved_mask, b->base + CPU_MASK_SET);
> + __raw_writel(~b->saved_mask, b->base + CPU_MASK_CLEAR);
> + irq_gc_unlock(gc);
> +}
> +
> +int __init brcmstb_l2_intc_of_init(struct device_node *np,
> + struct device_node *parent)
> +{
> + unsigned int clr = IRQ_NOREQUEST | IRQ_NOPROBE | IRQ_NOAUTOEN;
> + struct brcmstb_l2_intc_data *data;
> + struct irq_chip_generic *gc;
> + struct irq_chip_type *ct;
> + int ret;
> +
> + data = kzalloc(sizeof(*data), GFP_KERNEL);
> + if (!data)
> + return -ENOMEM;
> +
> + data->base = of_iomap(np, 0);
> + if (!data->base) {
> + pr_err("failed to remap intc L2 registers\n");
> + return -ENOMEM;
> + }
You're mixing error handling techniques here. Please choose one or the
other. If you choose goto's, you need to jump to free(data), below.
> +
> + /* Disable all interrupts by default */
> + __raw_writel(0xffffffff, data->base + CPU_MASK_SET);
> + __raw_writel(0xffffffff, data->base + CPU_CLEAR);
> +
> + data->parent_irq = irq_of_parse_and_map(np, 0);
> + if (data->parent_irq < 0) {
> + pr_err("failed to find parent interrupt\n");
> + ret = data->parent_irq;
> + goto out_unmap;
> + }
> +
> + data->domain = irq_domain_add_linear(np, 32,
> + &irq_generic_chip_ops, NULL);
> + if (!data->domain) {
> + ret = -ENOMEM;
> + goto out_unmap;
> + }
> +
> + /* Allocate a single Generic IRQ chip for this node */
> + ret = irq_alloc_domain_generic_chips(data->domain, 32, 1,
> + np->full_name, handle_level_irq, clr, 0, 0);
> + if (ret) {
> + pr_err("failed to allocate generic irq chip\n");
> + goto out_free_domain;
> + }
> +
> + /* Set the IRQ chaining logic */
> + irq_set_handler_data(data->parent_irq, data);
> + irq_set_chained_handler(data->parent_irq, brcmstb_l2_intc_irq_handle);
> +
> + gc = irq_get_domain_generic_chip(data->domain, 0);
> + gc->reg_base = data->base;
> + gc->private = data;
> + ct = gc->chip_types;
> +
> + ct->chip.irq_ack = irq_gc_ack_set_bit;
> + ct->regs.ack = CPU_CLEAR;
> +
> + ct->chip.irq_mask = irq_gc_mask_disable_reg;
> + ct->regs.disable = CPU_MASK_SET;
> +
> + ct->chip.irq_unmask = irq_gc_unmask_enable_reg;
> + ct->regs.enable = CPU_MASK_CLEAR;
> +
> + ct->chip.irq_suspend = brcmstb_l2_intc_suspend;
> + ct->chip.irq_resume = brcmstb_l2_intc_resume;
> +
> + if (of_property_read_bool(np, "brcm,irq-can-wake")) {
> + data->can_wake = true;
> + /* This IRQ chip can wake the system, set all child interrupts
> + * in wake_enabled mask
> + */
> + gc->wake_enabled = 0xffffffff;
> + ct->chip.irq_set_wake = irq_gc_set_wake;
> + }
> +
> + pr_info("registered L2 intc (mem: 0x%p, parent irq: %d)\n",
> + data->base, data->parent_irq);
> +
> + return 0;
> +
> +out_free_domain:
> + irq_domain_remove(data->domain);
> +out_unmap:
> + iounmap(data->base);
out_free_data:
> + kfree(data);
> + return ret;
> +}
> +IRQCHIP_DECLARE(brcmstb_l2_intc, "brcm,l2-intc", brcmstb_l2_intc_of_init);
thx,
Jason.
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists