备赛电赛学习STM32篇(九):ADC

目录

一、ADC的简介

二、逐次逼近型ADC

2.1、逐次逼近型ADC框图 

2.2、STM32 ADC内部介绍

2.2.1、STM32ADC的通道以及存储数据的寄存器 

2.2.2、触发方式

2.2.3、STM32ADC时钟部分

2.2.4 

三、ADC基本结构框图

四、另外的细节问题

4.1、输入通道

4.2、转换模式

4.2.1、单次转换、非扫描模式

4.2.2、连续转换、非扫描模式

4.2.3、单次转换、扫描模式

4.2.4、连续转换、扫描模式

4.3、触发控制

4.4、数据对齐

​编辑

4.5、转换时间

4.6、校准

五、实例部分

5.1、ADC的初始化步骤:

5.2、ADC相关库函数介绍

5.2.1、ADCCLK配置函数

5.2.2、ADC库函数中基本功能和规则组的配置

5.2.3、注入组函数(不多了解)

5.2.4、模拟看门狗配置

5.2.5、内部两个通道 

5.2.6、标志位、中断挂起位相关相关

5.3、实验部分

5.3.1、AD单通道(单次扫描、非扫描模式) 

5.3.2、AD多通道


一、ADC的简介

  • 12位的ADC范围是0~2^(12)-1就是量化结果为0~4095,位数越高,量化结果越精细,对应的分辨率越高
  • AD转换是需要时间的,这里的1us就表示从AD转换开始到产生结果需要1us的时间,对应AD转换的频率就是1MHZ(STM32ADC的最快转换频率)
  • 0~3.3V线性对应0~4095
  • 16个外部信号源就是16个GPIO口,在引脚上直接接模拟信号就好
  • 2个内部信号源是内部信号传感器内部参考电压,温度传感器可以测量CPU的温度,比如你电脑可以显示一个CPU温度就可以用ADC读取这个温度传感器来测量,内部参考电压是一个1.2V左右的基准电压,这个基准电压是不随外部供电变化而变化的,所以如果你的芯片的供电不是标准的3.3V,那测量外部引脚的电压可能就不对,这时就可以读取这个基准电压进行校准电压进行校准,这样就能得到正确的电压值了
  • 普通的AD转换流程是:启动一次转换,读一次值,然后再启动再读值。但是STM32的ADC就比较高级可以列一个组,一次性启动一个组,连续转换多个值,并且有两个组,一个是用于常规使用的规则组,另一个用于突发事件的注入组
  • 二、逐次逼近型ADC

    2.1、逐次逼近型ADC框图 

    2.2、STM32 ADC内部介绍

    2.2.1、STM32ADC的通道以及存储数据的寄存器 

  •  STM32的ADC在选择通道选择中可以同时选择多个,而且在转换的时候,还分成了两个组,规则通道组和注入通道组,其中规则通道组可以一次性最多选择16个通道,注入组最多可以选择4个通道,需要注意的是规则通道组只有一个寄存器,运行到最后,前面15个数据都会被挤掉,所以对于规则组来说,最好搭配DMA来使用(DMA是一个数据转运小帮手,他可以在每一个数据来的时候挪到其他地方,防止被覆盖),而注入组中有四个寄存器正好对应四个通道,所以不用担心数据被覆盖的问题
  • 2.2.2、触发方式

  • 对于STM32的ADC,触发ADC开始转换的信号有两种,一种是软件触发,就是在程序中手动调用一条代码,就可以启动转换了,另一种是硬件触发,左边是注入组的触发源,右边是规则组的触发源
  •  正常的思路是用定时器,每隔1ms申请一次中断,在中断里手动开始一次转换,但是频繁进入中断会对我们程序的运行会有一定的影响,比如你有很多中断都要频繁进入,那么肯定会影响主程序的执行,并且不同中断之间,由于优先级的不同,也会导致某些中断不能及时响应,如果触发ADC的中断不能及时响应,那我们ADC的转换频率就会收到影响,所以对于这种需要频繁进中断中断,并且在中断中只完成了简单工作的情况,一般都会有硬件的支持,比如这里可以把TIM3定时为1us的时间,并且把TIM3的更新事件选择为TRGO输出,然后在ADC这里,选择开始触发信号为TIM3的TRGO,这样TIM3的更新事件就能通过硬件自动触发ADC转换了。 
  • 2.2.3、STM32ADC时钟部分

  • 对于ADC的时钟控制(用于驱动内部逐次比较的时钟)是来自ADC预分频器,这个ADC预分频器是来源于RCC-APB2的,APB2为72MHZ,而ADCCLK最大为14MHZ,所以预分频器参数要选择6(得到12M)或者8(得到9M)
  • 2.2.4 

  •  上面的模拟看门狗里面可以存一个阈值高限和阈值低限,如果启动了阈值看门狗,并指定了看门通道,那这个看门狗就会关注这个通道,一旦超过这个阈值范围了,就会申请一个模拟看门狗的中断,最后通向NVIC
  • 对于规则组和注入组而言,它们转换完成之后也会有一个EOC转换完成的信号,在这里EOC是规则组的完成信号,JEOC是注入组完成的信号,这两个信号会在状态寄存器里置一个标志位,我们读取这个标志位,就知道是不是转换结束了,同时这两个标志位也可以去NVIC,申请中断
  • 三、ADC基本结构框图

  • 左边是输入通道,16个GPIO口,外加两个内部的通道,
  • 然后进入AD转换器,规则组最多可以选择16个通道,注入组最多可以选择4个通道  
  • 然后转换的结果可以存在AD寄存器里,其中规则组只有规则组只有一个数据寄存器,注入组有四个
  • 下面有触发控制,提供了开始转换的START的信号,触发控制可以选择软件触发和硬件触发,硬件触发主要来自定时器,当然也可以选择外部中断的引脚,
  • 右边这里是来自RCC的ADCCLK,ADC的逐次比价就是由这个时钟推动的
  • 然后上面可以布置一个模拟看门狗的用于检测转换结果的范围,如果超过设定的阈值范围,就通过中断输出控制,向NVIC申请中断,另外,规则组和注入组转换完成后会有个EOC信号,它会置一个标志位,当然也可以通向NVIC
  • 最后右下角有个开关孔子,在库函数中,就是ADC_Cmd函数,用于给ADC上电的
  • 四、另外的细节问题

    4.1、输入通道

    也可以通过引脚定义图查找        

    4.2、转换模式

     在ADC初始化的结构体里有两个参数,一个是选择单次转换还是连续转换,另一个是选择选择扫描模式和非扫描模式的,这两个参数组合起来就有四种转换方式

    4.2.1、单次转换、非扫描模式

  • 上面的表示规则组里的菜单,有16个空位,分别是序列1到序列16,可以在里面写入你要转换的通道,在非扫描模式下,这个菜单只有第一个序列1 的位置有效,这时菜单同时选中一组的方式就退化为简单地选中一个的方式,在这里我们可以在序列1的位置上指定我们想转换的通道,比如通道2,然后我们就可以触发转换,ADC就会对这个通道进行模数转换,过一小段时间后,转换完成,转换结果放在数据寄存器里,同时给EOC标志位置1,这个转换过程就结束了,如果我们想再启动一次转换,我们就要再触发一次
  • 4.2.2、连续转换、非扫描模式

  •  首先,它还是非扫描模式,所以菜单列表就只用第一个,然后它与上一种单次转换不同的是,它在一次转换完成后是不会停止的,而是立刻开始下一轮的转换,然后一直持续下去,好处是不用判断是否结束,节约时间,想要读取AD值的时候,直接从数据寄存器读取即可
  • 4.2.3、单次转换、扫描模式

  • 单次转换:每触发一次,转换结束后,就会停下来,下次转换就要再次触发
  • 扫描模式:菜单列表会被使用,在这里每个位置是通道几可以任一指定,并且可以重复,然后初始化结构体里还会有个参数,就是通道数目,因为这16个位置你可以不用完,只用前几个,那你就需要给定一个通道数目
  • 4.2.4、连续转换、扫描模式

    同上

    在扫描模式的情况下,还可以有一种模式——间断模式,它的作用是在扫描的过程中,每隔几个转换就暂停一次,需要再次触发 

    4.3、触发控制

    4.4、数据对齐

    STM32的ADC是12位的,它的转换结果就是一个12位的数据,而数据寄存器是16位的,所以就存在一个数据对齐的问题

    1. 数据右对齐:12位的数据向右靠,高位多出来的补0
    2. 数据左对齐:12位的数据向左靠,低位多出来的补0 

    平时使用的都是数据右对齐,这样读取出来的数据就是真实的结果,而左对齐会比实际的值大

    4.5、转换时间

    4.6、校准

    这个校准过程是固定的,我们只需要在ADC初始化的最后,加上几条代码就行 

    五、实例部分

    5.1、ADC的初始化步骤:

    1. 开启RCC时钟,包括ADC和GPIO,另外ADCCLK的分频器也要配置一下
    2. 配置GPIO,把需要用的GPIO配置成模拟输入的模式
    3. 配置多路开关把左边的通道接入到右边的规则组列表里
    4. 配置ADC转换器
    5. 如果你想开启模拟看门狗,那么会有几个函数来配置阈值和监测通道,如果你想开启中断,那就在中断输出控制里用ITConfig函数来开启对应的中断输出,然后在NVIC里配置一下优先级,这样就能触发中断了
    6. 最后是开关控制,调用一下ADC_Cmd函数开启ADC

    注:我们还可以在开启ADC后,对ADC进行一下校准,这样可以减小误差 

    5.2、ADC相关库函数介绍

    5.2.1、ADCCLK配置函数

    //配置ADC时钟,参数为分频系数(6或8)
    void RCC_ADCCLKConfig(uint32_t RCC_PCLK2);
    

    5.2.2、ADC库函数中基本功能和规则组的配置

    //恢复缺省配置
    void ADC_DeInit(ADC_TypeDef* ADCx);
    
    //初始化
    void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);
    
    //结构体初始化
    void ADC_StructInit(ADC_InitTypeDef* ADC_InitStruct);
    
    //ADC最后使能,上电,开关控制
    void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);
    
    //开启DMA输出信号
    void ADC_DMACmd(ADC_TypeDef* ADCx, FunctionalState NewState);、
    
    //ADC中断输出控制
    void ADC_ITConfig(ADC_TypeDef* ADCx, uint16_t ADC_IT, FunctionalState NewState);
    
    //以下四个函数分别为复位校准、获取复位校准状态、开始校准、获取校准状态
    //用于控制校准的函数,在ADC初始化完成之后,依次调用即可
    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);
    
    //ADC获取软件开始转换状态(因为返回值SWSTART在ADC开始转换时就置0了,并不能知道是否转换)
    FlagStatus ADC_GetSoftwareStartConvStatus(ADC_TypeDef* ADCx);
    
    //获取标志位状态,参数给EOC的标志位,判断EOC标志位是不是置1
    FlagStatus ADC_GetFlagStatus(ADC_TypeDef* ADCx, uint8_t ADC_FLAG);
    
    
    //这两个函数是用来配置间断模式的
    //第一个函数是,每隔几个通道间断一次
    //第二个函数是,是否启用间断模式
    void ADC_DiscModeChannelCountConfig(ADC_TypeDef* ADCx, uint8_t Number);
    void ADC_DiscModeCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
    
    //ADC规则组通道配置
    //参数:ADCx(哪个ADC)、ADC_Channel(你想指定的通道)、Rank(序列几的位置)、ADC_SampleTime(采样时间)
    void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime);
    
    //ADC外部触发转换控制,就是是否允许外部触发转换
    void ADC_ExternalTrigConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
    
    //ADC获取转换值
    uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx);
    
    //ADC获取双模式转换值
    //这个是双ADC模式读取转换结果的函数
    uint32_t ADC_GetDualModeConversionValue(void);
    

    5.2.3、注入组函数(不多了解)

    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);
    

    5.2.4、模拟看门狗配置

    //是否启动模拟看门狗
    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);
    

    5.2.5、内部两个通道 

    //ADC内部温度传感器、内部参考电压控制
    //这个是用来开启内部的两个通道的
    void ADC_TempSensorVrefintCmd(FunctionalState NewState);
    

    5.2.6、标志位、中断挂起位相关相关

    //获取标志位状态
    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);
    

    5.3、实验部分

    5.3.1、AD单通道(单次扫描、非扫描模式) 

  • 接线图
  • AD.c
  • #include "stm32f10x.h"                  // Device header
    
    void AD_Init()
    {
    	//开启ADC时钟
    	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
    	//开启GPIO时钟
    	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    	
    	//配置ADC时钟
    	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
    	
    	//配置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);
    	
    	//配置多路开关把左边的通道接入到右边的规则组列表里
    	ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_55Cycles5);
    	
    	//ADC配置
    	ADC_InitTypeDef ADC_InitStructure;
    	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
    	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
    	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
    	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
    	ADC_InitStructure.ADC_NbrOfChannel = 1;
    	ADC_InitStructure.ADC_ScanConvMode = DISABLE;
    	ADC_Init(ADC1,&ADC_InitStructure);
    
    	//ADC使能
    	ADC_Cmd(ADC1,ENABLE);
    	
    	//ADC校准
    	ADC_ResetCalibration(ADC1);
    	while(ADC_GetResetCalibrationStatus(ADC1)==SET);
    	ADC_StartCalibration(ADC1);
    	while(ADC_GetCalibrationStatus(ADC1)==SET);
    
    }
    
    //获取ADC规则寄存器的值
    uint16_t AD_GetValue()
    {
    	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
    	while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC == RESET));
    	return ADC_GetConversionValue(ADC1);
    	
    }
    
  • main.c
  • #include "stm32f10x.h"                  // Device header
    #include "Delay.h"
    #include "OLED.h"
    #include "AD.h"
    
    uint16_t AD_Value;
    float Voltage;
    
    int main(void)
    {
    	OLED_Init();
    	AD_Init();
    	OLED_ShowString(1, 1, "AD_Value");
    	OLED_ShowString(1, 1, "Voltage:0.00");
    	while (1)
    	{
    		AD_Value = AD_GetValue();
    		Voltage = (float)(AD_Value/4095) * 3.3;
    		
    		OLED_ShowNum(1, 9, AD_Value, 4);
    		OLED_ShowNum(2, 9, Voltage, 1);
    		OLED_ShowNum(2, 11, (uint16_t)(Voltage/100) % 100, 2);		
    		
    		Delay_ms(100);
    
    	}
    }
    

    5.3.2、AD多通道

  • 接线图
  •  AD.c
  • #include "stm32f10x.h"                  // Device header
    
    void AD_Init()
    {
    	//开启ADC时钟
    	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
    	//开启GPIO时钟
    	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    	
    	//配置ADC时钟
    	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
    	
    	//配置GPIO
    	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_ContinuousConvMode = DISABLE;
    	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
    	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
    	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
    	ADC_InitStructure.ADC_NbrOfChannel = 1;
    	ADC_InitStructure.ADC_ScanConvMode = DISABLE;
    	ADC_Init(ADC1,&ADC_InitStructure);
    
    	//ADC使能
    	ADC_Cmd(ADC1,ENABLE);
    	
    	//ADC校准
    	ADC_ResetCalibration(ADC1);
    	while(ADC_GetResetCalibrationStatus(ADC1)==SET);
    	ADC_StartCalibration(ADC1);
    	while(ADC_GetCalibrationStatus(ADC1)==SET);
    
    }
    
    //获取ADC规则寄存器的值
    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);
    	
    }
    
  • main.c
  • #include "stm32f10x.h"                  // Device header
    #include "Delay.h"
    #include "OLED.h"
    #include "AD.h"
    
    uint16_t AD0, AD1, AD2, AD3;
    float Voltage;
    
    int main(void)
    {
    	OLED_Init();
    	AD_Init();
    	OLED_ShowString(1, 1, "AD0:");
    	OLED_ShowString(2, 1, "AD1:");
    	OLED_ShowString(3, 1, "AD2:");
    	OLED_ShowString(4, 1, "AD3:");
    	while (1)
    	{
    		AD0 = AD_GetValue(ADC_Channel_0);
    		AD1 = AD_GetValue(ADC_Channel_1);
    		AD2 = AD_GetValue(ADC_Channel_2);
    		AD3 = AD_GetValue(ADC_Channel_3);
    		
    		OLED_ShowNum(1, 5, AD0, 4);
    		OLED_ShowNum(2, 5, AD1, 4);
    		OLED_ShowNum(3, 5, AD2, 4);
    		OLED_ShowNum(4, 5, AD3, 4);
    		
    		Delay_ms(100);
    
    	}
    }
    

    物联沃分享整理
    物联沃-IOTWORD物联网 » 备赛电赛学习STM32篇(九):ADC

    发表评论