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
Vref+ 与 Vdda的关系:
1.Vdda和Vssa分别为ADC转换电路的电源引脚和电源地
2.
3.电压范围:
--------------------------------------------------------------------------------------------------------------------------