STM32 HAL库实战:高效整合DMA与ADC开发指南

STM32 HAL库实战:高效整合DMA与ADC开发指南

一、DMA与ADC基础介绍

1. DMA:解放CPU的"数据搬运工"

DMA(Direct Memory Access) 是STM32中用于在外设与内存之间直接传输数据的硬件模块。其核心优势在于无需CPU干预,可显著提升系统效率。

  • 功能特点:

    • 支持存储器↔外设、存储器↔存储器的数据传输。

    • 多通道管理,每个通道独立控制一个外设的数据传输。

    • 传输完成后触发中断通知CPU。

  • 典型应用场景:

    • ADC多通道连续采样数据的自动存储。

    • 串口大数据收发、SPI/I2C通信等。

2. ADC:模拟世界的"数字翻译器"

ADC(Analog-to-Digital Converter) 负责将模拟信号(如电压、温度)转换为数字信号。STM32的ADC模块支持多通道、高精度采样。

  • 关键参数:

    • 分辨率:12位(0~4095)。

    • 采样速率:最高1MHz(STM32F1系列)。

    • 输入通道:16个外部通道 + 2个内部通道(温度传感器、VREF)。

  • 工作模式:

    • 单次转换、连续转换、扫描模式(多通道轮询)。

二、DMA+ADC整合开发步骤

1. 硬件与工程配置

硬件连接示例
  • ADC通道:PA1(通道1)接模拟输入(如MQ2烟雾传感器、电位器等)。
  • DMA通道:ADC1使用DMA1通道1(见STM32参考手册)。
硬件代码配置
ADC基础配置
c 复制代码
void adc_config(void)
{
    adc_handle.Instance = ADC1;
    adc_handle.Init.DataAlign = ADC_DATAALIGN_RIGHT;
    adc_handle.Init.ScanConvMode = ADC_SCAN_DISABLE;
    adc_handle.Init.ContinuousConvMode = ENABLE;
    adc_handle.Init.NbrOfConversion = 1;
    adc_handle.Init.DiscontinuousConvMode = DISABLE;
    adc_handle.Init.NbrOfDiscConversion = 0;
    adc_handle.Init.ExternalTrigConv = ADC_SOFTWARE_START;
    HAL_ADC_Init(&adc_handle);
    
    HAL_ADCEx_Calibration_Start(&adc_handle);
}
ADC硬件配置
c 复制代码
void HAL_ADC_MspInit(ADC_HandleTypeDef* hadc)
{
    if(hadc->Instance == ADC1)
    {
        RCC_PeriphCLKInitTypeDef adc_clk_init = {0};
        GPIO_InitTypeDef gpio_init_struct = {0};
        
        __HAL_RCC_ADC1_CLK_ENABLE();
        __HAL_RCC_GPIOA_CLK_ENABLE();
        
        gpio_init_struct.Pin = GPIO_PIN_1;
        gpio_init_struct.Mode = GPIO_MODE_ANALOG;
        HAL_GPIO_Init(GPIOA, &gpio_init_struct);
        
        adc_clk_init.PeriphClockSelection = RCC_PERIPHCLK_ADC;
        adc_clk_init.AdcClockSelection = RCC_ADCPCLK2_DIV6;
        HAL_RCCEx_PeriphCLKConfig(&adc_clk_init);
    }
}
ADC通道配置
c 复制代码
void adc_channel_config(ADC_HandleTypeDef* hadc, uint32_t ch, uint32_t rank, uint32_t stime)
{
    ADC_ChannelConfTypeDef adc_ch_config = {0};
    
    adc_ch_config.Channel = ch;
    adc_ch_config.Rank = rank;
    adc_ch_config.SamplingTime = stime;
    HAL_ADC_ConfigChannel(hadc, &adc_ch_config);
}
DMA配置:
c 复制代码
void dma_config(void)
{
    __HAL_RCC_DMA1_CLK_ENABLE();
    dma_handle.Instance = DMA1_Channel1;
    dma_handle.Init.Direction = DMA_PERIPH_TO_MEMORY;
    
    //内存相关配置
    dma_handle.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
    dma_handle.Init.MemInc = DMA_MINC_ENABLE;
    
    //外设相关配置
    dma_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
    dma_handle.Init.PeriphInc = DMA_PINC_DISABLE;
    
    dma_handle.Init.Priority = DMA_PRIORITY_MEDIUM;
    dma_handle.Init.Mode = DMA_CIRCULAR;
    HAL_DMA_Init(&dma_handle);
    
    __HAL_LINKDMA(&adc_handle, DMA_Handle, dma_handle);
}

2. 关键代码实现

初始化并启动DMA
c 复制代码
void adc_dma_init(uint32_t *mar)
{
    adc_config();
    adc_channel_config(&adc_handle, ADC_CHANNEL_1, ADC_REGULAR_RANK_1, ADC_SAMPLETIME_239CYCLES_5);
    dma_config();
    
    HAL_ADC_Start_DMA(&adc_handle, mar, 1);
}

在主函数中,调用该函数,DMA将会一直将值存放在mar所指的内存上。

3. 优化技巧与常见问题

提升ADC精度
  • 硬件优化:

    • 添加0.1μF滤波电容到ADC输入引脚。

    • 单独为VDDA和VSSA供电。

  • 软件优化:

    • 丢弃前几次采样值(避免电源不稳定)。
常见问题排查
  • 数据错位:检查DMA数据宽度(需与ADC对齐方式一致)。

  • 采样值跳动:增加采样时间或添加软件滤波(如滑动平均)。

  • DMA不触发:确认DMA通道与ADC的映射关系(参考数据手册)。

三、项目实战:DMA单通道采集(烟雾)

功能需求

  • 使用PA1烟雾报警模拟输入。
  • 每3秒计算一次并通过串口输出。

核心代码片段

c 复制代码
#include "sys.h"
#include "delay.h"
#include "led.h"
#include "uart1.h"
#include "adc.h"


#define CAL_PPM  10  // 校准环境中PPM值
#define RL	     10  // RL阻值
#define R0	     97  // R0阻值

uint16_t adc_result = 0;
int main(void)
{
	 float RS ;
	 float ppm;
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9);     /* 设置时钟, 72Mhz */
    led_init();                         /* 初始化LED灯 */
    uart1_init(115200);
    adc_dma_init((uint32_t  *)&adc_result);
    printf("hello world!\r\n");

    while(1)
    { 
//        printf("adc result: %f\r\n", (float)adc_result / 4096 * 3.3);
	  RS = (3.3f - (float)adc_result / 4096 * 3.3) / (float)adc_result / 4096 * 3.3 * RL;
	  ppm = 98.322f * pow(RS/R0, -1.458f);
		printf("adc result: %f\r\n", ppm);
        delay_ms(3000);
    }
}

四、总结

通过DMA+ADC的高效整合,开发者可以实现低CPU占用率的模拟信号采集系统。关键点在于:

  1. 合理配置ADC的扫描模式与DMA循环传输。
  2. 利用HAL库的中断回调机制处理数据。
  3. 通过硬件与软件优化提升信号质量。

掌握这一技术后,可轻松应对传感器数据采集、工业控制等高实时性场景的需求。

相关推荐
百锦再5 分钟前
《C#上位机开发从门外到门内》3-3:基于USB的设备管理系统
开发语言·单片机·数码相机·c#·上位机·下位机
OKay_J5 小时前
使用VSCode开发STM32补充(Debug调试)
ide·经验分享·笔记·vscode·stm32·学习·编辑器
爱学习的张哥8 小时前
UDP协议栈之整体架构处理
单片机·架构·udp
天天爱吃肉821810 小时前
【恒流源cc与恒压源cv典型电路解析】
自动化测试·单片机·嵌入式硬件·汽车
九鼎创展科技10 小时前
LGA封装 Z3588开发板,8K视频编解码
arm开发·人工智能·嵌入式硬件
电子艾号哲11 小时前
STC89C52单片机学习——第20节: [8-2]串口向电脑发送数据&电脑通过串口控制LED
单片机·学习·电脑
万能的小裴同学11 小时前
给单片机生成字库的方案
单片机·嵌入式硬件
c-u-r-ry3013 小时前
011---UART之RS232通信接口标准(二)
嵌入式硬件·fpga开发
二年级程序员14 小时前
51单片机的工作过程
单片机·嵌入式硬件·51单片机