Commit 72a8be5b authored by Anssi Hannula's avatar Anssi Hannula Committed by Kleber Sacilotto de Souza

can: xilinx_can: fix incorrect clear of non-processed interrupts

BugLink: https://bugs.launchpad.net/bugs/1791942

commit 2f4f0f33 upstream.

xcan_interrupt() clears ERROR|RXOFLV|BSOFF|ARBLST interrupts if any of
them is asserted. This does not take into account that some of them
could have been asserted between interrupt status read and interrupt
clear, therefore clearing them without handling them.

Fix the code to only clear those interrupts that it knows are asserted
and therefore going to be processed in xcan_err_interrupt().

Fixes: b1201e44 ("can: xilinx CAN controller support")
Signed-off-by: default avatarAnssi Hannula <anssi.hannula@bitwise.fi>
Cc: Michal Simek <michal.simek@xilinx.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: default avatarMarc Kleine-Budde <mkl@pengutronix.de>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarStefan Bader <stefan.bader@canonical.com>
Signed-off-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
parent 8a27759d
...@@ -937,6 +937,7 @@ static irqreturn_t xcan_interrupt(int irq, void *dev_id) ...@@ -937,6 +937,7 @@ static irqreturn_t xcan_interrupt(int irq, void *dev_id)
struct net_device *ndev = (struct net_device *)dev_id; struct net_device *ndev = (struct net_device *)dev_id;
struct xcan_priv *priv = netdev_priv(ndev); struct xcan_priv *priv = netdev_priv(ndev);
u32 isr, ier; u32 isr, ier;
u32 isr_errors;
/* Get the interrupt status from Xilinx CAN */ /* Get the interrupt status from Xilinx CAN */
isr = priv->read_reg(priv, XCAN_ISR_OFFSET); isr = priv->read_reg(priv, XCAN_ISR_OFFSET);
...@@ -955,11 +956,10 @@ static irqreturn_t xcan_interrupt(int irq, void *dev_id) ...@@ -955,11 +956,10 @@ static irqreturn_t xcan_interrupt(int irq, void *dev_id)
xcan_tx_interrupt(ndev, isr); xcan_tx_interrupt(ndev, isr);
/* Check for the type of error interrupt and Processing it */ /* Check for the type of error interrupt and Processing it */
if (isr & (XCAN_IXR_ERROR_MASK | XCAN_IXR_RXOFLW_MASK | isr_errors = isr & (XCAN_IXR_ERROR_MASK | XCAN_IXR_RXOFLW_MASK |
XCAN_IXR_BSOFF_MASK | XCAN_IXR_ARBLST_MASK)) { XCAN_IXR_BSOFF_MASK | XCAN_IXR_ARBLST_MASK);
priv->write_reg(priv, XCAN_ICR_OFFSET, (XCAN_IXR_ERROR_MASK | if (isr_errors) {
XCAN_IXR_RXOFLW_MASK | XCAN_IXR_BSOFF_MASK | priv->write_reg(priv, XCAN_ICR_OFFSET, isr_errors);
XCAN_IXR_ARBLST_MASK));
xcan_err_interrupt(ndev, isr); xcan_err_interrupt(ndev, isr);
} }
......
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