This thread has been locked.
If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.
An old design is using TIMER_A PWM (OUTMOD_7) to generate low-bandwidth analog signal. However, TACCR registers are not double-buffered, so because we are generating a variable-duty-cycle PWM signal, care must be taken when changing the duty cycle (TACCR value).
The most suggestions to avoid occasional 100% cycles in the output are to do the TACCR update in the corresponding TACCR interrupt. However, because this is an old design, I wanted to keep the code changes as small a possible; especially I did not want to start adding new interrupt handlers.
I am wondering if OUTMOD_0 could be used to manually force the output signal low after the TACCR value is updated and the TAR value is tested to be larger than the new TACCR. See the code example below:
monitor void set_pwm(unsigned int uValue)
{
#define SAFE_MARGIN 11 // Number of timer cycles used for the comparison and execution of the if block (check from compiled code)
TA0CCR1 = uValue;
// Ensure that output is set low even if the value is decreased skipping over the current timer value
// Do not touch the output if it is close to the wrap-around (avoid too late change)
if( TA0R >= uValue && TA0R < 0x7FFF /* fixed TA0CCR0 value */ - SAFE_MARGIN ) {
TA0CCTL1 = (CM_0 | CCIS_2 | OUTMOD_0); // Force output to zero
TA0CCTL1 = (CM_0 | CCIS_2 | OUTMOD_7); // Back to PWM mode
}
}
Description:
Could this simple solution avoid the extra 100% cycles without any interrupts or waiting loops? In my quick test there are no bumps the filtered analog output signal anymore.
With this, you're forcing an additional output reset event. So yes, it will prevent an otherwise overlong high cycle. However, you might just encounter the critical point while you are doing so, getting an overlong low cycle instead. (less likely/less critical than the other version).Panu Tuominen said:Could this simple solution avoid the extra 100% cycles without any interrupts or waiting loops?
Also, SAFE_MARGIN needs to be recalculated each time you change the tiemer tick or the PWM frequency etc.
Not so complicated, just the count the maximum number of TIMER_A counts possible during execution of the if-block (including the comparison itself). If the timer is running at MCLK frequency (usually is, because usually we like to have as high PWM frequency as possible), it is just cycle counting. Making safe margin slightly too large does not affect the output a lot. There might be a small disruption in the PWM: we get a 100% cycle if the counter was close (closer than SAFE_MARGIN) to the wrap-around but it is possible only if the duty cycle setting was previously 99.9% (or larger), so the effect of the extra 100% cycle will be negligible in the filtered analog signal.The SAFE_MARGIN may help, but the logic to calculate it may be complex and, honestly, I don't want to twist my brain that much after a day of work :)
Yes, that's true. And there is no way to make the calculation automatic, even compiler optimization changes may affect it, so we must be careful.Also, SAFE_MARGIN needs to be recalculated each time you change the tiemer tick or the PWM frequency etc.
In general, similar risks are present when using TACCR interrupt to update the TACCR value - you might still end to an extra 100% between a "99.9%" cycle followed by a "0.1%" cycle. So no method is fault-proof.
Actually my note was that the OUTMOD_0 can be used to latch a new value to the timer output - it is not mentioned explicitly in the documentation. It might have been possible that the OUT bit value would have been present on the output only while the OUTMOD is 0, and the previous value to reappear when OUTMOD_7 was restored. But luckily the TIMER_A circuit was not working such way.
You're right. Usually, you don't change the DC on every cycle. (well, I've seen some code that changed the DC hundreds of times during one single PWM period - no wonder it didn't work)Panu Tuominen said:we get a 100% cycle if the counter was close (closer than SAFE_MARGIN) to the wrap-around but it is possible only if the duty cycle setting was previously 99.9% (or larger), so the effect of the extra 100% cycle will be negligible in the filtered analog signal.
Indeed, you're right. Especially if using LPMs or a MCLK as timer clock. This is why I prefer using the TimerB if available. It offers better sync functionality for multiple PWMs and reduces this. However, a synced 'load on TAR->0' or 'load on external trigger' would be great.Panu Tuominen said:similar risks are present when using TACCR interrupt to update the TACCR value
You're right and it should. It's a good catch and surely worth to be added to the docs.Panu Tuominen said:Actually my note was that the OUTMOD_0 can be used to latch a new value to the timer output - it is not mentioned explicitly in the documentation.
**Attention** This is a public forum