一、STM32 ADC
1.1参考资料
《STM32不完全手册》第20章 《STM32中文参考手册》第11章 芯片数据手册ADC部分+GPIO部分
1.2 什么是ADC
ADC是Analog-to- Converter 的缩写,指模/数转换器,将连续的模拟信号转换为离散的数字信号的器件,一般使用电压值来表示数字化。
1.3 怎么学
实话说,看完中文参考手册ADC部分,就是一个感觉,听君一席话,就是听君一席话… 学习的时候管制一下这几个问题
- 几个ADC控制器
- 几位什么型的AD
- 那些引脚有ADC功能
- 参考电压问题
- 注入通道/规则通道是什么,有什么区别
- 有哪些转换模式(单次、连续、扫描、间断)
- 时钟
- 寄存器
1、基本信息
2、引脚对应表
3、电压问题
4、框图
注释: 1、规则通道正常进入,按照代码顺序执行的AD采集,最多16个通道,注入通道通过中断事件进入的AD采集,最多4个通道
5、关于模式
1、单次转换 2、连续转换 3、扫描模式 4、间断模式 设置不同的序列后将他们组成一个小组,然后进行小组的AD采集
二、寄存器
1、ADC_CR1
2、ADC_CR2
3、ADC_SMPR1和ADC_SMPR2
可以独立设置每一个通道的采用时间 ADC转换时间
4、ADC_SQR1、ADC_SQR2、ADC_SQR3和ADC_JSQR
5、ADC_JDRx和ADC_DR
6、ADC_SR
三、库函数配置
1、void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);
不同参数的有效输入可以查看固件库函数说明
2、void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);
3、void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState)
使用软件转换函数
4、ADC_RegularChannelConfig
规则通道配置函数
5、uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx)
获取ADC转换结果的值
四、应用
目标:实现4个AD采集并且显示
1、初始化
2、读取转换值
3、多次采集优化结果
4、主函数
5、代码
(1)主函数
#include "led.h"
#include "delay.h"
#include "sys.h"
#include "usart.h"
#include "lcd.h"
#include "adc.h"
int main(void)
{
u16 adcx;
float temp;
u16 adcx2;
float temp2;
u16 adcx3;
float temp3;
u16 adcx4;
float temp4;
delay_init();
uart_init(9600);
LED_Init();
LCD_Init();
Adc_Init();
POINT_COLOR=RED;
LCD_ShowString(60,50,200,16,16,"Mini STM32");
LCD_ShowString(60,70,200,16,16,"ADC TEST");
LCD_ShowString(60,90,200,16,16,"lixingye");
LCD_ShowString(60,110,200,16,16,"2022/04/28");
POINT_COLOR=BLUE;
LCD_ShowString(60,130,200,16,16,"ADC_CH1_VAL:");
LCD_ShowString(60,150,200,16,16,"ADC_CH1_VOL:0.000V");
LCD_ShowString(60,170,200,16,16,"ADC_CH2_VAL:");
LCD_ShowString(60,190,200,16,16,"ADC_CH2_VOL:0.000V");
LCD_ShowString(60,210,200,16,16,"ADC_CH3_VAL:");
LCD_ShowString(60,230,200,16,16,"ADC_CH3_VOL:0.000V");
LCD_ShowString(60,250,200,16,16,"ADC_CH4_VAL:");
LCD_ShowString(60,270,200,16,16,"ADC_CH4_VOL:0.000V");
while(1)
{
adcx=Get_Adc_Average(ADC_Channel_1,10);
LCD_ShowxNum(156,130,adcx,4,16,0);
temp=(float)adcx*(3.3/4096);
adcx=temp;
LCD_ShowxNum(156,150,adcx,1,16,0);
temp-=adcx;
temp*=1000;
LCD_ShowxNum(172,150,temp,3,16,0X80);
LED0=!LED0;
delay_ms(250);
adcx2=Get_Adc_Average(ADC_Channel_2,10);
LCD_ShowxNum(156,170,adcx2,4,16,0);
temp2=(float)adcx2*(3.3/4096);
adcx2=temp2;
LCD_ShowxNum(156,190,adcx2,1,16,0);
temp2-=adcx2;
temp2*=1000;
LCD_ShowxNum(172,190,temp2,3,16,0X80);
LED0=!LED0;
delay_ms(250);
adcx3=Get_Adc_Average(ADC_Channel_3,10);
LCD_ShowxNum(156,210,adcx3,4,16,0);
temp3=(float)adcx3*(3.3/4096);
adcx3=temp3;
LCD_ShowxNum(156,230,adcx3,1,16,0);
temp3-=adcx3;
temp3*=1000;
LCD_ShowxNum(172,230,temp3,3,16,0X80);
LED0=!LED0;
delay_ms(250);
adcx4=Get_Adc_Average(ADC_Channel_4,10);
LCD_ShowxNum(156,250,adcx4,4,16,0);
temp4=(float)adcx4*(3.3/4096);
adcx4=temp4;
LCD_ShowxNum(156,270,adcx4,1,16,0);
temp4-=adcx4;
temp4*=1000;
LCD_ShowxNum(172,270,temp4,3,16,0X80);
LED0=!LED0;
delay_ms(250);
}
}
(2)adc.c
#include "adc.h"
#include "delay.h"
void Adc_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1 , ENABLE );
RCC_ADCCLKConfig(RCC_PCLK2_Div6);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOA, &GPIO_InitStructure);
ADC_DeInit(ADC1);
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_InitStructure.ADC_NbrOfChannel = 2;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_InitStructure.ADC_NbrOfChannel = 3;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_InitStructure.ADC_NbrOfChannel = 4;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
}
u16 Get_Adc(u8 ch)
{
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));
return ADC_GetConversionValue(ADC1);
}
u16 Get_Adc_Average(u8 ch,u8 times)
{
u32 temp_val=0;
u8 t;
for(t=0;t<times;t++)
{
temp_val+=Get_Adc(ch);
delay_ms(5);
}
return temp_val/times;
}
(3)adc.h
#ifndef __ADC_H
#define __ADC_H
#include "sys.h"
void Adc_Init(void);
u16 Get_Adc(u8 ch);
u16 Get_Adc_Average(u8 ch,u8 times);
#endif
五、总结
这里使用的一个是最简单的模式了,还是要继续去研究其他模式怎么样使用的!
|