This is an old revision of the document!
This scenario presents how to control the fan's rotation speed with a PWM signal. You can observe the rotation via the camera, but it would be rather hard to notice the change in the rotation speed. The fan is mounted on the top of the pressure tube and blows the air onto the pressure sensor. You can observe changes in sensor readings according to different fan speeds. How to use the sensor is shown in another scenario.
A good understanding of the PWM signal and duty cycle is necessary. We also use built-in timers to control the PWM hardware channels of the STM32WB55 chip. In this case, we do not use an external library; instead, we use the hardware timer library built in the Arduino Core STM32 framework (stm32duino)[1] for STM32WB55 so that no additional external libraries will be included in the project.
Implement a program that will change the fan's rotation speed up from 0% to 100% and down from 100% to 0%. Cotrol the rotation speed with the PWM signal.
The hardware timer library implements functions which allow us to control the duty cycle of the PWM signal and express it in different formats, including percentages. In the laboratory equipment, the fan is connected to the hardware timer instance shared with the servo motor. It means that both elements use the same base frequency of the PWM signal. If you use a fan and servo in the same project the frequency of the PMW signal need to match the servo requirements. If you use the fan only the frequency can be freely chosen. In the case of setting the PWM duty cycle expressed in percentages, the minimum value is 0, and the max (full brightness) is 100. Note that low values of the duty cycle values do not provide enough energy to the fan motor so it does not start rotating from 1%.
To use PWM in STM32WB55, it is best to use a built-in hardware timer library.
#include <HardwareTimer.h>
The hardware timer uses internal timer modules and allows us to define channels attached to the timer. We will use 1 channel per colour (R, G and B, so 3 in total). A PWM frequency is controlled with the timer to be shared for all 3 R, G and B channels. Channels control the PWM duty cycle.
Include the library, and define PIN assignments to channels and PWM frequency (100Hz):
#include <HardwareTimer.h> // Pins definition for RGB LED #define LED_PWM_GREEN D3 //Arduino numbering D3, STM numbering PA_10 #define LED_PWM_BLUE D6 //Arduino numbering D6, STM numbering PA_8 #define LED_PWM_RED D9 //Arduino numbering D9, STM numbering PA_9 #define PWM_freq 100
GPIO pins controlling LEDS are D9 (Red), D3 (Green) and D6 (Blue), respectively.
Define variables for the timer object and handlers of channels.
// PWM variables definitions HardwareTimer *MyTimLED; //Hardware Timer for PWM uint32_t channelR, channelG, channelB; //RGB channels
Instantiate the timer object and initialise the 3 channels for PWM and make them dark (1% duty cycle):
//Create Timer instance type based on the chosen LED pin (the same timer is used for all LEDs). TIM_TypeDef *Instance = (TIM_TypeDef *)pinmap_peripheral(digitalPinToPinName(LED_PWM_RED), PinMap_PWM); //Define three separate channels for LEDs channelR = STM_PIN_CHANNEL(pinmap_function(digitalPinToPinName(LED_PWM_RED), PinMap_PWM)); channelG = STM_PIN_CHANNEL(pinmap_function(digitalPinToPinName(LED_PWM_GREEN), PinMap_PWM)); channelB = STM_PIN_CHANNEL(pinmap_function(digitalPinToPinName(LED_PWM_BLUE), PinMap_PWM)); // Instantiate HardwareTimer object. Thanks to 'new' instantiation, HardwareTimer is not destructed when setup() function is finished. MyTimLED = new HardwareTimer(Instance); // Configure and start PWM MyTimLED->setPWM(channelR,LED_PWM_RED,PWM_freq,1); /* 100 Hertz, 1% dutycycle. */ MyTimLED->setPWM(channelG,LED_PWM_GREEN,PWM_freq,1); /* 100 Hertz, 1% dutycycle. */ MyTimLED->setPWM(channelB,LED_PWM_BLUE,PWM_freq,1); /* 100 Hertz, 1% dutycycle. */
To modify the intensity of the chosen LED after initialisation use the setCaptureCompare function. In the following example, the duty_cycle_value can vary between 0 and 100.
MyTimLED->setCaptureCompare(channelR, duty_cycle_value, PERCENT_COMPARE_FORMAT); //modify duty cycle
Write a loop for each colour (R, G, then B) to light the colour from dark to max value.
Mind to compose code to increase and decrease each colour. A hint is below (for channelR):
// Increase brightness for (int duty_cycle_value = 0; duty_cycle_value <= 50; duty_cycle_value++) { // Gradually increase duty cycle for Red LED MyTimLED->setCaptureCompare(channelR, duty_cycle_value, PERCENT_COMPARE_FORMAT); delay(20); // Delay for smooth transition } delay(100); // Decrease brightness for (int duty_cycle_value = 50; duty_cycle_value >= 0; duty_cycle_value--) { // Gradually decrease duty cycle for Red LED MyTimLED->setCaptureCompare(channelR, duty_cycle_value, PERCENT_COMPARE_FORMAT); delay(20); // Delay for smooth transition }
You should be able to observe the pulsing colours of the RGB LED, increasing and decreasing brightness linearly.
What is the maximum number of channels?: the MCU we use here is STM32WB55 on the Nucleo board. The pins available are connected to timer 1 with three PWM channels (connected to our RGB LED) and timer 2 with 4 PWM channels (with two of them connected to the servo and fan described in other scenarios). A single timer can generate PWM signals with independent duty cycles and identical frequency.
What is the maximum bit resolution for PWM?: Maximum resolution is 16 bits. Note that we can express the duty cycle in a variety of formats. In the presented example we expressed it in percentage so it varies between 1 and 100. It can be also expressed in ticks, Hertz, microseconds, and 1-16 bits numbers.
What PWM frequency should I use?: there is no straightforward answer to this question: assuming you observe LED remotely with a camera, even 50Hz would be enough. But it would give a severe flickering experience to the live user, on the other hand. In the example above, we propose 100Hz, but this MCU can easily handle higher frequencies.
This Intellectual Output was implemented under the Erasmus+ KA2.
Project IOT-OPEN.EU Reloaded – Education-based strengthening of the European universities, companies and labour force in the global IoT market.
Project number: 2022-1-PL01-KA220-HED-000085090.
Erasmus+ Disclaimer
This project has been funded with support from the European Commission.
This publication reflects the views of only the author, and the Commission cannot be held responsible for any use that may be made of the information contained therein.
Copyright Notice
This content was created by the IOT-OPEN.EU Reloaded consortium, 2022,2024.
The content is Copyrighted and distributed under CC BY-NC Creative Commons Licence, free for Non-Commercial use.