aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorStephen Warren <swarren@nvidia.com>2014-01-07 17:00:12 -0500
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2014-02-06 14:08:14 -0500
commit8d353b6d2f264168b820b2124c3db5018430fd04 (patch)
tree04c48ed6f02c067787c42286733f18dc410b0364
parent2f8cef8c248cf56daefe3c2fb98f13cc626377f5 (diff)
serial: 8250: enable UART_BUG_NOMSR for Tegra
commit 3685f19e07802ec4207b52465c408f185b66490e upstream. Tegra chips have 4 or 5 identical UART modules embedded. UARTs C..E have their MODEM-control signals tied off to a static state. However UARTs A and B can optionally route those signals to/from package pins, depending on the exact pinmux configuration. When these signals are not routed to package pins, false interrupts may trigger either temporarily, or permanently, all while not showing up in the IIR; it will read as NO_INT. This will eventually lead to the UART IRQ being disabled due to unhandled interrupts. When this happens, the kernel may print e.g.: irq 68: nobody cared (try booting with the "irqpoll" option) In order to prevent this, enable UART_BUG_NOMSR. This prevents UART_IER_MSI from being enabled, which prevents the false interrupts from triggering. In practice, this is not needed under any of the following conditions: * On Tegra chips after Tegra30, since the HW bug has apparently been fixed. * On UARTs C..E since their MODEM control signals are tied to the correct static state which doesn't trigger the issue. * On UARTs A..B if the MODEM control signals are routed out to package pins, since they will then carry valid signals. However, we ignore these exceptions for now, since they are only relevant if a board actually hooks up more than a 4-wire UART, and no currently supported board does this. If we ever support a board that does, we can refine the algorithm that enables UART_BUG_NOMSR to take those exceptions into account, and/or read a flag from DT/... that indicates that the board has hooked up and pinmux'd more than a 4-wire UART. Reported-by: Olof Johansson <olof@lixom.net> # autotester Signed-off-by: Stephen Warren <swarren@nvidia.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--drivers/tty/serial/8250/8250_core.c4
1 files changed, 4 insertions, 0 deletions
diff --git a/drivers/tty/serial/8250/8250_core.c b/drivers/tty/serial/8250/8250_core.c
index 86c00b1c5583..d28d7afc128a 100644
--- a/drivers/tty/serial/8250/8250_core.c
+++ b/drivers/tty/serial/8250/8250_core.c
@@ -2670,6 +2670,10 @@ static void serial8250_config_port(struct uart_port *port, int flags)
2670 if (port->type == PORT_16550A && port->iotype == UPIO_AU) 2670 if (port->type == PORT_16550A && port->iotype == UPIO_AU)
2671 up->bugs |= UART_BUG_NOMSR; 2671 up->bugs |= UART_BUG_NOMSR;
2672 2672
2673 /* HW bugs may trigger IRQ while IIR == NO_INT */
2674 if (port->type == PORT_TEGRA)
2675 up->bugs |= UART_BUG_NOMSR;
2676
2673 if (port->type != PORT_UNKNOWN && flags & UART_CONFIG_IRQ) 2677 if (port->type != PORT_UNKNOWN && flags & UART_CONFIG_IRQ)
2674 autoconfig_irq(up); 2678 autoconfig_irq(up);
2675 2679