diff options
author | Marti Bolivar <mbolivar@leaflabs.com> | 2011-07-20 13:48:30 -0400 |
---|---|---|
committer | Marti Bolivar <mbolivar@leaflabs.com> | 2011-07-20 14:02:30 -0400 |
commit | e65a05a9f556b96b9d061dc394c022ba21c1c7b4 (patch) | |
tree | ab24d8530d74fefaa543abd242418339fc488680 /libmaple | |
parent | 87e774db58f9e4432fd5f8db0358f0432047a31c (diff) | |
download | librambutan-e65a05a9f556b96b9d061dc394c022ba21c1c7b4.tar.gz librambutan-e65a05a9f556b96b9d061dc394c022ba21c1c7b4.zip |
timer.c: Optimize IRQ dispatch routines.
Remove dispatch_irq() and dispatch_cc_irqs().
For IRQs which handle exactly one timer interrupt, add new
dispatch_single_irq(). The mere fact that the IRQ has been called
suffices to prove that its interrupt enable bit (in TIMx_DIER) and
interrupt flag (in TIMx_SR) are set. These facts are combined in
dispatch_single_irq(), which only needs to check if the timer_dev
handler is non-null before calling it and clearing the SR flag.
For IRQs which serve multiple timer interrupts, replace the
composition of calls to dispatch_irq() and dispatch_cc_irqs() with
individualized routines. These eliminate unnecessary timer register
reads/writes, and, in the case of capture/compare interrupts, have a
loop unrolling performed.
Diffstat (limited to 'libmaple')
-rw-r--r-- | libmaple/timer.c | 125 |
1 files changed, 66 insertions, 59 deletions
diff --git a/libmaple/timer.c b/libmaple/timer.c index 377bab7..951baf4 100644 --- a/libmaple/timer.c +++ b/libmaple/timer.c @@ -309,85 +309,92 @@ void __irq_tim8_cc(void) { } #endif -static inline void dispatch_irq(timer_dev *dev, - timer_interrupt_id iid, - uint32 irq_mask); -static inline void dispatch_cc_irqs(timer_dev *dev); +/* Note: the following dispatch routines make use of the fact that + * DIER interrupt enable bits and SR interrupt flags have common bit + * positions. Thus, ANDing DIER and SR lets us check if an interrupt + * is enabled and if it has occurred simultaneously. + */ + +/* A special-case dispatch routine for single-interrupt NVIC lines. + * This function assumes that the interrupt corresponding to `iid' has + * in fact occurred (i.e., it doesn't check DIER & SR). */ +static inline void dispatch_single_irq(timer_dev *dev, + timer_interrupt_id iid, + uint32 irq_mask) { + timer_bas_reg_map *regs = (dev->regs).bas; + void (*handler)(void) = dev->handlers[iid]; + if (handler) { + handler(); + regs->SR &= ~irq_mask; + } +} + +/* For dispatch routines which service multiple interrupts. */ +#define handle_irq(dier_sr, irq_mask, handlers, iid, handled_irq) do { \ + if ((dier_sr) & (irq_mask)) { \ + void (*__handler)(void) = (handlers)[iid]; \ + if (__handler) { \ + __handler(); \ + handled_irq |= (irq_mask); \ + } \ + } \ + } while (0) static inline void dispatch_adv_brk(timer_dev *dev) { - dispatch_irq(dev, TIMER_BREAK_INTERRUPT, TIMER_SR_BIF); + dispatch_single_irq(dev, TIMER_BREAK_INTERRUPT, TIMER_SR_BIF); } static inline void dispatch_adv_up(timer_dev *dev) { - dispatch_irq(dev, TIMER_UPDATE_INTERRUPT, TIMER_SR_UIF); + dispatch_single_irq(dev, TIMER_UPDATE_INTERRUPT, TIMER_SR_UIF); } static inline void dispatch_adv_trg_com(timer_dev *dev) { - dispatch_irq(dev, TIMER_TRG_INTERRUPT, TIMER_SR_TIF); - dispatch_irq(dev, TIMER_COM_INTERRUPT, TIMER_SR_COMIF); -} + timer_adv_reg_map *regs = (dev->regs).adv; + void (**hs)(void) = dev->handlers; + uint32 dsr = regs->DIER & regs->SR; + uint32 handled = 0; /* Logical OR of SR interrupt flags we end up + * handling. We clear these. User handlers + * must clear overcapture flags, to avoid + * wasting time in output mode. */ -static inline void dispatch_adv_cc(timer_dev *dev) { - dispatch_cc_irqs(dev); -} + handle_irq(dsr, TIMER_SR_TIF, hs, TIMER_TRG_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_COMIF, hs, TIMER_COM_INTERRUPT, handled); -static inline void dispatch_general(timer_dev *dev) { - dispatch_irq(dev, TIMER_TRG_INTERRUPT, TIMER_SR_TIF); - dispatch_cc_irqs(dev); - dispatch_irq(dev, TIMER_UPDATE_INTERRUPT, TIMER_SR_UIF); + regs->SR &= ~handled; } -static inline void dispatch_basic(timer_dev *dev) { - dispatch_irq(dev, TIMER_UPDATE_INTERRUPT, TIMER_SR_UIF); -} - -/* Note: The following dispatch routines play some tricks that depend - * on the positions of the relevant interrupt-related bits in TIMx_SR - * and TIMx_DIER. */ +static inline void dispatch_adv_cc(timer_dev *dev) { + timer_adv_reg_map *regs = (dev->regs).adv; + void (**hs)(void) = dev->handlers; + uint32 dsr = regs->DIER & regs->SR; + uint32 handled = 0; -/* TODO (optimization): since the callers all know the timer - * statically, it may be possible to speed these dispatch routines up - * by skipping the timer_dev and accessing timer_reg_map/handlers - * directly */ + handle_irq(dsr, TIMER_SR_CC4IF, hs, TIMER_CC4_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC3IF, hs, TIMER_CC3_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC2IF, hs, TIMER_CC2_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC1IF, hs, TIMER_CC1_INTERRUPT, handled); -static inline void dispatch_irq(timer_dev *dev, - timer_interrupt_id iid, - uint32 irq_mask) { - timer_bas_reg_map *regs = (dev->regs).bas; - void (*handler)(void) = dev->handlers[iid]; - if ((regs->DIER & irq_mask) && (regs->SR & irq_mask) && handler) { - handler(); - regs->SR &= ~irq_mask; - } + regs->SR &= ~handled; } -#define DIER_CCS (TIMER_DIER_CC4IE | \ - TIMER_DIER_CC3IE | \ - TIMER_DIER_CC2IE | \ - TIMER_DIER_CC1IE) - -static inline void dispatch_cc_irqs(timer_dev *dev) { +static inline void dispatch_general(timer_dev *dev) { timer_gen_reg_map *regs = (dev->regs).gen; - uint32 dier = regs->DIER; - uint32 sr; - uint32 sr_clear; /* cuts down on writes to a volatile register */ - uint32 b; + void (**hs)(void) = dev->handlers; + uint32 dsr = regs->DIER & regs->SR; + uint32 handled = 0; - if (!(dier & DIER_CCS)) { - return; - } + handle_irq(dsr, TIMER_SR_TIF, hs, TIMER_TRG_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC4IF, hs, TIMER_CC4_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC3IF, hs, TIMER_CC3_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC2IF, hs, TIMER_CC2_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_CC1IF, hs, TIMER_CC1_INTERRUPT, handled); + handle_irq(dsr, TIMER_SR_UIF, hs, TIMER_UPDATE_INTERRUPT, handled); - sr = regs->SR; - sr_clear = 0; - for (b = TIMER_SR_CC1IF_BIT; b <= TIMER_SR_CC4IF_BIT; b++) { - uint32 mask = BIT(b); - if ((dier & mask) && (sr & mask) && dev->handlers[b]) { - (dev->handlers[b])(); - sr_clear |= mask; - } - } + regs->SR &= ~handled; +} - regs->SR &= ~sr_clear; +static inline void dispatch_basic(timer_dev *dev) { + dispatch_single_irq(dev, TIMER_UPDATE_INTERRUPT, TIMER_SR_UIF); } /* |