tty: xuartps: Move request_irq to after setting up the HW
authorSoren Brinkmann <soren.brinkmann@xilinx.com>
Tue, 12 Jan 2016 01:41:36 +0000 (17:41 -0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sun, 7 Feb 2016 06:52:23 +0000 (22:52 -0800)
Request_irq() should be _after_ h/w programming, otherwise an
interrupt could be triggered and in-progress before the h/w has been
setup.

Reported-by: Peter Hurley <peter@hurleysoftware.com>
Signed-off-by: Soren Brinkmann <soren.brinkmann@xilinx.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/tty/serial/xilinx_uartps.c

index db9e23eaf300fed2f4fbb8d445ad7cb46aeaa544..5da1c51e9e4054f2462c415853099619ef3451c5 100644 (file)
@@ -762,13 +762,9 @@ static void cdns_uart_set_termios(struct uart_port *port,
  */
 static int cdns_uart_startup(struct uart_port *port)
 {
+       int ret;
        unsigned long flags;
-       unsigned int retval = 0, status = 0;
-
-       retval = request_irq(port->irq, cdns_uart_isr, 0, CDNS_UART_NAME,
-                                                               (void *)port);
-       if (retval)
-               return retval;
+       unsigned int status = 0;
 
        spin_lock_irqsave(&port->lock, flags);
 
@@ -814,15 +810,22 @@ static int cdns_uart_startup(struct uart_port *port)
        writel(readl(port->membase + CDNS_UART_ISR_OFFSET),
                        port->membase + CDNS_UART_ISR_OFFSET);
 
+       spin_unlock_irqrestore(&port->lock, flags);
+
+       ret = request_irq(port->irq, cdns_uart_isr, 0, CDNS_UART_NAME, port);
+       if (ret) {
+               dev_err(port->dev, "request_irq '%d' failed with %d\n",
+                       port->irq, ret);
+               return ret;
+       }
+
        /* Set the Interrupt Registers with desired interrupts */
        writel(CDNS_UART_IXR_TXEMPTY | CDNS_UART_IXR_PARITY |
                CDNS_UART_IXR_FRAMING | CDNS_UART_IXR_OVERRUN |
                CDNS_UART_IXR_RXTRIG | CDNS_UART_IXR_TOUT,
                port->membase + CDNS_UART_IER_OFFSET);
 
-       spin_unlock_irqrestore(&port->lock, flags);
-
-       return retval;
+       return 0;
 }
 
 /**