对于STM32F103来说,它的ADC是12位,一共18个通道,其中16个外部通道,2个内部通道。支持单次,连续以及间断模式扫描。
指的是STM32F103的ADC分辨率具有12位,位数越高采集到的ADC越精准。12位是相对于二进制数来说,也就是“111111111111”,转换为十进制就是4095,其实是0-4095,实际上是4096个数,STM32F103的引脚电压是0-3.3V,12位的ADC就会把0-3.3V切割成4096份。这样转换器得到的ADC值便可以转换为相应电压,设转换器采集到的ADC值为x,实际所求电压为y。那么公式为:y=x/4096*3.3V。
f103的芯片上有16个引脚是接到模拟电压上可以进行电压检测的,这16个通道会分给3个转换器,这三个转换器是独立的,也就是我们常见的ADC1,ADC2,ADC3。引脚和通道的对应关系可通过手册或者芯片引脚定义查询。
因为手册的引脚定义表太大,我这里就给出一个开发板原理图上的芯片引脚图。在图中可以看到哪个引脚有ADC通道。
比如PC0 标注的是ADC123_IN10,说明该引脚可以作为ADC引脚,而123说明该引脚可以被ADC1.ADC2,ADC3三个转化器共用。10对应上面所说的18个通道中的10通道。
我在使用该应用的时候是因为项目需要有一个有线遥控器连接控制板,遥控器上按键很多,如果采用外部中断或者按键扫描这种模式,那么每一个按键都需要连接控制板,造成IO口资源的浪费,同时连接线也会更粗,更贵。于是我们采用双排按键,每一排按键对应一个ADC的IO口,根据每个按键按下的电压不同来判断哪个按键被按下。
我们采用固件库的方式编写代码,那么就先初始化ADC的端口:
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA ,ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
首先还是走老路,初始化所用的GPIO,此次所用的是PA4和PA5。通过手册可以看到这两个口对应的是ADC1和ADC2,对应通道是通道4和通道5。
于是选择ADC1的通道4和通道5对ADC进行初始化:
ADC_InitTypeDef ADC_InitStructure;
RCC_APB2PeriphClockCmd( RCC_APB2Periph_ADC1 ,ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div6);
ADC_DeInit(ADC1);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 2;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_RegularChannelConfig(ADC1, ADC_Channel_4,1, ADC_SampleTime_71Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_5,2, ADC_SampleTime_71Cycles5);
ADC_Cmd(ADC1, ENABLE);
ADC_DMACmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
DMA_Cmd(DMA1_Channel1, ENABLE);
参数 ADC_Mode :是用来设置 ADC 的模式 ,ADC的模式很多(如下图),这里选择独立模式:ADC_Mode_Independent
参数ADC_ScanConvMode :规定了模数转换工作在扫描模式(多通道)还是单次(单通道)模式。可以设置这个参数为 ENABLE 或者 DISABLE。
参数ADC_ContinuousConvMode :规定了模数转换工作在连续还是单次模式。可以设置这个参数为 ENABLE 或者 DISABLE。
参数ADC_ExternalTrigConv :定义了使用外部触发来启动规则通道的模数转换 ,可取值如下图
参数ADC_DataAlign :规定了 ADC 数据向左边对齐还是向右边对齐。
参数ADC_NbreOfChannel :规定了顺序进行规则转换的 ADC 通道的数目 。我这里只用了双通道,所以为2.
函数ADC_RegularChannelConfig():这一步比较重要,上面我们初始化了ADC,但是并未指定引脚,也就是规则组通道,这里设置有规则组通道和转化顺序以及采样时间。根据上面所说,我们选择ADC1,通道4和5,顺序是通道4第一个采样,通道5第二个采样。ADC_SampleTime 设定了通道的 ADC 采样时间,可选参数如下:
DMA_InitTypeDef DMA_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr =(u32)&ADC1->DR;
DMA_InitStructure.DMA_MemoryBaseAddr =(u32)&ADCConvertedValue;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = 100*2;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
DMA_Cmd(DMA1_Channel1, ENABLE);
因为我在代码中做了备注,这里踢几个人比较重要的点:
参数DMA_BufferSize :设置DMA在传输时缓冲区的长度 ,这里要提一下我的数据储存数组u16 ADCConvertedValue[100][2];因为采用两个通道,所以需要两个缓冲保存数据,100*2=200,DMA在运行是就会采集200个ADC值保存在数组中,100个通道4的值,100个通道5的值。取值处理会在后面介绍。
此项设置完之后DMA就回不停地采集这两个ADC通道中的ADC值并保存在数组中。我们只需要在需要的时候读取数组中的值便可。
int sum;
int sum1=0;
int sum2=0;
int value1[10];
int value2[10];
int isSorted,temp1,temp2,k,p;
float a;
int i,j,c;
float ADC_Value[2];
for(c=0;c<10;c++){
for(i=0;i<2;i++)
{
sum=0;
for(j=0;j<100;j++)
{
sum+=ADCConvertedValue[j][i];
}
ADC_Value[i]=(float)sum*3.3/4096/100;
}
value1[c]=(int)(ADC_Value[0]*100);
value2[c]=(int)(ADC_Value[1]*100);
}