In this tutorial, we’ll discuss the STM32 ADC Multi-Channel Scan Mode with DMA & Polling techniques for reading the ADC conversion results. You’ll learn how STM32 ADC Multi-Channel Scan mode works and how to use it to read a regular group of multiple ADC channels and get the conversion data using Polling and DMA with the STM32 HAL API functions.
The practical example we’ll implement in this tutorial is an STM32 LED dimmer using multiple potentiometers on multiple analog input pins and multiple PWM outputs to control the brightness of some LEDs. We’ll read the multiple analog input channels (regular group) in single-conversion mode using the DMA and Polling methods. Without further ado, let’s get right into it!
Table of Contents
- STM32 ADC Multi-Channel Scan (Single-Conversion)
- STM32 ADC Multi-Channel Discontinuous Mode Example
- STM32 STM32 ADC Multi-Channel Scan Mode Polling (Single-Conversion)
- STM32 ADC Multi-Channel Scan Interrupt Example (Single-Conversion)
- STM32 ADC Multi-Channel Scan DMA Example (Single-Conversion)
- Wrap Up
STM32 ADC Multi-Channel Scan (Single-Conversion)
In this tutorial, we’ll explore the STM32 ADC Multi-Channel Scan Mode in single-conversion (one-shot) mode. In this mode, the ADC will start converting the configured regular group of channels one by one according to the channel ranks (from low to high) till the end of the group where it stops and generates an interrupt signal indicating the end of the regular group’s single-conversion process.
The STM32 ADC Multi-Channel Scan Mode can also be used with Polling or DMA to get the ADC reading results as soon as the conversion is completed. And that’s what we’ll do in the practical example project hereafter in this tutorial.
1. STM32 ADC Multi-Channel Polling (Discontinuous Conversion)
In the polling method, the ADC is started in software at least once. We’ll poll for the conversion completion for the regular group of channels one by one until we get all the channels’ readings into the corresponding buffer variables. This is obviously a blocking way to read the ADC as the CPU will be busy waiting for the completion of all the conversions.
However, there is no guarantee that we can reliably get the correct channel reading at the right time. The CPU may be executing some higher-priority interrupt handler when the ADC EOC flag is triggered, therefore it can miss reading the channel before the buffer is overwritten by the next ADC channel conversion in the regular group. This will cause your system to mix all the subsequent ADC readings unless you reset everything at the end of the regular group conversion.
2. STM32 ADC Multi-Channel Interrupt (Discontinuous Conversion)
Using interrupts with ADC Multi-Channel scan mode is just impossible as stated in the documentation of the STM32 microcontroller’s datasheet. Because the interrupt signal is fired at the end of the regular group conversion.
This means the ADC result buffer will keep getting overwritten by the regular group channels one by one from the lowest rank to the highest. At the time of interrupt firing, you’ll only find the last group’s channel reading in the ADC buffer and all the previous channels are gone forever.
3. STM32 ADC Multi-Channel DMA (Discontinuous Conversion)
And here comes the third method which is using the DMA unit that can directly transfer the ADC result from the peripheral to the memory in the manner that you want and program it to. All that is done without any CPU intervention and upon DMA transfer completion, it can notify the CPU to process the data or whatever.
To sum things up, the STM32 ADC in Multi-Channel Scan Mode (Single-Conversion) can only be reliably used with DMA. The interrupt-based reading is impossible to implement and the polling method can potentially cause your system to fail (halt forever) or get the channels’ readings mixed up in more serious applications.
STM32 ADC Multi-Channel Discontinuous Mode Example
In this example project, we’ll create an STM32 LED Dimmer using ADC & PWM to read multiple analog inputs from 4x potentiometers to control the brightness of 4x PWM outputs going to 4x LEDs. This demo will run the STM32 ADC in multi-channel single-conversion mode using 3 different ADC reading techniques (DMA, Interrupt, and Polling).
- Set up a new project as usual with system clock @ 72MHz
- Set up 4x Analog Input Pins (Channel 6, 7, 8, and 9) In Discontinuous-Conversion Mode (The 4x Pot. Pin)
- Set up Timer2 in PWM mode with output on channels 1-4 (The 4x LED Pins)
The Application will have 3 versions each does the same thing which is to read the ADC result and move it to the timer CCR register which decides the PWM duty cycle percentage on the output LED pin.
Example 1, ADC Multi-Channel Discontinuous-Conversion is used in blocking mode (polling)
Example 2, ADC Multi-Channel Discontinuous-Conversion is used in non-blocking mode (interrupt)
Example 3, ADC Multi-Channel Discontinuous-Conversion is used in non-blocking mode (DMA)
STM32 STM32 ADC Multi-Channel Scan Mode Polling (Single-Conversion)
In this LAB, our goal is to build a system that initializes multiple ADC analog input pins (4x channels: 6-9). And also configure a timer module to operate in PWM mode with 4x outputs on channels 1-4 (4x LED pins). Therefore, we can start an ADC regular group conversion, get & map the results to the PWM duty cycle registers, and repeat the whole process over and over again.
And now, let’s build this system step-by-step
Step #1
Open STM32CubeMX, create a new project, and select the target microcontroller.
Step #2
Configure The ADC1 Peripheral, Enable the Channels: 6-9, and enable the discontinuous mode. You’ll find that the analog channel has these default configurations which happens to be ok for us in this example project.
Step #3
Configure Timer2 To Operate In PWM Mode With Output On CH1-4.
Step #4
Go to the RCC clock configuration page and enable the HSE external crystal oscillator input.
Step #5
Go to the clock configurations page, and select the HSE as a clock source, PLL output, and type in 72MHz for the desired output system frequency. Hit the “ Enter” key, and let the application solve for the required PLL dividers/multipliers to achieve the desired clock rate.
Step #6
Name & Generate The Project Initialization Code For CubeIDE or The IDE You’re Using.
STM32 ADC Multi-Channel Scan Single-Conversion (Polling) Example Code
Here is The Application Code For This LAB (main.c)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 | /* * LAB Name: STM32 ADC Multi-Channel (Scan) Single-Conversion Mode - Polling * Author: Khaled Magdy * For More Info Visit: www.DeepBlueMbedded.com */ #include "main.h" ADC_HandleTypeDef hadc1; TIM_HandleTypeDef htim2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_ADC1_Init(void); static void MX_TIM2_Init(void); int main(void) { uint16_t AD_RES = 0, i = 0; HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_ADC1_Init(); MX_TIM2_Init(); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_2); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_3); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_4); while (1) { for(i=0; i<4; i++) { HAL_ADC_Start(&hadc1); // Start ADC Conversion HAL_ADC_PollForConversion(&hadc1, 1); // Poll ADC1 Peripheral & TimeOut = 1mSec AD_RES = HAL_ADC_GetValue(&hadc1); // Read ADC Conversion Result switch(i) { case 0: TIM2->CCR1 = (AD_RES<<4); // Map The ADC_RES To PWM DutyCycle break; case 1: TIM2->CCR2 = (AD_RES<<4); // Map The ADC_RES To PWM DutyCycle break; case 2: TIM2->CCR3 = (AD_RES<<4); // Map The ADC_RES To PWM DutyCycle break; case 3: TIM2->CCR4 = (AD_RES<<4); // Map The ADC_RES To PWM DutyCycle break; } } HAL_Delay(1); } } |
Wiring
I know it looks messy, but there isn’t too much about it. These are only 4x LEDs and 4x potentiometers connected to the corresponding GPIO pins on the STM32 blue pill board.
STM32 ADC Multi-Channel Single-Conversion Example Testing
STM32 ADC Multi-Channel Scan Interrupt Example (Single-Conversion)
This application is impossible to build as stated in the STM32 ADC documentation (reference manual). We used to build this interrupt-based application for all the previous STM32 ADC tutorial parts, however, we won’t be doing it this time.
For ADC Multi-Channel regular group scan, reading the conversion can either be achieved by Polling ( HAL_ADC_Start() with Discontinuous mode and NbrOfDiscConversion=1) or by DMA ( HAL_ADC_Start_DMA()), but not by interruption ( HAL_ADC_Start_IT()).
In the STM32 ADC multi-channel scan mode, interruption is triggered only on the last conversion of the sequence. All previous conversions will be overwritten by the last one. The injected group used with scan mode doesn’t have this limitation. Each rank has its own result buffer register, no data is overwritten in injected mode as we’ll test it in a later tutorial in this multi-part STM32 ADC series.
To sum things up, we just can’t use the interrupt with the STM32 ADC in scan mode (multi-channel) conversion. The interrupt is fired only on the last conversion of the sequence (regular group of channels). All previous conversions will be overwritten by the last one.
STM32 ADC Multi-Channel Scan DMA Example (Single-Conversion)
Also The Exact Same Steps As The First Example Except For Step 2. The ADC Configuration Will Be As Follows:
This time the STM32 ADC interrupts are not activated and the DMA is configured instead, and the DMA interrupt is enabled by default in the NVIC controller tab. The STM32 DMA configurations for the ADC will be as shown below.
Generate The Project & Open It In The CubeIDE
STM32 ADC Multi-Channel DMA (Single-Conversion) Example Code
Here Is The Application Code Using The DMA Method
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 | /* * LAB Name: STM32 ADC Multi-Channel (Scan) Single-Conversion Mode - DMA * Author: Khaled Magdy * For More Info Visit: www.DeepBlueMbedded.com */ #include "main.h" uint32_t AD_RES_BUFFER[4]; ADC_HandleTypeDef hadc1; DMA_HandleTypeDef hdma_adc1; TIM_HandleTypeDef htim2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_DMA_Init(void); static void MX_ADC1_Init(void); static void MX_TIM2_Init(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_DMA_Init(); MX_ADC1_Init(); MX_TIM2_Init(); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_2); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_3); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_4); while (1) { // Start ADC Conversion in DMA Mode (Periodically Every 1ms) HAL_ADC_Start_DMA(&hadc1, AD_RES_BUFFER, 4); HAL_Delay(1); } } void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) { // Conversion Complete & DMA Transfer Complete As Well // So The AD_RES_BUFFER Is Now Updated & Let's Move Values To The PWM CCRx // Update The PWM Channels With Latest ADC Scan Conversion Results TIM2->CCR1 = (AD_RES_BUFFER[0] << 4); // ADC CH6 -> PWM CH1 TIM2->CCR2 = (AD_RES_BUFFER[1] << 4); // ADC CH7 -> PWM CH2 TIM2->CCR3 = (AD_RES_BUFFER[2] << 4); // ADC CH8 -> PWM CH3 TIM2->CCR4 = (AD_RES_BUFFER[3] << 4); // ADC CH9 -> PWM CH4 } |
ADC Multi-Channel DMA Code Explanation
Even though this tutorial is dedicated to the ADC Mutli-Channel Scan Mode in Single-Conversion operations, we need to keep the conversion going to smoothly control the LED brightness using the potentiometers.
Therefore, I had to “manually” trigger the ADC in DMA mode every 1ms to keep it going. But this doesn’t contradict the fact that the ADC is actually running in single-conversion (one-shot) mode as we intended to do in this tutorial.
Upon DMA transfer completion, the DMA interrupt is fired. In its ISR, we read the ADC_RES_BUFFER array to get the readings of the ADC Channels (6-9). Then we map those readings from 12-bit to 16-bit values to be written to the timer’s CCRx registers for PWM duty cycle control.
Using the DMA is the best way to read the STM32 ADC in multi-channel (scan) mode whether it’s a single-conversion (one-shot) or a continuous-conversion as we’ll see in the next tutorial.
Required Parts For STM32 Examples
All the example Code/LABs/Projects in this STM32 Series of Tutorials are done using the Dev boards & Electronic Parts Below:QTY. Component Name Amazon.com AliExpress eBay 1 STM32-F103 BluePill Board (ARM Cortex-M3 @ 72MHz) Amazon AliExpress eBay 1 Nucleo-L432KC (ARM Cortex-M4 @ 80MHz) Amazon AliExpress eBay 1 ST-Link V2 Debugger Amazon AliExpress eBay 2 BreadBoard Amazon AliExpress eBay 1 LEDs Kit Amazon & Amazon AliExpress eBay 1 Resistors Kit Amazon & Amazon AliExpress eBay 1 Capacitors Kit Amazon & Amazon AliExpress & AliExpress eBay & eBay 1 Jumper Wires Pack Amazon & Amazon AliExpress & AliExpress eBay & eBay 1 Push Buttons Amazon & Amazon AliExpress eBay 1 Potentiometers Amazon AliExpress eBay 1 Micro USB Cable Amazon AliExpress eBay
★ Check The Links Below For The Full Course Kit List & LAB Test Equipment Required For Debugging ★
Download Attachments
You can download all attachment files for this Article/Tutorial (project files, schematics, code, etc..) using the link below. Please consider supporting our work through the various support options listed in the link down below. Every small donation helps to keep this website up and running and ultimately supports the whole community.
Wrap Up
In conclusion, we’ve explored the various methods of reading the STM32 ADC multiple-channel in single-conversion mode using: (DMA & Polling). Be advised that you can’t use the interrupt method with the ADC in multi-channel scan mode. The DMA is the best option for this sort of application.
You can build on top of the examples provided in this tutorial and/or explore the other parts of the STM32 ADC tutorials series for more information about the other STM32 ADC operating modes and conversion schemes.