STM32 ADC模数转换器

一、ADC简介

  • ADC(Analog-Digital Converter)模拟-数字转换器
  • ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁
  • 12位逐次逼近型ADC,1us转换时间 输入电压范围:0~3.3V,转换结果范围:0~4095
  • 18个输入通道,可测量16个外部和2个内部信号源
  • 规则组和注入组两个转换单元
  • 模拟看门狗自动监测输入电压范围
  • STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道

二、ADC基本框图

三、基本结构

四、输入通道查看

五、触发模式

六、数据对齐,校准

<.对齐

由于ADC是12位的,但是数据寄存器是16位的,所以就需要数据对齐。一般选择右对齐从低到高

<.校准

  • ADC有一个内置自校准模式。校准可大幅减小因内部电容器组的变化而造成的准精度误差。校准期间,在每个电容器上都会计算出一个误差修正码(数字值),这个码用于消除在随后的转换中每个电容器上产生的误差
  • 建议在每次上电后执行一次校准
  • 启动校准前, ADC必须处于关电状态超过至少两个ADC时钟周期

校准过程是固定的配置几条代码

七、代码部分

cpp 复制代码
void RCC_ADCCLKConfig(uint32_t RCC_PCLK2);//配置ADCCLK分频器

void ADC_DeInit(ADC_TypeDef* ADCx);//恢复缺损配置 
void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);//初始化
void ADC_StructInit(ADC_InitTypeDef* ADC_InitStruct);//结构体初始化

void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);//adc上电
void ADC_DMACmd(ADC_TypeDef* ADCx, FunctionalState NewState);//开启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);//获取开始校准状态


//ADC软件开始转换控制,用于软件触发的函数(基本结构里的触发控制)
void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
FlagStatus ADC_GetSoftwareStartConvStatus(ADC_TypeDef* ADCx);//获取ADC软件开始状态,没啥用

//配置间断模式
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规则组通道配置,给16个序列填写指定的通道(1:ADCX,2:指定的通道,3:序列几的位置,4:指定通道的采样时间)

//外部触发转换控制,是否允许外部触发转换
void ADC_ExternalTrigConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx);//ADC获取转换值
uint32_t ADC_GetDualModeConversionValue(void);//获取双模式转换值




//******************对于注入组的配置****************//

void ADC_AutoInjectedConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
void ADC_InjectedDiscModeCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
void ADC_ExternalTrigInjectedConvConfig(ADC_TypeDef* ADCx, uint32_t ADC_ExternalTrigInjecConv);
void ADC_ExternalTrigInjectedConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
void ADC_SoftwareStartInjectedConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
FlagStatus ADC_GetSoftwareStartInjectedConvCmdStatus(ADC_TypeDef* ADCx);
void ADC_InjectedChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime);
void ADC_InjectedSequencerLengthConfig(ADC_TypeDef* ADCx, uint8_t Length);
void ADC_SetInjectedOffset(ADC_TypeDef* ADCx, uint8_t ADC_InjectedChannel, uint16_t Offset);
uint16_t ADC_GetInjectedConversionValue(ADC_TypeDef* ADCx, uint8_t ADC_InjectedChannel);
//*************************************************************************************//


//***********************对于模拟看门狗配置*************************//
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);
//获取标志位状态,参数给EOC的标志位,就可以判断EOC是不是置1了(判断转换是否结束)

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);//清除中断挂起位

整体配置代码:

cpp 复制代码
#include "stm32f10x.h"                  // Device header

void AD_init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6); 
	
	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;         //选择50mhz
	GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	//配置序列里的通道((1:ADCX,2:指定的通道,3:序列几的位置,4:指定通道的采样时间))
	ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_55Cycles5);
	
	ADC_InitTypeDef ADC_InitStruct;
	ADC_InitStruct.ADC_Mode = ADC_Mode_Independent;//ADC转换模式,这里选择独立模式,ADC1和ADC2各自转换
	ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;//左对齐还是右对齐
  ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	//外部触发转换选择,触发控制的触发源选择,这里的参数是外部触发none,也就是不使用外部触发,使用内部触发
	ADC_InitStruct.ADC_ContinuousConvMode = DISABLE;//连续转换模式,非扫描模式
	ADC_InitStruct.ADC_ScanConvMode =  DISABLE;      //扫描转换模式,单次转换
  ADC_InitStruct.ADC_NbrOfChannel =  1;           //通道数目,
  ADC_Init(ADC1,&ADC_InitStruct);
	
	//中断和看门狗如果有需要可以在下面配置//
	
	
	//*******************//
	
	ADC_Cmd(ADC1,ENABLE);
	
	//***********对ADC进行校准*****************//
	ADC_ResetCalibration(ADC1);//复位校准
  while (ADC_GetResetCalibrationStatus(ADC1) == SET);//返回复位校准的状态,一直循环等它硬件清0.说明复位校准完成
  ADC_StartCalibration(ADC1);//启动校准,内部电路自动进行校准
  while (ADC_GetCalibrationStatus(ADC1) == SET);//获取校准状态,等待校准是否完成


//	ADC_SoftwareStartConvCmd(ADC1,ENABLE); //软件触发转换的函数,ADC开始转换。
}

//*****************启动转换,获取结果*****************//

uint16_t AD_GetValue(void)
{
	ADC_SoftwareStartConvCmd(ADC1,ENABLE); //软件触发转换的函数,ADC开始转换。
	while (ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET);//获取标志位状态的函数,当EOC标志位等于0时,转换未完成。
	return ADC_GetConversionValue(ADC1);//ADC获取转换值

}

/*注:如果想使用连续转换函数,则24行使用enable。软件触发放在上面init就行了。并且获取函数使用下面的


uint16_t AD_GetValue(void)
{
	
//	while (ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET);//获取标志位状态的函数,当EOC标志位等于0时,转换未完成。
	
直接return就行了,不需要一直获取是否转换结束,因为连续转换是一直在进行的。
  return ADC_GetConversionValue(ADC1);//ADC获取转换值

}

主程序代码:

cpp 复制代码
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

uint16_t ADValue;
float Voltage;

int main(void)
{
  OLED_Init();
  AD_init();
	OLED_ShowString(1,1,"ADValue:");
	OLED_ShowString(2,1,"VOlatge:0.00V");
	while(1)
	{
		ADValue = AD_GetValue();
		Voltage = (float )ADValue / 4095 * 3.3;
		OLED_ShowNum(1,9,ADValue,4);
		OLED_ShowNum(2,9,Voltage,1);
		OLED_ShowNum(2,11,(uint16_t)(Voltage*100)%100,2);
		Delay_ms(100);
		
	}
 }


	
相关推荐
yuan1999721 小时前
STM32远程升级系统(Bootloader + 上位机)
stm32·单片机·嵌入式硬件
Heartache boy21 小时前
野火STM32_HAL库版课程笔记-ADC多通道采集热敏、光敏、反射传感器(轮询)
笔记·stm32·单片机
AI+程序员在路上1 天前
嵌入式软件技术大全
linux·开发语言·arm开发·单片机
秀秀更健康1 天前
STM32的程序下载不进去----VDDA悬空
stm32·单片机·嵌入式硬件
长安第一美人1 天前
AI辅助下的嵌入式UI系统设计与实践(二)[代码阅读理解]
c++·嵌入式硬件·ui·显示屏·工业应用
我在人间贩卖青春1 天前
DMA的应用
单片机·dma·gpdma
学嵌入式的小杨同学1 天前
STM32 进阶封神之路(二十五):ESP8266 深度解析 —— 从 WiFi 通信原理到 AT 指令开发(底层逻辑 + 实战基础)
c++·vscode·stm32·单片机·嵌入式硬件·mcu·智能硬件
树爷只认钱1 天前
ESP01S模块+串口底座 AT指令连接中移Onenet物联网全过程(第1篇)
单片机·嵌入式硬件·物联网·esp8266
学嵌入式的小杨同学1 天前
STM32 进阶封神之路(二十六):ESP8266 实战全攻略 ——TCP 通信 + 数据上传 + 远程控制 + 透传模式(库函数 + 代码落地)
stm32·单片机·嵌入式硬件·mcu·硬件架构·硬件工程·智能硬件
Nice__J1 天前
Mcu架构以及原理——7.寄存器编程与抽象
stm32·单片机·架构