LKML Archive on lore.kernel.org
help / color / mirror / Atom feed
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
[not found] <20080129224316.GA23155@gandalf.sssup.it>
@ 2008-01-29 23:12 ` michael
2008-01-30 9:41 ` Haavard Skinnemoen
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-01-29 23:12 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Remy Bohmer, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hi,
>> From: Remy Bohmer <linux@bohmer.net>
>>
>> This patch splits up the interrupt handler of the serial port
>> into a interrupt top-half and a tasklet.
>>
>> The goal is to get the interrupt top-half as short as possible to
>> minimize latencies on interrupts. But the old code also does some
>> calls in the interrupt handler that are not allowed on preempt-RT
>> in IRQF_NODELAY context. This handler is executed in this context
>> because of the interrupt sharing with the timer interrupt.
>> +static void
>> +atmel_buffer_rx_char(struct uart_port *port, unsigned int status,
>> + unsigned int ch)
>> +{
>> + struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
>> + struct circ_buf *ring = &atmel_port->rx_ring;
>> + struct atmel_uart_char *c;
>>
I'm testing this patch on an at91sam9260 on 2.6.24-rt. I'm using this
patch with the tclib support for hrtimer and the clocksource pit_clk.
These are the results:
- Voluntary Kernel Preemption the system (crash)
- Preemptible Kernel (crash)
/*
* Drop the lock here since it might end up calling
* uart_start(), which takes the lock.
spin_unlock(&port->lock);
*/
tty_flip_buffer_push(port->info->tty);
/*
spin_lock(&port->lock);
*/
The same code with this comments out runs
Complete Preemption (Real-Time) ok but the serials is just unusable due
to too many overruns (just using lrz)
The system is stable and doesn't crash reverting this patch.
I don't test with the thread hardirqs active.
>> +
>> + if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE))
>> + /* Buffer overflow, ignore char */
>> + return;
>> +
>> + c = &((struct atmel_uart_char *)ring->buf)[ring->head];
>> + c->status = status;
>> + c->ch = ch;
>> +
>> + /* Make sure the character is stored before we update head. */
>> + smp_wmb();
>> +
>> + ring->head = (ring->head + 1) & (ATMEL_SERIAL_RINGSIZE - 1);
>> +}
>> +
>>
...
>> +
>> port = &atmel_ports[pdev->id];
>> atmel_init_port(port, pdev);
>>
>> + ret = -ENOMEM;
>> + data = kmalloc(ATMEL_SERIAL_RINGSIZE, GFP_KERNEL);
>> + if (!data)
>> + goto err_alloc_ring;
>> + port->rx_ring.buf = data;
>> +
>> ret = uart_add_one_port(&atmel_uart, &port->uart);
>>
Is the kmalloc correct?
maybe:
data = kmalloc(ATMEL_SERIAL_RINGSIZE * sizeof(struct atmel_uart_char),
GFP_KERNEL);
>> if (ret)
>> goto err_add_port;
>> @@ -1013,6 +1142,9 @@ static int __devinit atmel_serial_probe(struct platform_device *pdev)
>> return 0;
>>
>> err_add_port:
>> + kfree(port->rx_ring.buf);
>> + port->rx_ring.buf = NULL;
>> +err_alloc_ring:
>> if (!atmel_is_console_port(&port->uart)) {
>> clk_disable(port->clk);
>> clk_put(port->clk);
>> @@ -1033,6 +1165,9 @@ static int __devexit atmel_serial_remove(struct platform_device *pdev)
>>
>> ret = uart_remove_one_port(&atmel_uart, port);
>>
>> + tasklet_kill(&atmel_port->tasklet);
>> + kfree(atmel_port->rx_ring.buf);
>> +
>>
Why the tasklet_kill is not done in atmel_shutdown?
Regards
Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-29 23:12 ` [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler michael
@ 2008-01-30 9:41 ` Haavard Skinnemoen
2008-01-30 10:21 ` Remy Bohmer
2008-01-30 10:29 ` michael
0 siblings, 2 replies; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-30 9:41 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Wed, 30 Jan 2008 00:12:23 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> I'm testing this patch on an at91sam9260 on 2.6.24-rt. I'm using this
> patch with the tclib support for hrtimer and the clocksource pit_clk.
> These are the results:
>
> - Voluntary Kernel Preemption the system (crash)
> - Preemptible Kernel (crash)
Ouch. I'm assuming this is with DMA disabled?
> /*
> * Drop the lock here since it might end up calling
> * uart_start(), which takes the lock.
> spin_unlock(&port->lock);
> */
> tty_flip_buffer_push(port->info->tty);
> /*
> spin_lock(&port->lock);
> */
> The same code with this comments out runs
Now, _that_ is strange. I can't see anything that needs protection
across that call; in fact, I think we can lock a lot less than what we
currently do.
> Complete Preemption (Real-Time) ok but the serials is just unusable due
> to too many overruns (just using lrz)
Is it worse than before? IIRC Remy mentioned something about
IRQF_NODELAY being the reason for moving all this code to softirq
context in the first place; does the interrupt handler run in hardirq
context?
> The system is stable and doesn't crash reverting this patch.
> I don't test with the thread hardirqs active.
Ok.
> >> + ret = -ENOMEM;
> >> + data = kmalloc(ATMEL_SERIAL_RINGSIZE, GFP_KERNEL);
> >> + if (!data)
> >> + goto err_alloc_ring;
> >> + port->rx_ring.buf = data;
> >> +
> >> ret = uart_add_one_port(&atmel_uart, &port->uart);
> >>
> Is the kmalloc correct?
> maybe:
> data = kmalloc(ATMEL_SERIAL_RINGSIZE * sizeof(struct atmel_uart_char),
> GFP_KERNEL);
I think you're right. Can you change it and see if it helps?
I guess I didn't test it thoroughly enough with DMA
disabled...slub_debug ought to catch such things, but not until we
receive enough data to actually overflow the buffer.
> >> @@ -1033,6 +1165,9 @@ static int __devexit atmel_serial_remove(struct platform_device *pdev)
> >>
> >> ret = uart_remove_one_port(&atmel_uart, port);
> >>
> >> + tasklet_kill(&atmel_port->tasklet);
> >> + kfree(atmel_port->rx_ring.buf);
> >> +
> >>
> Why the tasklet_kill is not done in atmel_shutdown?
Why should it be? If it should, we must move the call to tasklet_init
into atmel_startup too, and I don't really see the point.
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 9:41 ` Haavard Skinnemoen
@ 2008-01-30 10:21 ` Remy Bohmer
2008-01-30 10:34 ` Haavard Skinnemoen
2008-01-30 10:29 ` michael
1 sibling, 1 reply; 28+ messages in thread
From: Remy Bohmer @ 2008-01-30 10:21 UTC (permalink / raw)
To: Haavard Skinnemoen, michael
Cc: fabio, Andrew Victor, Chip Coldwell, Marc Pignat, David Brownell,
linux-kernel, Alan Cox
[-- Attachment #1: Type: text/plain, Size: 4705 bytes --]
Hello Haavard (and Michael),
First I want to mention that I did not found the time yet to test your
latest integration of these atmel_serial patches, and I only know that
your set of the end of December works fine. I do not know the changes
you made since posting that set and your latest patch-set.
But let me clear some things up:
The original patchset I posted, existed of a set of patches suitable
for the mainline kernel, plus an additional patch for Preempt-RT only.
So, the splitup of the interrupt handler was also needed for
Preempt-RT, but it was not the only patch that was needed on
preempt-rt.
Now looking at this problem:
> > * Drop the lock here since it might end up calling
> > * uart_start(), which takes the lock.
> > spin_unlock(&port->lock);
> > */
> > tty_flip_buffer_push(port->info->tty);
> > /*
> > spin_lock(&port->lock);
> > */
> > The same code with this comments out runs
I expect the UART generating the problem is the DBGU port. The DBGU
shares its interrupt line with the timer interrupt with the IRQF_TIMER
flag set, and thus the DBGU interrupt handler is running in
IRQF_NODELAY context. Within this context it is forbidden to lock a
normal spinlock, because a normal spinlock is converted to a mutex on
Preempt-RT; a mutex can sleep which is forbidden in interrupt context.
So, to get around this problem, this lock spinlock has to be of the
raw_spinlock_t type. The raw_spinlock_t is the normal mainline-kernel
spinlock, and as such it is not converted to a mutex, and will
therefor never sleep.
Attached a patch that changes this spinlock type. I used it in my
patchset, but your updates of December last year do not need this
patch anymore, so apparantly you changed something that has a
regression on Preempt-RT...
> > Complete Preemption (Real-Time) ok but the serials is just unusable due
> > to too many overruns (just using lrz)
This problem is not there in the December-set either. It works like a charm...
I believe I have to look at the latest set of patches, and try to find
any regressions. Do you have a location somewhere where I can download
the latest versions? Or do I need to dig through LKML to find the
latest... ;-)
Kind Regards,
Remy
2008/1/30, Haavard Skinnemoen <hskinnemoen@atmel.com>:
> On Wed, 30 Jan 2008 00:12:23 +0100
> michael <trimarchi@gandalf.sssup.it> wrote:
>
> > I'm testing this patch on an at91sam9260 on 2.6.24-rt. I'm using this
> > patch with the tclib support for hrtimer and the clocksource pit_clk.
> > These are the results:
> >
> > - Voluntary Kernel Preemption the system (crash)
> > - Preemptible Kernel (crash)
>
> Ouch. I'm assuming this is with DMA disabled?
>
> > /*
> > * Drop the lock here since it might end up calling
> > * uart_start(), which takes the lock.
> > spin_unlock(&port->lock);
> > */
> > tty_flip_buffer_push(port->info->tty);
> > /*
> > spin_lock(&port->lock);
> > */
> > The same code with this comments out runs
>
> Now, _that_ is strange. I can't see anything that needs protection
> across that call; in fact, I think we can lock a lot less than what we
> currently do.
>
> > Complete Preemption (Real-Time) ok but the serials is just unusable due
> > to too many overruns (just using lrz)
>
> Is it worse than before? IIRC Remy mentioned something about
> IRQF_NODELAY being the reason for moving all this code to softirq
> context in the first place; does the interrupt handler run in hardirq
> context?
>
> > The system is stable and doesn't crash reverting this patch.
> > I don't test with the thread hardirqs active.
>
> Ok.
>
> > >> + ret = -ENOMEM;
> > >> + data = kmalloc(ATMEL_SERIAL_RINGSIZE, GFP_KERNEL);
> > >> + if (!data)
> > >> + goto err_alloc_ring;
> > >> + port->rx_ring.buf = data;
> > >> +
> > >> ret = uart_add_one_port(&atmel_uart, &port->uart);
> > >>
> > Is the kmalloc correct?
> > maybe:
> > data = kmalloc(ATMEL_SERIAL_RINGSIZE * sizeof(struct atmel_uart_char),
> > GFP_KERNEL);
>
> I think you're right. Can you change it and see if it helps?
>
> I guess I didn't test it thoroughly enough with DMA
> disabled...slub_debug ought to catch such things, but not until we
> receive enough data to actually overflow the buffer.
>
> > >> @@ -1033,6 +1165,9 @@ static int __devexit atmel_serial_remove(struct platform_device *pdev)
> > >>
> > >> ret = uart_remove_one_port(&atmel_uart, port);
> > >>
> > >> + tasklet_kill(&atmel_port->tasklet);
> > >> + kfree(atmel_port->rx_ring.buf);
> > >> +
> > >>
> > Why the tasklet_kill is not done in atmel_shutdown?
>
> Why should it be? If it should, we must move the call to tasklet_init
> into atmel_startup too, and I don't really see the point.
>
> Haavard
>
[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #2: atmel_serial_irqf_nodelay.patch --]
[-- Type: text/x-patch; name=atmel_serial_irqf_nodelay.patch, Size: 1134 bytes --]
On PREEMPT-RT we may not block on a normal spinlock in IRQ/IRQF_NODELAY-context.
This patch fixes this by making the lock a raw_spinlock_t for the
Atmel serial console.
This patch demands the following patches to be installed first:
* atmel_serial_cleanup.patch
* atmel_serial_irq_splitup.patch
Signed-off-by: Remy Bohmer <linux@bohmer.net>
---
include/linux/serial_core.h | 7 ++++++-
1 file changed, 6 insertions(+), 1 deletion(-)
Index: linux-2.6.23/include/linux/serial_core.h
===================================================================
--- linux-2.6.23.orig/include/linux/serial_core.h 2007-12-13 13:31:27.000000000 +0100
+++ linux-2.6.23/include/linux/serial_core.h 2007-12-13 16:32:22.000000000 +0100
@@ -226,7 +226,12 @@ struct uart_icount {
typedef unsigned int __bitwise__ upf_t;
struct uart_port {
- spinlock_t lock; /* port lock */
+
+#ifdef CONFIG_SERIAL_ATMEL
+ raw_spinlock_t lock; /* port lock */
+#else
+ spinlock_t lock; /* port lock */
+#endif
unsigned int iobase; /* in/out[bwl] */
unsigned char __iomem *membase; /* read/write[bwl] */
unsigned int irq; /* irq number */
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 9:41 ` Haavard Skinnemoen
2008-01-30 10:21 ` Remy Bohmer
@ 2008-01-30 10:29 ` michael
2008-01-30 12:36 ` Haavard Skinnemoen
1 sibling, 1 reply; 28+ messages in thread
From: michael @ 2008-01-30 10:29 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hi
> On Wed, 30 Jan 2008 00:12:23 +0100
> michael <trimarchi@gandalf.sssup.it> wrote:
>
>
>> - Voluntary Kernel Preemption the system (crash)
>> - Preemptible Kernel (crash)
>>
>
> Ouch. I'm assuming this is with DMA disabled?
>
>
Yes, is with DMA disabled
>> /*
>> * Drop the lock here since it might end up calling
>> * uart_start(), which takes the lock.
>> spin_unlock(&port->lock);
>> */
>> tty_flip_buffer_push(port->info->tty);
>> /*
>> spin_lock(&port->lock);
>> */
>> The same code with this comments out runs
>>
>
> Now, _that_ is strange. I can't see anything that needs protection
> across that call; in fact, I think we can lock a lot less than what we
> currently do.
>
>
I explain it bad:
- with spin_lock the system seems, there is no problem with Valuntary
Preeption and
Preemptible Kernel
- with full preemption it runs but the serial line can't be used for
receiving at
high bit rate (using lrz)
>> Complete Preemption (Real-Time) ok but the serials is just unusable due
>> to too many overruns (just using lrz)
>>
>
> Is it worse than before? IIRC Remy mentioned something about
> IRQF_NODELAY being the reason for moving all this code to softirq
> context in the first place; does the interrupt handler run in hardirq
> context?
>
>
In the complete preemption yes.
>> The system is stable and doesn't crash reverting this patch.
>> I don't test with the thread hardirqs active.
>>
>
> Ok.
>
>
>> Is the kmalloc correct?
>> maybe:
>> data = kmalloc(ATMEL_SERIAL_RINGSIZE * sizeof(struct atmel_uart_char),
>> GFP_KERNEL);
>>
>
> I think you're right. Can you change it and see if it helps?
>
>
I just change it because I have corruption on receiving buffer. All
my test are done with this fix
> I guess I didn't test it thoroughly enough with DMA
> disabled...slub_debug ought to catch such things, but not until we
> receive enough data to actually overflow the buffer.
>
>
I just test it I don't have
buffer overflow.
I protect with a spinlock the access to the register when we sending
from the tasklet. It is correct?
>
> Why should it be? If it should, we must move the call to tasklet_init
> into atmel_startup too, and I don't really see the point.
>
>
Ok
Regards Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 10:21 ` Remy Bohmer
@ 2008-01-30 10:34 ` Haavard Skinnemoen
2008-01-30 11:05 ` Remy Bohmer
0 siblings, 1 reply; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-30 10:34 UTC (permalink / raw)
To: Remy Bohmer
Cc: michael, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Wed, 30 Jan 2008 11:21:49 +0100
"Remy Bohmer" <linux@bohmer.net> wrote:
> > > * Drop the lock here since it might end up calling
> > > * uart_start(), which takes the lock.
> > > spin_unlock(&port->lock);
> > > */
> > > tty_flip_buffer_push(port->info->tty);
> > > /*
> > > spin_lock(&port->lock);
> > > */
> > > The same code with this comments out runs
>
> I expect the UART generating the problem is the DBGU port. The DBGU
> shares its interrupt line with the timer interrupt with the IRQF_TIMER
> flag set, and thus the DBGU interrupt handler is running in
> IRQF_NODELAY context. Within this context it is forbidden to lock a
> normal spinlock, because a normal spinlock is converted to a mutex on
> Preempt-RT; a mutex can sleep which is forbidden in interrupt context.
> So, to get around this problem, this lock spinlock has to be of the
> raw_spinlock_t type. The raw_spinlock_t is the normal mainline-kernel
> spinlock, and as such it is not converted to a mutex, and will
> therefor never sleep.
>
> Attached a patch that changes this spinlock type. I used it in my
> patchset, but your updates of December last year do not need this
> patch anymore, so apparantly you changed something that has a
> regression on Preempt-RT...
The code above is from the tasklet, so I don't think that's the problem.
The interrupt handler doesn't take any locks.
Or are spinlocks not allowed in softirq context either?
> I believe I have to look at the latest set of patches, and try to find
> any regressions. Do you have a location somewhere where I can download
> the latest versions? Or do I need to dig through LKML to find the
> latest... ;-)
They are in -mm. You were Cc'ed I think...
Haavard
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 10:34 ` Haavard Skinnemoen
@ 2008-01-30 11:05 ` Remy Bohmer
2008-01-30 12:43 ` Haavard Skinnemoen
2008-01-30 15:25 ` michael
0 siblings, 2 replies; 28+ messages in thread
From: Remy Bohmer @ 2008-01-30 11:05 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: michael, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hello Haavard,
> The code above is from the tasklet, so I don't think that's the problem.
> The interrupt handler doesn't take any locks.
> Or are spinlocks not allowed in softirq context either?
They are allowed there...
So, it has to be something different.
> > I believe I have to look at the latest set of patches, and try to find
> > any regressions. Do you have a location somewhere where I can download
> > the latest versions? Or do I need to dig through LKML to find the
> > latest... ;-)
> They are in -mm. You were Cc'ed I think...
Yes, I saw them, but I did not know if there were any updates in the
mean time, because I had seen some discussions, which confused me a
bit about the current status of the complete set.
> - with full preemption it runs but the serial line can't be used for
> receiving at high bit rate (using lrz)
A few questions arise here to me:
* What serial port is used here? (DBGU, or something else)
* No DMA was used, was flow-control enabled? (cannot with DBGU)
* If some other UART, why not using DMA?
Notice that the DBGU has no flow control, and just a 1 byte FIFO (thus
no fifo at all).
At high speeds (e.g. >=115200) it is _likely_ that you will miss
characters, nothing can prevent that. DBGU should only be used at
lower speeds, or just as text console. 115200 is running fine here as
text-console.
I would not expect that the behaviour is worse than without the
patchset, because without it it does not work at all on Preempt-RT,
but also: there was done much more in interrupt context previously, so
the chance of buffer overruns was much more likely in the old
situation.
The real interrupt handler (doing the reading from the fifo) must be
as short as possible, to be able to keep up with the data flow.
A simple calculation: 115200bps results in approx. 11520 bytes per second.
This means that the interrupt handler must be capable of handling each
byte on DBGU within 87us. With a worst case interrupt latency of about
85us, and average between 2us and 54us (on Preempt-RT and AT91RM9200),
you can simply understand that this will not match, how good/fast the
interrupt handling will ever be.
So, I suggest to either use flow-control, or DMA for bulkdata... (thus not DBGU)
Kind Regards,
Remy
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 10:29 ` michael
@ 2008-01-30 12:36 ` Haavard Skinnemoen
2008-01-30 15:26 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-30 12:36 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Wed, 30 Jan 2008 11:29:59 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> > Now, _that_ is strange. I can't see anything that needs protection
> > across that call; in fact, I think we can lock a lot less than what we
> > currently do.
> >
> >
> I explain it bad:
> - with spin_lock the system seems, there is no problem with Valuntary
> Preeption and
> Preemptible Kernel
> - with full preemption it runs but the serial line can't be used for
> receiving at
> high bit rate (using lrz)
...but if you drop the spinlock across the call to
tty_flip_buffer_push, you get an Oops?
Could you post the Oops?
> >> Complete Preemption (Real-Time) ok but the serials is just unusable due
> >> to too many overruns (just using lrz)
> >>
> >
> > Is it worse than before? IIRC Remy mentioned something about
> > IRQF_NODELAY being the reason for moving all this code to softirq
> > context in the first place; does the interrupt handler run in hardirq
> > context?
> >
> >
> In the complete preemption yes.
Which question did you answer "yes" to? That it's worse than before or
that the interrupt handler runs in hardirq context (i.e. IRQF_NODELAY)?
> > I think you're right. Can you change it and see if it helps?
> >
> >
> I just change it because I have corruption on receiving buffer. All
> my test are done with this fix
Ok.
> > I guess I didn't test it thoroughly enough with DMA
> > disabled...slub_debug ought to catch such things, but not until we
> > receive enough data to actually overflow the buffer.
> >
> >
> I just test it I don't have
> buffer overflow.
No, I'd expect your allocation fix to take care of that. Or did you by
any chance test without the fix and with slub_debug enabled?
> I protect with a spinlock the access to the register when we sending
> from the tasklet. It is correct?
I have no idea. Could you post some more specifics about what you
modified, for example a diff?
Most of the tasklet is already protected by the spinlock, so you must
be careful to avoid any lock recursion.
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 11:05 ` Remy Bohmer
@ 2008-01-30 12:43 ` Haavard Skinnemoen
2008-01-30 15:25 ` michael
1 sibling, 0 replies; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-30 12:43 UTC (permalink / raw)
To: Remy Bohmer
Cc: michael, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Wed, 30 Jan 2008 12:05:40 +0100
"Remy Bohmer" <linux@bohmer.net> wrote:
> > > I believe I have to look at the latest set of patches, and try to find
> > > any regressions. Do you have a location somewhere where I can download
> > > the latest versions? Or do I need to dig through LKML to find the
> > > latest... ;-)
> > They are in -mm. You were Cc'ed I think...
>
> Yes, I saw them, but I did not know if there were any updates in the
> mean time, because I had seen some discussions, which confused me a
> bit about the current status of the complete set.
There's been one update, atmel_serial-add-dma-support-fix.patch, which
you were also Cc'ed on.
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 11:05 ` Remy Bohmer
2008-01-30 12:43 ` Haavard Skinnemoen
@ 2008-01-30 15:25 ` michael
1 sibling, 0 replies; 28+ messages in thread
From: michael @ 2008-01-30 15:25 UTC (permalink / raw)
To: Remy Bohmer
Cc: Haavard Skinnemoen, fabio, Andrew Victor, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hi,
> A few questions arise here to me:
> * What serial port is used here? (DBGU, or something else)
> * No DMA was used, was flow-control enabled? (cannot with DBGU)
> * If some other UART, why not using DMA?
>
>
DBGU, so no flow control
> Notice that the DBGU has no flow control, and just a 1 byte FIFO (thus
> no fifo at all).
> At high speeds (e.g. >=115200) it is _likely_ that you will miss
> characters, nothing can prevent that. DBGU should only be used at
> lower speeds, or just as text console. 115200 is running fine here as
> text-console.
>
Overrun are admitted using DBGU and UART1..n without flow control,
but with the old version of the driver I can send a file using lrz
and with the new and full preemption is impossible.
> I would not expect that the behaviour is worse than without the
> patchset, because without it it does not work at all on Preempt-RT,
> but also: there was done much more in interrupt context previously, so
> the chance of buffer overruns was much more likely in the old
> situation.
> The real interrupt handler (doing the reading from the fifo) must be
> as short as possible, to be able to keep up with the data flow.
>
> A simple calculation: 115200bps results in approx. 11520 bytes per second.
> This means that the interrupt handler must be capable of handling each
> byte on DBGU within 87us. With a worst case interrupt latency of about
> 85us, and average between 2us and 54us (on Preempt-RT and AT91RM9200),
> you can simply understand that this will not match, how good/fast the
> interrupt handling will ever be.
>
> So, I suggest to either use flow-control, or DMA for bulkdata... (thus not DBGU)
>
> Kind Regards,
>
> Remy
>
>
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 12:36 ` Haavard Skinnemoen
@ 2008-01-30 15:26 ` michael
2008-01-30 15:46 ` Haavard Skinnemoen
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-01-30 15:26 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hi,
> On Wed, 30 Jan 2008 11:29:59 +0100
> michael <trimarchi@gandalf.sssup.it> wrote:
>
>>> Now, _that_ is strange. I can't see anything that needs protection
>>> across that call; in fact, I think we can lock a lot less than what we
>>> currently do.
>>>
>>>
>>>
>> I explain it bad:
>> - with spin_lock the system seems, there is no problem with Valuntary
>> Preeption and
>> Preemptible Kernel
>> - with full preemption it runs but the serial line can't be used for
>> receiving at
>> high bit rate (using lrz)
>>
>
> ...but if you drop the spinlock across the call to
> tty_flip_buffer_push, you get an Oops?
>
> Could you post the Oops?
>
>
So this code
/*
* Drop the lock here since it might end up calling
* uart_start(), which takes the lock.
*/
spin_unlock(&port->lock);
tty_flip_buffer_push(port->info->tty);
spin_lock(&port->lock);
Works with:
CONFIG_PREEMPT_RT=y
CONFIG_PREEMPT=y
CONFIG_PREEMPT_SOFTIRQS=y
CONFIG_PREEMPT_HARDIRQS=y
CONFIG_PREEMPT_BKL=y
but crash with:
# CONFIG_PREEMPT_NONE is not set
CONFIG_PREEMPT_VOLUNTARY=y
# CONFIG_PREEMPT_DESKTOP is not set
# CONFIG_PREEMPT_RT is not set
CONFIG_PREEMPT_SOFTIRQS=y
# CONFIG_PREEMPT_HARDIRQS is not set
# CONFIG_PREEMPT_BKL is not set
CONFIG_CLASSIC_RCU=y
Seems to work in the last config if I comment the spin_lock &
spin_unlock call.
/*
* Drop the lock here since it might end up calling
* uart_start(), which takes the lock.
spin_unlock(&port->lock); */
tty_flip_buffer_push(port->info->tty);
/* spin_lock(&port->lock); */
It is not readable because I can't compile it with debugging information
(poor memory system)
>>>> Complete Preemption (Real-Time) ok but the serials is just unusable due
>>>> to too many overruns (just using lrz)
>>>>
>>>>
>>> Is it worse than before? IIRC Remy mentioned something about
>>> IRQF_NODELAY being the reason for moving all this code to softirq
>>> context in the first place; does the interrupt handler run in hardirq
>>> context?
>>>
>>>
>>>
>> In the complete preemption yes.
>>
>
> Which question did you answer "yes" to? That it's worse than before or
> that the interrupt handler runs in hardirq context (i.e. IRQF_NODELAY)?
>
>
The interrupt handler run in IRQF_NODELAY context.
>>> I think you're right. Can you change it and see if it helps?
>>>
>>>
>>>
>> I just change it because I have corruption on receiving buffer. All
>> my test are done with this fix
>>
>
> Ok.
>
>
>>> I guess I didn't test it thoroughly enough with DMA
>>> disabled...slub_debug ought to catch such things, but not until we
>>> receive enough data to actually overflow the buffer.
>>>
>>>
>>>
>> I just test it I don't have
>> buffer overflow.
>>
>
> No, I'd expect your allocation fix to take care of that. Or did you by
> any chance test without the fix and with slub_debug enabled?
>
>
I just meant that the buffer never fills up to its size.
>> I protect with a spinlock the access to the register when we sending
>> from the tasklet. It is correct?
>>
>
> I have no idea. Could you post some more specifics about what you
> modified, for example a diff?
>
>
...
/* The interrupt handler does not take the lock */
spin_lock_irqsave(&port->lock, flags);
atmel_tx_chars(port);
spin_unlock_irqrestore(&port->lock, flags);
spin_lock(&port->lock);
...
The atmel_tx_chars using the serial device registers like the interrupt
routine
and so I think that it is possible to have interference during send
operation.
> Most of the tasklet is already protected by the spinlock, so you must
> be careful to avoid any lock recursion.
>
> Haavard
>
>
Regards Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 15:26 ` michael
@ 2008-01-30 15:46 ` Haavard Skinnemoen
2008-01-31 1:53 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-30 15:46 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Wed, 30 Jan 2008 16:26:27 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> > I have no idea. Could you post some more specifics about what you
> > modified, for example a diff?
> >
> >
>
> ...
> /* The interrupt handler does not take the lock */
> spin_lock_irqsave(&port->lock, flags);
> atmel_tx_chars(port);
> spin_unlock_irqrestore(&port->lock, flags);
Sorry, this isn't going to work.
Please post a diff with the changes you did to the driver, and whatever
output you got when it crashed.
It's really difficult to help you when I don't know (a) what code
you're actually running, or (b) anything about the crash.
> The atmel_tx_chars using the serial device registers like the interrupt
> routine
> and so I think that it is possible to have interference during send
> operation.
No, it's only called from the tasklet, and the interrupt handler doesn't
touch the TX data register. There shouldn't be any need to disable
interrupts around the call to atmel_tx_chars(). In fact, this may very
well be the cause of the overruns you're seeing.
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-30 15:46 ` Haavard Skinnemoen
@ 2008-01-31 1:53 ` michael
2008-01-31 15:07 ` Haavard Skinnemoen
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-01-31 1:53 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hi,
Haavard Skinnemoen wrote:
> On Wed, 30 Jan 2008 16:26:27 +0100
> michael <trimarchi@gandalf.sssup.it> wrote:
>
>
>>> I have no idea. Could you post some more specifics about what you
>>> modified, for example a diff?
>>>
>>>
>>>
>> ...
>> /* The interrupt handler does not take the lock */
>> spin_lock_irqsave(&port->lock, flags);
>> atmel_tx_chars(port);
>> spin_unlock_irqrestore(&port->lock, flags);
>>
>
> Sorry, this isn't going to work.
>
> Please post a diff with the changes you did to the driver, and whatever
> output you got when it crashed.
>
> It's really difficult to help you when I don't know (a) what code
> you're actually running, or (b) anything about the crash.
>
>
Ok, but the problem is that I have some added code for using the uart with
smart card in iso mode, (is never called) and the patch is not so clean.
Now I return to the original patch without the spin_lock_irqsave and with
the fix of buffer allocation,and I don't see the crash anymore.
In full preemptive all works with threading hardirqs and sofirqs. I will
do other testing before posting again.
>> The atmel_tx_chars using the serial device registers like the interrupt
>> routine
>> and so I think that it is possible to have interference during send
>> operation.
>>
>
> No, it's only called from the tasklet, and the interrupt handler doesn't
> touch the TX data register. There shouldn't be any need to disable
> interrupts around the call to atmel_tx_chars(). In fact, this may very
> well be the cause of the overruns you're seeing.
>
> Haavard
>
>
The overrun still remain. An lrz receive session is impossible using
full preemption. I will try the dma patch too and test in iso mode for
smart card.
Thanks
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-31 1:53 ` michael
@ 2008-01-31 15:07 ` Haavard Skinnemoen
2008-01-31 19:36 ` Remy Bohmer
2008-02-04 19:01 ` michael
0 siblings, 2 replies; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-31 15:07 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Thu, 31 Jan 2008 02:53:50 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> The overrun still remain. An lrz receive session is impossible using
> full preemption. I will try the dma patch too and test in iso mode for
> smart card.
Hmm. Seems to work reasonably well on a non-rt kernel -- I get quite a
few overruns, but nothing more than the Zmodem protocol can handle.
It seems to be very sensitive to network traffic though...could it have
something to do with softirq scheduling? Could you try the patch below
and see if you can trigger the error message?
Haavard
diff --git a/drivers/serial/atmel_serial.c b/drivers/serial/atmel_serial.c
index 477950f..c61fcc3 100644
--- a/drivers/serial/atmel_serial.c
+++ b/drivers/serial/atmel_serial.c
@@ -337,9 +337,12 @@ atmel_buffer_rx_char(struct uart_port *port, unsigned int status,
struct circ_buf *ring = &atmel_port->rx_ring;
struct atmel_uart_char *c;
- if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE))
+ if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE)) {
+ dev_err(port->dev, "RX ring buffer full, dropping data\n");
+
/* Buffer overflow, ignore char */
return;
+ }
c = &((struct atmel_uart_char *)ring->buf)[ring->head];
c->status = status;
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-31 15:07 ` Haavard Skinnemoen
@ 2008-01-31 19:36 ` Remy Bohmer
2008-02-04 12:39 ` Haavard Skinnemoen
2008-02-04 19:01 ` michael
1 sibling, 1 reply; 28+ messages in thread
From: Remy Bohmer @ 2008-01-31 19:36 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: michael, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hello Haavard,
> It seems to be very sensitive to network traffic though...could it have
> something to do with softirq scheduling? Could you try the patch below
> and see if you can trigger the error message?
Funny that you mention this.
The largest latencies I currently have on RT (and rm9200) occur when
using a telnet session or NFS filesystems, thus while using network.
The impact on hardware Interrupt latencies are limited (<85us), so the
interrupt handler should still be able to keep up the receive buffer,
but context switches between threads can stall for a longer time under
some conditions.
A long shot, but can it be that the ringbuffer overflows, and that
therefor characters are lost?
Kind Regards,
Remy
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-31 19:36 ` Remy Bohmer
@ 2008-02-04 12:39 ` Haavard Skinnemoen
2008-02-04 19:44 ` Remy Bohmer
0 siblings, 1 reply; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-02-04 12:39 UTC (permalink / raw)
To: Remy Bohmer
Cc: michael, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Thu, 31 Jan 2008 20:36:25 +0100
"Remy Bohmer" <linux@bohmer.net> wrote:
> A long shot, but can it be that the ringbuffer overflows, and that
> therefor characters are lost?
That's what I was thinking too. If this is indeed the cause, the
dev_err() added by the debug patch I posted should trigger and we may
consider boosting the priority of the tasklet (using
tasklet_hi_schedule.)
Other solutions may involve trying to figure out what exactly is
blocking the tasklet from running. I have a patch series for the macb
driver that optimizes the RX processing a bit, but it obviously won't
help at91rm9200 since it uses a different driver...
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-31 15:07 ` Haavard Skinnemoen
2008-01-31 19:36 ` Remy Bohmer
@ 2008-02-04 19:01 ` michael
2008-02-04 20:25 ` Haavard Skinnemoen
1 sibling, 1 reply; 28+ messages in thread
From: michael @ 2008-02-04 19:01 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hi
> Haavard
>
> diff --git a/drivers/serial/atmel_serial.c b/drivers/serial/atmel_serial.c
> index 477950f..c61fcc3 100644
> --- a/drivers/serial/atmel_serial.c
> +++ b/drivers/serial/atmel_serial.c
> @@ -337,9 +337,12 @@ atmel_buffer_rx_char(struct uart_port *port, unsigned int status,
> struct circ_buf *ring = &atmel_port->rx_ring;
> struct atmel_uart_char *c;
>
> - if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE))
> + if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE)) {
> + dev_err(port->dev, "RX ring buffer full, dropping data\n");
> +
> /* Buffer overflow, ignore char */
> return;
> + }
>
> c = &((struct atmel_uart_char *)ring->buf)[ring->head];
> c->status = status;
>
>
I have already tried that but I have never seen the buffer full. So
tomorrow I can do other
tests with the serial device. I think the the atmel_interrupt handler
must check the
pass_counter before return IRQ_HANDLED.
Regards Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-04 12:39 ` Haavard Skinnemoen
@ 2008-02-04 19:44 ` Remy Bohmer
2008-02-06 12:19 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Remy Bohmer @ 2008-02-04 19:44 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: michael, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hello Haavard,
> That's what I was thinking too. If this is indeed the cause, the
> dev_err() added by the debug patch I posted should trigger and we may
> consider boosting the priority of the tasklet (using
> tasklet_hi_schedule.)
Notice that we are talking about Preempt-RT here. Everything is
running in thread context, even tasklets, softirqs etc. They are _all_
preemptible, and if Michael has some RT-thread in the system that has
a higher priority than this tasklet or softirq, than the buffer will
eventually overflow.
I wonder also if Michael has set the RT-priorities correctly, on RT
_every_ softirq/irq thread starts by default on priority 50,
SCHED_FIFO. If they are still at 50, any other softirq/tasklet, or irq
can make this behavior worse.
Notice that the default 50 thingy rarely gives a decent behaving
system. (But any other default will also give problems anyway, so it
_has_ to be customized/tuned by the end user)
Kind Regards,
Remy
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-04 19:01 ` michael
@ 2008-02-04 20:25 ` Haavard Skinnemoen
2008-02-05 12:29 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-02-04 20:25 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Mon, 04 Feb 2008 20:01:26 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> I think the the atmel_interrupt handler
> must check the
> pass_counter before return IRQ_HANDLED.
I'm not sure if it helps in this particular case but yeah, since the
interrupt may be shared, it's definitely wrong to always return
IRQ_HANDLED.
Nice catch. Could you try the patch below?
Haavard
diff --git a/drivers/serial/atmel_serial.c b/drivers/serial/atmel_serial.c
index cb70cc5..f310a80 100644
--- a/drivers/serial/atmel_serial.c
+++ b/drivers/serial/atmel_serial.c
@@ -552,7 +552,7 @@ static irqreturn_t atmel_interrupt(int irq, void *dev_id)
atmel_handle_transmit(port, pending);
} while (pass_counter++ < ATMEL_ISR_PASS_LIMIT);
- return IRQ_HANDLED;
+ return pass_counter ? IRQ_HANDLED : IRQ_NONE;
}
/*
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-04 20:25 ` Haavard Skinnemoen
@ 2008-02-05 12:29 ` michael
2008-02-06 12:30 ` Haavard Skinnemoen
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-02-05 12:29 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
hi,
> diff --git a/drivers/serial/atmel_serial.c b/drivers/serial/atmel_serial.c
> index cb70cc5..f310a80 100644
> --- a/drivers/serial/atmel_serial.c
> +++ b/drivers/serial/atmel_serial.c
> @@ -552,7 +552,7 @@ static irqreturn_t atmel_interrupt(int irq, void *dev_id)
> atmel_handle_transmit(port, pending);
> } while (pass_counter++ < ATMEL_ISR_PASS_LIMIT);
>
> - return IRQ_HANDLED;
> + return pass_counter ? IRQ_HANDLED : IRQ_NONE;
> }
>
> /*
>
>
Just one question:
Receiving with hardware handshake works without PDC?
Regards Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-04 19:44 ` Remy Bohmer
@ 2008-02-06 12:19 ` michael
2008-02-06 19:41 ` Remy Bohmer
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-02-06 12:19 UTC (permalink / raw)
To: Remy Bohmer
Cc: Haavard Skinnemoen, fabio, Andrew Victor, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hi,
the serial driver works fine. The problem seems to be related to the
tclib, when I use
it as a clocksource. The numbers of overruns depends on the type of
files too.
It is possible?
Regards
Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-05 12:29 ` michael
@ 2008-02-06 12:30 ` Haavard Skinnemoen
2008-02-06 13:41 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-02-06 12:30 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Tue, 05 Feb 2008 13:29:35 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> Just one question:
> Receiving with hardware handshake works without PDC?
I don't know...I haven't tried. These patches shouldn't change anything
though.
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-06 12:30 ` Haavard Skinnemoen
@ 2008-02-06 13:41 ` michael
2008-02-06 15:22 ` Haavard Skinnemoen
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-02-06 13:41 UTC (permalink / raw)
To: Haavard Skinnemoen
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
Hi,
Haavard Skinnemoen wrote:
> On Tue, 05 Feb 2008 13:29:35 +0100
> michael <trimarchi@gandalf.sssup.it> wrote:
>
>
>> Just one question:
>> Receiving with hardware handshake works without PDC?
>>
>
> I don't know...I haven't tried. These patches shouldn't change anything
> though.
>
> Haavard
> --
> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
> Please read the FAQ at http://www.tux.org/lkml/
>
>
I refer to this part of documentation:
"The USART behavior when hardware handshaking is enabled is the same as
the behavior in
standard synchronous or asynchronous mode, except that the receiver
drives the RTS pin as
described below and the level on the CTS pin modifies the behavior of
the transmitter as
described below. Using this mode requires using the PDC channel for
reception. The transmitter
can handle hardware handshaking in any case."
Regards Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-06 13:41 ` michael
@ 2008-02-06 15:22 ` Haavard Skinnemoen
0 siblings, 0 replies; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-02-06 15:22 UTC (permalink / raw)
To: michael
Cc: Remy Bohmer, fabio, Andrew Victor, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox
On Wed, 06 Feb 2008 14:41:09 +0100
michael <trimarchi@gandalf.sssup.it> wrote:
> I refer to this part of documentation:
>
> "The USART behavior when hardware handshaking is enabled is the same as
> the behavior in
> standard synchronous or asynchronous mode, except that the receiver
> drives the RTS pin as
> described below and the level on the CTS pin modifies the behavior of
> the transmitter as
> described below. Using this mode requires using the PDC channel for
> reception. The transmitter
> can handle hardware handshaking in any case."
Oh. I guess the answer is no, then.
Haavard
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-06 12:19 ` michael
@ 2008-02-06 19:41 ` Remy Bohmer
2008-02-13 11:07 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Remy Bohmer @ 2008-02-06 19:41 UTC (permalink / raw)
To: michael
Cc: Haavard Skinnemoen, fabio, Andrew Victor, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hello Michael,
> the serial driver works fine. The problem seems to be related to the
> tclib, when I use it as a clocksource.
tclib as a clocksource should be no problem on Preempt-RT, do not use
it as clockevent device, it will not work properly on preempt-rt on
at91* yet, especially with the NO_HZ config.
> The numbers of overruns depends on the type of
> files too. It is possible?
I am still very curious to your (RT-) thread-prio layout. The problems
you mention can also be caused by a weird configuration of these
threads on preempt-rt.
Remy
2008/2/6, michael <trimarchi@gandalf.sssup.it>:
> Hi,
>
> the serial driver works fine. The problem seems to be related to the
> tclib, when I use
> it as a clocksource. The numbers of overruns depends on the type of
> files too.
> It is possible?
>
> Regards
> Michael
>
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
> Please read the FAQ at http://www.tux.org/lkml/
>
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-06 19:41 ` Remy Bohmer
@ 2008-02-13 11:07 ` michael
2008-02-13 20:13 ` Remy Bohmer
0 siblings, 1 reply; 28+ messages in thread
From: michael @ 2008-02-13 11:07 UTC (permalink / raw)
To: Remy Bohmer
Cc: Haavard Skinnemoen, fabio, Andrew Victor, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hi,
All works now for me with preempt-rt. The problem is using hrtimer.
I think that hrtimer are executed with interrupts disabled so, if
this happen when I must receive a char, i have an overrun. The only
solution was the dma support to serial device.
Regards Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-13 11:07 ` michael
@ 2008-02-13 20:13 ` Remy Bohmer
2008-02-14 7:30 ` michael
0 siblings, 1 reply; 28+ messages in thread
From: Remy Bohmer @ 2008-02-13 20:13 UTC (permalink / raw)
To: michael
Cc: Haavard Skinnemoen, fabio, Andrew Victor, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hello All,
> All works now for me with preempt-rt. The problem is using hrtimer.
> I think that hrtimer are executed with interrupts disabled so, if
> this happen when I must receive a char, i have an overrun.
No, they share the same interrupt line...
So, while the timer interrupt handler is running, the serial handler
has to wait until the timer interrupt handler has finished.
Notice that the HRT interrupt handler is quite heavy and takes a long
time to complete.
And, as I already mentioned, related to the 1 byte FIFO and a
interrupt latency of about 85us (without HRT), it is logical that you
can get an overrun at the higher serial speeds... (>=115200bps)
> The only solution was the dma support to serial device.
Or, use flow control?
Kind Regards,
Remy
^ permalink raw reply [flat|nested] 28+ messages in thread
* Re: [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-02-13 20:13 ` Remy Bohmer
@ 2008-02-14 7:30 ` michael
0 siblings, 0 replies; 28+ messages in thread
From: michael @ 2008-02-14 7:30 UTC (permalink / raw)
To: Remy Bohmer
Cc: Haavard Skinnemoen, fabio, Andrew Victor, Chip Coldwell,
Marc Pignat, David Brownell, linux-kernel, Alan Cox
Hi,
Remy Bohmer wrote:
> Hello All,
>
>
>> All works now for me with preempt-rt. The problem is using hrtimer.
>> I think that hrtimer are executed with interrupts disabled so, if
>> this happen when I must receive a char, i have an overrun.
>>
>
> No, they share the same interrupt line...
>
I think that the hrtimer use and other interrupt line. The
AT91SAM9260_ID_TC2.
> So, while the timer interrupt handler is running, the serial handler
> has to wait until the timer interrupt handler has finished.
> Notice that the HRT interrupt handler is quite heavy and takes a long
> time to complete.
>
The problem is the heavy of HRT interrupt handler of course.
> And, as I already mentioned, related to the 1 byte FIFO and a
> interrupt latency of about 85us (without HRT), it is logical that you
> can get an overrun at the higher serial speeds... (>=115200bps)
>
>
I don't have the same problem without the hrtimer, I suppose the the
timer latency
is not so heavy.
>> The only solution was the dma support to serial device.
>>
>
> Or, use flow control?
>
>
>
Yes :)
Michael
^ permalink raw reply [flat|nested] 28+ messages in thread
* [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler
2008-01-24 12:41 ` [PATCH -mm v4 5/9] atmel_serial: Fix bugs in probe() error path and remove() Haavard Skinnemoen
@ 2008-01-24 12:41 ` Haavard Skinnemoen
0 siblings, 0 replies; 28+ messages in thread
From: Haavard Skinnemoen @ 2008-01-24 12:41 UTC (permalink / raw)
To: Andrew Morton
Cc: Andrew Victor, Remy Bohmer, Chip Coldwell, Marc Pignat,
David Brownell, linux-kernel, Alan Cox, Haavard Skinnemoen
From: Remy Bohmer <linux@bohmer.net>
This patch splits up the interrupt handler of the serial port
into a interrupt top-half and a tasklet.
The goal is to get the interrupt top-half as short as possible to
minimize latencies on interrupts. But the old code also does some
calls in the interrupt handler that are not allowed on preempt-RT
in IRQF_NODELAY context. This handler is executed in this context
because of the interrupt sharing with the timer interrupt.
The timer interrupt on Preempt-RT runs in IRQF_NODELAY context.
The tasklet takes care of handling control status changes, pushing
incoming characters to the tty layer, handling break and other errors.
It also handles pushing TX data into the data register.
Reading the complete receive queue is still done in the top-half
because we never want to miss any incoming character.
[hskinnemoen@atmel.com: misc cleanups and simplifications]
Signed-off-by: Remy Bohmer <linux@bohmer.net>
Signed-off-by: Haavard Skinnemoen <hskinnemoen@atmel.com>
---
drivers/serial/atmel_serial.c | 245 +++++++++++++++++++++++++++++++---------
1 files changed, 190 insertions(+), 55 deletions(-)
diff --git a/drivers/serial/atmel_serial.c b/drivers/serial/atmel_serial.c
index 0e715f4..0e65e98 100644
--- a/drivers/serial/atmel_serial.c
+++ b/drivers/serial/atmel_serial.c
@@ -104,6 +104,13 @@
static int (*atmel_open_hook)(struct uart_port *);
static void (*atmel_close_hook)(struct uart_port *);
+struct atmel_uart_char {
+ u16 status;
+ u16 ch;
+};
+
+#define ATMEL_SERIAL_RINGSIZE 1024
+
/*
* We wrap our port structure around the generic uart_port.
*/
@@ -112,6 +119,12 @@ struct atmel_uart_port {
struct clk *clk; /* uart clock */
unsigned short suspended; /* is port suspended? */
int break_active; /* break being received */
+
+ struct tasklet_struct tasklet;
+ unsigned int irq_status;
+ unsigned int irq_status_prev;
+
+ struct circ_buf rx_ring;
};
static struct atmel_uart_port atmel_ports[ATMEL_MAX_UART];
@@ -241,22 +254,42 @@ static void atmel_break_ctl(struct uart_port *port, int break_state)
}
/*
+ * Stores the incoming character in the ring buffer
+ */
+static void
+atmel_buffer_rx_char(struct uart_port *port, unsigned int status,
+ unsigned int ch)
+{
+ struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
+ struct circ_buf *ring = &atmel_port->rx_ring;
+ struct atmel_uart_char *c;
+
+ if (!CIRC_SPACE(ring->head, ring->tail, ATMEL_SERIAL_RINGSIZE))
+ /* Buffer overflow, ignore char */
+ return;
+
+ c = &((struct atmel_uart_char *)ring->buf)[ring->head];
+ c->status = status;
+ c->ch = ch;
+
+ /* Make sure the character is stored before we update head. */
+ smp_wmb();
+
+ ring->head = (ring->head + 1) & (ATMEL_SERIAL_RINGSIZE - 1);
+}
+
+/*
* Characters received (called from interrupt handler)
*/
static void atmel_rx_chars(struct uart_port *port)
{
struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
- struct tty_struct *tty = port->info->tty;
- unsigned int status, ch, flg;
+ unsigned int status, ch;
status = UART_GET_CSR(port);
while (status & ATMEL_US_RXRDY) {
ch = UART_GET_CHAR(port);
- port->icount.rx++;
-
- flg = TTY_NORMAL;
-
/*
* note that the error handling code is
* out of the main execution path
@@ -264,17 +297,14 @@ static void atmel_rx_chars(struct uart_port *port)
if (unlikely(status & (ATMEL_US_PARE | ATMEL_US_FRAME
| ATMEL_US_OVRE | ATMEL_US_RXBRK)
|| atmel_port->break_active)) {
+
/* clear error */
UART_PUT_CR(port, ATMEL_US_RSTSTA);
+
if (status & ATMEL_US_RXBRK
&& !atmel_port->break_active) {
- /* ignore side-effect */
- status &= ~(ATMEL_US_PARE | ATMEL_US_FRAME);
- port->icount.brk++;
atmel_port->break_active = 1;
UART_PUT_IER(port, ATMEL_US_RXBRK);
- if (uart_handle_break(port))
- goto ignore_char;
} else {
/*
* This is either the end-of-break
@@ -287,52 +317,30 @@ static void atmel_rx_chars(struct uart_port *port)
status &= ~ATMEL_US_RXBRK;
atmel_port->break_active = 0;
}
- if (status & ATMEL_US_PARE)
- port->icount.parity++;
- if (status & ATMEL_US_FRAME)
- port->icount.frame++;
- if (status & ATMEL_US_OVRE)
- port->icount.overrun++;
-
- status &= port->read_status_mask;
-
- if (status & ATMEL_US_RXBRK)
- flg = TTY_BREAK;
- else if (status & ATMEL_US_PARE)
- flg = TTY_PARITY;
- else if (status & ATMEL_US_FRAME)
- flg = TTY_FRAME;
}
- if (uart_handle_sysrq_char(port, ch))
- goto ignore_char;
-
- uart_insert_char(port, status, ATMEL_US_OVRE, ch, flg);
-
-ignore_char:
+ atmel_buffer_rx_char(port, status, ch);
status = UART_GET_CSR(port);
}
- tty_flip_buffer_push(tty);
+ tasklet_schedule(&atmel_port->tasklet);
}
/*
- * Transmit characters (called from interrupt handler)
+ * Transmit characters (called from tasklet with TXRDY interrupt
+ * disabled)
*/
static void atmel_tx_chars(struct uart_port *port)
{
struct circ_buf *xmit = &port->info->xmit;
- if (port->x_char) {
+ if (port->x_char && UART_GET_CSR(port) & ATMEL_US_TXRDY) {
UART_PUT_CHAR(port, port->x_char);
port->icount.tx++;
port->x_char = 0;
- return;
}
- if (uart_circ_empty(xmit) || uart_tx_stopped(port)) {
- atmel_stop_tx(port);
+ if (uart_circ_empty(xmit) || uart_tx_stopped(port))
return;
- }
while (UART_GET_CSR(port) & ATMEL_US_TXRDY) {
UART_PUT_CHAR(port, xmit->buf[xmit->tail]);
@@ -345,8 +353,8 @@ static void atmel_tx_chars(struct uart_port *port)
if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
uart_write_wakeup(port);
- if (uart_circ_empty(xmit))
- atmel_stop_tx(port);
+ if (!uart_circ_empty(xmit))
+ UART_PUT_IER(port, ATMEL_US_TXRDY);
}
/*
@@ -372,14 +380,18 @@ atmel_handle_receive(struct uart_port *port, unsigned int pending)
}
/*
- * transmit interrupt handler.
+ * transmit interrupt handler. (Transmit is IRQF_NODELAY safe)
*/
static void
atmel_handle_transmit(struct uart_port *port, unsigned int pending)
{
+ struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
+
/* Interrupt transmit */
- if (pending & ATMEL_US_TXRDY)
- atmel_tx_chars(port);
+ if (pending & ATMEL_US_TXRDY) {
+ UART_PUT_IDR(port, ATMEL_US_TXRDY);
+ tasklet_schedule(&atmel_port->tasklet);
+ }
}
/*
@@ -389,18 +401,13 @@ static void
atmel_handle_status(struct uart_port *port, unsigned int pending,
unsigned int status)
{
- /* TODO: All reads to CSR will clear these interrupts! */
- if (pending & ATMEL_US_RIIC)
- port->icount.rng++;
- if (pending & ATMEL_US_DSRIC)
- port->icount.dsr++;
- if (pending & ATMEL_US_DCDIC)
- uart_handle_dcd_change(port, !(status & ATMEL_US_DCD));
- if (pending & ATMEL_US_CTSIC)
- uart_handle_cts_change(port, !(status & ATMEL_US_CTS));
+ struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
+
if (pending & (ATMEL_US_RIIC | ATMEL_US_DSRIC | ATMEL_US_DCDIC
- | ATMEL_US_CTSIC))
- wake_up_interruptible(&port->info->delta_msr_wait);
+ | ATMEL_US_CTSIC)) {
+ atmel_port->irq_status = status;
+ tasklet_schedule(&atmel_port->tasklet);
+ }
}
/*
@@ -427,6 +434,114 @@ static irqreturn_t atmel_interrupt(int irq, void *dev_id)
return IRQ_HANDLED;
}
+static void atmel_rx_from_ring(struct uart_port *port)
+{
+ struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
+ struct circ_buf *ring = &atmel_port->rx_ring;
+ unsigned int flg;
+ unsigned int status;
+
+ while (ring->head != ring->tail) {
+ struct atmel_uart_char c;
+
+ /* Make sure c is loaded after head. */
+ smp_rmb();
+
+ c = ((struct atmel_uart_char *)ring->buf)[ring->tail];
+
+ ring->tail = (ring->tail + 1) & (ATMEL_SERIAL_RINGSIZE - 1);
+
+ port->icount.rx++;
+ status = c.status;
+ flg = TTY_NORMAL;
+
+ /*
+ * note that the error handling code is
+ * out of the main execution path
+ */
+ if (unlikely(status & (ATMEL_US_PARE | ATMEL_US_FRAME
+ | ATMEL_US_OVRE | ATMEL_US_RXBRK))) {
+ if (status & ATMEL_US_RXBRK) {
+ /* ignore side-effect */
+ status &= ~(ATMEL_US_PARE | ATMEL_US_FRAME);
+
+ port->icount.brk++;
+ if (uart_handle_break(port))
+ continue;
+ }
+ if (status & ATMEL_US_PARE)
+ port->icount.parity++;
+ if (status & ATMEL_US_FRAME)
+ port->icount.frame++;
+ if (status & ATMEL_US_OVRE)
+ port->icount.overrun++;
+
+ status &= port->read_status_mask;
+
+ if (status & ATMEL_US_RXBRK)
+ flg = TTY_BREAK;
+ else if (status & ATMEL_US_PARE)
+ flg = TTY_PARITY;
+ else if (status & ATMEL_US_FRAME)
+ flg = TTY_FRAME;
+ }
+
+
+ if (uart_handle_sysrq_char(port, c.ch))
+ continue;
+
+ uart_insert_char(port, status, ATMEL_US_OVRE, c.ch, flg);
+ }
+
+ /*
+ * Drop the lock here since it might end up calling
+ * uart_start(), which takes the lock.
+ */
+ spin_unlock(&port->lock);
+ tty_flip_buffer_push(port->info->tty);
+ spin_lock(&port->lock);
+}
+
+/*
+ * tasklet handling tty stuff outside the interrupt handler.
+ */
+static void atmel_tasklet_func(unsigned long data)
+{
+ struct uart_port *port = (struct uart_port *)data;
+ struct atmel_uart_port *atmel_port = (struct atmel_uart_port *)port;
+ unsigned int status;
+ unsigned int status_change;
+
+ /* The interrupt handler does not take the lock */
+ spin_lock(&port->lock);
+
+ atmel_tx_chars(port);
+
+ status = atmel_port->irq_status;
+ status_change = status ^ atmel_port->irq_status_prev;
+
+ if (status_change & (ATMEL_US_RI | ATMEL_US_DSR
+ | ATMEL_US_DCD | ATMEL_US_CTS)) {
+ /* TODO: All reads to CSR will clear these interrupts! */
+ if (status_change & ATMEL_US_RI)
+ port->icount.rng++;
+ if (status_change & ATMEL_US_DSR)
+ port->icount.dsr++;
+ if (status_change & ATMEL_US_DCD)
+ uart_handle_dcd_change(port, !(status & ATMEL_US_DCD));
+ if (status_change & ATMEL_US_CTS)
+ uart_handle_cts_change(port, !(status & ATMEL_US_CTS));
+
+ wake_up_interruptible(&port->info->delta_msr_wait);
+
+ atmel_port->irq_status_prev = status;
+ }
+
+ atmel_rx_from_ring(port);
+
+ spin_unlock(&port->lock);
+}
+
/*
* Perform initialization and enable port for reception
*/
@@ -758,6 +873,11 @@ static void __devinit atmel_init_port(struct atmel_uart_port *atmel_port,
port->mapbase = pdev->resource[0].start;
port->irq = pdev->resource[1].start;
+ tasklet_init(&atmel_port->tasklet, atmel_tasklet_func,
+ (unsigned long)port);
+
+ memset(&atmel_port->rx_ring, 0, sizeof(atmel_port->rx_ring));
+
if (data->regs)
/* Already mapped by setup code */
port->membase = data->regs;
@@ -998,11 +1118,20 @@ static int atmel_serial_resume(struct platform_device *pdev)
static int __devinit atmel_serial_probe(struct platform_device *pdev)
{
struct atmel_uart_port *port;
+ void *data;
int ret;
+ BUILD_BUG_ON(!is_power_of_2(ATMEL_SERIAL_RINGSIZE));
+
port = &atmel_ports[pdev->id];
atmel_init_port(port, pdev);
+ ret = -ENOMEM;
+ data = kmalloc(ATMEL_SERIAL_RINGSIZE, GFP_KERNEL);
+ if (!data)
+ goto err_alloc_ring;
+ port->rx_ring.buf = data;
+
ret = uart_add_one_port(&atmel_uart, &port->uart);
if (ret)
goto err_add_port;
@@ -1013,6 +1142,9 @@ static int __devinit atmel_serial_probe(struct platform_device *pdev)
return 0;
err_add_port:
+ kfree(port->rx_ring.buf);
+ port->rx_ring.buf = NULL;
+err_alloc_ring:
if (!atmel_is_console_port(&port->uart)) {
clk_disable(port->clk);
clk_put(port->clk);
@@ -1033,6 +1165,9 @@ static int __devexit atmel_serial_remove(struct platform_device *pdev)
ret = uart_remove_one_port(&atmel_uart, port);
+ tasklet_kill(&atmel_port->tasklet);
+ kfree(atmel_port->rx_ring.buf);
+
/* "port" is allocated statically, so we shouldn't free it */
clk_disable(atmel_port->clk);
--
1.5.3.8
^ permalink raw reply [flat|nested] 28+ messages in thread
end of thread, other threads:[~2008-02-14 7:31 UTC | newest]
Thread overview: 28+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
[not found] <20080129224316.GA23155@gandalf.sssup.it>
2008-01-29 23:12 ` [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler michael
2008-01-30 9:41 ` Haavard Skinnemoen
2008-01-30 10:21 ` Remy Bohmer
2008-01-30 10:34 ` Haavard Skinnemoen
2008-01-30 11:05 ` Remy Bohmer
2008-01-30 12:43 ` Haavard Skinnemoen
2008-01-30 15:25 ` michael
2008-01-30 10:29 ` michael
2008-01-30 12:36 ` Haavard Skinnemoen
2008-01-30 15:26 ` michael
2008-01-30 15:46 ` Haavard Skinnemoen
2008-01-31 1:53 ` michael
2008-01-31 15:07 ` Haavard Skinnemoen
2008-01-31 19:36 ` Remy Bohmer
2008-02-04 12:39 ` Haavard Skinnemoen
2008-02-04 19:44 ` Remy Bohmer
2008-02-06 12:19 ` michael
2008-02-06 19:41 ` Remy Bohmer
2008-02-13 11:07 ` michael
2008-02-13 20:13 ` Remy Bohmer
2008-02-14 7:30 ` michael
2008-02-04 19:01 ` michael
2008-02-04 20:25 ` Haavard Skinnemoen
2008-02-05 12:29 ` michael
2008-02-06 12:30 ` Haavard Skinnemoen
2008-02-06 13:41 ` michael
2008-02-06 15:22 ` Haavard Skinnemoen
2008-01-24 12:41 [PATCH -mm v4 0/9] atmel_serial cleanups and improvements Haavard Skinnemoen
2008-01-24 12:41 ` [PATCH -mm v4 1/9] MAINTAINERS: Add myself as maintainer of the atmel_serial driver Haavard Skinnemoen
2008-01-24 12:41 ` [PATCH -mm v4 2/9] atmel_serial: Clean up the code Haavard Skinnemoen
2008-01-24 12:41 ` [PATCH -mm v4 3/9] atmel_serial: Use cpu_relax() when busy-waiting Haavard Skinnemoen
2008-01-24 12:41 ` [PATCH -mm v4 4/9] atmel_serial: Use existing console options only if BRG is running Haavard Skinnemoen
2008-01-24 12:41 ` [PATCH -mm v4 5/9] atmel_serial: Fix bugs in probe() error path and remove() Haavard Skinnemoen
2008-01-24 12:41 ` [PATCH -mm v4 6/9] atmel_serial: Split the interrupt handler Haavard Skinnemoen
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).