Skip to content
Snippets Groups Projects
Commit a8092c02 authored by Haavard Skinnemoen's avatar Haavard Skinnemoen
Browse files

avr32: Fix theoretical race in udelay()


If the specified delay is very short, the cycle counter may go past the
"end" time we are waiting for before we get around to reading it.

Fix it by checking the different between the cycle count "now" and the
cycle count at the beginning. This will work as long as the delay
measured in number of cycles is below 2^31.

Signed-off-by: default avatarHaavard Skinnemoen <haavard.skinnemoen@atmel.com>
parent 48ea623e
No related branches found
No related tags found
No related merge requests found
...@@ -98,18 +98,16 @@ void set_timer(unsigned long t) ...@@ -98,18 +98,16 @@ void set_timer(unsigned long t)
*/ */
void udelay(unsigned long usec) void udelay(unsigned long usec)
{ {
unsigned long now, end; unsigned long cycles;
unsigned long base;
unsigned long now;
now = sysreg_read(COUNT); base = sysreg_read(COUNT);
cycles = ((usec * (get_tbclk() / 10000)) + 50) / 100;
end = ((usec * (get_tbclk() / 10000)) + 50) / 100; do {
end += now;
while (now > end)
now = sysreg_read(COUNT);
while (now < end)
now = sysreg_read(COUNT); now = sysreg_read(COUNT);
} while ((now - base) < cycles);
} }
static int set_interrupt_handler(unsigned int nr, void (*handler)(void), static int set_interrupt_handler(unsigned int nr, void (*handler)(void),
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment