STM32——ADC

来源:互联网 发布:优化发展环境情况汇报 编辑:程序博客网 时间:2024/05/23 10:29
12位ADC是一种逐次逼近型模拟数字数字转换器。它有多达18个通道,可测量16个外部和2个内部信号源。

ADC的输入时钟不得超过14MHZ,它是由PCLK2经分频产生。

如果被ADC转换的模拟电压低于低阀值或高于高阀值,AWD模拟看门狗状态位被设置。

ADC通常要与DMA一起使用 这里只是简单的用库配置ADC 不断扫描来实现ADC的应用。

首先配置GPIO与ADC的时钟:

ADC_InitTypeDef  ADC_InitStructure; 
GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE); 
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

GPIO_InitStructure.GPIO_Pin  =GPIO_Pin_1; 
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_AIN; 
GPIO_Init(GPIOB,&GPIO_InitStructure); //默认速度为两兆

配置ADC的运行:

ADC_InitStructure.ADC_Mode              = ADC_Mode_Independent;  //独立模式 
ADC_InitStructure.ADC_ScanConvMode      =DISABLE;      //连续多通道模式 
ADC_InitStructure.ADC_ContinuousConvMode =ENABLE;      //连续转换 
ADC_InitStructure.ADC_ExternalTrigConv  = ADC_ExternalTrigConv_None; //转换不受外界决定 
ADC_InitStructure.ADC_DataAlign         =ADC_DataAlign_Right;   //右对齐 
ADC_InitStructure.ADC_NbrOfChannel      =1;       //扫描通道数 
ADC_Init(ADC1,&ADC_InitStructure); 
ADC_RegularChannelConfig(ADC1,ADC_Channel_9, 1,ADC_SampleTime_1Cycles5); //通道X,采样时间为1.5周期,1代表规则通道第1个这个1是啥意思我不太清楚只有是1的时候我的ADC才正常。 
ADC_Cmd  (ADC1,ENABLE);             //使能或者失能指定的ADC 
ADC_SoftwareStartConvCmd(ADC1,ENABLE);//使能或者失能指定的ADC的软件转换启动功能

STM32的ADC通道与GPIO对照表:(STM32不完全手册)


这里我用的是ADC1的9通道 PB1引脚。

也用一些默认的配置函数同GPIO 的一样例如: ADC_StructInit

ADC_InitTypeDef structureADC_InitTypeDef定义于文件“stm32f10x_adc.h”:

typedef struct

{

u32 ADC_Mode; FunctionalState ADC_ScanConvMode; FunctionalStateADC_ContinuousConvMode; u32 ADC_ExternalTrigConv; u32ADC_DataAlign; u8 ADC_NbrOfChannel;

} ADC_InitTypeDef

注意:为了能够正确地配置每一个ADC通道,用户在调用ADC_Init()之后,必须调用ADC_ChannelConfig()来配置每个所使用通道的转换次序和采样时间。

然后就是不停的读;

u16 TestAdc(void) 

u16 adc; 
while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC)==RESET); //检查制定ADC标志位置1与否 ADC_FLAG_EOC 转换结束标志位 
adc=ADC_GetConversionValue(ADC1);

returnadc;//返回最近一次ADCx规则组的转换结果 
}

这个程序的8位单片机风格很重,真正的ADC一定要放在DMA或是中断之中。


转自:http://blog.csdn.net/wangxiaobupt/article/details/18562111

ADC干嘛的就不多说了,按照平常习惯走。

补:有关ADC的通道

STM32中有三个ADC,每个ADC最多有16各通道,16个通道是以时分复用的方式共用一个ADC转换器的,就好比内部有很多个电子开关,每个电子开关控制一路信号的接入,这样ADC的使用效率就提高了很多

一、初始化

1.定义结构体

typedef struct    //ADC结构体的组成
{
uint32_t ADC_Mode; 
FunctionalState ADC_ScanConvMode;   
FunctionalState ADC_ContinuousConvMode; 
uint32_t ADC_ExternalTrigConv;         
uint32_t ADC_DataAlign;                
uint8_t ADC_NbrOfChannel;  
}ADC_InitTypeDef;


ADC_InitTypeDef ADC_InitStructure; 
GPIO_InitTypeDef GPIO_InitStructure;

2.开启ADCx时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE ); //使能ADC1通道时钟
 
3.设置ADC采集的速率
RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M


4.初始化相应的GPIO       //设置为模拟输入                      
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;  //模拟输入引脚
GPIO_Init(GPIOA, &GPIO_InitStructure);

5.复位ADC
ADC_DeInit(ADC1);  //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值

6.初始化ADC
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;  //ADC工作模式:ADC1和ADC2工作在独立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE;  //模数转换工作在单通道模式
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//模数转换工作在单次转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//转换由软件而不是外部触发启动
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//ADC数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目
ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   

7.使能 

ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );//ADC1,ADC通道,采样时间为239.5周期
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1


8.复位与校准
ADC_ResetCalibration(ADC1); //使能复位校准   
while(ADC_GetResetCalibrationStatus(ADC1));  //等待复位校准结束
ADC_StartCalibration(ADC1); //开启AD校准 
while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束


二、采集

1.软件启动 //软件启动就是软件开启转换  也可以设置为外部开启,外部电平变化时自动开启转换

ADC_SoftwareStartConvCmd(ADC1, ENABLE);//使能指定的ADC1的软件转换启动功能 

2.等待转换结束之后返回结果

while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束
return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果


转自:http://blog.csdn.net/liuxizhen2009/article/details/17094757

[cpp] view plaincopyprint?
  1. #include "led.h"  
  2. #include "delay.h"  
  3. #include "key.h"  
  4. #include "sys.h"  
  5. #include "lcd.h"  
  6. #include "usart.h"     
  7. #include "adc.h"  
  8. //ALIENTEK战舰STM32开发板实验17  
  9. //ADC 实验    
  10. //技术支持:www.openedv.com  
  11. //广州市星翼电子科技有限公司  
  12. /* Private define ------------------------------------------------------------*/  
  13. #define ADC1_DR_Address    ((u32)0x4001244C)   
  14. extern void DMA_Configuration(void);   
  15. extern void Uart1Putc(char c);  
  16. u16 AD_Value=0xff;   
  17. void  Adc_Init(void)  
  18. {     
  19.     ADC_InitTypeDef ADC_InitStructure;   
  20.     GPIO_InitTypeDef GPIO_InitStructure;  
  21.   
  22.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1    , ENABLE );   //使能ADC1通道时钟  
  23.    
  24.   
  25.     RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M  
  26.   
  27.     //PA1 作为模拟通道输入引脚                           
  28.     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;  
  29.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;       //模拟输入引脚  
  30.     GPIO_Init(GPIOA, &GPIO_InitStructure);    
  31.   
  32.     ADC_DeInit(ADC1);  //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值  
  33.   
  34.     ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;  //ADC工作模式:ADC1和ADC2工作在独立模式  
  35.     ADC_InitStructure.ADC_ScanConvMode = DISABLE;   //模数转换工作在单通道模式  
  36.     ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;  //模数转换工作在单次转换模式  
  37.     ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动  
  38.     ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;  //ADC数据右对齐  
  39.     ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目  
  40.     ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器     
  41.     //设置指定ADC的规则组通道,一个序列,采样时间  
  42.     ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_13Cycles5 );    //ADC1,ADC通道,采样时间为239.5周期                     
  43.   
  44.     
  45.     ADC_Cmd(ADC1, ENABLE);  //使能指定的ADC1  
  46.     // 开启ADC的DMA支持(要实现DMA功能,还需独立配置DMA通道等参数)  
  47.     ADC_DMACmd(ADC1, ENABLE);  
  48.       
  49.     ADC_ResetCalibration(ADC1); //使能复位校准    
  50.        
  51.     while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束  
  52.       
  53.     ADC_StartCalibration(ADC1);  //开启AD校准  
  54.    
  55.     while(ADC_GetCalibrationStatus(ADC1));   //等待校准结束  
  56.    
  57. //  ADC_SoftwareStartConvCmd(ADC1, ENABLE);     //使能指定的ADC1的软件转换启动功能  
  58.   
  59. }                   
  60. //获得ADC值  
  61. //ch:通道值 0~3  
  62. u16 Get_Adc(u8 ch)     
  63. {  
  64.     //设置指定ADC的规则组通道,一个序列,采样时间  
  65.     ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );  //ADC1,ADC通道,采样时间为239.5周期                     
  66.     
  67.     ADC_SoftwareStartConvCmd(ADC1, ENABLE);     //使能指定的ADC1的软件转换启动功能      
  68.        
  69.     while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束  
  70.   
  71.     return ADC_GetConversionValue(ADC1);    //返回最近一次ADC1规则组的转换结果  
  72. }  
  73.   
  74. u16 Get_Adc_Average(u8 ch,u8 times)  
  75. {  
  76.     u32 temp_val=0;  
  77.     u8 t;  
  78.     for(t=0;t<times;t++)  
  79.     {  
  80.         temp_val+=Get_Adc(ch);  
  81.         delay_ms(5);  
  82.     }  
  83.     return temp_val/times;  
  84. }      
  85. /******************************************************************************* 
  86. * Function Name  : DMA_Configuration 
  87. * Description    : DMA设置:从ADC模块自动读转换结果至内存 
  88. * Input          : None 
  89. * Output         : None 
  90. * Return         : None 
  91. *******************************************************************************/  
  92. void DMA_Configuration(void)  
  93. {  
  94.     DMA_InitTypeDef DMA_InitStructure;  
  95.         //启动DMA时钟  
  96.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);  
  97.      
  98.     DMA_DeInit(DMA1_Channel1);  
  99.     DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;  
  100.     DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&AD_Value;  
  101.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  
  102.     //BufferSize=2,因为ADC转换序列有2个通道  
  103.     //如此设置,使序列1结果放在AD_Value[0],序列2结果放在AD_Value[1]  
  104.     DMA_InitStructure.DMA_BufferSize = 1;  
  105.     DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  
  106.     DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;  
  107.     DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;  
  108.     DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;  
  109.     //循环模式开启,Buffer写满后,自动回到初始地址开始传输  
  110.     DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;  
  111.     DMA_InitStructure.DMA_Priority = DMA_Priority_High;  
  112.     DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  
  113.     DMA_Init(DMA1_Channel1, &DMA_InitStructure);  
  114.     //配置完成后,启动DMA通道  
  115.     DMA_Cmd(DMA1_Channel1, ENABLE);  
  116. }  
  117.   
  118.  int main(void)  
  119.  {     
  120.     u16 adcx;  
  121.     float temp;  
  122.     delay_init();            //延时函数初始化      
  123.     NVIC_Configuration();    //设置NVIC中断分组2:2位抢占优先级,2位响应优先级  
  124.     uart_init(9600);        //串口初始化为9600  
  125.     LED_Init();              //LED端口初始化  
  126.     LCD_Init();   
  127.     DMA_Configuration();              
  128.     Adc_Init();             //ADC初始化  
  129.   
  130.     POINT_COLOR=RED;//设置字体为红色   
  131.     LCD_ShowString(60,50,200,16,16,"WarShip STM32");      
  132.     LCD_ShowString(60,70,200,16,16,"ADC TEST");   
  133.     LCD_ShowString(60,90,200,16,16,"ATOM@ALIENTEK");  
  134.     LCD_ShowString(60,110,200,16,16,"2012/9/7");      
  135.     //显示提示信息  
  136.     POINT_COLOR=BLUE;//设置字体为蓝色  
  137.     LCD_ShowString(60,130,200,16,16,"ADC_CH0_VAL:");            
  138.     LCD_ShowString(60,150,200,16,16,"ADC_CH0_VOL:0.000V");  
  139.         //启动第一次AD转换  
  140.     ADC_SoftwareStartConvCmd(ADC1, ENABLE);  
  141.     //因为已经配置好了DMA,接下来AD自动连续转换,结果自动保存在AD_Value处   
  142.       
  143.     while(1){  
  144.         printf("%04x\n",AD_Value);  
  145.         LCD_ShowxNum(156,130,AD_Value,4,16,0);//显示ADC的值  
  146.         delay_ms(200);  
  147.     }            
  148. //  while(1)  
  149. //  {  
  150. //      adcx=Get_Adc_Average(ADC_Channel_1,10);  
  151. //      LCD_ShowxNum(156,130,adcx,4,16,0);//显示ADC的值  
  152. //      temp=(float)adcx*(3.3/4096);  
  153. //      adcx=temp;  
  154. //      LCD_ShowxNum(156,150,adcx,1,16,0);//显示电压值  
  155. //      temp-=adcx;  
  156. //      temp*=1000;  
  157. //      LCD_ShowxNum(172,150,temp,3,16,0X80);  
  158. //      LED0=!LED0;  
  159. //      delay_ms(250);    
  160. //  }  
  161.  }  
0 0