Re: [Patch v7 3/7] irqchip: xilinx: restructure and use jump label api

From: Michal Simek
Date: Tue Nov 15 2016 - 07:49:29 EST


On 14.11.2016 13:13, Zubair Lutfullah Kakakhel wrote:
> Add a global structure to house various variables.
> And cleanup read/write handling by using jump label api.
>
> Signed-off-by: Zubair Lutfullah Kakakhel <Zubair.Kakakhel@xxxxxxxxxx>
>
> ---
> V6 -> V7
> Restructure and use jump label api
> Better commit log
>
> V5 -> V6
> Removed __func__ from printk
> Rebase to v4.9-rc3
>
> V4 -> V5
> Rebased to v4.9-rc1
> Better error handling
>
> V3 -> V4
> Better error handling for kzalloc
> Erroring out if the axi intc is probed twice as that isn't
> supported
>
> V2 -> V3
> New patch. Cleans up driver structure
> ---
> drivers/irqchip/irq-xilinx-intc.c | 118 +++++++++++++++++++++-----------------
> 1 file changed, 66 insertions(+), 52 deletions(-)
>
> diff --git a/drivers/irqchip/irq-xilinx-intc.c b/drivers/irqchip/irq-xilinx-intc.c
> index 096c1ed..7331d8c 100644
> --- a/drivers/irqchip/irq-xilinx-intc.c
> +++ b/drivers/irqchip/irq-xilinx-intc.c
> @@ -14,10 +14,9 @@
> #include <linux/irqchip.h>
> #include <linux/of_address.h>
> #include <linux/io.h>
> +#include <linux/jump_label.h>
> #include <linux/bug.h>
>
> -static void __iomem *intc_baseaddr;
> -
> /* No one else should require these constants, so define them locally here. */
> #define ISR 0x00 /* Interrupt Status Register */
> #define IPR 0x04 /* Interrupt Pending Register */
> @@ -31,27 +30,30 @@ static void __iomem *intc_baseaddr;
> #define MER_ME (1<<0)
> #define MER_HIE (1<<1)
>
> -static unsigned int (*read_fn)(void __iomem *);
> -static void (*write_fn)(u32, void __iomem *);
> +static DEFINE_STATIC_KEY_FALSE(xintc_is_be);
>
> -static void intc_write32(u32 val, void __iomem *addr)
> -{
> - iowrite32(val, addr);
> -}
> +struct xintc_irq_chip {
> + void __iomem *base;
> + struct irq_domain *root_domain;
> + u32 intr_mask;
> +};
>
> -static unsigned int intc_read32(void __iomem *addr)
> -{
> - return ioread32(addr);
> -}
> +static struct xintc_irq_chip *xintc_irqc;
>
> -static void intc_write32_be(u32 val, void __iomem *addr)
> +static void xintc_write(int reg, u32 data)
> {
> - iowrite32be(val, addr);
> + if (static_branch_unlikely(&xintc_is_be))
> + iowrite32be(data, xintc_irqc->base + reg);
> + else
> + iowrite32(data, xintc_irqc->base + reg);
> }
>
> -static unsigned int intc_read32_be(void __iomem *addr)
> +static unsigned int xintc_read(int reg)
> {
> - return ioread32be(addr);
> + if (static_branch_unlikely(&xintc_is_be))
> + return ioread32be(xintc_irqc->base + reg);
> + else
> + return ioread32(xintc_irqc->base + reg);
> }
>
> static void intc_enable_or_unmask(struct irq_data *d)
> @@ -65,21 +67,21 @@ static void intc_enable_or_unmask(struct irq_data *d)
> * acks the irq before calling the interrupt handler
> */
> if (irqd_is_level_type(d))
> - write_fn(mask, intc_baseaddr + IAR);
> + xintc_write(IAR, mask);
>
> - write_fn(mask, intc_baseaddr + SIE);
> + xintc_write(SIE, mask);
> }
>
> static void intc_disable_or_mask(struct irq_data *d)
> {
> pr_debug("irq-xilinx: disable: %ld\n", d->hwirq);
> - write_fn(1 << d->hwirq, intc_baseaddr + CIE);
> + xintc_write(CIE, 1 << d->hwirq);
> }
>
> static void intc_ack(struct irq_data *d)
> {
> pr_debug("irq-xilinx: ack: %ld\n", d->hwirq);
> - write_fn(1 << d->hwirq, intc_baseaddr + IAR);
> + xintc_write(IAR, 1 << d->hwirq);
> }
>
> static void intc_mask_ack(struct irq_data *d)
> @@ -87,8 +89,8 @@ static void intc_mask_ack(struct irq_data *d)
> unsigned long mask = 1 << d->hwirq;
>
> pr_debug("irq-xilinx: disable_and_ack: %ld\n", d->hwirq);
> - write_fn(mask, intc_baseaddr + CIE);
> - write_fn(mask, intc_baseaddr + IAR);
> + xintc_write(CIE, mask);
> + xintc_write(IAR, mask);
> }
>
> static struct irq_chip intc_dev = {
> @@ -99,15 +101,13 @@ static struct irq_chip intc_dev = {
> .irq_mask_ack = intc_mask_ack,
> };
>
> -static struct irq_domain *root_domain;
> -
> unsigned int get_irq(void)
> {
> unsigned int hwirq, irq = -1;
>
> - hwirq = read_fn(intc_baseaddr + IVR);
> + hwirq = xintc_read(IVR);
> if (hwirq != -1U)
> - irq = irq_find_mapping(root_domain, hwirq);
> + irq = irq_find_mapping(xintc_irqc->root_domain, hwirq);
>
> pr_debug("irq-xilinx: hwirq=%d, irq=%d\n", hwirq, irq);
>
> @@ -116,9 +116,7 @@ unsigned int get_irq(void)
>
> static int xintc_map(struct irq_domain *d, unsigned int irq, irq_hw_number_t hw)
> {
> - u32 intr_mask = (u32)d->host_data;
> -
> - if (intr_mask & (1 << hw)) {
> + if (xintc_irqc->intr_mask & (1 << hw)) {
> irq_set_chip_and_handler_name(irq, &intc_dev,
> handle_edge_irq, "edge");
> irq_clear_status_flags(irq, IRQ_LEVEL);
> @@ -138,59 +136,75 @@ static const struct irq_domain_ops xintc_irq_domain_ops = {
> static int __init xilinx_intc_of_init(struct device_node *intc,
> struct device_node *parent)
> {
> - u32 nr_irq, intr_mask;
> + u32 nr_irq;
> int ret;
> + struct xintc_irq_chip *irqc;
>
> - intc_baseaddr = of_iomap(intc, 0);
> - BUG_ON(!intc_baseaddr);
> + if (xintc_irqc) {
> + pr_err("irq-xilinx: Multiple instances aren't supported\n");
> + return -EINVAL;
> + }

I don't agree with this.
Pretty long time ago we were added support for multiple instances in
xilinx private tree.
You can look here.

https://github.com/Xilinx/linux-xlnx/blob/master/drivers/irqchip/irq-xilinx-intc.c

Not sure if this the latest way how to do it but as you can see
we were setting up
irq_set_handler_data(irq, intc);

and then when you need that structure we were calling
struct intc *local_intc = irq_data_get_irq_chip_data(d);

And that should be it to support multiple instance of this driver.

Based on 5/7 you are describing your interrupt subsystem like this.

Peripherals --> xilinx_intcontroller -> mips_cpu_int controller
If mips_cpu_int has more than one input you can connect more xilinx intc
controllers.
If not you still have an option to connect
xilinx_intcontroller(up to 32 peripherals) -> xilinx_intcontroller(one
intc + up to 31 peripherals) -> mips_cpu_int controller

Thanks,
Michal