diff mbox series

[v4,1/4] irqchip: qcom-pdc: Fix phantom irq when changing between rising/falling

Message ID 20201211141514.v4.1.I2702919afc253e2a451bebc3b701b462b2d22344@changeid
State Accepted
Commit 2f5fbc4305d07725bfebaedb09e57271315691ef
Headers show
Series [v4,1/4] irqchip: qcom-pdc: Fix phantom irq when changing between rising/falling | expand

Commit Message

Doug Anderson Dec. 11, 2020, 10:15 p.m. UTC
We have a problem if we use gpio-keys and configure wakeups such that
we only want one edge to wake us up.  AKA:
  wakeup-event-action = <EV_ACT_DEASSERTED>;
  wakeup-source;

Specifically we end up with a phantom interrupt that blocks suspend if
the line was already high and we want wakeups on rising edges (AKA we
want the GPIO to go low and then high again before we wake up).  The
opposite is also problematic.

Specifically, here's what's happening today:
1. Normally, gpio-keys configures to look for both edges.  Due to the
   current workaround introduced in commit c3c0c2e18d94 ("pinctrl:
   qcom: Handle broken/missing PDC dual edge IRQs on sc7180"), if the
   line was high we'd configure for falling edges.
2. At suspend time, we change to look for rising edges.
3. After qcom_pdc_gic_set_type() runs, we get a phantom interrupt.

We can solve this by just clearing the phantom interrupt.

NOTE: it is possible that this could cause problems for a client with
very specific needs, but there's not much we can do with this
hardware.  As an example, let's say the interrupt signal is currently
high and the client is looking for falling edges.  The client now
changes to look for rising edges.  The client could possibly expect
that if the line has a short pulse low (and back high) that it would
always be detected.  Specifically no matter when the pulse happened,
it should either have tripped the (old) falling edge trigger or the
(new) rising edge trigger.  We will simply not trip it.  We could
narrow down the race a bit by polling our parent before changing
types, but no matter what we do there will still be a period of time
where we can't tell the difference between a real transition (or more
than one transition) and the phantom.

Fixes: f55c73aef890 ("irqchip/pdc: Add PDC interrupt controller for QCOM SoCs")
Signed-off-by: Douglas Anderson <dianders@chromium.org>
Reviewed-by: Maulik Shah <mkshah@codeaurora.org>
Tested-by: Maulik Shah <mkshah@codeaurora.org>
Reviewed-by: Stephen Boyd <swboyd@chromium.org>
---
There are no dependencies between this patch and patch #2/#3.  It can
go in by itself.  Patches are only grouped together in one series
because they address similar issues.

Maulik has got confirmation from hardware guys and understands the
problem.  This patch is ready to land.

Changes in v4:
- No changes, this patch on its own ready to land.

Changes in v3:
- Adjusted the comment as per Maulik.

Changes in v2:
- 0 => false
- If irq_chip_set_type_parent() fails don't bother clearing.
- Add Fixes tag.

 drivers/irqchip/qcom-pdc.c | 21 ++++++++++++++++++++-
 1 file changed, 20 insertions(+), 1 deletion(-)

Comments

Stephen Boyd Dec. 17, 2020, 4:54 a.m. UTC | #1
Quoting Douglas Anderson (2020-12-11 14:15:37)
> In Linux, if a driver does disable_irq() and later does enable_irq()
> on its interrupt, I believe it's expecting these properties:
> * If an interrupt was pending when the driver disabled then it will
>   still be pending after the driver re-enables.
> * If an edge-triggered interrupt comes in while an interrupt is
>   disabled it should assert when the interrupt is re-enabled.
> 
> If you think that the above sounds a lot like the disable_irq() and
> enable_irq() are supposed to be masking/unmasking the interrupt
> instead of disabling/enabling it then you've made an astute
> observation.  Specifically when talking about interrupts, "mask"
> usually means to stop posting interrupts but keep tracking them and
> "disable" means to fully shut off interrupt detection.  It's
> unfortunate that this is so confusing, but presumably this is all the
> way it is for historical reasons.
> 
> Perhaps more confusing than the above is that, even though clients of
> IRQs themselves don't have a way to request mask/unmask
> vs. disable/enable calls, IRQ chips themselves can implement both.
> ...and yet more confusing is that if an IRQ chip implements
> disable/enable then they will be called when a client driver calls
> disable_irq() / enable_irq().
> 
> It does feel like some of the above could be cleared up.  However,
> without any other core interrupt changes it should be clear that when
> an IRQ chip gets a request to "disable" an IRQ that it has to treat it
> like a mask of that IRQ.
> 
> In any case, after that long interlude you can see that the "unmask
> and clear" can break things.  Maulik tried to fix it so that we no
> longer did "unmask and clear" in commit 71266d9d3936 ("pinctrl: qcom:
> Move clearing pending IRQ to .irq_request_resources callback"), but it
> only handled the PDC case (it also had problems, but that's the
> subject of another patch).  Let's fix this for the non-PDC case.
> 
> From my understanding the source of the phantom interrupt in the
> non-PDC case was the one that could have been introduced in
> msm_gpio_irq_set_type().  Let's handle that one and then get rid of
> the clear.
> 
> Fixes: 4b7618fdc7e6 ("pinctrl: qcom: Add irq_enable callback for msm gpio")
> Signed-off-by: Douglas Anderson <dianders@chromium.org>
> ---

Reviewed-by: Stephen Boyd <swboyd@chromium.org>

One comment clarification below.

> I don't have lots of good test cases here, so hopefully someone from
> Qualcomm can confirm that this works well for them and there isn't
> some other phantom interrupt source that I'm not aware of.
> 
> Changes in v4:
> - ("pinctrl: qcom: Don't clear pending interrupts when enabling") split for v4.
> 
>  drivers/pinctrl/qcom/pinctrl-msm.c | 32 +++++++++++++-----------------
>  1 file changed, 14 insertions(+), 18 deletions(-)
> 
> diff --git a/drivers/pinctrl/qcom/pinctrl-msm.c b/drivers/pinctrl/qcom/pinctrl-msm.c
> index 588df91274e2..f785646d1df7 100644
> --- a/drivers/pinctrl/qcom/pinctrl-msm.c
> +++ b/drivers/pinctrl/qcom/pinctrl-msm.c
> @@ -1046,6 +1032,16 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)
>         }
>         msm_writel_intr_cfg(val, pctrl, g);
>  
> +       /*
> +        * The first time we set RAW_STATUS_EN it could trigger an interrupt.
> +        * Clear it.  This is safe because we have IRQCHIP_SET_TYPE_MASKED.

Clear the interrupt? 'it' is ambiguous.

> +        */
> +       if (!was_enabled) {
> +               val = msm_readl_intr_status(pctrl, g);
> +               val &= ~BIT(g->intr_status_bit);
> +               msm_writel_intr_status(val, pctrl, g);
> +       }
> +
>         if (test_bit(d->hwirq, pctrl->dual_edge_irqs))
>                 msm_gpio_update_dual_edge_pos(pctrl, g, d);
>
Rajendra Nayak Dec. 18, 2020, 5:35 a.m. UTC | #2
On 12/12/2020 3:45 AM, Douglas Anderson wrote:
> In Linux, if a driver does disable_irq() and later does enable_irq()

> on its interrupt, I believe it's expecting these properties:

> * If an interrupt was pending when the driver disabled then it will

>    still be pending after the driver re-enables.

> * If an edge-triggered interrupt comes in while an interrupt is

>    disabled it should assert when the interrupt is re-enabled.

> 

> If you think that the above sounds a lot like the disable_irq() and

> enable_irq() are supposed to be masking/unmasking the interrupt

> instead of disabling/enabling it then you've made an astute

> observation.  Specifically when talking about interrupts, "mask"

> usually means to stop posting interrupts but keep tracking them and

> "disable" means to fully shut off interrupt detection.  It's

> unfortunate that this is so confusing, but presumably this is all the

> way it is for historical reasons.

> 

> Perhaps more confusing than the above is that, even though clients of

> IRQs themselves don't have a way to request mask/unmask

> vs. disable/enable calls, IRQ chips themselves can implement both.

> ...and yet more confusing is that if an IRQ chip implements

> disable/enable then they will be called when a client driver calls

> disable_irq() / enable_irq().

> 

> It does feel like some of the above could be cleared up.  However,

> without any other core interrupt changes it should be clear that when

> an IRQ chip gets a request to "disable" an IRQ that it has to treat it

> like a mask of that IRQ.

> 

> In any case, after that long interlude you can see that the "unmask

> and clear" can break things.  Maulik tried to fix it so that we no

> longer did "unmask and clear" in commit 71266d9d3936 ("pinctrl: qcom:

> Move clearing pending IRQ to .irq_request_resources callback"), but it

> only handled the PDC case (it also had problems, but that's the

> subject of another patch).  Let's fix this for the non-PDC case.

> 

>  From my understanding the source of the phantom interrupt in the

> non-PDC case was the one that could have been introduced in

> msm_gpio_irq_set_type().  Let's handle that one and then get rid of

> the clear.

> 

> Fixes: 4b7618fdc7e6 ("pinctrl: qcom: Add irq_enable callback for msm gpio")

> Signed-off-by: Douglas Anderson <dianders@chromium.org>

> ---

> I don't have lots of good test cases here, so hopefully someone from

> Qualcomm can confirm that this works well for them and there isn't

> some other phantom interrupt source that I'm not aware of.


I currently don;t have access to any non-PDC hardware, so could not really do
any real tests, but the changes seem sane, so

Reviewed-by: Rajendra Nayak <rnayak@codeaurora.org>


> 

> Changes in v4:

> - ("pinctrl: qcom: Don't clear pending interrupts when enabling") split for v4.

> 

>   drivers/pinctrl/qcom/pinctrl-msm.c | 32 +++++++++++++-----------------

>   1 file changed, 14 insertions(+), 18 deletions(-)

> 

> diff --git a/drivers/pinctrl/qcom/pinctrl-msm.c b/drivers/pinctrl/qcom/pinctrl-msm.c

> index 588df91274e2..f785646d1df7 100644

> --- a/drivers/pinctrl/qcom/pinctrl-msm.c

> +++ b/drivers/pinctrl/qcom/pinctrl-msm.c

> @@ -774,7 +774,7 @@ static void msm_gpio_irq_mask(struct irq_data *d)

>   	raw_spin_unlock_irqrestore(&pctrl->lock, flags);

>   }

>   

> -static void msm_gpio_irq_clear_unmask(struct irq_data *d, bool status_clear)

> +static void msm_gpio_irq_unmask(struct irq_data *d)

>   {

>   	struct gpio_chip *gc = irq_data_get_irq_chip_data(d);

>   	struct msm_pinctrl *pctrl = gpiochip_get_data(gc);

> @@ -792,17 +792,6 @@ static void msm_gpio_irq_clear_unmask(struct irq_data *d, bool status_clear)

>   

>   	raw_spin_lock_irqsave(&pctrl->lock, flags);

>   

> -	if (status_clear) {

> -		/*

> -		 * clear the interrupt status bit before unmask to avoid

> -		 * any erroneous interrupts that would have got latched

> -		 * when the interrupt is not in use.

> -		 */

> -		val = msm_readl_intr_status(pctrl, g);

> -		val &= ~BIT(g->intr_status_bit);

> -		msm_writel_intr_status(val, pctrl, g);

> -	}

> -

>   	val = msm_readl_intr_cfg(pctrl, g);

>   	val |= BIT(g->intr_raw_status_bit);

>   	val |= BIT(g->intr_enable_bit);

> @@ -822,7 +811,7 @@ static void msm_gpio_irq_enable(struct irq_data *d)

>   		irq_chip_enable_parent(d);

>   

>   	if (!test_bit(d->hwirq, pctrl->skip_wake_irqs))

> -		msm_gpio_irq_clear_unmask(d, true);

> +		msm_gpio_irq_unmask(d);

>   }

>   

>   static void msm_gpio_irq_disable(struct irq_data *d)

> @@ -837,11 +826,6 @@ static void msm_gpio_irq_disable(struct irq_data *d)

>   		msm_gpio_irq_mask(d);

>   }

>   

> -static void msm_gpio_irq_unmask(struct irq_data *d)

> -{

> -	msm_gpio_irq_clear_unmask(d, false);

> -}

> -

>   /**

>    * msm_gpio_update_dual_edge_parent() - Prime next edge for IRQs handled by parent.

>    * @d: The irq dta.

> @@ -936,6 +920,7 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)

>   	struct msm_pinctrl *pctrl = gpiochip_get_data(gc);

>   	const struct msm_pingroup *g;

>   	unsigned long flags;

> +	bool was_enabled;

>   	u32 val;

>   

>   	if (msm_gpio_needs_dual_edge_parent_workaround(d, type)) {

> @@ -997,6 +982,7 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)

>   	 * could cause the INTR_STATUS to be set for EDGE interrupts.

>   	 */

>   	val = msm_readl_intr_cfg(pctrl, g);

> +	was_enabled = val & BIT(g->intr_raw_status_bit);

>   	val |= BIT(g->intr_raw_status_bit);

>   	if (g->intr_detection_width == 2) {

>   		val &= ~(3 << g->intr_detection_bit);

> @@ -1046,6 +1032,16 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)

>   	}

>   	msm_writel_intr_cfg(val, pctrl, g);

>   

> +	/*

> +	 * The first time we set RAW_STATUS_EN it could trigger an interrupt.

> +	 * Clear it.  This is safe because we have IRQCHIP_SET_TYPE_MASKED.

> +	 */

> +	if (!was_enabled) {

> +		val = msm_readl_intr_status(pctrl, g);

> +		val &= ~BIT(g->intr_status_bit);

> +		msm_writel_intr_status(val, pctrl, g);

> +	}

> +

>   	if (test_bit(d->hwirq, pctrl->dual_edge_irqs))

>   		msm_gpio_update_dual_edge_pos(pctrl, g, d);

>   

> 


-- 
QUALCOMM INDIA, on behalf of Qualcomm Innovation Center, Inc. is a member
of Code Aurora Forum, hosted by The Linux Foundation
Maulik Shah Dec. 21, 2020, 4 p.m. UTC | #3
Hi Doug,

On 12/12/2020 3:45 AM, Douglas Anderson wrote:
> In Linux, if a driver does disable_irq() and later does enable_irq()
> on its interrupt, I believe it's expecting these properties:
> * If an interrupt was pending when the driver disabled then it will
>    still be pending after the driver re-enables.
> * If an edge-triggered interrupt comes in while an interrupt is
>    disabled it should assert when the interrupt is re-enabled.
>
> If you think that the above sounds a lot like the disable_irq() and
> enable_irq() are supposed to be masking/unmasking the interrupt
> instead of disabling/enabling it then you've made an astute
> observation.  Specifically when talking about interrupts, "mask"
> usually means to stop posting interrupts but keep tracking them and
> "disable" means to fully shut off interrupt detection.  It's
> unfortunate that this is so confusing, but presumably this is all the
> way it is for historical reasons.
>
> Perhaps more confusing than the above is that, even though clients of
> IRQs themselves don't have a way to request mask/unmask
> vs. disable/enable calls, IRQ chips themselves can implement both.
> ...and yet more confusing is that if an IRQ chip implements
> disable/enable then they will be called when a client driver calls
> disable_irq() / enable_irq().
>
> It does feel like some of the above could be cleared up.  However,
> without any other core interrupt changes it should be clear that when
> an IRQ chip gets a request to "disable" an IRQ that it has to treat it
> like a mask of that IRQ.
>
> In any case, after that long interlude you can see that the "unmask
> and clear" can break things.  Maulik tried to fix it so that we no
> longer did "unmask and clear" in commit 71266d9d3936 ("pinctrl: qcom:
> Move clearing pending IRQ to .irq_request_resources callback"), but it
> only handled the PDC case (it also had problems, but that's the
> subject of another patch).  Let's fix this for the non-PDC case.
>
>  From my understanding the source of the phantom interrupt in the
> non-PDC case was the one that could have been introduced in
> msm_gpio_irq_set_type().  Let's handle that one and then get rid of
> the clear.
>
> Fixes: 4b7618fdc7e6 ("pinctrl: qcom: Add irq_enable callback for msm gpio")
> Signed-off-by: Douglas Anderson <dianders@chromium.org>
> ---
> I don't have lots of good test cases here, so hopefully someone from
> Qualcomm can confirm that this works well for them and there isn't
> some other phantom interrupt source that I'm not aware of.
>
> Changes in v4:
> - ("pinctrl: qcom: Don't clear pending interrupts when enabling") split for v4.
>
>   drivers/pinctrl/qcom/pinctrl-msm.c | 32 +++++++++++++-----------------
>   1 file changed, 14 insertions(+), 18 deletions(-)
>
> diff --git a/drivers/pinctrl/qcom/pinctrl-msm.c b/drivers/pinctrl/qcom/pinctrl-msm.c
> index 588df91274e2..f785646d1df7 100644
> --- a/drivers/pinctrl/qcom/pinctrl-msm.c
> +++ b/drivers/pinctrl/qcom/pinctrl-msm.c
> @@ -774,7 +774,7 @@ static void msm_gpio_irq_mask(struct irq_data *d)
>   	raw_spin_unlock_irqrestore(&pctrl->lock, flags);
>   }
>   
> -static void msm_gpio_irq_clear_unmask(struct irq_data *d, bool status_clear)
> +static void msm_gpio_irq_unmask(struct irq_data *d)
>   {
>   	struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
>   	struct msm_pinctrl *pctrl = gpiochip_get_data(gc);
> @@ -792,17 +792,6 @@ static void msm_gpio_irq_clear_unmask(struct irq_data *d, bool status_clear)
>   
>   	raw_spin_lock_irqsave(&pctrl->lock, flags);
>   
> -	if (status_clear) {
> -		/*
> -		 * clear the interrupt status bit before unmask to avoid
> -		 * any erroneous interrupts that would have got latched
> -		 * when the interrupt is not in use.
> -		 */
> -		val = msm_readl_intr_status(pctrl, g);
> -		val &= ~BIT(g->intr_status_bit);
> -		msm_writel_intr_status(val, pctrl, g);
> -	}
> -
Removing above does not cover the case where GPIO IRQ do not have parent 
PDC.

Specifically, for edge IRQs during masking we donot clear 
intr_raw_status_bit.
see below at msm_gpio_irq_mask()

         if (irqd_get_trigger_type(d) & IRQ_TYPE_LEVEL_MASK)
                 val &= ~BIT(g->intr_raw_status_bit);

we have to keep the bit set anyway so that edges are latched while the 
line is masked.

The problem is even when GPIO is set to some other function like 
"mi2s_2" it can still sense the line at make
interrupt pending depending on the line toggle if intr_raw_status_bit is 
left set.

I have thought of solution to this,

1) During msm_gpio_irq_mask() we keep intr_raw_status_bit set already in 
today's code
This will make edges to latch when the line is masked.
so no change required for this.

2) During msm_pinmux_set_mux() if we set GPIO to anyother function than 
GPIO interrupt mode,
we clear intr_raw_status_bit, so the interrupt cannot latch when GPIO is 
used in other function.
Below snippet can be inserted in msm_pinmux_set_mux()

         val |= i << g->mux_bit;
         msm_writel_ctl(val, pctrl, g);

+        if (i != gpio_func) {
+                val = msm_readl_intr_cfg(pctrl, g);
+                val &= ~BIT(g->intr_raw_status_bit);
+                msm_writel_intr_cfg(val, pctrl, g);
+        }
+
         raw_spin_unlock_irqrestore(&pctrl->lock, flags);

3) During msm_gpio_irq_unmask(), if the intr_raw_status_bit is not set, 
then clear the pending IRQ.
specifically setting this bit itself can cause the error IRQ, so clear 
it when setting this.

for edge IRQ, intr_raw_status_bit can only be cleared in 
msm_pinmux_set_mux() so clearing pending
IRQ should not loose any edges since we know GPIO was used in other 
function mode like mi2s_2 for
which we do not need to latch IRQs.
Below snippet can be inserted in msm_gpio_irq_unmask()

+       was_enabled = val & BIT(g->intr_raw_status_bit);
         val |= BIT(g->intr_raw_status_bit);
         val |= BIT(g->intr_enable_bit);
         msm_writel_intr_cfg(val, pctrl, g);

+       if (!was_enabled) {
+               val = msm_readl_intr_status(pctrl, g);
+               val &= ~BIT(g->intr_status_bit);
+               msm_writel_intr_status(val, pctrl, g);
+       }
+
         set_bit(d->hwirq, pctrl->enabled_irqs);

This can cover the cases for which the GPIO do not have parent.

Thanks,
Maulik

>   	val = msm_readl_intr_cfg(pctrl, g);
>   	val |= BIT(g->intr_raw_status_bit);
>   	val |= BIT(g->intr_enable_bit);
> @@ -822,7 +811,7 @@ static void msm_gpio_irq_enable(struct irq_data *d)
>   		irq_chip_enable_parent(d);
>   
>   	if (!test_bit(d->hwirq, pctrl->skip_wake_irqs))
> -		msm_gpio_irq_clear_unmask(d, true);
> +		msm_gpio_irq_unmask(d);
>   }
>   
>   static void msm_gpio_irq_disable(struct irq_data *d)
> @@ -837,11 +826,6 @@ static void msm_gpio_irq_disable(struct irq_data *d)
>   		msm_gpio_irq_mask(d);
>   }
>   
> -static void msm_gpio_irq_unmask(struct irq_data *d)
> -{
> -	msm_gpio_irq_clear_unmask(d, false);
> -}
> -
>   /**
>    * msm_gpio_update_dual_edge_parent() - Prime next edge for IRQs handled by parent.
>    * @d: The irq dta.
> @@ -936,6 +920,7 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)
>   	struct msm_pinctrl *pctrl = gpiochip_get_data(gc);
>   	const struct msm_pingroup *g;
>   	unsigned long flags;
> +	bool was_enabled;
>   	u32 val;
>   
>   	if (msm_gpio_needs_dual_edge_parent_workaround(d, type)) {
> @@ -997,6 +982,7 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)
>   	 * could cause the INTR_STATUS to be set for EDGE interrupts.
>   	 */
>   	val = msm_readl_intr_cfg(pctrl, g);
> +	was_enabled = val & BIT(g->intr_raw_status_bit);
>   	val |= BIT(g->intr_raw_status_bit);
>   	if (g->intr_detection_width == 2) {
>   		val &= ~(3 << g->intr_detection_bit);
> @@ -1046,6 +1032,16 @@ static int msm_gpio_irq_set_type(struct irq_data *d, unsigned int type)
>   	}
>   	msm_writel_intr_cfg(val, pctrl, g);
>   
> +	/*
> +	 * The first time we set RAW_STATUS_EN it could trigger an interrupt.
> +	 * Clear it.  This is safe because we have IRQCHIP_SET_TYPE_MASKED.
> +	 */
> +	if (!was_enabled) {
> +		val = msm_readl_intr_status(pctrl, g);
> +		val &= ~BIT(g->intr_status_bit);
> +		msm_writel_intr_status(val, pctrl, g);
> +	}
> +
>   	if (test_bit(d->hwirq, pctrl->dual_edge_irqs))
>   		msm_gpio_update_dual_edge_pos(pctrl, g, d);
>
Doug Anderson Jan. 8, 2021, 5:37 p.m. UTC | #4
Hi,

On Mon, Dec 21, 2020 at 8:01 AM Maulik Shah <mkshah@codeaurora.org> wrote:
>

> Hi Doug,

>

> On 12/12/2020 3:45 AM, Douglas Anderson wrote:

> > In Linux, if a driver does disable_irq() and later does enable_irq()

> > on its interrupt, I believe it's expecting these properties:

> > * If an interrupt was pending when the driver disabled then it will

> >    still be pending after the driver re-enables.

> > * If an edge-triggered interrupt comes in while an interrupt is

> >    disabled it should assert when the interrupt is re-enabled.

> >

> > If you think that the above sounds a lot like the disable_irq() and

> > enable_irq() are supposed to be masking/unmasking the interrupt

> > instead of disabling/enabling it then you've made an astute

> > observation.  Specifically when talking about interrupts, "mask"

> > usually means to stop posting interrupts but keep tracking them and

> > "disable" means to fully shut off interrupt detection.  It's

> > unfortunate that this is so confusing, but presumably this is all the

> > way it is for historical reasons.

> >

> > Perhaps more confusing than the above is that, even though clients of

> > IRQs themselves don't have a way to request mask/unmask

> > vs. disable/enable calls, IRQ chips themselves can implement both.

> > ...and yet more confusing is that if an IRQ chip implements

> > disable/enable then they will be called when a client driver calls

> > disable_irq() / enable_irq().

> >

> > It does feel like some of the above could be cleared up.  However,

> > without any other core interrupt changes it should be clear that when

> > an IRQ chip gets a request to "disable" an IRQ that it has to treat it

> > like a mask of that IRQ.

> >

> > In any case, after that long interlude you can see that the "unmask

> > and clear" can break things.  Maulik tried to fix it so that we no

> > longer did "unmask and clear" in commit 71266d9d3936 ("pinctrl: qcom:

> > Move clearing pending IRQ to .irq_request_resources callback"), but it

> > only handled the PDC case (it also had problems, but that's the

> > subject of another patch).  Let's fix this for the non-PDC case.

> >

> >  From my understanding the source of the phantom interrupt in the

> > non-PDC case was the one that could have been introduced in

> > msm_gpio_irq_set_type().  Let's handle that one and then get rid of

> > the clear.

> >

> > Fixes: 4b7618fdc7e6 ("pinctrl: qcom: Add irq_enable callback for msm gpio")

> > Signed-off-by: Douglas Anderson <dianders@chromium.org>

> > ---

> > I don't have lots of good test cases here, so hopefully someone from

> > Qualcomm can confirm that this works well for them and there isn't

> > some other phantom interrupt source that I'm not aware of.

> >

> > Changes in v4:

> > - ("pinctrl: qcom: Don't clear pending interrupts when enabling") split for v4.

> >

> >   drivers/pinctrl/qcom/pinctrl-msm.c | 32 +++++++++++++-----------------

> >   1 file changed, 14 insertions(+), 18 deletions(-)

> >

> > diff --git a/drivers/pinctrl/qcom/pinctrl-msm.c b/drivers/pinctrl/qcom/pinctrl-msm.c

> > index 588df91274e2..f785646d1df7 100644

> > --- a/drivers/pinctrl/qcom/pinctrl-msm.c

> > +++ b/drivers/pinctrl/qcom/pinctrl-msm.c

> > @@ -774,7 +774,7 @@ static void msm_gpio_irq_mask(struct irq_data *d)

> >       raw_spin_unlock_irqrestore(&pctrl->lock, flags);

> >   }

> >

> > -static void msm_gpio_irq_clear_unmask(struct irq_data *d, bool status_clear)

> > +static void msm_gpio_irq_unmask(struct irq_data *d)

> >   {

> >       struct gpio_chip *gc = irq_data_get_irq_chip_data(d);

> >       struct msm_pinctrl *pctrl = gpiochip_get_data(gc);

> > @@ -792,17 +792,6 @@ static void msm_gpio_irq_clear_unmask(struct irq_data *d, bool status_clear)

> >

> >       raw_spin_lock_irqsave(&pctrl->lock, flags);

> >

> > -     if (status_clear) {

> > -             /*

> > -              * clear the interrupt status bit before unmask to avoid

> > -              * any erroneous interrupts that would have got latched

> > -              * when the interrupt is not in use.

> > -              */

> > -             val = msm_readl_intr_status(pctrl, g);

> > -             val &= ~BIT(g->intr_status_bit);

> > -             msm_writel_intr_status(val, pctrl, g);

> > -     }

> > -

> Removing above does not cover the case where GPIO IRQ do not have parent

> PDC.

>

> Specifically, for edge IRQs during masking we donot clear

> intr_raw_status_bit.

> see below at msm_gpio_irq_mask()

>

>          if (irqd_get_trigger_type(d) & IRQ_TYPE_LEVEL_MASK)

>                  val &= ~BIT(g->intr_raw_status_bit);

>

> we have to keep the bit set anyway so that edges are latched while the

> line is masked.

>

> The problem is even when GPIO is set to some other function like

> "mi2s_2" it can still sense the line at make

> interrupt pending depending on the line toggle if intr_raw_status_bit is

> left set.


Ah, so it's the same problem as we have with the PDC.  Makes sense.


> I have thought of solution to this,

>

> 1) During msm_gpio_irq_mask() we keep intr_raw_status_bit set already in

> today's code

> This will make edges to latch when the line is masked.

> so no change required for this.

>

> 2) During msm_pinmux_set_mux() if we set GPIO to anyother function than

> GPIO interrupt mode,

> we clear intr_raw_status_bit, so the interrupt cannot latch when GPIO is

> used in other function.

> Below snippet can be inserted in msm_pinmux_set_mux()

>

>          val |= i << g->mux_bit;

>          msm_writel_ctl(val, pctrl, g);

>

> +        if (i != gpio_func) {

> +                val = msm_readl_intr_cfg(pctrl, g);

> +                val &= ~BIT(g->intr_raw_status_bit);

> +                msm_writel_intr_cfg(val, pctrl, g);

> +        }

> +

>          raw_spin_unlock_irqrestore(&pctrl->lock, flags);

>

> 3) During msm_gpio_irq_unmask(), if the intr_raw_status_bit is not set,

> then clear the pending IRQ.

> specifically setting this bit itself can cause the error IRQ, so clear

> it when setting this.

>

> for edge IRQ, intr_raw_status_bit can only be cleared in

> msm_pinmux_set_mux() so clearing pending

> IRQ should not loose any edges since we know GPIO was used in other

> function mode like mi2s_2 for

> which we do not need to latch IRQs.

> Below snippet can be inserted in msm_gpio_irq_unmask()

>

> +       was_enabled = val & BIT(g->intr_raw_status_bit);

>          val |= BIT(g->intr_raw_status_bit);

>          val |= BIT(g->intr_enable_bit);

>          msm_writel_intr_cfg(val, pctrl, g);

>

> +       if (!was_enabled) {

> +               val = msm_readl_intr_status(pctrl, g);

> +               val &= ~BIT(g->intr_status_bit);

> +               msm_writel_intr_status(val, pctrl, g);

> +       }

> +

>          set_bit(d->hwirq, pctrl->enabled_irqs);

>

> This can cover the cases for which the GPIO do not have parent.


I think your solution can be made to work, but I think also we can
just use the exact same solution that I already came up with in patch
#4.  We can leave the "raw" bit alone and just mask the interrupt when
we switch the mux, then clear the interrupt when we switch back.

I've now combined the PDC/non-PDC cases and it actually turned out
fairly clean I think.  See what you think about v5.

-Doug
diff mbox series

Patch

diff --git a/drivers/irqchip/qcom-pdc.c b/drivers/irqchip/qcom-pdc.c
index bd39e9de6ecf..5dc63c20b67e 100644
--- a/drivers/irqchip/qcom-pdc.c
+++ b/drivers/irqchip/qcom-pdc.c
@@ -159,6 +159,8 @@  static int qcom_pdc_gic_set_type(struct irq_data *d, unsigned int type)
 {
 	int pin_out = d->hwirq;
 	enum pdc_irq_config_bits pdc_type;
+	enum pdc_irq_config_bits old_pdc_type;
+	int ret;
 
 	if (pin_out == GPIO_NO_WAKE_IRQ)
 		return 0;
@@ -187,9 +189,26 @@  static int qcom_pdc_gic_set_type(struct irq_data *d, unsigned int type)
 		return -EINVAL;
 	}
 
+	old_pdc_type = pdc_reg_read(IRQ_i_CFG, pin_out);
 	pdc_reg_write(IRQ_i_CFG, pin_out, pdc_type);
 
-	return irq_chip_set_type_parent(d, type);
+	ret = irq_chip_set_type_parent(d, type);
+	if (ret)
+		return ret;
+
+	/*
+	 * When we change types the PDC can give a phantom interrupt.
+	 * Clear it.  Specifically the phantom shows up when reconfiguring
+	 * polarity of interrupt without changing the state of the signal
+	 * but let's be consistent and clear it always.
+	 *
+	 * Doing this works because we have IRQCHIP_SET_TYPE_MASKED so the
+	 * interrupt will be cleared before the rest of the system sees it.
+	 */
+	if (old_pdc_type != pdc_type)
+		irq_chip_set_parent_state(d, IRQCHIP_STATE_PENDING, false);
+
+	return 0;
 }
 
 static struct irq_chip qcom_pdc_gic_chip = {