Commit 1dbf7f29 authored by Linus Walleij's avatar Linus Walleij

gpio: pl061: detail IRQ trigger handling

I couldn't follow this code flow. Make it dirt simple to figure
out what is going on and get proper debug prints. Warn if we
set up an IRQ without any trigger. Should make no semantic
difference.
Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
parent f881bab0
...@@ -158,24 +158,63 @@ static int pl061_irq_type(struct irq_data *d, unsigned trigger) ...@@ -158,24 +158,63 @@ static int pl061_irq_type(struct irq_data *d, unsigned trigger)
gpiois = readb(chip->base + GPIOIS); gpiois = readb(chip->base + GPIOIS);
gpioibe = readb(chip->base + GPIOIBE); gpioibe = readb(chip->base + GPIOIBE);
if ((trigger & (IRQ_TYPE_LEVEL_HIGH | IRQ_TYPE_LEVEL_LOW)) &&
(trigger & (IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING)))
{
dev_err(gc->dev,
"trying to configure line %d for both level and edge "
"detection, choose one!\n",
offset);
return -EINVAL;
}
if (trigger & (IRQ_TYPE_LEVEL_HIGH | IRQ_TYPE_LEVEL_LOW)) { if (trigger & (IRQ_TYPE_LEVEL_HIGH | IRQ_TYPE_LEVEL_LOW)) {
bool polarity = trigger & IRQ_TYPE_LEVEL_HIGH;
/* Disable edge detection */
gpioibe &= ~bit;
/* Enable level detection */
gpiois |= bit; gpiois |= bit;
if (trigger & IRQ_TYPE_LEVEL_HIGH) /* Select polarity */
if (polarity)
gpioiev |= bit; gpioiev |= bit;
else else
gpioiev &= ~bit; gpioiev &= ~bit;
} else
gpiois &= ~bit;
if ((trigger & IRQ_TYPE_EDGE_BOTH) == IRQ_TYPE_EDGE_BOTH) dev_dbg(gc->dev, "line %d: IRQ on %s level\n",
/* Setting this makes GPIOEV be ignored */ offset,
polarity ? "HIGH" : "LOW");
} else if ((trigger & IRQ_TYPE_EDGE_BOTH) == IRQ_TYPE_EDGE_BOTH) {
/* Disable level detection */
gpiois &= ~bit;
/* Select both edges, setting this makes GPIOEV be ignored */
gpioibe |= bit; gpioibe |= bit;
else {
dev_dbg(gc->dev, "line %d: IRQ on both edges\n", offset);
} else if ((trigger & IRQ_TYPE_EDGE_RISING) ||
(trigger & IRQ_TYPE_EDGE_FALLING)) {
bool rising = trigger & IRQ_TYPE_EDGE_RISING;
/* Disable level detection */
gpiois &= ~bit;
/* Clear detection on both edges */
gpioibe &= ~bit; gpioibe &= ~bit;
if (trigger & IRQ_TYPE_EDGE_RISING) /* Select edge */
if (rising)
gpioiev |= bit; gpioiev |= bit;
else if (trigger & IRQ_TYPE_EDGE_FALLING) else
gpioiev &= ~bit; gpioiev &= ~bit;
dev_dbg(gc->dev, "line %d: IRQ on %s edge\n",
offset,
rising ? "RISING" : "FALLING");
} else {
/* No trigger: disable everything */
gpiois &= ~bit;
gpioibe &= ~bit;
gpioiev &= ~bit;
dev_warn(gc->dev, "no trigger selected for line %d\n",
offset);
} }
writeb(gpiois, chip->base + GPIOIS); writeb(gpiois, chip->base + GPIOIS);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment