Maybe I'll have to create a launchpad software to show the point, but here's an attempt in words:
- A PWM signal is received in a TCCP pin
- The period of the signal is about 1ms, and it is non-stop
- Timer A0 is configured to capture the signal on BOTH EDGES. DMAEventSet is properly configured to CAPEVENT
TimerConfigure(TIMER0_BASE,(TIMER_CFG_SPLIT_PAIR | TIMER_CFG_A_CAP_TIME_UP)); //TimerPrescaleSet(TIMER0_BASE, TIMER_A, ENC_PRESCALER); TimerPrescaleSet(TIMER0_BASE, TIMER_A, prescale); TimerLoadSet(TIMER0_BASE, TIMER_A, 0); // Was 0xFFFF TimerControlEvent(TIMER0_BASE, TIMER_A, TIMER_EVENT_BOTH_EDGES); TimerDMAEventSet(TIMER0_BASE, TIMER_DMA_CAPEVENT_A); TimerIntEnable(TIMER0_BASE, TIMER_TIMA_DMA | TIMER_TIMA_TIMEOUT); IntPrioritySet(INT_TIMER0A, 0); // The highest possible interrupt IntEnable(INT_TIMER0A);
Note that there is an attempt to enable TIMEOUT as an interrupt source here - which does not happen - but that's still not the discussion right now, I'll get back to it.
When I want to measure the PWM, I configure the DMA channel:
uDMAChannelAssign(UDMA_CH18_TIMER0A); uDMAChannelAttributeDisable(UDMA_CH18_TIMER0A, UDMA_ATTR_ALTSELECT | UDMA_ATTR_USEBURST | UDMA_ATTR_HIGH_PRIORITY | UDMA_ATTR_REQMASK); uDMAChannelControlSet((UDMA_CH18_TIMER0A|UDMA_PRI_SELECT),(UDMA_SIZE_32|UDMA_SRC_INC_NONE|UDMA_DST_INC_32|UDMA_ARB_1)); uDMAChannelTransferSet((UDMA_CH18_TIMER0A|UDMA_PRI_SELECT), UDMA_MODE_BASIC,(void *)(TIMER0_BASE + 0x048),encoderPulses,4); uDMAChannelEnable(UDMA_CH18_TIMER0A); // Enable this particular channel HWREG(TIMER0_BASE + TIMER_O_TAV) = 0; // Zeroes timer counter ROM_TimerEnable(TIMER0_BASE, TIMER_A); // Enables timer0 so that signal is captured
This is enough to get the internal gears running - and 4 edge moments should be transferred via DMA from a fixed TMER0 TAR register into a incrementing array encoderPulses.
It works as expected, but very often I get weird sequences, like:
2 350 396, 2 397 921, 54 350, 101 754
Clearly, the timer counter rolled over. That's (supposedly) impossible on this system, the signal remains running all the time, and the first 4 pulses should be recorded. If I add the timeout load to the smaller values (in this particular configuration, that's 37*2^16), I obtain perfect continuous values that reflect the actual signal. A proper sequence is something like:
1 084 102, 1 131 633, 1 212 768, 1 260 302
Still, the sequence above shows that the 4 recorded values were DMA-transferred kind of LATE! If the signal is ~1ms, the first event must have happened in less than 120000 clicks after TAV is loaded to zero and TimerEnable is set.
Is there some DMA mistake on my configuration or errata that I am missing?
Regards
Bruno