Re: [PATCH] atmel_serial: Split the interrupt handler
From: Remy Bohmer
Date: Tue Dec 18 2007 - 10:23:28 EST
Hello Haavard,
A few remarks:
> From: Remy Bohmer <hskinnemoen@xxxxxxxxx>
My name, at your address ;-)))
> This patch splits up the interrupt handler of the serial port
> into a interrupt top-half and a tasklet.
I see you moved the handling of the sysrq-key to the tasklet. This was
actually a very nice feature in the IRQ-top half on preempt-RT. This
helps debugging running away RT-processes.
> In this version of the patch, we try to only do things that are
> absolutely necessary in the interrupt handler, storing away the
> status register along with the received character and letting the
> tasklet handle break, sysrq, error flags, etc.
Preempt-RT now absolutely requires my (4th) IRQ_NODELAY patch, because
the spinlock now is always inside the code, and not only in
theexception path, and thus without my NO_DELAY patch we have a panic
during boot.
On preempt-RT this spinlock must be a raw-spinlock. (If this type is
known in the mainline kernel, you can apply that patch it anyway)
BTW: Attached I have added a 2nd patch that I use for Preempt-RT. (For
cleaner startup, and to get rid of useless IRQ-threads.
> This patch should apply on top of the cleanup patch I sent earlier
For the cleanup patch:
Acked-by: Remy Bohmer <linux@xxxxxxxxxx>
> today. Or at least I think so...I'll send the full series once
> everyone are happy.
So, for this patch: I am almost happy ;-)
Kind Regards,
Remy
This patch prevents starting up a useless IRQ-thread for IRQ-1,
because this IRQ is running in IRQF_NODELAY context due to the
timer interrupt.
Signed-off-by: Remy Bohmer <linux@xxxxxxxxxx>
---
drivers/serial/atmel_serial.c | 10 +++++++++-
1 file changed, 9 insertions(+), 1 deletion(-)
Index: linux-2.6.23/drivers/serial/atmel_serial.c
===================================================================
--- linux-2.6.23.orig/drivers/serial/atmel_serial.c 2007-12-18 15:49:02.000000000 +0100
+++ linux-2.6.23/drivers/serial/atmel_serial.c 2007-12-18 15:56:37.000000000 +0100
@@ -549,6 +549,7 @@ static int atmel_startup(struct uart_por
{
struct atmel_uart_port *atmel_port = (struct atmel_uart_port *) port;
int retval;
+ unsigned long irqflags = IRQF_SHARED;
/*
* Ensure that no interrupts are enabled otherwise when
@@ -557,10 +558,17 @@ static int atmel_startup(struct uart_por
*/
UART_PUT_IDR(port, -1);
+#ifdef CONFIG_PREEMPT_RT
+ /* IRQ1 is the System IRQ, which is shared with the Timer-IRQ, and
+ thus it is running in IRQF_NODELAY context. By setting this flag
+ here also, we prevent starting up a useless IRQ-thread.*/
+ if (port->irq == 1)
+ irqflags |= IRQF_NODELAY;
+#endif
/*
* Allocate the IRQ
*/
- retval = request_irq(port->irq, atmel_interrupt, IRQF_SHARED,
+ retval = request_irq(port->irq, atmel_interrupt, irqflags,
"atmel_serial", port);
if (retval) {
printk("atmel_serial: atmel_startup - Can't get irq\n");