-
-
Notifications
You must be signed in to change notification settings - Fork 284
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
PWM signal from analogWrite does not stop when pin is set to different mode #169
Comments
@carlosperate thanks for reporting! Any suggestions on fixing this? |
The arduino core checks for this in digitalWrite. Looking at the core it would be easy to fix (maybe as little as 3 lines), i can have a look into it if you wish. |
@micooke that would be great! |
Hmm, its turning out to be a bit trickier than i expected. I am working on it though. I need to move a few static globals around so that they are accessible by wiring_digital.h. Ive implemented what i think should work, but it just doesnt at the moment. Ill keep you posted |
I think i fixed it locally, i will need someone to check on a nrf52832 though. I also added a fallback i found in the avr core that will use digitalWrite if all pwm channels are exhausted. Another note, there are only 3 PWM channels. |
@sandeepmistry - ill wait on #186 and rebase my repo before i send a PR for this. |
To summarise, analogWrite allocates a pwm channel and digitialWrite should deallocate a pwm channel if a write is registered to a pwm channel allocated to a (now) digital pin. The static pwm pin allocation / timer active references were visible to wiring_digital.c but their state is invalid so i needed to move the struct defines to a global include (wiring_private.h) that is visible to both wiring_digital.c and wiring_analog_.c, then instantiate a struct in wiring_analog_.c which is referenced via extern in wiring_digital.c. While i was modifying these 4 files i semi-standardised wiring_analog_*.c to use the same structs, and (as mentioned earlier) added a fallback i found in the avr core that will use digitalWrite if all pwm channels are exhausted. The last thing to explain, the NRF51 could use timer2 to allocate more pwm pins (Timer1 is used by the softcore). I didnt explore this but it is the reason that the irqNumber variable was added to the PWMStatus struct (allocation for future expansion). struct PWMStatus {
int8_t numActive;
int8_t irqNumber;
}; |
Needs testing against the nRF52 - Im sorry but i dont have one of these to test against. |
wiring_analog_* : fallback to digitalWrite if no available PWM channel (copied from AVR core) wiring_analog_nRF52.c : convert pwmChannelPins & pwmChannelSequence -> pwmContext to semi-standardise pwm pin allocation and pwm status between nRF51 and nRF52 wiring_private.h : Move pwm structures defines out of wiring_analog_* into wiring_private and make the instantiation of these structure externs instead of statics wiring_digital.c : disable the appropriate pwm timer when a digitalWrite is sent to an allocated (from analogWrite) pwm pin, and free up the pwm channel for re-allocation
Since I need a very specific and stable PWM frequency without interrupt jitter, I implemented an alternative version which works entirely in hardware, based on the PPI and GPIOTE peripherals. I think that would also solve the issue with the PWM channels on the side. If there's interest, I'll see if I can prepare a PR. |
@micooke didn't see you already have a fix ready. Will this be merged at some point? |
@floe - yeah sorry, i should have made reference to PR#195 instead of the last commit. If you have any comments on my implementation please comment on the PR as the merge seems to have stalled. And just to confirm here, the PR works for both the NRF51 and NRF52 😊 |
@floe this seems interesting - so please prepare a PR when you get a chance. Is this only slightly related to PR #195? |
It will probably conflict with #195 as it would be a rewrite of the |
wiring_analog_* : fallback to digitalWrite if no available PWM channel (copied from AVR core) wiring_analog_nRF52.c : convert pwmChannelPins & pwmChannelSequence -> pwmContext to semi-standardise pwm pin allocation and pwm status between nRF51 and nRF52 wiring_private.h : Move pwm structures defines out of wiring_analog_* into wiring_private and make the instantiation of these structure externs instead of statics wiring_digital.c : disable the appropriate pwm timer when a digitalWrite is sent to an allocated (from analogWrite) pwm pin, and free up the pwm channel for re-allocation
Looks the
analogWrite()
"pwm" signal on the nRF51 (now PWM hardware, so uses a timer instead) does not stop operation even afterpinMode()
anddigitalWrite()
has been applied to the same pin.Simple sketch for testing:
Expected behaviour:
Recorded behaviour:
The text was updated successfully, but these errors were encountered: