mfd: ab8500: prepare to handle AB8500 GPIO's IRQs correctly
authorLee Jones <lee.jones@linaro.org>
Thu, 20 Dec 2012 11:23:42 +0000 (11:23 +0000)
committerLinus Walleij <linus.walleij@linaro.org>
Fri, 8 Feb 2013 13:25:30 +0000 (14:25 +0100)
In an upcoming patch, the gpio-ab8500 driver will relinquish all
IRQ handling capability and pass it back into the AB8500 core
driver. This will aid in reducing massive code duplication within
the kernel. Also, most of the functionality is already in the
AB8500 core driver, as the GPIO IRQs are actually sandwiched
between lots of other IRQs which the core driver already handles.

All we're doing here is providing the core driver with knowledge
that each GPIO has two IRQs assigned to it; one for rising and
a separate one for falling.

Cc: Samuel Ortiz <sameo@linux.intel.com>
Signed-off-by: Lee Jones <lee.jones@linaro.org>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
drivers/mfd/ab8500-core.c

index 4778bb124efe09b27da77446896dfd6ae9e97f0e..3d5da51b6a94f71f642988d5ea99ea37d6d3a407 100644 (file)
@@ -368,16 +368,40 @@ static void ab8500_irq_mask(struct irq_data *data)
        int mask = 1 << (offset % 8);
 
        ab8500->mask[index] |= mask;
+
+       /* The AB8500 GPIOs have two interrupts each (rising & falling). */
+       if (offset >= AB8500_INT_GPIO6R && offset <= AB8500_INT_GPIO41R)
+               ab8500->mask[index + 2] |= mask;
+       if (offset >= AB9540_INT_GPIO50R && offset <= AB9540_INT_GPIO54R)
+               ab8500->mask[index + 1] |= mask;
+       if (offset == AB8540_INT_GPIO43R || offset == AB8540_INT_GPIO44R)
+               ab8500->mask[index] |= (mask >> 1);
 }
 
 static void ab8500_irq_unmask(struct irq_data *data)
 {
        struct ab8500 *ab8500 = irq_data_get_irq_chip_data(data);
+       unsigned int type = irqd_get_trigger_type(data);
        int offset = data->hwirq;
        int index = offset / 8;
        int mask = 1 << (offset % 8);
 
-       ab8500->mask[index] &= ~mask;
+       if (type & IRQ_TYPE_EDGE_RISING)
+               ab8500->mask[index] &= ~mask;
+
+       /* The AB8500 GPIOs have two interrupts each (rising & falling). */
+       if (type & IRQ_TYPE_EDGE_FALLING) {
+               if (offset >= AB8500_INT_GPIO6R && offset <= AB8500_INT_GPIO41R)
+                       ab8500->mask[index + 2] &= ~mask;
+               else if (offset >= AB9540_INT_GPIO50R && offset <= AB9540_INT_GPIO54R)
+                       ab8500->mask[index + 1] &= ~mask;
+               else if (offset == AB8540_INT_GPIO43R || offset == AB8540_INT_GPIO44R)
+                       ab8500->mask[index] &= ~(mask >> 1);
+               else
+                       ab8500->mask[index] &= ~mask;
+       } else
+               /* Satisfies the case where type is not set. */
+               ab8500->mask[index] &= ~mask;
 }
 
 static struct irq_chip ab8500_irq_chip = {