Skip to content
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

Open
carlosperate opened this issue Jun 19, 2017 · 13 comments

Comments

@carlosperate
Copy link
Collaborator

Looks the analogWrite() "pwm" signal on the nRF51 (now PWM hardware, so uses a timer instead) does not stop operation even after pinMode() and digitalWrite() has been applied to the same pin.

Simple sketch for testing:

const int analogOutPin = 9;

void setup() {
  // put your setup code here, to run once:
  
}

void loop() {
  // Increase PWM until maximun value, takes around a couple of seconds
  for (byte i = 0; i<255; i++) {
    analogWrite(analogOutPin, i);
    delay(10);
  }

  // Toggle the same pin high and low a couple of times
  delay(1000);
  pinMode(analogOutPin, OUTPUT);
  digitalWrite(analogOutPin, HIGH);
  delay(1000);
  digitalWrite(analogOutPin, LOW);
  delay(1000);
  digitalWrite(analogOutPin, HIGH);
  delay(1000);
  digitalWrite(analogOutPin, LOW);
  delay(1000);
}

Expected behaviour:

  • PWM signal increasing from minimum to maximum
  • Same pin signal toggled high and low a couple of times
  • loop iterates

Recorded behaviour:

  • PWM signal increasing from minimum to maximum
  • PWM signal stays at maximum value
  • loop iterates
@sandeepmistry
Copy link
Owner

@carlosperate thanks for reporting! Any suggestions on fixing this?

@micooke
Copy link
Contributor

micooke commented Sep 9, 2017

The arduino core checks for this in digitalWrite.
This adds a small overhead, but its probably best to be consistent.

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.

@sandeepmistry
Copy link
Owner

@micooke that would be great!

@micooke
Copy link
Contributor

micooke commented Sep 11, 2017

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

@micooke
Copy link
Contributor

micooke commented Sep 11, 2017

I think i fixed it locally, i will need someone to check on a nrf52832 though.
Linkage is a little 'special', so if you digitalWrite on all allocated pwm pins timer1 will stay on and interrupt but won't toggle any pins (not a biggie). Fixed (only needed for NRF51)

I also added a fallback i found in the avr core that will use digitalWrite if all pwm channels are exhausted.
@sandeepmistry - it looks like pwm channel 0 is unavailable. Do you know if this is used by another peripheral?

Another note, there are only 3 PWM channels.
For the NRF51 : TIMER1 is used, TIMER2 could also be used to expand the number of PWM channels
For the NRF52 : I havent looked, but its a totally different structure

@micooke
Copy link
Contributor

micooke commented Sep 12, 2017

@sandeepmistry - ill wait on #186 and rebase my repo before i send a PR for this.

@micooke
Copy link
Contributor

micooke commented Sep 18, 2017

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;
};

@micooke
Copy link
Contributor

micooke commented Sep 18, 2017

Needs testing against the nRF52 - Im sorry but i dont have one of these to test against.

micooke added a commit to micooke/arduino-nRF5 that referenced this issue Sep 29, 2017
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
@floe
Copy link

floe commented Sep 6, 2018

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.

@floe
Copy link

floe commented Sep 6, 2018

@micooke didn't see you already have a fix ready. Will this be merged at some point?

@micooke
Copy link
Contributor

micooke commented Sep 6, 2018

@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 😊

@sandeepmistry
Copy link
Owner

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.

@floe this seems interesting - so please prepare a PR when you get a chance. Is this only slightly related to PR #195?

@floe
Copy link

floe commented Sep 7, 2018

It will probably conflict with #195 as it would be a rewrite of the analogWrite() function. But as I can only test on the nRF51 currently, it's probably not going to be ready for immediate merging anyway. I'll see what I can put together for the nRF51 alone as a discussion starter.

JonesChi pushed a commit to JonesChi/arduino-nRF5 that referenced this issue May 3, 2023
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
# for free to join this conversation on GitHub. Already have an account? # to comment
Projects
None yet
Development

No branches or pull requests

4 participants