Patch "can: m_can: enable NAPI before enabling interrupts" has been added to the 6.1-stable tree

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



This is a note to let you know that I've just added the patch titled

    can: m_can: enable NAPI before enabling interrupts

to the 6.1-stable tree which can be found at:
    http://www.kernel.org/git/?p=linux/kernel/git/stable/stable-queue.git;a=summary

The filename of the patch is:
     can-m_can-enable-napi-before-enabling-interrupts.patch
and it can be found in the queue-6.1 subdirectory.

If you, or anyone else, feels it should not be added to the stable tree,
please let <stable@xxxxxxxxxxxxxxx> know about it.



commit 51667faf4b1a36691140d247b447b41f8db15df0
Author: Jake Hamby <Jake.Hamby@xxxxxxxxxxxx>
Date:   Fri Sep 6 23:19:51 2024 +0000

    can: m_can: enable NAPI before enabling interrupts
    
    [ Upstream commit 801ad2f87b0c6d0c34a75a4efd6bfd3a2d9f9298 ]
    
    If an interrupt (RX-complete or error flag) is set when bringing up
    the CAN device, e.g. due to CAN bus traffic before initializing the
    device, when m_can_start() is called and interrupts are enabled,
    m_can_isr() is called immediately, which disables all CAN interrupts
    and calls napi_schedule().
    
    Because napi_enable() isn't called until later in m_can_open(), the
    call to napi_schedule() never schedules the m_can_poll() callback and
    the device is left with interrupts disabled and can't receive any CAN
    packets until rebooted.
    
    This can be verified by running "cansend" from another device before
    setting the bitrate and calling "ip link set up can0" on the test
    device. Adding debug lines to m_can_isr() shows it's called with flags
    (IR_EP | IR_EW | IR_CRCE), which calls m_can_disable_all_interrupts()
    and napi_schedule(), and then m_can_poll() is never called.
    
    Move the call to napi_enable() above the call to m_can_start() to
    enable any initial interrupt flags to be handled by m_can_poll() so
    that interrupts are reenabled. Add a call to napi_disable() in the
    error handling section of m_can_open(), to handle the case where later
    functions return errors.
    
    Also, in m_can_close(), move the call to napi_disable() below the call
    to m_can_stop() to ensure all interrupts are handled when bringing
    down the device. This race condition is much less likely to occur.
    
    Tested on a Microchip SAMA7G54 MPU. The fix should be applicable to
    any SoC with a Bosch M_CAN controller.
    
    Signed-off-by: Jake Hamby <Jake.Hamby@xxxxxxxxxxxx>
    Fixes: e0d1f4816f2a ("can: m_can: add Bosch M_CAN controller support")
    Link: https://patch.msgid.link/20240910-can-m_can-fix-ifup-v3-1-6c1720ba45ce@xxxxxxxxxxxxxx
    Signed-off-by: Marc Kleine-Budde <mkl@xxxxxxxxxxxxxx>
    Signed-off-by: Sasha Levin <sashal@xxxxxxxxxx>

diff --git a/drivers/net/can/m_can/m_can.c b/drivers/net/can/m_can/m_can.c
index ead8321ed3bfc..d8e6a081118b7 100644
--- a/drivers/net/can/m_can/m_can.c
+++ b/drivers/net/can/m_can/m_can.c
@@ -1555,9 +1555,6 @@ static int m_can_close(struct net_device *dev)
 
 	netif_stop_queue(dev);
 
-	if (!cdev->is_peripheral)
-		napi_disable(&cdev->napi);
-
 	m_can_stop(dev);
 	m_can_clk_stop(cdev);
 	free_irq(dev->irq, dev);
@@ -1567,6 +1564,8 @@ static int m_can_close(struct net_device *dev)
 		destroy_workqueue(cdev->tx_wq);
 		cdev->tx_wq = NULL;
 		can_rx_offload_disable(&cdev->offload);
+	} else {
+		napi_disable(&cdev->napi);
 	}
 
 	close_candev(dev);
@@ -1784,6 +1783,8 @@ static int m_can_open(struct net_device *dev)
 
 	if (cdev->is_peripheral)
 		can_rx_offload_enable(&cdev->offload);
+	else
+		napi_enable(&cdev->napi);
 
 	/* register interrupt handler */
 	if (cdev->is_peripheral) {
@@ -1815,9 +1816,6 @@ static int m_can_open(struct net_device *dev)
 	if (err)
 		goto exit_start_fail;
 
-	if (!cdev->is_peripheral)
-		napi_enable(&cdev->napi);
-
 	netif_start_queue(dev);
 
 	return 0;
@@ -1831,6 +1829,8 @@ static int m_can_open(struct net_device *dev)
 out_wq_fail:
 	if (cdev->is_peripheral)
 		can_rx_offload_disable(&cdev->offload);
+	else
+		napi_disable(&cdev->napi);
 	close_candev(dev);
 exit_disable_clks:
 	m_can_clk_stop(cdev);




[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Index of Archives]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux