STM32学习——ADC模数转换器

ADC模数转换器

简介

ADC(Analog-Digital Converter)模拟-数字转换器【DAC可以将数字变量转换为模拟电压—>PWM】

ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁

12位逐次逼近型ADC,1us转换时间

输入电压范围:0-3.3V,转换结果范围:0~4095

18个输入通道,可测量16个外部和2个内部信号源

最快转换频率1MHz

规则组和注入组两个转换单元

模拟看门狗自动监测输入电压范围:模拟看门狗可以监测指定的某些通道,当AD值高于它设定的上阈值或者低于下阈值时,就会申请中断,就可以在中断函数中执行相应的操作

STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道

逐次逼近型ADC

8位逐次逼近型芯片

在这里插入图片描述
在这里插入图片描述

STM32ADC框图

在这里插入图片描述

ADC基本结构图

在这里插入图片描述

转换模式

单次转换,非扫描模式

在这里插入图片描述

左表:16个通道,在非扫描模式下,只有序列1有效,此时,注入组同时选中16个的方式转换为只能选中一个的方式,可以在序列1的位置指定想要转换的通道,然后触发转换,ADC就会对通道2进行模数转换,转换完成后,转换结果放在数据寄存器里,同时给EOC标志位置1,判断EOC标志位,如果转换完成,就可以在数据寄存器里读取想要的结果,如果想再读一次数据就要再启动一次转换

连续转换,非扫描模式

在这里插入图片描述

在一次转换结束后,不会停止,而是立刻进行下一轮的转换,然后一直持续下去,这样只需要最开始触发一次转换,然后就可以一直转换。好处:开始转换后不需要等待一段时间,因此不用判断是否结束,想要读AD值的时候,直接从数据寄存器中读取

单次转换,扫描模式

在这里插入图片描述

连续转换,扫描模式

在这里插入图片描述

数据对齐

在这里插入图片描述

因为ADC是12位的,转换结果就是12位的数据,但是数据寄存器是16位的,因此存在数据对齐的问题,一种是数据右对齐,就是12位的数据向右靠,高位补零,一般用右对齐,这样读取16位寄存器,直接就是转换结果

左对齐作用:可以只选择高八位,将12位ADC转换成8位ADC,精度更低

转换时间

AD转换的步骤:采样,保持,量化,编码

STM32 ADC的总转换时间为:

TCONV = 采样时间 + 12.5个ADC周期

例如:当ADCCLK=14MHz,采样时间为1.5个ADC周期

TCONV = 1.5 + 12.5 = 14个ADC周期 = 1μs

校准

ADC有一个内置自校准模式。校准可大幅减小因内部电容器组的变化而造成的准精度误差。校准期间,在每个电容器上都会计算出一个误差修正码(数字值),这个码用于消除在随后的转换中每个电容器上产生的误差

建议在每次上电后执行一次校准

启动校准前, ADC必须处于关电状态超过至少两个ADC时钟周期

配置步骤

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

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

第三步:配置多路开关,把左边的通道接入到右边的规则组里面

第四步:配置ADC转换器,用结构体来配置

第五步:开关控制,调用cmd函数,开启ADC

第六步:对ADC进行校准,减小误差

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

库函数介绍

void RCC_ADCCLKConfig(uint32_t RCC_PCLK2);

用来配置ADCCLK分频器,可以对APB2的72MHz时钟选择2,4,6,8分频,输入到ADCCLK

void ADC_DeInit(ADC_TypeDef* ADCx);
void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);
void ADC_StructInit(ADC_InitTypeDef* ADC_InitStruct);

分别是恢复缺省配置、Init初始化、StructInit结构体初始化

void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);

用于给ADC上电

void ADC_DMACmd(ADC_TypeDef* ADCx, FunctionalState NewState);

用于开启DMA输出信号,如果使用DMA转运数据,需要调用这个函数

void ADC_ITConfig(ADC_TypeDef* ADCx, uint16_t ADC_IT, FunctionalState NewState);

中断输出控制,用于控制某个中断能不能通往NVIC

void ADC_ResetCalibration(ADC_TypeDef* ADCx);

复位校准

FlagStatus ADC_GetResetCalibrationStatus(ADC_TypeDef* ADCx);

获取复位校准状态

void ADC_StartCalibration(ADC_TypeDef* ADCx);

开始校准

FlagStatus ADC_GetCalibrationStatus(ADC_TypeDef* ADCx);

获取开始校准状态

void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);

ADC软件开始转换控制,用于软件触发的函数,也就是触发控制

FlagStatus ADC_GetSoftwareStartConvStatus(ADC_TypeDef* ADCx);

ADC获取软件开始转换状态,只能判断转换是否正在进行,不能判断转换结束

FlagStatus ADC_GetFlagStatus(ADC_TypeDef* ADCx, uint8_t ADC_FLAG);

获取标志位状态,参数给EOC的标志位,可以判断EOC标志位是否置1,如果转换结束,EOC标志位置1

void ADC_DiscModeChannelCountConfig(ADC_TypeDef* ADCx, uint8_t Number);

配置间断模式:每隔几个通道间断一次

void ADC_DiscModeCmd(ADC_TypeDef* ADCx, FunctionalState NewState);

配置间断模式:是不是启用间断模式

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

ADC规则组通道配置,给序列的每个位置填写指定的通道,ADC_Channel指定的通道,Rank是序列几的位置,ADC_SampleTime指定通道的采样时间

void ADC_ExternalTrigConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);

ADC外部触发转换控制,是否允许外部触发转换

uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx);

ADC获取转换值,就是获取AD转换的数据寄存器,读取转换结果需要用到这个函数

uint32_t ADC_GetDualModeConversionValue(void);

ADC获取双模式转换值,双模式ADC模式读取转换结果的函数

void ADC_AnalogWatchdogCmd(ADC_TypeDef* ADCx, uint32_t ADC_AnalogWatchdog);
void ADC_AnalogWatchdogThresholdsConfig(ADC_TypeDef* ADCx, uint16_t HighThreshold, uint16_t LowThreshold);
void ADC_AnalogWatchdogSingleChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel);

对模拟看门狗进行配置,第一个是是否启动看门狗,第二个是配置高低阈值,第三个是配置看门的通道

void ADC_TempSensorVrefintCmd(FunctionalState NewState);

ADC内部两个通道的配置

FlagStatus ADC_GetFlagStatus(ADC_TypeDef* ADCx, uint8_t ADC_FLAG);
void ADC_ClearFlag(ADC_TypeDef* ADCx, uint8_t ADC_FLAG);
ITStatus ADC_GetITStatus(ADC_TypeDef* ADCx, uint16_t ADC_IT);
void ADC_ClearITPendingBit(ADC_TypeDef* ADCx, uint16_t ADC_IT);

获取标志位状态、清除标志位、获取中断状态、清除中断挂起位

代码示例

代码文件:AD.h(单次转换,非扫描模式)【单通道】

#include "stm32f10x.h"                  // Device header

void AD_Init(void)
{
	//第一步:开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	//第二步:配置GPIO
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	
	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);
	
	//第三步:选择规则组的输入通道
	ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_55Cycles5);
	
	//第四步:用结构体初始化ADC
	ADC_InitTypeDef ADC_InitStructure;
	ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;
	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_Cmd(ADC1,ENABLE);
	
	//校准
	ADC_ResetCalibration(ADC1);
	while(ADC_GetResetCalibrationStatus(ADC1)==SET);
	ADC_StartCalibration(ADC1);
	while(ADC_GetCalibrationStatus(ADC1)==SET);
}

uint16_t AD_GetValue(void)
{
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
	while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET);
	return ADC_GetConversionValue(ADC1);
}

代码文件:AD.h【多通道】

#include "stm32f10x.h"                  // Device header

void AD_Init(void)
{
	//第一步:开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	//第二步:配置GPIO
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;//模拟输入模式
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	
	
	//第四步:用结构体初始化ADC
	ADC_InitTypeDef ADC_InitStructure;
	ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;
	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_Cmd(ADC1,ENABLE);
	
	//校准
	ADC_ResetCalibration(ADC1);
	while(ADC_GetResetCalibrationStatus(ADC1)==SET);
	ADC_StartCalibration(ADC1);
	while(ADC_GetCalibrationStatus(ADC1)==SET);
}

//在调用AD_GetValue时,只需要指定一个转换的通道,返回值就是指定通道的结果
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);
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值