STM32通过DMA进行ADC采集(HAL库)

本文介绍了如何使用CubeMX配置STM32F103C8T6的ADC和DMA进行多通道模拟信号(如遥杆电位器)的连续转换。配置包括启用ADC1的IN1和IN2通道,选择扫描和连续转换模式,并通过DMA进行循环传输。在遇到ADC DMA接收中断过于频繁导致主函数无法运行的问题时,可以通过增加采样周期来解决。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一. 前言

请添加图片描述

ADC是指模数转化,即将模拟信号转化为数字信号。 世界的信号本质都是模拟信号,如速度、温度、声音,在本质上都是模拟的。但是微控制器和微处理器上运行的电子设备不能直接读取或者解释这些模拟值,因为它们只在0和1上运行。所以我们需要将这些模拟值转换成0和1,以便微控制器和微处理器能够理解它们。这就是我们所说的模拟信号转化为数字信号。与ADC相对应的是DAC,即将数字信号转化为模拟信号。

二. CubeMX配置

开发环境:Keil 、 CubeMX
软件版本:5.31 、6.2.1
使用芯片:STM32F103C8T6
开发语言:C

CubeMX的配置非常简单:Analog->ADCx->INx


此处选择多通道采集,选择ADC1 ,勾选IN1 IN2 。配置如下:

  • 由于是多通道采集,需开启Scan Conversion Mode 扫描模式
  • 需要多次读取,因此开启Continuous Conversion Mode 连续转换模式
  • 选配Rank,给IN1、IN2选择合适的采样通道和采样周期。

添加DMA通道 ,将模式改为Circular 循环模式 ,这样每次DMA传输成功后会主动开启下一次传输。


更多的ADC参数可以参考这篇文章:梳理ADC模数转换器在CubeMX上的配置

三. ADC读取

启动函数:

HAL_ADC_Start_DMA(ADC_HandleTypeDef* hadc, uint32_t* pData, uint32_t Length)

将ADC用于遥杆的读取: 遥杆的本质是一个电位器,通过ADC采集变化的电压可以得到遥杆X、Y轴的遥杆值。按照上面的配置,在生成的Keil工程中加入如下代码可以便可以得到采集的遥杆值。


uint16_t Joystick_ADC[2];

HAL_ADC_Start_DMA(&hadc1,(uint32_t*)&Joystick_ADC,2);


注意: 如果开启ADC的DMA接收后没办法继续运行主函数可能是因为设置的采样周期太短,频繁触发DMA接受中断导致的。可以在CubeMX中把Rank中通道的采样周期Samplink Time 设长一些,亲测有效。

<think>好的,我现在需要处理用户关于STM32 HAL中使用ADCDMA进行采集的模块化代码实现的问题。首先,我需要回顾用户提供的引用内容,看看是否有相关的代码示例或知识点可以整合。 引用1提到标准ADCDMA知识点,但用户需要的是HAL的实现,所以可能需要更多的HAL资料。引用2和3都涉及HALADCDMA整合,特别是引用2提到了中断和DMA的应用场景,引用3则强调了ADC的多通道和高精度采样。因此,我需要结合这些信息,构建一个模块化的代码示例。 接下来,我需要确保代码符合HAL的结构,模块化设计通常包括初始化配置、启动转换、数据处理回调函数等部分。DMAHAL中通常用于自动传输ADC转换结果到内存,避免CPU干预,提高效率。 用户可能需要的是分步骤的代码实现,包括ADCDMA的初始化、启动转换,以及如何处理转换完成的数据。同时,需要注意DMA的配置,比如循环模式、数据对齐等参数。此外,模块化意味着代码应可重用,可能需要将配置部分封装成函数,便于调用。 然后,我需要检查是否有遗漏的部分,比如中断处理。根据引用2,使用中断+DMA的方式,可能需要配置中断回调函数,但DMA传输完成中断可能已经足够。HAL中,DMA传输完成会触发相应的回调函数,比如HAL_ADC_ConvCpltCallback,这里可以处理数据。 另外,用户可能关心如何确保代码的稳定性和正确性,比如防止数据溢出或配置错误。需要提醒用户根据具体型号调整ADCDMA的通道配置,时钟设置等。 最后,生成相关问题时要围绕ADCDMA的使用,可能涉及优化、多通道处理、错误处理等方向。需要确保问题覆盖用户可能感兴趣的扩展话题。</think>以下是一个基于STM32 HALADC通过DMA采集的模块化代码实现,包含初始化配置、数据采集和回调处理三部分: --- ### 1. ADCDMA初始化配置 ```c // 在头文件中声明全局变量 #define ADC_BUFFER_SIZE 256 extern uint16_t adc_buffer[ADC_BUFFER_SIZE]; extern ADC_HandleTypeDef hadc1; extern DMA_HandleTypeDef hdma_adc1; // 初始化函数 void ADC_DMA_Init(void) { // ADC1基本配置 hadc1.Instance = ADC1; hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT; hadc1.Init.ScanConvMode = ENABLE; // 扫描模式使能 hadc1.Init.ContinuousConvMode = ENABLE; // 连续转换模式 hadc1.Init.NbrOfConversion = 1; // 转换通道数 hadc1.Init.DiscontinuousConvMode = DISABLE; hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START; HAL_ADC_Init(&hadc1); // DMA配置(循环模式) hdma_adc1.Instance = DMA1_Channel1; hdma_adc1.Init.Direction = DMA_PERIPH_TO_MEMORY; hdma_adc1.Init.PeriphInc = DMA_PINC_DISABLE; hdma_adc1.Init.MemInc = DMA_MINC_ENABLE; hdma_adc1.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD; hdma_adc1.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD; hdma_adc1.Init.Mode = DMA_CIRCULAR; // 循环缓冲模式 hdma_adc1.Init.Priority = DMA_PRIORITY_HIGH; HAL_DMA_Init(&hdma_adc1); // 关联ADCDMA __HAL_LINKDMA(&hadc1, DMA_Handle, hdma_adc1); // 校准ADC并启动 HAL_ADCEx_Calibration_Start(&hadc1); HAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_buffer, ADC_BUFFER_SIZE); } ``` --- ### 2. 中断回调处理 ```c // 在main.c中重写转换完成回调 void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) { if(hadc->Instance == ADC1) { // 此处添加数据处理逻辑 // adc_buffer包含最新采集ADC_BUFFER_SIZE个数据 } } ``` --- ### 3. 模块化使用示例 ```c int main(void) { HAL_Init(); SystemClock_Config(); ADC_DMA_Init(); while(1) { // 主循环可添加其他任务 // ADC数据通过DMA自动更新到adc_buffer数组 } } ``` --- ### 关键配置说明: 1. **扫描模式**:支持多通道采集,需配合`ADC_ChannelConfTypeDef`配置多个通道[^3] 2. **循环缓冲**:DMA自动循环覆盖缓冲区,避免数据溢出[^2] 3. **校准操作**:HAL_ADCEx_Calibration_Start()可提高ADC精度[^1] 4. **触发方式**:本例使用软件触发,也可配置定时器触发 ---
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

冬瓜~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值