zoukankan      html  css  js  c++  java
  • STM32.ADC

    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("
     The current AD value = 0x%04X 
    ", ADC_ConvertedValue); 
            printf("
     The current AD value = %f V 
    ",ADC_ConvertedValueLocal); 
    
            Delay(0xffffee);  
        }

    为什么是4096

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

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

     Vref+ 与 Vdda的关系:

    1.Vdda和Vssa分别为ADC转换电路的电源引脚和电源地

    2.

    3.电压范围:

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

     

  • 相关阅读:
    如何理解volatile关键字
    Spring 4+JMS+ActiveMQ Example with Annotations
    Spring MVC如何获得客户端IP地址
    PDFBox –如何读取PDF的内容
    Spring MVC 使用Ehcache作为缓存的例子
    自定义了一个data table控件
    Linux基本配置
    位操作学习
    QT——QPainter类详解
    QT——设计文件的和控制类的关联问题
  • 原文地址:https://www.cnblogs.com/Ph-one/p/4155008.html
Copyright © 2011-2022 走看看