The abbreviation PWM stands for “Pulse Width Modulation” and is also often referred to in German as pulse width modulation or pulse duration modulation. This technology is used, among other things, to control servomotors and is also used, for example, for the fans of a regular computer.
With PWM, it is possible to control a component such as a motor no longer purely binary, i.e. off (0% power) or on (100% power), but to control them almost at will. The functionality of PWM works in such a way that the component is switched off and on again and again within a certain period of time.
When using the pigpio-pwm provider two different types of PWM are available on the Raspberry Pi, specifically a software and a hardware implementation. Both basically offer the same options, but the software version cannot achieve precise or particularly fast frequencies. When using the linuxfs-pwm provider only hardware PWM is available.
The reason for this is that in the software implementation for each individual cycle (on / off) a new control command must be transmitted from the JVM (Java Virtual Machine) to the corresponding component, while in the hardware implementation of the Raspberry Pi notices the desired frequency and regulates it independently directly on the board.
The Raspberry Pi supports 2 hardware based PWM channels. You can access these two channels via 2 separate sets of 4 GPIO header pins, but still limited to only 2 channels (2 unique PWM timing configurations).
The same PWM channel is available on multiple GPIOs. The latest frequency and dutycycle setting will be used by all GPIO which share a PWM channel.
The GPIO must be one of the following:
12 PWM channel 0 All models but A and B
13 PWM channel 1 All models but A and B
18 PWM channel 0 All models
19 PWM channel 1 All models but A and B
40 PWM channel 0 Compute module only
41 PWM channel 1 Compute module only
45 PWM channel 1 Compute module only
52 PWM channel 0 Compute module only
53 PWM channel 1 Compute module only
The GPIO number in the above chart is supplied as the buildPwmConfig config value address
.
As Pi4J is using PiGPIO “under the hood”, you can take advantage of the additional
PWM functionalities of it. PiGPIO is providing additional (soft) PWM support to any
of the GPIO pins (0-31) and its using some hardware timing technique to optimize
performance — but its not the same as the actual hardware PWM pins natively on the
RaspberryPi. In the Pi4J API, we call this “Software” PWM and you would need to set
.pwmType(PwmType.SOFTWARE)
. We consider this software-based PWM because its being
provided at a software layer, in this case by the PIGPIO library.
If you need more than 2 PWM pins, use the software PWM functionality, it may be perfectly fine for your application. If they are not good enough, then you will probably need a PWM expander board/chip (controlled by I2C/SPI) to provide additional PWM support.
As of version 2.6.0 of Pi4J, linuxfs-pwm
also supports hardware PMW on the Raspberry Pi 5. More information and an example implementation is available in the blog post PWM Hardware Support on Raspberry Pi5.
Only hardware PWM is supported.
The user must modify config.txt
to enable PWM.
/boot/config.txt
/boot/firmware/config.txt
To take effect after file modification the Raspberry Pi must be rebooted.
Use one of the following configurations:
[all]
dtoverlay=pwm
# GPIO 18 = channel 0
[all]
dtoverlay=pwm-2chan
# GPIO 18 = channel 0
# GPIO 19 = channel 1
[all]
dtoverlay=pwm-2chan,pin=12,func=4,pin2=13,func2=4
# GPIO 12 = channel 0
# GPIO 13 = channel 1
Use one of the following configurations:
[all]
dtoverlay=pwm
# GPIO 18 = channel 2
[all]
dtoverlay=pwm-2chan
# GPIO 18 = channel 2
# GPIO 19 = channel 3
[all]
Dtoverlay=pwm-2chan,pin=12,func=4,pin2=13,func2=4
# GPIO 12 = channel 0
# GPIO 13 = channel 1
The statement added to config.txt
will determine which GPIOs will exhibit the PWM behavior.
The channel number in the above charts are supplied as the buildPwmConfig
value for the address
.
For the technical control of a component with PWM, two values must be defined:
These two values can be controlled via the Pi4J library and are also used internally by this project.
The following example is an extract of the CrowPi example project which includes a component to control a buzzer with PWM. Of importance, this example executes on a Raspberry Pi4, the buildPwmConfig(Context pi4j, int address) example code uses pigpio-pwm and the value passed for ‘address’ is the BCM pin number.
public class BuzzerComponent extends Component {
protected final Pwm pwm;
/**
* Creates a new buzzer component with a custom BCM pin.
*
* @param pi4j Pi4J context
* @param address Custom BCM pin address
*/
public BuzzerComponent(Context pi4j, int address) {
this.pwm = pi4j.create(buildPwmConfig(pi4j, address));
}
/**
* Plays a tone with the given frequency in Hz indefinitely.
* This method is non-blocking and returns immediately.
* A frequency of zero causes the buzzer to play silence.
*
* @param frequency Frequency in Hz
*/
public void playTone(int frequency) {
playTone(frequency, 0);
}
/**
* Plays a tone with the given frequency in Hz for a specific duration.
* This method is blocking and will sleep until the specified duration has passed.
* A frequency of zero causes the buzzer to play silence.
* A duration of zero to play the tone indefinitely and return immediately.
*
* @param frequency Frequency in Hz
* @param duration Duration in milliseconds
*/
public void playTone(int frequency, int duration) {
if (frequency > 0) {
// Activate the PWM with a duty cycle of 50% and the given frequency in Hz.
// This causes the buzzer to be on for half of the time during each cycle, resulting in the desired frequency.
pwm.on(50, frequency);
// If the duration is larger than zero, the tone should be automatically stopped after the given duration.
if (duration > 0) {
sleep(duration);
this.playSilence();
}
} else {
this.playSilence(duration);
}
}
/**
* Silences the buzzer and returns immediately.
*/
public void playSilence() {
pwm.off();
}
/**
* Silences the buzzer and waits for the given duration.
* This method is blocking and will sleep until the specified duration has passed.
*
* @param duration Duration in milliseconds
*/
public void playSilence(int duration) {
this.playSilence();
sleep(duration);
}
/**
* Returns the created PWM instance for the buzzer
*
* @return PWM instance
*/
protected Pwm getPwm() {
return this.pwm;
}
/**
* Builds a new PWM configuration for the buzzer using pigpio-pwm
*
* @param pi4j Pi4J context
* @param address BCM pin address
* @return PWM configuration
*/
protected static PwmConfig buildPwmConfig(Context pi4j, int address) {
return Pwm.newConfigBuilder(pi4j)
.id("BCM" + address)
.name("Buzzer")
.address(address)
.pwmType(PwmType.HARDWARE)
.provider("pigpio-pwm")
.initial(0)
.shutdown(0)
.build();
}
/**
* Builds a new PWM configuration for the buzzer using linuxfs-pwm
* @param pi4j Pi4J context
* @param address Channel
* @return PWM configuration
*/
protected static PwmConfig buildPwmConfig(Context pi4j, int address) {
return Pwm.newConfigBuilder(pi4j)
.id("Channel" + address)
.name("Buzzer")
.address(address)
.pwmType(PwmType.HARDWARE)
.provider("linuxfs-pwm")
.initial(0)
.shutdown(0)
.build();
}
}