ADC实验

原理图:

STM32.ADC-LMLPHP

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模式

STM32.ADC-LMLPHP

ADC1是挂载到DMA1的通道1的

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

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

STM32.ADC-LMLPHP

便宜地址:0x4c

STM32.ADC-LMLPHP

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

那么对于ADC自身配置:

STM32.ADC-LMLPHP

采样总时间:

STM32.ADC-LMLPHP

STM32.ADC-LMLPHP

对于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 = ;     //每次只发送一个数据
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
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, , 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 ()
{
ADC_ConvertedValueLocal =(float) ADC_ConvertedValue/*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

Vref+ 与 Vdda的关系:

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

2.

STM32.ADC-LMLPHP

3.电压范围:

STM32.ADC-LMLPHP

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

 

05-11 22:12