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单通道函数设计的基础内容。


一、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 配置多路开关

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

代码示例:

ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_55Cycles5);

        用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转换函数设计

        上文描述的代码设置使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(void)
{ADC_SoftwareStartConvCmd(ADC1,ENABLE);while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET);return ADC_GetConversionValue(ADC1);
}

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


总结

        以上就是今天要讲的内容,本文仅仅简单介绍了AD单通道函数设计的步骤和代码含义。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/765097.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

STM32不使用中断实现定时器微秒级精确延时

我们在写代码的时候避免不了要使用延时函数,很多延时函数都是使用中断或者tick来实现的,tick的方式最大到毫秒ms级别,通过中断方式的通用定时器来实现,如果实现1us的延时那么每1us就来一次中断,很影响cpu的效率。 本文…

程序员表白

啥?!你说程序员老实,认真工作,根本不会什么表白!那你就错了!(除了我) 那今天我们就来讲一下这几个代码!赶紧复制下来,这些代码肯定有你有用的时候! 1.Python爱心代码 im…

Google XSS Game Level 6 通关方式

文章目录 链接:[Google XSS Game](#https://xss-game.appspot.com/)Level 6 - Follow the 🐇思路1 (当然,我使用这个方式没有成功,所以才来记录下)解法2 【最简单的解法】需要注意的一个小问题 链接&#x…

单相桥式全控整流电路

1仿真目的 通过对单相桥式全控整流电路的仿真研究,分析电路带电阻负载与阻感负载的不同工作情况。研究对电路的影响 2仿真原理 2.1单相桥式 如图所示为单相桥式全控电路的框图,设负载为电阻负载。在桥式逆变电路中,桥臂的上下两个开关器件…

RabbitMQ在Java中使用 SpringBoot 从基础到高级

充分利用每一个监听者 需要充分利用每一个消费者,需要在配置文件中加上prefetch配置并设置为1 rabbitmq:listener:simple:prefetch: 1 # 每次只能获取一条消息,处理完成才能获取下一个消息创建交换机和队列 创建队列 "fanout.queue1"&…

超融合服务器:企业转型的助推器?

在当今快速发展的数字化时代,企业需要灵活、高效且可靠的IT基础设施来支撑其业务运营。传统的存储环境由于其复杂性和局限性,已经难以满足现代企业的需求。而超融合设备的出现,为企业提供了一个全新的选择。本文将深入探讨超融合服务器的优势…

C++一维数组练习oj(2)

这时上次的C一维数组练习:C一维数组练习oj-CSDN博客 这到题目我承认非常难!当然这只是我认为,因为我只学到了一维数组! 对于你们来说可能不难。 好了我不客套了。 这题我们可以将他理解为一条时间轴: 时间轴上小李每1…

【Linux 08】进程概念

文章目录 🌈 01. 基本概念🌈 02. 描述进程 PCB🌈 03. 使用 ./ 的方式创建进程🌈 04. ps 查看进程🌈 05. getpid / getppid 获取进程标识符🌈 06. kill 终止指定进程🌈 07. fork 创建子进程&…

tftp使用

下载 sudo apt-get install tftpd-hpa 创建文件夹 mkdir /home/ljl/work/tftpd mkdir /home/ljl/tftpd chmod 777 tftpd/编辑 sudo vim /etc/default/tftpd-hpa //服务器端 sudo apt-get install tftp-hpa //客户端编辑权限 sudo vi /etc/default/tftpd-hpa 内容&#xff1…

SQL Server 2008R2 日志文件大小设置及查询

SQL Server 2008R2 建立数据库存在日志无限增长问题,造成磁盘内存不足。本文解决这个问题,如下: 1.设置日志文件的最大大小 USE master; GO ALTER DATABASE [D_total] MODIFY FILE (NAME D_total_log, -- 日志文件的逻辑名称MAXSIZE 200…

【黄啊码】如何用GPT和向量数据库做问答型AI

知识库服务依赖该数据库,Embedding 形式个性化训练 ChatGPT,必不可少的就是向量数据库 因为 qdrant 向量数据库只支持 Docker 部署,所以需要先安装好 Docker 服务。 命令行安装 拉取镜像 docker pull qdrant/qdrant 运行服务 docker run -…

飞桨AI应用@riscv OpenKylin

在riscv编译安装飞桨PaddlePaddle参见: 算能RISC-V通用云编译飞桨paddlepaddleopenKylin留档_在riscv下进行paddlelite源码编译-CSDN博客 安装好飞桨,就可以用飞桨进行推理了。刚开始计划用ONNX推理,但是在算能云没有装上,所以最…

【正点原子Linux连载】第十七章 异步通知实验 摘自【正点原子】ATK-DLRK3568嵌入式Linux驱动开发指南

1)实验平台:正点原子ATK-DLRK3568开发板 2)平台购买地址:https://detail.tmall.com/item.htm?id731866264428 3)全套实验源码手册视频下载地址: http://www.openedv.com/docs/boards/xiaoxitongban 第十七…

Java发送请求-get源码

发送请求 配置依赖-pom.xml Welcome! - The Apache HTTP Server Project 官网解释这个源码httpClient是执行httpget和httppost 步骤: 查看httpClient源码,源码和方法都没有有用的解释 查看CloseableHttpClient源码类 这个抽象类实现2个接口&#xf…

docker 哲学 - 网络桥接器、容器网络接口 、容器间的通信方式

1、解释 docker0 veth eth 2、vethXX 和 ethXX 是肯定一一对应吗 比如 eth1 对应 veth1 3、如果 A容器使用 默认创建方式 。定义他内部网络为 eth0,容器B使用 --network 连上 已创建的网络 172.89.2.1 。此时假设 B的 ip是 172.89.2.2 ,容器网络接口是 e…

2024年企业级通用人工智能的关键技术趋势

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗?订阅我们的简报,深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同,从行业内部的深度分析和实用指南中受益。不要错过这个机会,成为AI领…

腾讯三面被问到有没有参加过CTF_我反手就是一套军体拳打得面试官哑口无言!

目录 ​ 前言: 正文: 什么是CTF? 什么是PWN? 为什么要学CTF? CTF竞赛模式: CTF各大题型简介: 学之前的思考:分析赛题情况 常规做法 CTF比赛需要的知识储备 CTF比赛的神器&#xff…

51单片机中断信号的种类及应用场景

在嵌入式系统中,中断是一种重要的事件处理机制,它可以在程序执行的任何时候暂停当前任务,转而执行与之相关的特殊任务或事件。51单片机作为一种常见的微控制器,其中断功能在各种应用中起着关键作用。然而,对于初学者和…

Jmeter-基础元件使用(二)-属性及对数据库简单操作

一、Jmeter属性 当我们想要在不同线程组中使用某变量,就需要使用属,此时Jmeter属性的设置需要函数来进行set和get操作 1.创建set函数 2.然后采用Beanshell取样器进行函数执行 3.调用全局变量pro_id 4.将上面生成的函数字符串粘贴到另一个线程组即可…

二、阅读器的开发(初始)-- 2、阅读器开发

1、epubjs核心工作原理 1.1 epubjs的核心工作原理解析 epub电子书,会通过epubjs去实例化一个Book对象,Book对象会对电子书进行解析。Book对象可以通过renderTo方法去生成一个Rendition对象,Rendition主要负责电子书的渲染,通过R…