STM32 AD多通道函数设计

2024-03-25 22:36
文章标签 设计 函数 stm32 ad 多通道

本文主要是介绍STM32 AD多通道函数设计,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

单片机学习!

目录

文章目录

前言

一、ADC配置步骤

二、详细步骤

2.1 开启RCC时钟

2.2 配置GPIO

2.3 配置多路开关

2.4 配置ADC转换器

2.5 开启ADC电源

2.6 ADC进行校准

        2.6.1 复位校准

        2.6.2 等待复位校准完成

        2.6.3 开始校准

2.6.4 等待校准完成

三、启动AD转换函数设计

3.1 软件触发转换

3.2 等待转换完成

3.3 读取ADC数据寄存器

3.4 启动AD转换函数块总结

总结


前言

        之前博文介绍AD单通道函数设计的基础内容,本文在之前博文的基础上做一修改,用简单的方法实现AD多通道函数设计:利用单次转换,非扫描的模式来实现AD多通道的设计。这样只需要在每次触发之前手动更改一下列表第一个位置的通道就行,如第一次转换,先写入通道0,之后触发、等待、读值;第二次转换,先把通道0改成通道1,之后触发、等待、读值;第三次转换,再改为通道2,之后触发、等待、读值,以此类推。这样在转换前先指定一下通道,再启动转换就可以简单轻松地实现多通道转换的功能了。


一、ADC配置步骤

        参照结构框图,在原理上将ADC外设运转起来。

第一步:开启RCC时钟,包括ADC和GPIO的时钟。另外ADCCLK的分频器也需要配置。

第二步:配置GPIO,把需要用到的GPIO配置成模拟输入的模式。

第三步:配置多路开关,把通道接入到规则组列表里。

第四步:配置ADC转换器,库函数中用结构体来配置电路参数,包括ADC是单次转换还是连续转换;扫描还是非扫描;有几个通道;触发源是什么;数据对齐是左对齐还是右对齐。

  • 如果需要模拟看门狗,可以用几个库函数来配置阈值和监测通道;
  • 如果需要开启中断,在中断输出控制里用 ITConfig 函数开启对应的中断输出,然后再在NVIC里配置一下优先级。这样就能触发中断了。

最后:开关控制,调用一下ADC_Cmd函数,开启ADC.

        在开启ADC之后,根据STM32手册的建议,还可以对ADC进行一下校准,这样可以减小误差。

        在ADC工作的时候,如果想要软件触发转换,有库函数可以触发;如果想读取转换结果,也会有函数可以读取结果。

二、详细步骤

2.1 开启RCC时钟

        第一步:开启RCC时钟。包括ADC和GPIO的时钟。另外ADCCLK的分频器也需要配置。

代码示例:

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);//开启ADC1的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//还需要开启PA0口的时钟RCC_ADCCLKConfig(RCC_PCLK2_Div6);//配置ADCCLK

        RCC_APB2PeriphClockCmd函数开启ADC1的时钟,ADC都是APB2上的设备,所以这里用APB2开启时钟的函数。

        RCC_APB2PeriphClockCmd函数用于开启PA0口的时钟。

        以上时钟就配置好了,还需要配置ADCCLK,用RCC_ADCCLKConfig/函数来配置,函数有四个参数分别是2、4、6、8分频:

  • RCC_PCLK2_Div2: ADC clock = PCLK2/2
  • RCC_PCLK2_Div4: ADC clock = PCLK2/4
  • RCC_PCLK2_Div6: ADC clock = PCLK2/6
  • RCC_PCLK2_Div8: ADC clock = PCLK2/8

        函数参数配置好之后,ADC的CLOCK=PCLK2/2、PCLK2/4、PCLK2/6、PCLK2/8,参数的PCLK2就是APB2时钟的意思。

        代码示例中选择6分频,分频之后,ADCCLK=72MHz/6=12MHz

2.2 配置GPIO

        第二步:配置GPIO,把需要用到的GPIO配置成模拟输入的模式。

代码示例:

    GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.GPIO_Mode= GPIO_Mode_AIN;//选择模拟输入GPIO_InitStruct.GPIO_Pin= GPIO_Pin_0;GPIO_InitStruct.GPIO_Speed= GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStruct);

        代码示例中选择GPIO_Mode_AIN模拟输入这个模式,在GPIO_Mode_AIN模式下,GPIO是无效的,断开GPIO口,防止GPIO口的输入输出对模拟电压造成干扰。GPIO_Mode_AIN模式就是ADC的专属模式。

2.3 配置多路开关

        第三步:配置多路开关,把通道接入到规则组列表里。但是这里为了实现AD多通道的功能,在ADC的配置步骤中这一步的代码不放在这里,而是放在本文的 “三、启动AD转换函数设计” 中。

        用ADC_RegularChannelConfig函数可选择规则组的输入通道。

函数参数:

  • 第1个参数:选择ADC。
  • 第2个参数:指定通道,通道0~通道17.
  • 第3个参数:Rank,规则组序列器里的次序,在1~16之间。若只有PA0一个通道,使用的是非扫描模式,那指定的通道就放在第一个序列1的位置。
  • 第4个参数:指定通道的采样时间,采样时间参数根据需求调整,需要更快的转换,就选择小的参数;需要更稳定的转换,就选择大的参数。

        代码示例的配置是:再ADC1中,在规则组菜单列表的第一个位置,写入通道0这个通道,指定通道的采样时间参数选择的采样时间为55.5个ADCCLK的周期。

        如果想在序列2的位置写入其他通道,就可以复制一下这个代码,把序列数改成2,然后指定你想要的通道,若还需要继续填充序列,可以再复制这个函数,修改序列和通道,另外每个通道也可以设置不同的采样时间,在函数最后一个参数修改即可。

2.4 配置ADC转换器

        第四步、用结构体初始化ADC。

代码示例:

    ADC_InitTypeDef ADC_InitStructure;ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;//ADC的工作模式ADC_InitStructure.ADC_DataAlign=ADC_DataAlign_Right;//数据对齐ADC_InitStructure.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None;//外部触发转换选择ADC_InitStructure.ADC_ContinuousConvMode=DISABLE;//连续/单次转换模式ADC_InitStructure.ADC_ScanConvMode=DISABLE;//扫描/非扫描转换模式ADC_InitStructure.ADC_NbrOfChannel=1;//通道数目ADC_Init(ADC1,&ADC_InitStructure);

        ADC_Mode 是ADC的工作模式,这个参数是配置ADC是工作在独立模式还是双ADC模式,代码示例选择独立模式。

        ADC_DataAlign 数据对齐,这里介绍是指定ADC数据是左对齐还是右对齐。

  • ADC_DataAlign_Right右对齐;
  • ADC_DataAlign_Left左对齐 

        ADC_ExternalTrigConv 外部触发转换选择,就是触发控制的触发源,定义用于启动规则组转换的外部触发源。参数对应结构框图中的外部触发源选择。代码示例选择 ADC_ExternalTrigConv_None 参数,就是不使用外部触发,也就是使用内部软件触发的意思。

        ADC_ContinuousConvMode 连续/单次转换模式,这个参数可以选择是ENABLE连续转换模式还是DISABLE单次转换模式。

        ADC_ScanConvMode 扫描/非扫描转换模式,这个参数可以选择是ENABLE扫描模式(多通道)还是DISABLE非扫描模式(单通道)。

        ADC_NbrOfChannel 通道数目,这个是在指定扫描模式下,总共会有几个通道需要扫描,参数必须在1~16之间。这个参数仅在扫描模式下使用,因为非扫描模式整个列表就只有第一个序列有效,无论写多少数目,最终都只有序列1的位置有效。

        后三个参数设置可以对应四种模式:

  • 单次转换非扫描。
  • 连续转换非扫描。
  • 单次转换扫描。
  • 连续转换扫描。

2.5 开启ADC电源

        第五步、开关控制,调用一下ADC_Cmd函数,开启ADC.

代码示例:

ADC_Cmd(ADC1,ENABLE);

        以上配置完后ADC准备就绪。

2.6 ADC进行校准

        在开启ADC电源之后,根据手册的建议,还需要对ADC进行校准,校准分为以下四步。

  1. 复位校准
  2. 等待复位校准完成
  3. 开始校准
  4. 等待校准完成

        2.6.1 复位校准

代码示例:

ADC_ResetCalibration(ADC1);//复位校准

        2.6.2 等待复位校准完成

代码示例:

while(ADC_GetResetCalibrationStatus(ADC1)==SET);

        ADC_GetResetCalibrationStatus函数是返回复位校准的状态,要等待复位完成的话,还需要加一个while循环,若没校准完成的话,就在这个while空循环里一直等待。

        获取的标志位和是否校准完成的对应关系需参考函数定义寄存器说明

函数定义:

        ADC_GetResetCalibrationStatus函数定义中返回值的说明是,ADC复位校准寄存器的状态,SET或RESET。

函数代码:

/*** @brief  Gets the selected ADC reset calibration registers status.* @param  ADCx: where x can be 1, 2 or 3 to select the ADC peripheral.* @retval The new state of ADC reset calibration registers (SET or RESET).*/
FlagStatus ADC_GetResetCalibrationStatus(ADC_TypeDef* ADCx)
{FlagStatus bitstatus = RESET;/* Check the parameters */assert_param(IS_ADC_ALL_PERIPH(ADCx));/* Check the status of RSTCAL bit */if ((ADCx->CR2 & CR2_RSTCAL_Set) != (uint32_t)RESET){/* RSTCAL bit is set */bitstatus = SET;}else{/* RSTCAL bit is reset */bitstatus = RESET;}/* Return the RSTCAL bit status */return  bitstatus;
}

        在此函数代码中也可以看出来,它获取的就是CR2寄存器里的RSTCAL标志位。

寄存器说明:

        在ADC的CR2寄存器里RSTCAL复位校准位的说明是该位由软件设置并由硬件清除,在校准寄存器被初始化后该位将被清除。

        所以该标志位的用法就是软件置改位为1,那硬件就会开始复位校准,当复位校准完成后,该位就会由硬件自动清0.

        因为校准的第一条代码ADC_ResetCalibration(ADC1);开始复位校准,就是将RSTCAL标志位置1,然后获取复位校准状态,就是读取RSTCAL标志位这一位,所以在读取这一位的时候:

  • 如果它是1,那就需要一直空循环等待;
  • 如果它变为0了,那就说明复位校准完成,可以跳出等待了。

        所以校准第二条代码while(ADC_GetResetCalibrationStatus(ADC1)==SET);中while的条件就是,获取标志位函数 ADC_GetResetCalibrationStatus 的返回值是不是==SET,如果等于SET,while条件为真,就会一直空循环。一旦标志位被硬件清0了,这个空循环就会自动跳出来。这样就实现了等待复位校准完成的效果。这里==SET也是可以省略的,因为返回值SET直接作为条件和是不是==SET作为条件效果是一样的。

        2.6.3 开始校准

代码示例:

ADC_StartCalibration(ADC1);

        调用ADC_StartCalibration函数就开始校准了,之后内部电路就会自动进行校准。

2.6.4 等待校准完成

    while(ADC_GetCalibrationStatus(ADC1)==SET);//调用函数获取校准状态

        调用ADC_GetCalibrationStatus函数获取校准状态,也需要将函数放于while循环内,和校准的第二步同理,循环条件是,校准标志位是不是==SET,这样就可以等待校准是否完成了。

三、启动AD转换函数设计

        在启动AD转换前,为了实现AD多通道的转换,将上文中的 “2.3 配置多路开关” 的代码放于启动函数设计的第一句。

代码示例:

    ADC_RegularChannelConfig(ADC1,ADC_Channel,1,ADC_SampleTime_55Cycles5);

        代码中第二个参数,指定通道,设置为函数源代码中的参数。

函数源代码:

void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime)

        上文描述的代码设置使ADC已处于准备就绪的状态,以下设计启动转换,获取转换结果的函数块。

        函数块里执行以下流程:

  • 第一步、软件触发转换。
  • 第二步、等待转换完成。
  • 第三步、读取ADC数据寄存器。

3.1 软件触发转换

        调用ADC_SoftwareStartConvCmd函数,实现软件触发。

代码示例:

ADC_SoftwareStartConvCmd(ADC1,ENABLE);

        调用ADC_SoftwareStartConvCmd函数之后就可以触发,ADC就已经开始进行转换了。转换需要一段时间。

3.2 等待转换完成

        等待转换完成,也就是等待EOC标志位置1.

代码示例:

    while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET);

        ADC_GetFlagStatus是获取标志位状态的函数,函数的第二个参数可选择以下参数:

  • ADC_FLAG_AWD: 模拟看门狗标志位
  • ADC_FLAG_EOC: 规则组转换完成标志位
  • ADC_FLAG_JEOC:注入组转换完成标志位
  • ADC_FLAG_JSTRT: 注入组开始转换标志位
  • ADC_FLAG_STRT: 规则组开始转换标志位

        这一步也需要套一个while空循环来实现一个等待的过程,返回标志位与转换是否完成的对应关系需参考函数定义和STM32手册的寄存器说明.

函数定义:

        ADC_GetFlagStatus函数定义中返回值的说明是,ADC状态寄存器的状态,SET或RESET。

函数代码:

/*** @brief  Checks whether the specified ADC flag is set or not.* @param  ADCx: where x can be 1, 2 or 3 to select the ADC peripheral.* @param  ADC_FLAG: specifies the flag to check. *   This parameter can be one of the following values:*     @arg ADC_FLAG_AWD: Analog watchdog flag*     @arg ADC_FLAG_EOC: End of conversion flag*     @arg ADC_FLAG_JEOC: End of injected group conversion flag*     @arg ADC_FLAG_JSTRT: Start of injected group conversion flag*     @arg ADC_FLAG_STRT: Start of regular group conversion flag* @retval The new state of ADC_FLAG (SET or RESET).*/
FlagStatus ADC_GetFlagStatus(ADC_TypeDef* ADCx, uint8_t ADC_FLAG)
{FlagStatus bitstatus = RESET;/* Check the parameters */assert_param(IS_ADC_ALL_PERIPH(ADCx));assert_param(IS_ADC_GET_FLAG(ADC_FLAG));/* Check the status of the specified ADC flag */if ((ADCx->SR & ADC_FLAG) != (uint8_t)RESET){/* ADC_FLAG is set */bitstatus = SET;}else{/* ADC_FLAG is reset */bitstatus = RESET;}/* Return the ADC_FLAG status */return  bitstatus;
}

寄存器说明:

        在ADC状态寄存器里有EOC转换结束标志位,这里获取的就是EOC标志位,EOC标志位由硬件在(规则或注入)通道组转换结束时设置。

        意思是EOC标志位在规则组或注入组完成时都会置1,这一位由软件清除或由读取ADC_DR时清除。

        ADC_DR是数据寄存器,一般EOC标志位置1,程序就会来读取数据,所以EOC标志位就多设计了一个功能,这一位可以在读取数据寄存器之后自动清除,无需手动清除,可以省一条代码。

  • 当EOC标志位为0时,表示转换未完成;
  • 当EOC标志位为1时,表示转换完成。

        所以在代码中,当ADC_GetFlagStatus函数的返回值,也就是EOC标志位==RESET时,转换未完成,while条件为真,执行空循环。转换完成后,EOC由硬件自动置1,while循环自动跳出。这样等待转换完成的代码就OK了。

        具体等待的时间:ADC配置时,通道的采样周期是55.5,转换周期是固定的12.5,加在一起就是68个周期。ADCS时钟配置的ADCCLK是72MHz的6分频,就是12MHz,12MHz进行68个周期,转换才能完成。最终的时间1/12M再×68,结果大概是5.6us,可参考之前博文算转换时间的内容:STM32 ADC数模转换器-CSDN博客

3.3 读取ADC数据寄存器

        等待完成之后就可以取结果了,调用ADC_GetConversionValue函数可取得结果。

代码示例:

    return ADC_GetConversionValue(ADC1);

        ADC_GetConversionValue函数,也就是ADC获取转换值函数的返回值就是AD转换的结果。函数直接读取ADC的DR数据寄存器,因为读取DR寄存器会自动清除EOC标志位,所以在函数之后就不需要手动清除标志位了。

 3.4 启动AD转换函数块总结

代码示例:

uint16_t AD_GetValue(uint8_t ADC_Channel)
{ADC_RegularChannelConfig(ADC1,ADC_Channel,1,ADC_SampleTime_55Cycles5);ADC_SoftwareStartConvCmd(ADC1,ENABLE);while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET);return ADC_GetConversionValue(ADC1);
}

        调用一次这个函数块就相当于执行了一次图单次转换,非扫描模式的流程。


总结

        以上就是今天要讲的内容,本文仅仅简单介绍了AD多通道函数设计的步骤和代码含义。相较于之前博文 “AD单通道函数设计” 本文只是在红色加粗地方有所改动,简单实现AD多通道的功能。

这篇关于STM32 AD多通道函数设计的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/846455

相关文章

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

怎么让1台电脑共享给7人同时流畅设计

在当今的创意设计与数字内容生产领域,图形工作站以其强大的计算能力、专业的图形处理能力和稳定的系统性能,成为了众多设计师、动画师、视频编辑师等创意工作者的必备工具。 设计团队面临资源有限,比如只有一台高性能电脑时,如何高效地让七人同时流畅地进行设计工作,便成为了一个亟待解决的问题。 一、硬件升级与配置 1.高性能处理器(CPU):选择多核、高线程的处理器,例如Intel的至强系列或AMD的Ry

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

SprinBoot+Vue网络商城海鲜市场的设计与实现

目录 1 项目介绍2 项目截图3 核心代码3.1 Controller3.2 Service3.3 Dao3.4 application.yml3.5 SpringbootApplication3.5 Vue 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍:CSDN认证博客专家,CSDN平台Java领域优质创作者,全网30w+

STM32(十一):ADC数模转换器实验

AD单通道: 1.RCC开启GPIO和ADC时钟。配置ADCCLK分频器。 2.配置GPIO,把GPIO配置成模拟输入的模式。 3.配置多路开关,把左面通道接入到右面规则组列表里。 4.配置ADC转换器, 包括AD转换器和AD数据寄存器。单次转换,连续转换;扫描、非扫描;有几个通道,触发源是什么,数据对齐是左对齐还是右对齐。 5.ADC_CMD 开启ADC。 void RCC_AD