STM32.ADC

来源:互联网 发布:免费工资管理软件 编辑:程序博客网 时间:2024/05/16 07:45

ADC实验

原理图:

 

 

1.ADC配置函数

/* enable adc1 and config adc1 to dma mode */
ADC1_Init();

/**  * @brief  ADC1初始化  * @param  无  * @retval 无  */void ADC1_Init(void){ADC1_GPIO_Config();    //端口初始化ADC1_Mode_Config();}

 对于配置ADC1的工作模式为MDA模式

ADC1是挂载到DMA1的通道1的

#define ADC1_DR_Address    ((u32)0x40012400+0x4c)//模拟量转换成数字量后存放到该地址中

看如下存储器映像 寄存器组起始地址(起始地址)

 

便宜地址:0x4c

----------------------------------------------------------------------------------------------------------------------

那么对于ADC自身配置:

采样总时间:

对于STM32最快的ADC转换时间也就是1US了,打死都不会变的了

-----------------------------------------------------------------------------------------------------------------------

代码:

/**  * @brief  配置ADC1的工作模式为MDA模式  * @param  无  * @retval 无  */static void ADC1_Mode_Config(void){    DMA_InitTypeDef DMA_InitStructure;    ADC_InitTypeDef ADC_InitStructure;        /* DMA channel1 configuration */    DMA_DeInit(DMA1_Channel1);//重新配置        DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;         //ADC地址    DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;    //内存地址    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //方向从#define DMA_DIR_PeripheralSRC                                     //((uint32_t)0x00000000)开始发    DMA_InitStructure.DMA_BufferSize = 1;            //每次只发送一个数据    DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;    //外设地址固定    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;          //内存地址固定    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);        /* Enable DMA channel1 */    DMA_Cmd(DMA1_Channel1, ENABLE);        /* ADC1 configuration */        ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;    //独立ADC模式(因为咱们只用一个)    ADC_InitStructure.ADC_ScanConvMode = DISABLE ;          //禁止扫描模式,扫描模式用于多通道采集                              //(ADC有3个通道,若几个通道同时采集一个模拟量则用扫描)                          //若同时采(交叉采集),速率更高,对于32的示波器就是三路同时采集       ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;  //开启连续转换模式,即不停地进行ADC转换    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;    //不使用外部触发转换    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                             //采集数据右对齐,数据时12位,保存数据的寄存器是16位    ADC_InitStructure.ADC_NbrOfChannel = 1;             //要转换的通道数目1    ADC_Init(ADC1, &ADC_InitStructure);        /*配置ADC时钟,为PCLK2的8分频,即9MHz(最大是14M)68/9MS*/    RCC_ADCCLKConfig(RCC_PCLK2_Div8);     /*配置ADC1的通道11为55.5个采样周期(模拟量转换成数字量需要多长时间),     序列为1(有三个通道它排第一) */     ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 1, ADC_SampleTime_55Cycles5);        /* Enable ADC1 DMA */    ADC_DMACmd(ADC1, ENABLE);        /* Enable ADC1 */    ADC_Cmd(ADC1, ENABLE);        /*复位校准寄存器 */       ADC_ResetCalibration(ADC1);    /*等待校准寄存器复位完成 */    while(ADC_GetResetCalibrationStatus(ADC1));        /* ADC校准 */    ADC_StartCalibration(ADC1);    /* 等待校准完成*/    while(ADC_GetCalibrationStatus(ADC1));        /* 由于没有采用外部触发,所以使用软件触发ADC转换 */     ADC_SoftwareStartConvCmd(ADC1, ENABLE);}

 

对于main函数

    while (1)    {        ADC_ConvertedValueLocal =(float) ADC_ConvertedValue/4096*3.3; // 读取转换的AD值            printf("\r\n The current AD value = 0x%04X \r\n", ADC_ConvertedValue);         printf("\r\n The current AD value = %f V \r\n",ADC_ConvertedValueLocal);         Delay(0xffffee);      }

为什么是4096

设V为我们转换出来的电压 

V/3.3  = 读取的寄存器值/4096  (寄存器满值2^12 = 4096) 当然此时Vref- = 0V  ;Vref+ = 3.3V;若 Vref+ = (x)V  V/(x)  = 读取的寄存器值/4096  

 

 

 

--------------------------------------------------------------------------------------------------------------------------

 

 

 

 

 

 

 

 

1 0
原创粉丝点击