STM32之九:ADC模数转换器

目录

1. 简介

2. ADC

2.1 逐次逼近型寄存器SAR

2.2 ADC转换时间

3  ADC框图

3.1 8 bit ADC0809芯片内部框图

3.2 ADC框图

3.2.1 注入通道和规则通道

3.2.2 单次/连续转换模式

3.2.3 扫描模式

3.2.4 外部触发转换

3.2.5 数据对齐

3.2.6 模拟看门狗

4. 总结和ADC驱动代码


1. 简介

12bitADC(Analog-To-Digital converter)是一个逐次逼近型的模拟数字转换器,有18个通道,可测16个外部和2个内部信号源。各通道的A/D转换可以单次、连续、扫描或间断模式执行。ADC的结果可以左对齐或右 对齐方式存储在16位数据寄存器中。

2. ADC

STM32的ADC是一个逐次逼近型的ADC,那么在了解STM32的ADC之前,我们先来了解下逐次逼近型的含义,这会帮助我们理解ADC。

2.1 逐次逼近型寄存器SAR

逐次逼近型SAR ADC实质上是一种二进制搜索算法,其结构如下:

模拟输入电压(VIN)由采样/保持电路保持。N位寄存器的MSB(最高有效位设置为1,即1000 000),N位寄存器中的数据输入到N位DAC中,DAC输出(VDAC)会被设置为VREF/2,VREF是提供位ADC的基准电压。

VIN和VDAC都被输入到比较器中,用来比较判断VIN是小于还是大于VDAC。如果VIN大于VDAC,则比较器输出逻辑高电平,N位寄存器的MSB保持为1。相反,如果VIN小于VDAC,则比较器输出逻辑低电平,N位比寄存器的MSB清0。随后SAR控制逻辑移至下一位,并将该位设置为高电平,进行下一次的比较,这个过程一直持续到LSB(最低有效位)。上述操作结束后,也就完成了转换,N位转换结果储存在寄存器内。这就是逐次逼近型寄存器ADC的工作原理。

下图是一个4bit ADC转换示意图,其中y轴和图中粗线表示VDAC,即DAC的输出电压。

在下图中,第一次比较,N位寄存器的值为1000,此时VIN<VDAC,所以bit3=0,即0000.

之后移至下一位,即0100,进行第二次比较,此时VIN>VDAC,所以bit2=1,即0100.

之后再一致下一位,即0110,进行第三次比较,此时VIN<VDAC,所以bit1=0,即0100.

之后移至最后一位,即0101,进行最后一次比较,此时VIN<VDAC,所以bit0=0,即0100.

至此,比较完毕,ADC的转换结果也已得出,即0100.

2.2 ADC转换时间

接上文,4bitADC需要4个比较周期,则NbitADC需要N个比较周期,这个过程在ADC转换过程中也叫做量化编码时间。上面说过一个采样/保持电路,出现该电路是由于ADC转换时候需要量化编码时间,所以需要保证在进行量化编码时VIN的值是一定的,而不是变化的,所以需要一个采样/保持电路。当采样开关打开时,采样一段时间,送入到保持电路,并将采样开关关闭,之后进行量化编码,完成一次ADC转换。所以,ADC转换需要4个步骤:采样、保持、量化、编码。在STM32中,采样保持叫做采样时间,由于STM32是12bit的ADC,其量化编码需要12.5个ADC周期,所以STM32 ADC的总转换时间为: TCONV = 采样时间 + 12.5个ADC周期

在STM32时钟树章节,我们了解到ADC时钟ADCCLK最大为14MHz。

当ADCCLK=14MHz,采样时间为1.5个ADC周期     TCONV = 1.5 + 12.5 = 14个ADC周期 = 1μs

所以ADC最快转换时间为1us。

3  ADC框图

3.1 8 bit ADC0809芯片内部框图

了解了逐次逼近型ADC,我们看下8bit ADC0809芯片的内部框图:

先看左侧红色部分,该部分为ADC的输入(采样)通道,一共有8位输入通道,即8bit的ADC。下方的地址锁存和译码可以决定选择那位通道,之后将该通道的模拟信号输入到比较器中。

接着我们看绿色部分,这里有一个逐次逼近型寄存器SAR,下方是一个DAC,DAC输出的信号也会输入到比较器中,下面的过程我们就熟悉了,依次从MSB进行比较,直至比较到LSB,这样便完成了ADC转换。

我们注意下图的上方有一个CLOCK,这个CLOCK主要用来控制转换周期;START信号,表示ADC开始转换;EOC,即 end of conver,表示转换结束,为转换结束标志位。图下方有一个Vcc、GND,为芯片的电压输入引脚;另外VREF(+)和VREF(-)为DAC的参考电压。例如Nbit寄存器中的值为255,那这个值转换为模拟信号是代表3.3V还是5V,即由此电压来决定。一般情况下将VREF(+)和Vcc相连,VREF(-)和GND相连。

所以,ADC内部结构主要包括:输入通道、逐次逼近型寄存器、DAC、时钟、转换结束信号,参考电压。

有了这个基础,下面我们来介绍ADC框图。

3.2 ADC框图

初看该图,会觉得无从下手,有了前面的铺垫,再看这个图,清晰多了。红色框框为输入通道选择,STM32共有18个ADC通道,其中16个为外部GPIO通道,剩下2个分别为温度传感器和VREFINT。绿色框框为逐次逼近型ADC部分。淡黄色同样为输出结果,在STM32中将ADC的转换结果存放到数据寄存器中。橙色部分为输入电压,可看下表:

3.2.1 注入通道和规则通道

观察红框部分,ADC有一个模拟多路开关,该开关输入为18个输入通道,其输出也可以为多个通道。STM32的ADC可以一次性配置多个通道进行转换,多个通道可以分配为注入组和规则组。规则组一次性可以选择16个通道,但规则通道的数据寄存器只有1个,所以,在完成规则组内一个通道的ADC转换之后,需要立马将其数据寄存器中的数据取出,防止其被覆盖,因此规则组通常会和DMA联合使用,提高效率。

注入组一次性只能选择4个通道,但是我们看ADC框图发现注入通道数据寄存器为4*16bit,即共有4个,所以注入组不需要担心数据被覆盖的问题。

一般情况下我们只使用规则组即可。

3.2.2 单次/连续转换模式

单次转换模式:

        在单次转换模式下,ADC只执行一次转换,转换完成后则停止ADC。

        可以通过配置ADC_CR2寄存器的ADON位来选择

连续转换模式:

        在连续转换模式下,ADC转换结束后立马启动另一次转换。

3.2.3 扫描模式

扫描模式是用来扫描一组模拟通道,经常将其和转换模式组合起来,共有4种组合。

3.2.4 外部触发转换

ADC转换也可以由外部事件触发(例如定时器捕获、外部中断等),可通过配置EXTTRIG位来控制。

这部分和ADC框图的粉色框框对应,可以在定时器中设置一个中断,当定时器中断时,产生ADC转换。此处主要说明一点TRGO事件,在TIM定时器章节,3.2 PWM输入模式 中我们曾介绍到,“定时器的主模式可以将定时器内部的信号映射到TRGO引脚,用于触发其他外设的操作”。此处便是一个例子,可以配置TIM3_TRGO事件为触发ADC转换,则硬件会自动在产生TRGO事件后主动触发ADC转换,节省了软件资源。

3.2.5 数据对齐

数据寄存器为16位的,但是ADC是12位的,因此ADC转换结果存放有两种方式:数据右对齐和数据左对齐。

一般情况下选择数据右对齐,这种情况相下,数据寄存器中的值即为ADC的结果,直接读出即可。

那么数据左对齐有什么作用呢,可以在调整分辨率的时候使用,例如在使用8bit的ADC时,可以选择数据左对齐,则选取时值只选取D11-D4部分即可。

可以在ADC_CR2寄存器中配置ALIGN位来实现数据对齐方式的选择:

3.2.6 模拟看门狗
模拟看门狗可以用来关注某些通道,当其电压值不在正常范围内时,看门狗可以产生中断。
如果被ADC转换的模拟电压低于低阀值或高于高阀值,AWD模拟看门狗状态位被设置。阀值位
于ADC_HTR和ADC_LTR寄存器的最低12个有效位中。通过设置ADC_CR1寄存器的AWDIE位
以允许产生相应中断。

4. 总结和ADC驱动代码

最后,根据江协大大的ADC框图,我们看着这个框图,做下总结和复习:

18个输入通道,之后多路输出,选择规则组或者注入组,可以通过控制触发信号开始ADC转换,RCC的ADCCLK来控制ADC周期,转换结束后会将结果放置到数据寄存器,同时EOC标志位也被置位,之后输入到NVIC中断,表示ADC转换结束。ADC也可以配置看门狗,当ADC的值超过设置阈值后,ADC也会触发中断,输入到NVIC中断中。

了解了理论之后,来看ADC配置的代码,就会一目了然。

//adc.c#include "stm32f10x.h"                  // Device header/*** 函    数:AD初始化* 参    数:无* 返 回 值:无*/
void AD_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);	//开启ADC1的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);	//开启GPIOA的时钟/*设置ADC时钟*/RCC_ADCCLKConfig(RCC_PCLK2_Div6);						//选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz/*GPIO初始化*/GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);					//将PA0引脚初始化为模拟输入/*规则组通道配置*/ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);		//规则组序列1的位置,配置为通道0/*ADC初始化*/ADC_InitTypeDef ADC_InitStructure;						//定义结构体变量ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;		//模式,选择独立模式,即单独使用ADC1ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;	//数据对齐,选择右对齐ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;	//外部触发,使用软件触发,不需要外部触发ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;		//连续转换,失能,每转换一次规则组序列后停止ADC_InitStructure.ADC_ScanConvMode = DISABLE;			//扫描模式,失能,只转换规则组的序列1这一个位置ADC_InitStructure.ADC_NbrOfChannel = 1;					//通道数,为1,仅在扫描模式下,才需要指定大于1的数,在非扫描模式下,只能是1ADC_Init(ADC1, &ADC_InitStructure);						//将结构体变量交给ADC_Init,配置ADC1/*ADC使能*/ADC_Cmd(ADC1, ENABLE);									//使能ADC1,ADC开始运行/*ADC校准*/ADC_ResetCalibration(ADC1);								//固定流程,内部有电路会自动执行校准while (ADC_GetResetCalibrationStatus(ADC1) == SET);ADC_StartCalibration(ADC1);while (ADC_GetCalibrationStatus(ADC1) == SET);
}/*** 函    数:获取AD转换的值* 参    数:无* 返 回 值:AD转换的值,范围:0~4095*/
uint16_t AD_GetValue(void)
{ADC_SoftwareStartConvCmd(ADC1, ENABLE);					//软件触发AD转换一次while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);	//等待EOC标志位,即等待AD转换结束return ADC_GetConversionValue(ADC1);					//读数据寄存器,得到AD转换的结果
}

参考:

1. 江协科技STM32

2. 理解逐次逼近寄存器型ADC:与其它类型ADC的架构对比 | Analog Devices

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

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

相关文章

前端写得好和写得差有什么区别?

前端开发的质量对用户体验、页面性能、代码维护性以及整个项目的成功有着直接的影响。前端写得好与写得差之间的区别可以从以下几个方面来看&#xff1a; 用户体验&#xff1a; 写得好&#xff1a;页面加载速度快&#xff0c;交互流畅&#xff0c;布局合理&#xff0c;响应迅速…

MYSQL ODBC驱动安装时的注意事项

今天想使用MYSQL的ODBC驱动连接数据库。 安装的时候遇到一个大坑&#xff0c;在这里记录一下。 window 64位的操作&#xff0c;要安装64位驱动&#xff0c;这个大家都知道了。 有以下的问题要注意区别的。 1 、windows是64位的&#xff0c;但是开发软件是32位的。 这个时候…

OpenStack Yoga版安装笔记(七)glance练习补充

1、练习场景说明 在OpenStack Yoga版安装笔记&#xff08;五&#xff09;中&#xff0c;glance已经在controller node虚拟机上安装完成&#xff0c;并且已经成功拍摄了快照。 此时&#xff0c;controller node虚机已经安装了keystone、keystone DB、glance、glance DB、OpenSta…

决策树的概念

决策树的概念 决策树是一种监督学习算法&#xff0c;主要用于分类任务。它通过构建一棵树结构模型来进行预测&#xff0c;其中每个内部节点表示一个特征属性上的判断条件&#xff0c;每条边代表一个判断结果对应的分支&#xff0c;而叶节点则代表最终的类别标签。 应用领域 …

MySQL中的MVCC(多版本并发控制)

MySQL中的MVCC&#xff08;多版本并发控制&#xff09; MySQL中的多版本并发控制&#xff08;MVCC&#xff09;是一种重要的机制&#xff0c;它允许多个事务并发地读取和修改数据库&#xff0c;同时保持数据的一致性和隔离性。MVCC通过维护数据的多个版本&#xff0c;使事务能…

Leetcode滑动窗口的使用

1.滑动窗口 文章目录 1.滑动窗口1.1 什么是滑动窗口&#xff1f;1.2 解题思路1.3 扩展 1.1 什么是滑动窗口&#xff1f; 滑动窗口是一种处理数组或序列数据时常用的数据结构和算法思想。在计算机科学中&#xff0c;它通常涉及在数组上设置一个可变的窗口&#xff0c;该窗口可以…

PCL-基于FPFH的SAC-IA结合ICP的点云配准方法

目录 一、相关方法原理1.凸包方法2.FPFH特征描述3.SAC-IA概述4.ICP概述 二、实验代码三、实验结果 一、相关方法原理 点云是在同一空间参考系下表达目标空间分布和目标表面特性的海量点集合&#xff0c;在获取物体表面每个采样点的空间坐标后&#xff0c;得到的是点的集合&…

【java技术】xxl-job的实现

Xxl-Job 是一个轻量级的分布式任务调度平台&#xff0c;它支持定时任务的创建、管理、执行和监控。Xxl-Job 的设计理念是简单易用、轻量级、高性能&#xff0c;适合于微服务架构下的任务调度场景。 Xxl-Job 的实现原理涉及到几个关键组件和技术细节。下面是 Xxl-Job 的核心组件…

构建智能运维系统:创新架构与效率优化

随着信息技术的迅猛发展&#xff0c;企业对于运维效率和服务质量的要求越来越高。智能运维系统的设计和实施&#xff0c;不仅能够提升系统可靠性和响应速度&#xff0c;还能有效降低成本和人力投入。本文将深入探讨智能运维系统的架构设计原则和关键技术&#xff0c;为企业在运…

数据结构重置版(概念篇)

本篇文章是对数据结构的重置&#xff0c;且只涉及概念 顺序表与链表的区别 不同点 顺序表 链表 存储空间上 物理上一定连续 逻辑上连续&#xff0c;但物理上不一定连续…

MYSQL(2) 高级查询

文章目录 概述高级查询基础查询条件查询范围查询判空查询模糊查询分页查询查询后排序分组查询 小结 概述 接上篇&#xff0c;上篇写到增删改查。这篇继续。 高级查询 基础查询 -- 全部查询 select * from student; -- 只查询部分字段 select sname, class_id from student;…

.env.local 配置本地环境变量 用于团队开发

.env.local 用途&#xff1a;.env.local 通常用于存储本地开发环境中的环境变量。这些变量可能包括敏感数据或特定于单个开发者的设置&#xff0c;不应该被提交到版本控制系统中。优先级&#xff1a;在大多数框架中&#xff0c;.env.local 文件中的变量会覆盖其他 .env 文件中…

Java唯一订单编号生成

在Java中生成唯一的订单编号通常需要结合时间戳、随机数和/或序列号等元素来确保唯一性。下面是一个简单的示例&#xff0c;使用当前时间的毫秒值加上一个随机数来生成订单号。为了简化&#xff0c;我们将使用​​java.util.UUID​​类来生成一个全局唯一的UUID&#xff0c;并将…

分类模型的完整流程及Python实现

1、加载函数和数据集 import numpy as np from sklearn.datasets import load_breast_cancer from sklearn.svm import SVC from sklearn.model_selection import train_test_split from sklearn.preprocessing import StandardScaler import matplotlib.pyplot as plt cancer…

linux系统查历史cpu使用数据(使用sar 查询cpu和网络占用最近1个月历史数据)。

一 sar 指令介绍 在 Linux 系统中&#xff0c;sar 是 System Activity Reporter 的缩写&#xff0c;是一个用于收集、报告和保存系统活动信息的工具。它是 sysstat 软件包的一部分&#xff0c;提供了丰富的系统性能数据&#xff0c;包括 CPU、内存、网络、磁盘等使用情况&am…

Jdk有哪些版本

JDK(Java Development Kit)是Java编程语言的软件开发工具包,其版本随着Java语言的不断发展而更新。以下是JDK的一些主要版本及其相关信息: JDK 8(发布于2014年3月):引入了一系列新功能,如Lambda表达式、函数式接口、Stream API和新的日期/时间API等。是Java历史上一个…

SQL中的LEFT JOIN、RIGHT JOIN和INNER JOIN

在SQL中&#xff0c;JOIN操作是连接两个或多个数据库表&#xff0c;并根据两个表之间的共同列&#xff08;通常是主键和外键&#xff09;返回数据的重要方法。其中&#xff0c;LEFT JOIN&#xff08;左连接&#xff09;、RIGHT JOIN&#xff08;右连接&#xff09;和INNER JOIN…

《JavaEE篇》--多线程(2)

《JavaEE篇》--多线程(1) 线程安全 线程不安全 我们先来观察一个线程不安全的案例&#xff1a; public class Demo {private static int count 0;public static void main(String[] args) throws InterruptedException {Thread t1 new Thread(() -> {//让count自增5W次…

HarmonyOS网络请求的简单用法,HttpUtil简单封装

请求网络获取数据 点击按钮发送一个post请求&#xff0c;发送一条string由于此处的返回result.data本身就是一个string&#xff0c;因此不需要转换类型 Button(请求网络).margin({ top: 10 }).fontSize(24).fontWeight(FontWeight.Bold).onClick(() > {httpRequestPost(http…

JVM 内存结构、垃圾回收机制与并发容器

目录 一、JVM 内存结构 1. 程序计数器&#xff08;Program Counter Register&#xff09;&#xff1a; 2. Java 虚拟机栈&#xff08;JVM Stack&#xff09;&#xff1a; 3.本地方法栈&#xff08;Native Method Stack&#xff09;&#xff1a; 4.堆&#xff08;Heap&#xff…