Re: [PATCH v3 02/14] irqchip: irq-pic32-evic: Add support for PIC32 interrupt controller

From: Thomas Gleixner
Date: Sun Jan 10 2016 - 05:11:46 EST


Joshua,

On Fri, 8 Jan 2016, Joshua Henderson wrote:
> On 01/08/2016 12:04 PM, Thomas Gleixner wrote:
> > On Thu, 7 Jan 2016, Joshua Henderson wrote:
> >> +/* acknowledge an interrupt */
> >> +static void ack_pic32_irq(struct irq_data *irqd)
> >> +{
> >> + u32 reg, mask;
> >> + unsigned int hwirq = irqd_to_hwirq(irqd);
> >> +
> >> + BIT_REG_MASK(hwirq, reg, mask);
> >> + writel(mask, &evic_base->ifs[reg].clr);
> >
> > So you invented an open coded variant of the generic irq chip. Just with the
> > difference that the generic chip caches the mask and the register offsets ....
> >
>
> On PIC32 we have 4 different register offsets in many cases, including the interrupt
> controller registers, to write to one hardware register. The PIC32 has special
> write only registers for set/clear/invert and which one is used is dependent on
> the logic at the time of writel(). Point being, there is no obvious value in
> caching when using these registers. We don't have to perform a readl() at any
> time beforehand to write a mask to a register to update it atomically.

The generic chip has functions which handle the seperate register case.

void irq_gc_mask_disable_reg(struct irq_data *d);
void irq_gc_unmask_enable_reg(struct irq_data *d);
void irq_gc_mask_disable_reg_and_ack(struct irq_data *d);
void irq_gc_eoi(struct irq_data *d);

> >> +static int set_type_pic32_irq(struct irq_data *data, unsigned int flow_type)
> >> +{
> >> + int index;
> >> +
> >> + switch (flow_type) {
> >> +
> >> + case IRQ_TYPE_EDGE_RISING:
> >> + case IRQ_TYPE_EDGE_FALLING:
> >> + irq_set_handler_locked(data, handle_edge_irq);
> >> + break;
> >> +
> >> + case IRQ_TYPE_LEVEL_HIGH:
> >> + case IRQ_TYPE_LEVEL_LOW:
> >> + irq_set_handler_locked(data, handle_fasteoi_irq);
> >> + break;
> >> +
> >> + default:
> >> + pr_err("Invalid interrupt type !\n");
> >> + return -EINVAL;
> >> + }
> >> +
> >> + /* set polarity for external interrupts only */
> >> + index = get_ext_irq_index(data->hwirq);
> >> + if (index >= 0)
> >> + evic_set_ext_irq_polarity(index, flow_type);
> >
> > So for the non external interrupts you set a different handler and be
> > done. How is that supposed to work? They switch magically from one mode to the
> > other?
> >
>
> It's all the same handlers (depending on whether it's persistent or
> non-persistent) irrelevant of it being an external interrupt or not. It's all
> the same hardware interrupt controller. Some pins on the chip can be configured
> as an interrupt source through pin configuration and those have dedicated
> interrupts associated with them. The only thing "special" about these external
> interrupts is they must be explicitly configured as edge rising or edge falling
> in hardware- which is what is being handled here. Non-external interrupts don't
> need this configuration.

I really cannot follow here. The code tells me that I can set
EDGE_RISING/FALLING/LEVEL_HIGH/LOW for any of those interrupts.

So that makes two questions:

1) Can the non-external mode handle all type variants automagically? I
seriously doubt that. If the type cannot be set, then it makes no sense
to pretend that it can and allow to switch the handler from fasteoi to
edge mode.

2) The external irqs do not support level according to your
evic_set_ext_irq_polarity() function. But you return success if set_type
is called with a level type and gladly switch the handler. You merily
pr_warn in evic_set_ext_irq_polarity().

That's just crap.

> >> +int pic32_get_c0_compare_int(void)
> >> +{
> >> + int virq;
> >> +
> >> + virq = irq_create_mapping(evic_irq_domain, CORE_TIMER_INTERRUPT);
> >> + irq_set_irq_type(virq, IRQ_TYPE_EDGE_RISING);
> >> + return virq;
> >
> > Why isn't that information retrieved via device tree?
> >
>
> I suppose it could be. We took a lesson from irq-mips-gic.c on this one.

You copied it, right? That does not make it any better.

> > Why don't you use a seperate irq chip for the ext irqs? You can do that with
> > the generic chip as well.
> >
> > irq_alloc_domain_generic_chips(domain, 32, 2, ....)
> >
> > And then you assign the alternate chip (2) to your ext irqs and have the set
> > type function only for the alternate chip.
> >
>
> We have one interrupt controller. All interrupts have a hardware hard-coded
> flow type (edge, level) ...

And that's exactly the point. They are hardcoded, but you still allow any
random driver to change the type and therefor the handler. How is that
supposed to work?

> ... with the exception of what we are calling "external interrupts".
> These are essentially gpio interrupts that can be software
> configured as edge rising or edge falling. Otherwise, there is no difference
> between any of the interrupts.

And again. Here you can change the type, but only edge rising and falling are
supported. And you still allow setting a level type.

So for both types you allow the driver/DT writer to get it wrong. And of
course this happens without a single line of comment which explains the
oddities of your driver, so a causual reader will stumble over it and ask
exactly the questions I'm asking. We want understandable and maintainable code
and not some 'work for me' hackery.

Thanks,

tglx