serial: msm_serial: Fix kgdb continue
authorStephen Boyd <sboyd@codeaurora.org>
Wed, 6 Aug 2014 01:37:24 +0000 (18:37 -0700)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 8 Sep 2014 22:55:25 +0000 (15:55 -0700)
Frank reports that after continuing in kgdb the RX stale event
doesn't occur until after the RX fifo is filled up with exactly
the amount of characters programmed for the RX watermark (in this
case it's 48). To read a single character from the uartdm
hardware we force a stale event so that any characters in the RX
packing buffer are flushed into the RX fifo immediately instead
of waiting for a stale timeout or for the fifo to fill. Forcing
that stale event asserts the stale interrupt but we never clear
that interrupt via UART_CR_CMD_RESET_STALE_INT in the polling
functions. So when kgdb continues the stale interrupt is left
pending in the hardware and we don't timeout with a stale event,
like we usually would if a user typed one character on the
console, until the reset stale interrupt and stale event commands
are sent. Frank could get things working again by running
handle_rx_dm(). By putting enough characters into the fifo he
could trigger a watermark interrupt, and thus cause
handle_rx_dm() to run finally resetting the stale interrupt
and enabling the stale event so that single characters would
cause timeouts again.

The fix is to just do what the interrupt routine was doing all
along and clear the stale interrupt and enable the event again.
Doing this also smooths over any differences in the fifo behavior
between v1.3 and v1.4 hardware allowing us to skip forcing the
uart into single character mode.

Reviewed-by: Frank Rowand <frank.rowand@sonymobile.com>
Tested-by: Frank Rowand <frank.rowand@sonymobile.com>
Fixes: f7e54d7ad743 "msm_serial: Add support for poll_{get,put}_char()"
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/tty/serial/msm_serial.c

index 26df03878b7e49dc65d86f2335a3de96ae50bb3e..4f9640d0b1bbc4d077b1625c32a51732e3ba7681 100644 (file)
@@ -682,17 +682,6 @@ static void msm_power(struct uart_port *port, unsigned int state,
 }
 
 #ifdef CONFIG_CONSOLE_POLL
-static int msm_poll_init(struct uart_port *port)
-{
-       struct msm_port *msm_port = UART_TO_MSM(port);
-
-       /* Enable single character mode on RX FIFO */
-       if (msm_port->is_uartdm >= UARTDM_1P4)
-               msm_write(port, UARTDM_DMEN_RX_SC_ENABLE, UARTDM_DMEN);
-
-       return 0;
-}
-
 static int msm_poll_get_char_single(struct uart_port *port)
 {
        struct msm_port *msm_port = UART_TO_MSM(port);
@@ -704,7 +693,7 @@ static int msm_poll_get_char_single(struct uart_port *port)
        return msm_read(port, rf_reg) & 0xff;
 }
 
-static int msm_poll_get_char_dm_1p3(struct uart_port *port)
+static int msm_poll_get_char_dm(struct uart_port *port)
 {
        int c;
        static u32 slop;
@@ -728,6 +717,10 @@ static int msm_poll_get_char_dm_1p3(struct uart_port *port)
                        slop = msm_read(port, UARTDM_RF);
                        c = sp[0];
                        count--;
+                       msm_write(port, UART_CR_CMD_RESET_STALE_INT, UART_CR);
+                       msm_write(port, 0xFFFFFF, UARTDM_DMRX);
+                       msm_write(port, UART_CR_CMD_STALE_EVENT_ENABLE,
+                                 UART_CR);
                } else {
                        c = NO_POLL_CHAR;
                }
@@ -751,8 +744,8 @@ static int msm_poll_get_char(struct uart_port *port)
        imr = msm_read(port, UART_IMR);
        msm_write(port, 0, UART_IMR);
 
-       if (msm_port->is_uartdm == UARTDM_1P3)
-               c = msm_poll_get_char_dm_1p3(port);
+       if (msm_port->is_uartdm)
+               c = msm_poll_get_char_dm(port);
        else
                c = msm_poll_get_char_single(port);
 
@@ -809,7 +802,6 @@ static struct uart_ops msm_uart_pops = {
        .verify_port = msm_verify_port,
        .pm = msm_power,
 #ifdef CONFIG_CONSOLE_POLL
-       .poll_init = msm_poll_init,
        .poll_get_char  = msm_poll_get_char,
        .poll_put_char  = msm_poll_put_char,
 #endif