天天看点

【STM32标准库】【基础知识】ADC转换,非DMA的单通道和多通道ADC概述ADC初始化模拟多通道采集(非DMA)总例程

文章目录

  • ADC概述
  • ADC初始化
    • GPIO初始化
    • 时钟
    • 全局ADC设置
      • 定义结构体
      • 是否使用DMA
      • 工作模式
      • 分频
      • 采样延迟
      • 例程
    • 单独ADC设置
      • 定义结构体
      • 连续转换
      • 数据对齐
      • 外部触发
      • 通道数量
      • 分辨率
      • 扫描模式
      • 例程
    • 设置规则
    • 打开软件调用ADC
  • 模拟多通道采集(非DMA)
    • 思路
    • 函数
  • 总例程

文章基于适用于STM32F4系列,作者使用STM32F401CCU6开发板。

本文章基于此系列和开发板展开讨论。

ADC概述

A是指模拟信号,D是指数字信号,C是指变换

故名思意,ADC即将模拟信号转换为数字信号的一种变换。

STM32F4的ADC为逐次逼近型ADC

原理是利用DAC产生不同大小的电压,与输入的电压比较,得到输入电压的大体范围,这个范围的误差即为分辨率

如为8位分辨率的ADC,则分度值为

3.3/2^8

,即与真实值的误差不会超过12.89mV

具体原理请看微机原理

ADC初始化

流程

  1. GPIO初始化
  2. 打开时钟
  3. 全局ADC设置
  4. 单独ADC设置

GPIO初始化

之前文章中介绍过,传送门,需要设置为模拟模式

例程

GPIO_InitTypeDef GPIO_InitStruct;						//GPIO初始化结构体
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);	//打开时钟
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AN;				//模拟模式
GPIO_InitStruct.GPIO_OType = GPIO_OType_OD;				//随意设置
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;		//端口
GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_NOPULL;			//随意设置
GPIO_InitStruct.GPIO_Speed = GPIO_High_Speed;			//高速
GPIO_Init(GPIOA, &GPIO_InitStruct);	
           

时钟

本系列单片机只有1个ADC,ADC1,端口对应请看这篇文章,传送门

使用这句命令打开ADC的时钟

全局ADC设置

这部分的设置是对全部的ADC有效的(本单片机即ADC1)

定义结构体

是否使用DMA

ADC_CommonInitStruct.ADC_DMAAccessMode
           

可以的选择使用或不使用,本文为不使用的情况。

工作模式

可以选择外部触发源或者不使用触发

本文为不使用外部触发(独立工作模式)

分频

ADC需要频率在12MHz以下,以AHB1的频率(与系统频率相等)为基准进行分频

本文选择8分频

采样延迟

这个只有在多ADC时才有效,作为两个ADC采样之间的时间间隔

例程

ADC_CommonInitTypeDef ADC_CommonInitStruct;
ADC_CommonInitStruct.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;	   //不使用DMA
ADC_CommonInitStruct.ADC_Mode = ADC_Mode_Independent;					   //独立工作模式
ADC_CommonInitStruct.ADC_Prescaler = ADC_Prescaler_Div8;				   //分频,建议分频后小于12MHZ
ADC_CommonInitStruct.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_20Cycles; //采样延迟,多ADC使用才有效
ADC_CommonInit(&ADC_CommonInitStruct);
           

单独ADC设置

下面的设置是针对单独的ADC的

定义结构体

连续转换

将所有开启的通道采集转换完一轮后自动开启下一轮的采集转换,这是连续模式

数据对齐

选择右对齐即可,其他基本上用不到

外部触发

可以选择边沿触发或者定时器中断触发,或者不使用外部触发

这里选择的不使用外部触发

ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC1;		 //选择外部触发事件,这里随意即可,因为使用了不允许外部触发
	ADC_InitStruct.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None; //软件触发
           

通道数量

即选择打开了几个通道

分辨率

即最大误差范围,这个数据的选择与后期计算电压值相关

可以是

分辨率 分割数量
6bit 64
8bit 256
10bit 1024
12bit 4096

扫描模式

用于多通道扫描,完成全部打开的通道的扫描后中断标志置位

但是后接受的数据会被覆盖,需要使用DMA

这里是单通道(非DMA多通道是软件模拟的),因此此处设置与不设置均可

例程

ADC_InitTypeDef ADC_InitStruct;
ADC_InitStruct.ADC_ContinuousConvMode = ENABLE;							 //连续模式
ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;						 //右对齐
ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC1;		 //选择外部触发事件,这里随意即可,因为使用了不允许外部触发
ADC_InitStruct.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None; //软件触发
ADC_InitStruct.ADC_NbrOfConversion = 1;									 //通道数量
ADC_InitStruct.ADC_Resolution = ADC_Resolution_12b;						 //分辨率
ADC_InitStruct.ADC_ScanConvMode = ENABLE;								 //扫描模式
ADC_Init(ADC1, &ADC_InitStruct);
ADC_Cmd(ADC1, ENABLE);
           

设置规则

使用此函数设置对应规则

分别输入ADC号(本单片机只有ADC1),ADC通道号,标记号,采样时间

标记号只能小于等于通道数,从1开始,采样时间是会将这段时间内的数据平均后输出

打开软件调用ADC

如果使用软件读取ADC,需要事先调用此函数

需要在设置后才可以使用软件调用ADC

因此可以使用这段代码确保已经设置成功

while (!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC))
		;
           

模拟多通道采集(非DMA)

思路

每次调用此读取ADC时,改变ADC对应的通道,设置成功后读取ADC并返回数据

函数

u16 ADC1_Read(u8 ch)
{
	ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_144Cycles); //设置通道规则
	ADC_SoftwareStartConv(ADC1);									 //开启软件触发
	while (!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC))
		;
	return ADC_GetConversionValue(ADC1); //软件触发
}
           

总例程

void ADC_init()
{
	GPIO_InitTypeDef GPIO_InitStruct;
	ADC_InitTypeDef ADC_InitStruct;
	ADC_CommonInitTypeDef ADC_CommonInitStruct;
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AN;
	GPIO_InitStruct.GPIO_OType = GPIO_OType_OD;
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;
	GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_NOPULL;
	GPIO_InitStruct.GPIO_Speed = GPIO_High_Speed;
	GPIO_Init(GPIOA, &GPIO_InitStruct);

	ADC_CommonInitStruct.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;	   //不使用DMA
	ADC_CommonInitStruct.ADC_Mode = ADC_Mode_Independent;					   //独立工作模式
	ADC_CommonInitStruct.ADC_Prescaler = ADC_Prescaler_Div8;				   //分频,建议分频后小于12MHZ
	ADC_CommonInitStruct.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_20Cycles; //采样延迟,多ADC使用才有效
	ADC_CommonInit(&ADC_CommonInitStruct);

	ADC_InitStruct.ADC_ContinuousConvMode = ENABLE;							 //连续模式
	ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;						 //右对齐
	ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC1;		 //选择外部触发事件,这里随意即可,因为使用了不允许外部触发
	ADC_InitStruct.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None; //软件触发
	ADC_InitStruct.ADC_NbrOfConversion = 1;									 //通道数量
	ADC_InitStruct.ADC_Resolution = ADC_Resolution_12b;						 //分辨率
	ADC_InitStruct.ADC_ScanConvMode = ENABLE;								 //扫描模式
	ADC_Init(ADC1, &ADC_InitStruct);
	ADC_Cmd(ADC1, ENABLE);
}

           
u16 ADC1_Read(u8 ch)
{
	ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_144Cycles); //设置通道规则
	ADC_SoftwareStartConv(ADC1);									 //开启软件触发
	while (!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC))
		;
	return ADC_GetConversionValue(ADC1); //软件触发
}
           
需要读取时只需要调用此函数即可,更改通道号即可读取多通道数据,例如这样
int main()
{
	int i = 0;
	ADC_init();
	Usart_init();
	while (1)
	{
		i = 0;
		i = ADC1_Read(ADC_Channel_1);
		i = (i * 3300 / 0xfff);
		printf("ch1=%d mV \r\n", i);
		Delay_ms(300);
		i = 0;
		i = ADC1_Read(ADC_Channel_0);
		i = (i * 3300 / 0xfff);
		printf("ch0=%d mV \r\n", i);
		Delay_ms(300);
	}
}