STM32手册中写的很详细:单次转换/连续转换,规则通道/注入通道,扫描模式。很多设置,但是比TIM简单多了感觉。。。。
双ADC模式还没有看
正点原子的例程网址,给出
http://openedv.com/posts/list/12176.htm
通过以上介绍,我们了解了STM32的单次转换模式下的相关设置,本章我们使用ADC1的通道1来进行AD转换,其详细设置步骤如下:
1)开启PA口时钟,设置PA1为模拟输入。
STM32F103ZET6的ADC通道1在PA1上,所以,我们先要使能PORTA的时钟,然后设置PA1为模拟输入。
2)使能ADC1时钟,并设置分频因子。
要使用ADC1,第一步就是要使能ADC1的时钟,在使能完时钟之后,进行一次ADC1的复位。接着我们就可以通过RCC_CFGR设置ADC1的分频因子。分频因子要确保ADC1的时钟(ADCCLK)不要超过14Mhz。
3)设置ADC1的工作模式。
在设置完分频因子之后,我们就可以开始ADC1的模式配置了,设置单次转换模式、触发方式选择、数据对齐方式等都在这一步实现。
4)设置ADC1规则序列的相关信息。
接下来我们要设置规则序列的相关信息,我们这里只有一个通道,并且是单次转换的,所以设置规则序列中通道数为1,然后设置通道1的采样周期。
5)开启AD转换器,并校准。
在设置完了以上信息后,我们就开启AD转换器,执行复位校准和AD校准,注意这两步是必须的!不校准将导致结果很不准确。
6)读取ADC值。
在上面的校准完成之后,ADC就算准备好了。接下来我们要做的就是设置规则序列1里面的通道,然后启动ADC转换。在转换结束后,读取ADC1_DR里面的值就是了。
根据正点原子的函数,在神舟做的验证,用的神舟的电位器引脚PC0,连到ADC1 CH9的pin:PB1 因为我们的PA1与外设有连接,数据不准。折腾了一番呼呼
1 //用电位器PC0 与 DAC CH9:PB1相连 2 //CH9 PB1 3 void Adc_Init(void) 4 { 5 ADC_InitTypeDef ADC_InitStructure; 6 GPIO_InitTypeDef GPIO_InitStructure; 7 8 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB |RCC_APB2Periph_ADC1, ENABLE ); //使能ADC1通道时钟 9 10 11 RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M 12 13 //PB1 作为模拟通道输入引脚 14 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; 15 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚 16 GPIO_Init(GPIOB, &GPIO_InitStructure); 17 18 ADC_DeInit(ADC1); //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值 19 20 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式 21 ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模数转换工作在单通道模式 22 ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模数转换工作在单次转换模式 23 ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动 24 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐 25 ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目 26 ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器 27 28 29 ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1 30 31 ADC_ResetCalibration(ADC1); //使能复位校准 32 33 while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束 34 35 ADC_StartCalibration(ADC1); //开启AD校准 36 37 while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束 38 39 // ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 40 41 }
设置为单次模式,不扫描,不连续。
以下是检测函数
1 //获得ADC值 2 //ch:通道值 0~3 3 u16 Get_Adc(u8 ch) 4 { 5 //设置指定ADC的规则组通道,一个序列,采样时间 6 ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期 7 8 ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 9 10 while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束 11 12 return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果 13 }
1 u16 Get_Adc_Average(u8 ch,u8 times) 2 { 3 u32 temp_val=0; 4 u8 t,i; 5 for(t=0;t<times;t++) 6 { 7 temp_val+=Get_Adc(ch); 8 for(i=0;i<200;i++) ; 9 } 10 return temp_val/times; 11 }
多测几次获得平均值
main函数中检测输出值就可以了
1 Adc_Init(); //ADC初始化 PB1 2 3 while(1) 4 { 5 6 adcx=Get_Adc_Average(ADC_Channel_9,10); 7 // adcx=Get_Adc(ADC_Channel_9); 8 temp=(float)adcx*(3.3/4096); 9 printf("\r\n PB1 ADC 9CH 当前AD转换结果为:0x%X, 百分比为:%d%%,电压值:%f V.\n\r", adcx, adcx*100/4096, temp); 10 }