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

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


前言

本章介绍使用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配置HAL库实现ADC转换的资料,特别提到可能与"Dir_xr"相关。虽然用户提供的两篇引用都不直接对应ADC(分别是DAC和GPIO输入),但值得注意其中一篇提到了DAC的"DHRx寄存器"——这个命名规则和ADC的"DR寄存器"很像,用户可能是记混了术语。 用户显然在开发STM32的模拟信号采集功能,重点需要知道三点:CubeMXADC配置流程、"Dir_xr"的正确含义、以及HAL库的代码实现逻辑。考虑到用户提到"Dir_xr"时加了括号说明"可能与Dir_xr相关",说明用户对这个术语也不完全确定,需要特别澄清。 从技术角度看,ADC配置的核心其实是:1)时钟和通道的基础配置 2)采样周期设置 3)触发方式选择(软件/定时器等)4)DMA或中断的数据传输。用户可能遇到的"Dir_xr"应该是ADC的DR(Data Register)寄存器,HAL库中通常使用HAL_ADC_GetValue()函数来读取它。 需要重点提醒用户避免两个常见错误:一是ADC通道与GPIO复用功能的对应关系容易配错,二是当使用规则组多通道采样时忘记设置扫描模式。在代码示例中应该演示单通道轮询这种最基础场景,同时补充DMA方式的优势说明。 用户之前没有提到具体型号,但STM32F4系列最典型,就以它为例。附加问题特意加入了校准和电压计算两个实用点——前者影响精度,后者是实际开发中最容易被忽略的转换步骤。 </think> 好的,关于使用 STM32CubeMX 配置 STM32 HAL 库进行 ADC (模数转换) 转换的资料,特别是与**数据寄存器 (DR)** 相关的部分(您提到的“Dir_xr”极有可能是指数据寄存器 `DR`,例如 `ADCx->DR`),以下是关键信息和步骤总结: **核心概念:ADC 与数据寄存器 (DR)** STM32ADC 模块将模拟信号(如电压)转换为数字值。转换完成后的数字结果存储在一个特定的寄存器中,称为 **数据寄存器 (Data Register, DR)**。在 STM32 HAL 库和参考手册中,这个寄存器通常表示为 `ADCx->DR`(其中 `x` 是 ADC 的编号,如 ADC1, ADC2, ADC3)。 * **作用:** `DR` 是 ADC 转换结果的最终存放位置。无论是单次转换、连续转换还是扫描模式,转换完成后的数字值都会被写入对应的 `DR`。 * **读取方式:** * **轮询 (Polling):** 在主循环中不断检查 ADC 转换完成标志位 (`EOC`),当标志置位时,读取 `DR` 的值。使用 HAL 库函数 `HAL_ADC_GetValue(&hadcx)` 本质上就是读取 `DR` 寄存器。 * **中断 (Interrupt):** 配置 ADC 转换完成中断 (`EOC`)。在中断服务程序 (ISR) 中,调用 `HAL_ADC_IRQHandler(&hadcx)`,该函数会处理中断标志并调用转换完成回调函数 `HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc)`。在这个回调函数中,您可以安全地调用 `HAL_ADC_GetValue(&hadcx)` 读取 `DR` 的值。 * **DMA (直接存储器访问):** 这是最常用且高效的方式。配置 DMA 通道,将其源地址设为 ADC 的 `DR` 寄存器地址 (`&ADCx->DR`),目标地址设为用户定义的数组。当每次 ADC 转换完成时,DMA 会自动把 `DR` 中的数据传输到目标数组,无需 CPU 干预。HAL 库函数 `HAL_ADC_Start_DMA(&hadcx, pData, Length)` 就是启动这种传输模式。 **使用 STM32CubeMX 配置 ADC 的步骤:** 1. **选择 ADC 通道:** * 在 CubeMX 的 `Pinout & Configuration` 视图下,找到您的 MCU 型号。 * 展开左侧 `Analog` 菜单下的 `ADCx` (ADC1, ADC2)。 * 在 `INx` (Input Channel x) 中选择您需要使用的模拟输入通道(如 `IN0`, `IN1` 等)。选择通道后,CubeMX 会自动配置对应的 GPIO 引脚为模拟输入模式 (`Analog`)。 2. **配置 ADC 参数:** * 在 `Configuration` 标签页下,点击 `ADCx` 进入参数配置。 * **Settings 选项卡:** * `Clock Prescaler`: ADC 时钟分频器 (选择适当的时钟频率,参考数据手册最大值)。 * `Resolution`: ADC 分辨率 (如 12 bits, 10 bits, 8 bits, 6 bits)。决定了转换结果的精度和范围 (0-4095 for 12bits)。 * `Data Alignment`: 数据对齐方式 (`Right` 或 `Left`)。通常选 `Right Alignment`。 * `Scan Conversion Mode`: 扫描模式。如果使用多个通道,需设为 `Enabled`。 * `Continuous Conversion Mode`: 连续转换模式。设为 `Enabled` 则 ADC 转换完成后立即开始下一次转换;`Disabled` 则为单次模式,需软件或触发启动每次转换。 * `Discontinuous Conversion Mode`: 间断模式。通常保持 `Disabled`。 * `DMA Continuous Requests`: 如果使用 DMA,且希望 DMA 在连续转换模式下一次配置就能传输所有数据,设为 `Enabled`。 * `End Of Conversion Selection`: 转换结束标志选择。通常保持默认 (`EOC flag at the end of single channel conversion`)。 * **Regular Conversion 选项卡 (规则组转换):** * 点击 `Add` 添加规则转换通道。 * 选择之前在 Pinout 视图选中的 `Channel`。 * 设置 `Rank` (转换顺序,1 为最先转换)。 * 设置 `Sampling Time`: 采样时间(通道采样保持周期)。根据信号源阻抗和精度要求选择(时间越长,精度通常越高,但转换速度越慢)。参考数据手册推荐值。 3. **配置触发源 (Trigger):** * 在 `Regular Conversion` 选项卡的 `External Trigger Conversion Source` 中,选择 ADC 转换的启动触发源: * `Software start`: 软件触发 (`HAL_ADC_Start(...)`, `HAL_ADC_Start_IT(...)`, `HAL_ADC_Start_DMA(...)`). * `Timer x Trigger Out event`: 定时器触发 (如 TIMx_TRGO)。常用于周期性采样。 4. **配置 DMA (推荐用于规则组多通道或连续转换):** * 在 `DMA Settings` 选项卡 (通常在 ADCx 配置窗口下方或 ADC 模块配置的独立选项卡),点击 `Add`。 * 选择 `DMA Request` 对应的 ADCx (如 `ADC1`). * 配置 DMA 流/通道 (Stream/Channel,根据 MCU 型号)。 * 设置传输方向:`Peripheral To Memory`。 * 设置优先级:`Medium` 或 `High`。 * 配置 DMA 参数: * `Mode`: `Circular` (循环模式,数据覆盖缓冲区末尾后回到开头) 或 `Normal` (传输指定长度后停止)。 * `Increment Address`: `Memory` 设为 `Enabled` (目标内存地址递增),`Peripheral` 设为 `Disabled` (源地址 `DR` 寄存器固定)。 * `Data Width`: `Word` (如果 ADC 分辨率 <= 16 bits,通常选 `Half Word` 就足够)。 5. **配置中断 (如果使用中断或 DMA):** * 在 `NVIC Settings` 选项卡中,使能对应的中断: * 如果使用轮询读取 `DR` 或仅软件触发,可能不需要中断。 * 如果使用转换完成中断读取 `DR`,需使能 `ADCx` 全局中断 (`ADCx` global interrupt)。 * 如果使用 DMA 传输,通常需要使能 `DMAx Streamy` 全局中断 (`DMAx Streamy global interrupt`),以便在 DMA 传输完成、传输一半或出错时产生中断。 6. **生成代码:** * 完成所有配置后,点击 `Project Manager` 选项卡配置项目名称、位置、IDE 等。 * 回到 `Code Generator` 选项卡,选择适当的配置。 * 点击 `Generate Code`。 **在代码中访问 ADC 转换结果 (`DR`):** CubeMX 生成的代码会初始化 ADC 和 DMA (如果配置)。 * **轮询方式:** ```c HAL_ADC_Start(&hadc1); // 启动ADC转换 (软件触发) if (HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY) == HAL_OK) { // 等待转换完成 uint16_t adc_value = HAL_ADC_GetValue(&hadc1); // 读取DR寄存器的值 // 处理 adc_value ... } ``` * **中断方式:** ```c void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) { if (hadc->Instance == ADC1) { // 确保是ADC1的回调 uint16_t adc_value = HAL_ADC_GetValue(hadc); // 读取DR寄存器的值 // 处理 adc_value ... } } // 在主程序初始化后启动带中断的转换 HAL_ADC_Start_IT(&hadc1); ``` * **DMA 方式 (最常见):** ```c #define ADC_BUFFER_LEN 100 uint16_t adc_buffer[ADC_BUFFER_LEN]; // 存储ADC结果的数组 int main(void) { // ... CubeMX生成的初始化代码 ... HAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_buffer, ADC_BUFFER_LEN); // 启动ADC DMA传输 while (1) { // 主循环。ADC结果会自动填充到adc_buffer数组中。 // 可以通过DMA传输完成中断 (HAL_ADC_ConvCpltCallback) 或半传输中断 (HAL_ADC_ConvHalfCpltCallback) 来通知数据处理, // 或者直接在循环中安全地访问已传输的数据部分。 } } // DMA传输完成回调函数示例 void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) { if (hadc->Instance == ADC1) { // 此时 adc_buffer[0] 到 adc_buffer[ADC_BUFFER_LEN-1] 已填充最新的ADC(来自DR寄存器) // 处理完整缓冲区数据 ... } } ``` **关键点总结:** 1. ADC 转换结果存储在 **数据寄存器 (DR)** 中 (`ADCx->DR`)。 2. **STM32CubeMX配置 ADC 硬件参数(通道、采样时间、触发、DMA)的图形化工具。** 3. **HAL 库提供了访问 ADC 和 `DR` 的抽象函数:** * `HAL_ADC_GetValue(&hadcx)` 是读取单个 `DR` 值的标准方法。 * DMA 方式是将 `DR` 寄存器 (`&ADCx->DR`) 直接链接到用户内存数组的最优方式。 4. **根据应用需求选择读取 `DR` 的方式:** 轮询 (简单低效)、中断 (实时性要求中等)、DMA (高效,多通道/连续转换必备)。 5. **触发源决定转换何时开始:** 软件启动或硬件定时器触发是常见方式。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

叶与花语

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

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

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

打赏作者

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

抵扣说明:

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

余额充值