ADC 转换:神舟IV实验

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与外设有连接,数据不准。折腾了一番呼呼

Adc_Init
 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 }                  

设置为单次模式,不扫描,不连续。

以下是检测函数

Get_Adc
 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 }
Get_Adc_Average
 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函数中检测输出值就可以了

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 }
原文地址:https://www.cnblogs.com/wwjdwy/p/2965142.html