sam3x8e/timer: Be careful of races in timer_set_next()

It's possible for sched_del_timer() to be called on a timer that fires
just after sched_del_timer disables irqs but before the next timer is
scheduled.  In this case be sure to clear the irq pending status flag
after scheduling the next timer so that a delayed irq doesn't cause
the wrong timer to be run.  For the same reason, make sure to check
the irq pending status flag at the start of the timer irq.

Also, as a safety check, make sure timer_set_next() isn't called from
within the timer irq dispatch loop.

Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
Kevin O'Connor 2017-02-14 22:15:51 -05:00
parent bdfdf7ef55
commit fec12030a9
1 changed files with 16 additions and 4 deletions

View File

@ -30,9 +30,10 @@ timer_from_us(uint32_t us)
void __visible void __visible
TC0_Handler(void) TC0_Handler(void)
{ {
TC0->TC_CHANNEL[0].TC_SR; // clear irq pending
irq_disable(); irq_disable();
sched_timer_kick(); uint32_t status = TC0->TC_CHANNEL[0].TC_SR; // read to clear irq pending
if (likely(status & TC_SR_CPAS))
sched_timer_kick();
irq_enable(); irq_enable();
} }
@ -42,6 +43,13 @@ timer_set(uint32_t value)
TC0->TC_CHANNEL[0].TC_RA = value; TC0->TC_CHANNEL[0].TC_RA = value;
} }
static void
timer_set_clear(uint32_t value)
{
TC0->TC_CHANNEL[0].TC_RA = value;
TC0->TC_CHANNEL[0].TC_SR; // read to clear irq pending
}
static void static void
timer_init(void) timer_init(void)
{ {
@ -83,12 +91,16 @@ uint8_t
timer_set_next(uint32_t next) timer_set_next(uint32_t next)
{ {
uint32_t cur = timer_read_time(); uint32_t cur = timer_read_time();
if (sched_is_before(TC0->TC_CHANNEL[0].TC_RA, cur)
&& !(TC0->TC_CHANNEL[0].TC_SR & TC_SR_CPAS))
// Already processing timer irqs
try_shutdown("timer_set_next called during timer dispatch");
uint32_t mintime = cur + TIMER_MIN_TICKS; uint32_t mintime = cur + TIMER_MIN_TICKS;
if (sched_is_before(mintime, next)) { if (sched_is_before(mintime, next)) {
timer_set(next); timer_set_clear(next);
return 0; return 0;
} }
timer_set(mintime); timer_set_clear(mintime);
return 1; return 1;
} }