引子
ADC算是整块板子比较重要的部分了,在往年的题目经常出现(虽然我参加的十二届省赛没考)。经常会和其他部分结合考察。
主板上的ADC对应PB0引脚(ADC1的通道8)
两路ADC需要用到扩展板
驱动编写
参考固件库STM32F10x_StdPeriph_Examples\ADC\ADC1_DMA main.c
复制其中的IO配置、时钟配置和ADC配置即可,DMA不需要
获取ADC数据
这里需要注意获取的流程: 使能开启软件转换 -> 等待软件转换结束标志位置1 -> 读取数据返回
用到的函数
void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);
FlagStatus ADC_GetFlagStatus(ADC_TypeDef* ADCx, uint8_t ADC_FLAG);
uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx);
两路ADC涉及通道选择还需要使用
void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime);
注意:标志位要选择ADC_FLAG_EOC
如图
float Adc_Get(void)
{
u16 value;
float vol;
ADC_SoftwareStartConvCmd(ADC1,ENABLE);
while(!ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC));
value = ADC_GetConversionValue(ADC1);
vol = (float)3.3*value/4095;
return vol;
}
一路ADC代码
#include "adc.h"
#include "lcd.h"
#include "stdio.h"
void Adc_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOB, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOB, &GPIO_InitStructure);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
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_InitStructure.ADC_NbrOfChannel = 1;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 1, ADC_SampleTime_55Cycles5);
ADC_Cmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
float Adc_Get(void)
{
u16 value;
float vol;
ADC_SoftwareStartConvCmd(ADC1,ENABLE);
while(!ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC));
value = ADC_GetConversionValue(ADC1);
vol = (float)3.3*value/4095;
return vol;
}
void Adc_Show(void)
{
u8 str[20];
float vol;
vol = Adc_Get();
sprintf((char*)str," V: %.2f V ",vol);
LCD_DisplayStringLine(Line2 ,str);
}
两路ADC代码
#include "adc.h"
#include "lcd.h"
#include "stdio.h"
extern float Volt_1,Volt_2;
void Adc_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
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_InitStructure.ADC_NbrOfChannel = 2;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_RegularChannelConfig(ADC1, ADC_Channel_4|ADC_Channel_5, 1, ADC_SampleTime_55Cycles5);
ADC_Cmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);
}
float Get_Adc(u8 channel)
{
u16 Adc_value;
float Volt;
ADC_RegularChannelConfig(ADC1,channel,1,ADC_SampleTime_55Cycles5);
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while(!ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC));
Adc_value=ADC_GetConversionValue(ADC1);
Volt = (float)3.3*Adc_value/4095;
return Volt;
}
void Adc_conv(void)
{
float Volt_1,Volt_2;
u8 str[30];
Volt_1=Get_Adc(ADC_Channel_4);
Volt_2=Get_Adc(ADC_Channel_5);
sprintf((char*)str," Vol_1: %.2fV ",Volt_1);
LCD_DisplayStringLine(Line5 ,str);
sprintf((char*)str," Vol_2: %.2fV ",Volt_2);
LCD_DisplayStringLine(Line7 ,str);
}
几个小问题
1.注意标志位的函数和参数传入不要和ADC_ClearFlag()、ADC_ClearITPendingBit()搞混,这两个函数传入值是中断类型。
2、读取数值以后通常要进行转换得到它的电压值,上面利用的是vol = (float)3.3value/4095 可以成功,但是博主尝试vol = (float)(3.3(value/4095))或者vol = (float)(3.3*value/4095)就失效显示电压一直是0(这个知道怎么改就行,我也还没想通,应该是类型转换问题)
3.第十一届省赛考了ADC电压等于某个值时候进行操作,但其实这是挖了一个坑。
通常理解会直接判断变量是否等于0或3.3,但实际float变量不能精确判断,可以采用一个接近的范围判断。 if(vol==3.3)可以改为if(vol>3.2)
4.如果用扩展板进行两路显示,需要更改驱动配置,ADC通道数要改成2,通道号和跳线帽也需要修改。