STM32开发(14)----CubeMX配置ADC

本文介绍了如何使用STM32CubeMX配置ADC,包括单通道、多通道(非DMA)和多通道(DMA)的ADC采样实现。STM32F103的ADC具有12位分辨率和多种工作模式,如规则转换、注入转换和中断。文章详细展示了配置过程和代码实现,帮助读者理解ADC的工作原理和配置细节。

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


前言

本章介绍使用STM32CubeMX对ADC进行配置的方法,ADC的原理、概念和特点,配置各个步骤的功能,并通过单通道,多通道,DMA三种方式实现采集。

一、什么是ADC?

ADC 即模拟数字转换器,英文详称 Analog-to-digital converter,可以将外部的模拟信号转换为数字信号。

以下是datasheet当中的内容,我就做个搬运工,简单翻一下,大家可以配合datasheet学习,这样理解会更加深刻。 STM32 的 ADC 是 12 位逐次逼近型的模拟数字转换器,它有 18 个通道,可测量 16 个外部和 2 个内部信号源,其中 ADC3 根据 CPU 引脚的不同其通道数也不同,一般有 8 个外部通道。 ADC 中的各个通道的 A/D 转换可以单次、连续、扫描或间断模式执行。 ADC 的结果可以以左对齐或者右对齐存储在 16 位数据寄存器中。

STM32F103 的 ADC 主要特性如下:
1、 12 位分辨率;
2、转换结束、注入转换结束和发生模拟看门狗事件时产生中断
3、单次和连续转换模式
4、自校准
5、带内嵌数据一致性的数据对齐
6、采样间隔可以按通道分别编程
7、规则转换和注入转换均有外部触发选项
8、间断模式
9、双重模式(带 2 个或以上 ADC 的器件)
10、 ADC 转换时间:时钟为 72MHz 为 1.17us
11、 ADC 供电要求: 2.4V 到 3.6V
12、 ADC 输入范围: VREF–≤VIN≤VREF+
13、规则通道转换期间有 DMA 请求产生

我们按照 ADC 的配置流程标记了七处位置,分别如下,理解不同的步骤请参考图和下列的介绍进行。
在这里插入图片描述

① ADC模块的输入电压
② 输入通道:电压输入后,外部输入的电压会通过通道输入到 ADC 转换器中,下面各个是ADC通道的列表
在这里插入图片描述
③ 转换顺序:ADC 多个通道以任意顺序输入,则需要进行顺序转换,转换的方式有两种,一种是规则组:即按照顺序进行,另一种是注入组:即打破原有顺序,有点类似中断的形式进行顺序转换、
④ 转换触发源:从图中可以看到,顺序转换是需要触发源的,这里就是对触发源的配置
⑤ 转换时间:即输入电压通过ADC的时间 计算公式:T = 采样时间 + 12.5 个周期
⑥数据寄存器:ADC 转换完成后的数据输出寄存器
⑦ 中断:ADC 中断可分为三种:规则组转换结束中断、注入组转换结束中断、设置了模拟看门狗状态位中断
⑧ 单次转换模式和连续转换模式
⑨ 扫描模式

二、实验过程

1.单通道ADC采集

STM32CubeMX配置

选择芯片stm32f103c6t6,新建工程

在这里插入图片描述

设置时钟源,最小系统外部晶振8Mhz,作为外部高速HSE时钟源。由于没有外接外部低速晶振,这里低速时钟源选择旁路时钟源。

在这里插入图片描述

配置时钟树,这里使用官方推荐的配置

在这里插入图片描述
为了展示内部温度的变化,我们配置USART1,打印获取温度的结果
在这里插入图片描述
USART1的参数配置如下,波特率115200,传输数据长度为8 Bit,奇偶检验无,停止位1.其他参数默认
在这里插入图片描述
使用ADC 通道0,参数中设置采样时间
在这里插入图片描述

Code Generator中设置只拷贝使用到的库,分离.c和.h文件
在这里插入图片描述

设置好项目名称和路径,点击GENERATE CODE即可,生成后使用keil5 IDE打开。

在这里插入图片描述

代码实现

在usart.c文件后面添加如下代码,代码中添加了#ifdef宏定义进行条件编译,如果使用GUNC编译,则PUTCHAR_PROTOTYPE 定义为int __io_putchar(int ch)函数,否则定义为int fputc(int ch, FILE *f)函数。

/* USER CODE BEGIN 0 */
#include "stdio.h"
#ifdef __GNUC__
  /* With GCC/RAISONANCE, small printf (option LD Linker->Libraries->Small printf
     set to 'Yes') calls __io_putchar() */
  #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
  #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif /* __GNUC__ */
/**
  * @brief  Retargets the C library printf function to the USART.
  * @param  None
  * @retval None
  */
PUTCHAR_PROTOTYPE
{
  /* Place your implementation of fputc here */
  /* e.g. write a character to the EVAL_COM1 and Loop until the end of transmission */
  HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, 0xFFFF);
 
  return ch;
}
/* USER CODE END 0 */

main函数如下:

int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */
  uint32_t ADC_Value;
  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
	HAL_ADC_Start(&hadc1);	
	HAL_ADC_PollForConversion(&hadc1,10);	
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
    /* USER CODE BEGIN 3 */
	  if(HAL_IS_BIT_SET(HAL_ADC_GetState(&hadc1), HAL_ADC_STATE_REG_EOC))
	  {
		ADC_Value = HAL_ADC_GetValue(&hadc1);
	  }
      printf(" ADC channel0 value = %1.3fV \r\n", ADC_Value*3.3f/4096);
      }

  }
  /* USER CODE END 3 */
}

2.多通道ADC采样(非DMA)

STM32CubeMX配置

为了避免冗余,这里省略掉CubeMX新建工程,配置时钟等步骤,直接展示ADC的配置
在这里插入图片描述

代码实现

为了避免冗余,这里省略掉printf重载,直接展示main函数如下:

int main(void)
{
  /* USER CODE BEGIN 1 */
  uint32_t ADC_Value[2];
  uint8_t i;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
	HAL_ADC_Start(&hadc1);	
	HAL_ADC_PollForConversion(&hadc1,10);	
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
    /* USER CODE BEGIN 3 */
	HAL_Delay(500);
	  for(i=0;i<2;i++)
	  {
			if(HAL_IS_BIT_SET(HAL_ADC_GetState(&hadc1), HAL_ADC_STATE_REG_EOC))
	  		{
				ADC_Value[i] = HAL_ADC_GetValue(&hadc1);
	 		}
	  		HAL_ADC_Stop (&hadc1);//打开ADC
      		printf(" ADC channel0 value = %1.3fV \r\n", ADC_Value[0]*3.3f/4096);
      		printf(" ADC channel1 value = %1.3fV \r\n", ADC_Value[1]*3.3f/4096);
  }
  /* USER CODE END 3 */
}

3.多通道ADC采样(DMA)

STM32CubeMX配置

为了避免冗余,这里省略掉CubeMX新建工程,配置时钟等步骤,直接展示DMA和ADC的配置

DMA配置如下,主要配置了通道,模式,字宽等信息
在这里插入图片描述
ADC配置如下
在这里插入图片描述

代码实现

为了避免冗余,这里省略掉printf重载,直接展示main函数如下:

int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */
  uint32_t ADC_Value[100];
  uint8_t i;
  uint32_t ad1,ad2;
  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
  HAL_ADC_Start_DMA(&hadc1, (uint32_t*)&ADC_Value, 50);
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
    /* USER CODE BEGIN 3 */
	HAL_Delay(500);
    for(i = 0,ad1 =0,ad2=0; i < 100;)
    {
        ad1 += ADC_Value[i++];
        ad2 += ADC_Value[i++];
    }
    ad1 /= 50;
    ad2 /= 50;
    printf("\r\n******** ADC DMA Example ********\r\n\r\n");//串口打印
    printf(" AD1 value = %1.3fV \r\n", ad1*3.3f/4096);
    printf(" AD2 value = %1.3fV \r\n", ad2*3.3f/4096);

}

总结

本章介绍使用STM32CubeMX对ADC进行配置的方法,ADC的原理、概念和特点,配置各个步骤的功能,并通过单通道,多通道,DMA三种方式实现采集,对于配置的细节,可以参照ADC实现原理图,一点点梳理,这样易于理解。

<think>好的,我需要帮助用户在STM32CubeMX配置ADC、定时器、多通道和DMA传输。首先,我得回忆一下相关配置步骤,并结合提供的引用内容来确保准确性。 用户的问题涉及多个外设的协同工作:ADC多通道采集,定时器触发,DMA传输数据。这可能涉及到STM32CubeMX中的各个配置界面,比如定时器、ADC模块和DMA设置。我需要分步骤讲解,确保每个部分配置正确。 首先,定时器的配置。根据引用中的内容,用户可能需要使用定时器来触发ADC转换,这样可以实现周期性的采样。在CubeMX中,定时器的触发输出需要配置为TRGO输出,并选择合适的触发源,比如更新事件。例如,设置TIM2的触发事件为更新事件,并调整预分频器和自动重载值以确定采样频率。 接下来是ADC配置。用户需要启用多个ADC通道,比如IN5、IN6、IN7,如引用1提到的。ADC的工作模式应为多通道扫描模式,并且连续转换应禁用,因为需要定时器触发。触发源应选择定时器的触发输出,如TIM2的TRGO。此外,DMA需要配置为循环模式,以持续传输数据。注意数据对齐和转换顺序,确保通道顺序正确。 DMA的设置也很关键。引用2和4提到使用HAL_ADC_Start_DMA函数启动传输。DMA需要设置为从外设(ADC)到内存,循环模式,并确保数据宽度匹配(半字或字)。同时,需要使能DMA中断,以便在传输完成时处理数据。 然后,用户可能需要生成代码后添加自己的处理逻辑,比如在DMA完成回调中处理数据,如引用4中的HAL_ADC_ConvCpltCallback函数。需要注意的是,如果定时器和DMA配置不当,可能会出现DMA中断频繁触发的问题,如引用3所述,这可能与定时器的配置或DMA模式有关。 在示例代码部分,需要展示初始化顺序,先启动定时器再启动ADC的DMA传输。同时,用户可能需要使用HAL库的函数来启动转换,并在回调函数中处理数据。例如,启动DMA传输后,定时器会周期性地触发ADC转换,转换结果通过DMA传输到内存数组,然后通过回调函数通知应用程序处理数据。 最后,需要提醒用户验证配置是否正确,比如检查定时器的触发信号是否正常,DMA传输是否按预期工作,以及数据是否正确。如果遇到问题,可以参考引用中的示例,特别是关于定时器触发和DMA配置的部分,避免配置错误导致的中断异常或数据错误。 总结步骤:配置定时器触发源→配置ADC多通道和触发方式→设置DMA循环传输→生成代码并添加处理逻辑→测试和调试。确保各部分协同工作,避免常见问题如DMA中断频率异常。</think>### STM32CubeMX配置ADC定时器触发多通道DMA传输教程 #### 一、配置流程 1. **定时器配置(以TIM2为例)** - 模式选择:`Internal Clock` - 触发事件(TRGO):选择`Update Event`作为ADC触发源[^2] - 预分频器(Prescaler):根据采样频率需求设置 - 自动重载值(Counter Period):计算采样周期 $$ \text{采样频率} = \frac{\text{定时器时钟}}{\text{(Prescaler+1) \times (Counter Period+1)}} $$ 2. **ADC配置** - 模式选择:`Multi-channel Scan Mode` - 触发源:选择`Timer 2 Trigger Out Event` - 通道设置:依次添加需要采集的ADC通道(如IN5/IN6/IN7)[^1] - DMA设置:`Add` DMA通道,模式为`Circular`,数据宽度`Word`或`Half Word`[^4] 3. **DMA参数** - 方向:`Peripheral To Memory` - 优先级:`High` - 内存地址增量:`Enable`(多通道必须开启) - 外设地址固定:`Enable` - 数据长度:`Word`(若ADC为12位分辨率,可用`Half Word`) #### 二、代码实现(示例) ```c // 变量定义(以3通道为例) #define ADC_CHANNEL_NUM 3 uint32_t adc_buffer[ADC_CHANNEL_NUM * 4]; // 每个通道采集4次 // 定时器启动 HAL_TIM_Base_Start(&htim2); // ADC DMA启动 HAL_ADC_Start_DMA(&hadc1, adc_buffer, ADC_CHANNEL_NUM * 4); // DMA传输完成回调函数 void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) { if(hadc == &hadc1) { // 数据处理(示例:取平均值) uint32_t ch1_avg = (adc_buffer[0] + adc_buffer[3] + adc_buffer[6] + adc_buffer[9]) / 4; // 其他处理逻辑... } } ``` #### 三、注意事项 1. **时序同步问题**:定时器触发周期应大于ADC转换时间,计算公式: $$ T_{conv} = (\text{采样周期} + 12.5) \times \frac{1}{ADC\_CLK} $$ 2. **内存对齐**:DMA缓冲区大小需为`通道数 × 采样次数`的整数倍 3. **中断冲突**:若出现DMA中断频繁触发,检查定时器配置是否产生意外触发事件[^3] #### 四、验证步骤 1. 使用逻辑分析仪检测TIM2输出波形 2. 在DMA完成中断设置断点观察数据更新 3. 通过串口输出ADC原始数据验证采集顺序
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

叶与花语

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

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

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

打赏作者

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

抵扣说明:

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

余额充值