开发板(正点原子stm32f407探索者开发板V2.4)
ST-LINK/V2驱动
STM32CubeMX软件(Version 6.10.0)
keil µVision5 IDE(MDK-Arm)
CH340G Windows系统驱动程序(CH341SER.EXE)
XCOM V2.6串口助手
1个滑动变阻器
使用STM32CubeMX软件配置STM32F407开发板的ADC实现单通道ADC采集,具体为使用ADC1_IN5通道通过软件/定时器触发采集滑动变阻器上的分压
ADC即模拟数字转换,是将模拟电压量转换为数字量的一种手段,如下图所示为STM32F407单个ADC的结构框图(注释1),其主要包括5个部分,分别为①ADC电源引脚、②ADC输入引脚、③ADC触发源、④ADC转换规则和⑤ADC中断触发,下面从这五个方面分别介绍STM32F407的ADC
①ADC电源引脚有VDDA、VSSA、VREF+和VREF-四个引脚,STM32的ADC采集范围为VREF- ≤ VIN ≤ VREF+,一般将ADC的负端参考电压与VSSA短接然后接地,将ADC的正端参考电压接VDDA即3.3V,这样ADC的采集范围就设置为0~3.3V,对于12位分辨率的ADC,其采集数字量范围为0-4095,这样就可以将0-3.3V的电压映射到0-4095的数字量,使其可以相互转化,ADC电源即参考电压引脚具体描述如下图所示(注释1)
②STM32F407有三个ADC可供使用,每个ADC又拥有16个通道ADCx_IN[15:0],其中ADC1还拥有Temperature Sensor Channel、Vrefint Channel和Vbat Channel三个内部通道,Temperature Sensor Channel通道用于测量芯片内部温度,范围为-40℃~125℃,精度为±1.5℃,Vrefint Channel用于测量内部参考电压,Vbat Channel用于测量备用电源电压的一半,ADC的转换主要依靠12位分辨率的片上模数转换器
③注入通道和规则通道均有16个触发源,可以选择定时器外部源触发/定时器比较捕获触发/软件常规触发,具体参看上图ADC框架中的紫色框框
④ADC启动转换时需要按照一定通道顺序转化,该顺序由规则通道和注入通道两者共同决定,其中注入通道其数据寄存器为4*16位,因此可以同时转换四个通道,但是规则通道的数据寄存器只有一个16位的寄存器,因此必须一个通道一个通道的转换,每转换完一个通道,就需要及时将转换完成的结果从规则通道数据寄存器中读出去,其中注入通道类似ADC通道转换过程的中断,如下图所示(注释2)
⑤ADC的中断事件有DMA溢出、ADC转换结束、注入转换结束和模拟看门狗事件共四个事件,前三个中断和其名字表述类似,当转换结束/溢出时就会产生中断,模拟看门狗可以设置ADC转换值的上限和下限,当超出限制之后就会产生中断,可以用于警报,如下表所示(注释1)
ADC有独立模式、二重和三重采集模式,当只有ADC1启动时只能使用独立模式,当ADC1/2启动时可以使用二重采集模式,当ADC1/2/3/全部启动时可以使用三重采集模式,本实验只介绍独立模式
本实验为ADC独立模式单通道单次转换模式,主要利用ADC1_IN5通道由软件/定时器启动ADC转换,如果是软件手动启动的ADC转换则在单次转换模式下每次转换完成一次之后均需要再次手动启动ADC转换,另外需要注意ADC在开始精确转换之前需要一段稳定时间tSTAB,如下图所示为ADC转化所经过的路径
请先阅读“STM32CubeMX教程1 工程建立”实验3.4.1小节配置RCC和SYS
系统时钟树配置均设置为STM32F407总线能达到的最高时钟频率,具体如下图所示
本实验需要需要初始化USART1作为输出信息渠道,具体配置步骤请阅读“STM32CubeMX教程9 USART/UART 异步通信”
设置TIM3通用定时器溢出时间100ms,外部触发事件选择更新事件,参数详解请阅读“STM32CubeMX教程6 TIM 通用定时器 - 生成PWM波”实验,具体配置如下图所示
在Pinout & Configuration页面左边功能分类栏目Analog中单击其中ADC1,在Mode中勾选需要使用的输入通道,本实验为单通道转换实验,因此任意勾选IN0~IN15之间的任一通道均可,笔者勾选了IN5
在Configuration中对ADC1_IN5的转换参数进行配置,下面介绍一些比较重要的参数
------------------ ADCs_Common_Settings ------------------
①Mode(模式):现在只能选择独立模式,当同时启用了ADC1/2或ADC1/2/3时这里会出现多重ADC采集的模式可选
------------------------- ADC_Settings -------------------------
②Clock Prescaler(时钟分频):决定ADC转换的频率,分频越少ADC转换的频率越高,最少2分频,一个通道一次ADC转换的总时间为N+12个ADC时钟周期,其中N为设置的采样次数Cycles
③Resolution(ADC转换精度):可以选择12/10/8/6位精度的转换值,精度选择不同需要的转换时钟周期也不同
④Data Alignment(数据对齐):由于规则数据寄存器为16位,但是最高转换精度为12位,因此数据可以选择以右对齐/左对齐的方式放入寄存器
⑤Scan Conversion Mode(扫描转换模式):规则通道同时只能转换一个通道,启用该参数后,当规则通道中有多个通道等待转换时其转换完当前通道会自动转换组内的下一个通道
⑥Continuous Conversion Mode(连续转换模式):启用该参数,ADC结束一个转换立即开始一个新的转换,与参数⑤共同启用,则组内最后一个通道转换完毕后会立即切换到第一个通道继续转换
⑦DMA Continuous Requests(DMA请求):需要在DMA Settings中添加DMA请求后,该参数才可以使能
⑧End Of Conversion Selection(结束转换标志):选择是一个通道转换完就产生EOC标志,还是一个组内所有通道全部转换完才产生EOC标志
------------------ ADC_Regular_ConversionMode------------------
⑨NumberOfConversion(转换通道数量):常规规则通道希望转换的通道数量
⑩External Trigger Conversion Source(外部触发转换源)
⑪External Trigger Conversion Edge(外部触发转换边沿)
⑫Rank(规则通道排序)
------------------ ADC_Injected_ConversionMode------------------
⑬NumberOfConversion(注入通道转换模式通道数量):注入通道希望转换的通道数量
具体参数配置如下图所示
在Pinout & Configuration页面左边System Core/NVIC中勾选ADC1/2/3全局中断,然后选择合适的中断优先级即可,步骤如下图所示
请先阅读“STM32CubeMX教程1 工程建立”实验3.4.3小节配置Project Manager
单击页面右上角GENERATE CODE生成工程
在生成的工程代码主函数main()中调用了MX_ADC1_Init()函数完成了对ADC1基本参数的配置,ADC常规规则通道/注入通道等参数配置
然后在ADC初始化函数HAL_ADC_Init()函数中调用了HAL_ADC_MspInit()函数对ADC1时钟和中断进行了使能,对中断优先级进行了配置,对ADC1_IN5输入引脚做了复用操作
如下图所示为ADC1单通道初始化的具体函数调用流程
使能ADC1/2/3全局中断后在stm32f4xx_it.c中自动生成了TIM4的中断处理函数ADC_IRQHandler()
ADC_IRQHandler()调用了HAL库的ADC中断处理函数HAL_ADC_IRQHandler(),该函数处理所有的ADC中断事件
在ADC转换完成之后最终调用了ADC转换完成中断回调函数HAL_ADC_ConvCpltCallback(),该函数为虚函数
如下图所示为ADC1单通道转换中断回调的具体函数调用流程
在adc.c中重新实现ADC转换完成中断回调函数HAL_ADC_ConvCpltCallback(),在该函数中获取ADC的转换值,然后将其计算为电压值*1000,并将这两个值通过串口输出,具体代码如下图所示
源代码如下
/*转换完成中断回调*/ void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc) { /*定时器中断启动单通道转换*/ if(hadc->Instance == ADC1) { uint32_t val=HAL_ADC_GetValue(&hadc1); uint32_t Volt=(3300*val)>>12; printf("val:%d, Volt:%d\r\n",val,Volt); } }
在主函数中以中断方式启动ADC转换,然后启动ADC的触发源TIM3定时器,具体代码如下图所示
/*启动ADC转换*/ HAL_StatusTypeDef HAL_ADC_Start(ADC_HandleTypeDef *hadc) /*停止ADC转换*/ HAL_StatusTypeDef HAL_ADC_Stop(ADC_HandleTypeDef *hadc) /*以中断方式启动ADC转换*/ HAL_StatusTypeDef HAL_ADC_Start_IT(ADC_HandleTypeDef *hadc) /*停止ADC转换*/ HAL_StatusTypeDef HAL_ADC_Stop_IT(ADC_HandleTypeDef *hadc) /*轮询ADC是否转换完毕*/ HAL_StatusTypeDef HAL_ADC_PollForConversion(ADC_HandleTypeDef *hadc, uint32_t Timeout) /*获取ADC转换值*/ uint32_t HAL_ADC_GetValue(ADC_HandleTypeDef *hadc) /*ADC转换完成中断回调函数*/ void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef *hadc)
“配置USART1用于输出信息 -> 配置通用定时器TIM3实现100ms定时 -> TIM3外部触发事件选择为更新事件 -> 启动ADC1的通道5并配置相关参数 -> 外部触发转换源选择TIM3的外部触发事件 -> 启动ADC1/2/3全局中断 -> 重新实现ADC转换完成中断回调HAL_ADC_ConvCpltCallback()函数 -> 在回调函数中读取ADC转换值并通过串口输出 -> 在主函数中启动定时器和ADC转换”
烧录程序,上电后打开串口,串口会每100ms传来一次ADC采集的数据,旋转滑动变阻器从一端到另一端,可以看到ADC采集到的值从0逐渐变为最大值4095
如果你希望不采用定时器触发中断采集的方式,而是想使用软件手动触发轮询采集的方式,需要先将ADC规则转换模式中触发源修改为软件触发,然后使用HAL_ADC_Start()启动转换,使用HAL_ADC_PollForConversion()轮询检测是否转换完成,转换完成后使用可以HAL_ADC_GetValue()获取转换后的ADC值,具体参考如下图所示
源代码如下
HAL_ADC_Start(&hadc1); if(HAL_ADC_PollForConversion(&hadc1,200)==HAL_OK) { uint32_t val=HAL_ADC_GetValue(&hadc1); uint32_t Volt=(3300*val)>>12; printf("val:%d, Volt:%d\r\n",val,Volt); } HAL_Delay(500);
注释1:图片来自STM32F4xx中文参考手册 RM0090
注释2:图片来自【STM32】HAL库 STM32CubeMX教程九---ADC_cubemx adc-CSDN博客
STM32Cube高效开发教程(基础篇)
更多内容请浏览 OSnotes的CSDN博客