Commit 6f3c3caf authored by Johan Hovold's avatar Johan Hovold Committed by Greg Kroah-Hartman

serial: qcom-geni: disable interrupts during console writes

Disable the GENI interrupts during console writes to reduce the risk of
having interrupt handlers spinning on the port lock on other cores for
extended periods of time.

This can, for example, reduce the total amount of time spent in the
interrupt handler during boot of the x1e80100 CRD by up to a factor nine
(e.g. from 274 ms to 30 ms) while the worst case processing time drops
from 19 ms to 8 ms.

Fixes: c4f52879 ("tty: serial: msm_geni_serial: Add serial driver support for GENI based QUP")
Reviewed-by: default avatarDouglas Anderson <dianders@chromium.org>
Tested-by: default avatarNícolas F. R. A. Prado <nfraprado@collabora.com>
Signed-off-by: default avatarJohan Hovold <johan+linaro@kernel.org>
Link: https://lore.kernel.org/r/20240906131336.23625-8-johan+linaro@kernel.orgSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent cc4a0e57
...@@ -477,6 +477,7 @@ static void qcom_geni_serial_console_write(struct console *co, const char *s, ...@@ -477,6 +477,7 @@ static void qcom_geni_serial_console_write(struct console *co, const char *s,
{ {
struct uart_port *uport; struct uart_port *uport;
struct qcom_geni_serial_port *port; struct qcom_geni_serial_port *port;
u32 m_irq_en, s_irq_en;
bool locked = true; bool locked = true;
unsigned long flags; unsigned long flags;
...@@ -492,6 +493,11 @@ static void qcom_geni_serial_console_write(struct console *co, const char *s, ...@@ -492,6 +493,11 @@ static void qcom_geni_serial_console_write(struct console *co, const char *s,
else else
uart_port_lock_irqsave(uport, &flags); uart_port_lock_irqsave(uport, &flags);
m_irq_en = readl(uport->membase + SE_GENI_M_IRQ_EN);
s_irq_en = readl(uport->membase + SE_GENI_S_IRQ_EN);
writel(0, uport->membase + SE_GENI_M_IRQ_EN);
writel(0, uport->membase + SE_GENI_S_IRQ_EN);
if (qcom_geni_serial_main_active(uport)) { if (qcom_geni_serial_main_active(uport)) {
/* Wait for completion or drain FIFO */ /* Wait for completion or drain FIFO */
if (!locked || port->tx_remaining == 0) if (!locked || port->tx_remaining == 0)
...@@ -504,6 +510,9 @@ static void qcom_geni_serial_console_write(struct console *co, const char *s, ...@@ -504,6 +510,9 @@ static void qcom_geni_serial_console_write(struct console *co, const char *s,
__qcom_geni_serial_console_write(uport, s, count); __qcom_geni_serial_console_write(uport, s, count);
writel(m_irq_en, uport->membase + SE_GENI_M_IRQ_EN);
writel(s_irq_en, uport->membase + SE_GENI_S_IRQ_EN);
if (locked) if (locked)
uart_port_unlock_irqrestore(uport, flags); uart_port_unlock_irqrestore(uport, flags);
} }
......
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