【蓝桥杯嵌入式备赛】5.ADC驱动快速编写(一路和两路)和问题总结

引子

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,通道号和跳线帽也需要修改。在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值