STM32(七):ADC电位检测 (标准库函数)

前言

上一篇文章已经介绍了如何用STM32单片机中的定时器的PWM波来实现LED的“呼吸”。这篇文章我们来介绍一下如何用STM32单片机中ADC进行电位检测,并发送到XCOM串口中显示。

一、实验原理

1.ADC模数转换的介绍

首先,我们先介绍一下AD模数模块(Analog-to-Digital Converter)。AD模数模块即模拟/数字转换器,主要功能是将连续变化的模拟信号转换为离散的数字信号。

由于单片机只能处理数字信号,因此,在对外部的模拟信号进行分析、处理的过程中,必须使用ADC模块将外部的模拟信号转换成单片机所能处理的数字信号。

A/D转换器的主要类型有以下几种:

  • 积分型
  • 逐次比较型
  • 并行比较型
  • \Sigma -\bigtriangleup调制型
  • 电容阵列逐次比较型
  • 压频变换型
A/D转换器的主要技术指标主要有以下几种:
  • 转换范围 :A/D转换器能够转换的模拟电压范围。
  • 分辨率:输出二进制数的位数表示,位数越多,分辨率越高。
  • 绝对精度:对应一个给定数字量的理论模拟输入与实际输入之差。
  • 转换速度:A/D转换器完成一次转换所需的时间。转 换时间是指从接到转换控制信号开始,到输出端得到稳定的 数字输出信号所经过的这段时间。

典型ADC结构性能比较如下:

2.STM32中的ADC

STM32拥有1~3个ADC(STM32F101/102系列只有1个ADC,STM32F103系列最少都拥有2个ADC ),这些ADC可以独立使用,也可以使用双重模式(提高采样率)。

STM32的ADC是12位逐次逼近型的模拟数字转换器。它有18个通道,可测量16个外部和2个内部信号源(温度传感器、内部参考电压)。ADC的输入时钟不得超过14MHz,其时钟频率由PCLK2分频产生。

ADC 特点
  • STM32单片机有2个独立的ADC控制器,有18个通道,可测量16个外部信号和2个内部信号源:内部温度传感器和内部参考电压(Bandgap voltage) 。
  • ADC 供电要求: 2.4V to 3.6 V 。
  • ADC 输入范围: VREF- ≤ VIN ≤ VREF+ (VREF+ and VREF- available only in LQFP100 package)
  • 精度:12位。结果可按左对齐或右对齐的方式存放在16位寄存器中。
  • A/D转换的过程:采用、保持、量化、编程。采样时间越长,转换结果越稳定。采样时间可设置为:1.5个/7.5个/13.5个/28.5个ADC时钟周期。
  • ADC转换时间 : 采用时间+转换时间(12.5个时钟周期)
  • DMA 功能 (ADC1)

要想搞清楚ADC的原理,那么首先还是先要学会看图,原理图如下图所示:

1.电压输入范围

ADC所能测量的电压范围就是VREF- ≤ VIN ≤ VREF+,把 VSSA 和 VREF-接地,把 VDDA和VREF+接 3.3V,得到ADC的输入电压范围为: 0~3.3V。

2.输入通道

STM32 16 个外部 ADC 通道可分为两组 : 规则的和注入的。
  • 每个组可以是这16个通道中的任意一些通道以任意顺序进行的组合。
  • 规则组最多有16个通道,通道和转换顺序在ADC规则系列寄存器x(ADC_JSQR)中选择。
  • 注入组最多有4个通道。通道和转换顺序在ADC注入系列寄存器(ADC_JSQR)中选择。

各通道的A/D 转换可以单次、连续、扫描或间断模 式执行, stm32f103zet6通道如下:

3.转换顺序

由于规则转换通道只有一个数据寄存器,使用多个通道进行转换需要考虑转换顺序。多个通道的使用顺序分为俩种情况:规则通道的转换顺序和注入通道的转换顺序。

可以看下参考博客,这边就不多加赘述。

4.触发源

触发有两种方式:配置寄存器和通过内部定时器或者外部IO触发转换。

  • 配置寄存器触发,通过配置控制寄存器CR2的ADON位,写1时开始转换,写0时停止转换。在程序运行过程中只要调用库函数,将CR2寄存器的ADON位置1就可以进行转换。
  • 通过内部定时器或者外部IO触发转换,也就是说可以利用内部时钟让ADC进行周期性的转换,也可以利用外部IO使ADC在需要时转换,具体的触发由控制寄存器CR2决定。

5.转换时间

转换时间由输入时钟和采样周期来决定。每个通道总的转换时间=TSampling+Tconversion Tsampling(采样时间)

转换时间=采样时间+12.5个周期

STM32的ADC最大的转换速率为1Mhz,也就是转换时间为1us(在ADCCLK=14M,采样周期为1.5个ADC时钟下得到)
不要让 ADC 的时钟超过 14M ,否则将导致结果准确度下降。

6.数据寄存器 

数据转换完成后的存放在2种数据寄存器中,分别是注入通道数据寄存器和规则通道数据寄存器。

注入通道数据寄存器

数据寄存器有4个,由于注入通道最多有4个,所以注入通道转换的数据都有固定的存放位置,不会跟规则寄存器那样产生数据覆盖的问题。

规则通道数据寄存器

负责存放规则通道转换的数据,通过32位寄存器ADC_DR来存放。

7.中断

规则和注入组转换结束时能产生中断,当模拟看门狗状态位被设置时也能产生中断。它们都有 独立的中断使能位。
数据转换完成之后可以产生中断,有三种情况,如下图所示:

二、实验步骤

1.配置NVIC

主要是配置ADC中断优先级,这边的对于优先级配置不懂的可以看之前的文章

STM32(三):外部中断 (标准库函数)_stm32外部中断的库函数程序怎么写-CSDN博客

static void ADC_NVIC_Config(void)
{
  NVIC_InitTypeDef NVIC_InitStructure;
   /* 配置中断优先级 */  
  NVIC_InitStructure.NVIC_IRQChannel = ADCx_IRQ;
	 /* 设置抢占式优先级为0 */
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
	 /* 设置子优先级为0 */
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
	 /* 使能外部中断通道 */
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	 /* 初始化配置嵌套向量中断控制器 */
  NVIC_Init(&NVIC_InitStructure);
}

2.GPIO 初始化

主要是对ADC IO口引脚的配置,同时还要使能端口时钟

static void ADCx_GPIO_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	/* 打开 ADC IO端口时钟 */
	ADC_APBxClock_FUN(ADC_GPIO_CLK, ENABLE );	
	/* 配置 ADC IO 引脚模式 */
	GPIO_InitStructure.GPIO_Pin = ADC_PIN;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;	
	/* 初始化 ADC IO */
	GPIO_Init(ADC_PORT, &GPIO_InitStructure);				
}

3.配置工作模式

static void ADCx_Mode_Config(void)
{
	ADC_InitTypeDef ADC_InitStructure;	
	/* 打开ADC时钟 */
	ADC_APBxClock_FUN(ADC_CLK,ENABLE );	
	/* ADC 模式配置 */
	/* 只使用一个ADC,属于单模式 */
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;	
	/* 禁止扫描模式,多通道才要,单通道不需要 */
	ADC_InitStructure.ADC_ScanConvMode = DISABLE ; 
	/* 连续转换模式 */
	ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
	/* 不用外部触发转换,软件开启即可 */
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	/* 转换结果右对齐 */
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;	
	/* 转换通道1个 */
	ADC_InitStructure.ADC_NbrOfChannel = 1;			
	/* 初始化ADC */
	ADC_Init(ADCx, &ADC_InitStructure);
	
	/* 配置ADC时钟为PCLK2的8分频,即9MHz */
	RCC_ADCCLKConfig(RCC_PCLK2_Div8);
	
	/* 配置 ADC 通道转换顺序为1,第一个转换,采样时间为55.5个时钟周期 */
	ADC_RegularChannelConfig(ADCx, ADC_CHANNEL, 1, ADC_SampleTime_55Cycles5);
	
	/* ADC 转换结束产生中断,在中断服务程序中读取转换值 */
	ADC_ITConfig(ADCx, ADC_IT_EOC, ENABLE);
	
	/* 开启ADC ,并开始转换 */
	ADC_Cmd(ADCx, ENABLE);
	
	/* 初始化ADC 校准寄存器   */
	ADC_ResetCalibration(ADCx);
	/*等待校准寄存器初始化完成 */
	while(ADC_GetResetCalibrationStatus(ADCx));	
	/* ADC开始校准*/
	ADC_StartCalibration(ADCx);
	/*等待校准完成 */
	while(ADC_GetCalibrationStatus(ADCx));
	
	/* 由于没有采用外部触发,所以使用软件触发ADC转换  */
	ADC_SoftwareStartConvCmd(ADCx, ENABLE);
}

记得配置ADC初始化函数,将上述配置放在一个函数ADCx_Init()

void ADCx_Init(void)
{
	ADCx_GPIO_Config();
  ADC_NVIC_Config();
	ADCx_Mode_Config();	
}

4.中断函数

但到这里我们的中断配置还没结束!!重点!!易踩坑!!

我们还需要在stm32f10x_it.c的文件里面加上新的中断服务函数 ADC_IRQHandler_FUN(),记得加上头文件,例如:

#include "bsp/adc/bsp_adc.h"

__IO uint16_t ADC_ConvertedValue;

void ADC_IRQHandler_FUN(void)
{
  if(ADC_GetITStatus(ADCx,ADC_IT_EOC)!=RESET)
  {
    ADC_ClearITPendingBit(ADCx,ADC_IT_EOC);
    /* 读取ADC的转换值 */
    ADC_ConvertedValue = ADC_GetConversionValue(ADCx);
    
  }
}

5.串口发送函数

主要是讲电压数据发送到XCOM串口助手中,运用到printf()函数,这边可以看发过的这篇文章

STM32(九):USART串口通信 (标准库函数)-CSDN博客

三、实操代码

程序分为3个文件:bsp_adc.c、bsp_adc.h、main.c

1.bsp_adc.c 

/* 包含头文件 ----------------------------------------------------------------*/
#include "bsp/adc/bsp_adc.h"

/**
  * 函数功能: ADC GPIO 初始化
  */
static void ADCx_GPIO_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	/* 打开 ADC IO端口时钟 */
	ADC_APBxClock_FUN(ADC_GPIO_CLK, ENABLE );	
	/* 配置 ADC IO 引脚模式 */
	GPIO_InitStructure.GPIO_Pin = ADC_PIN;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;	
	/* 初始化 ADC IO */
	GPIO_Init(ADC_PORT, &GPIO_InitStructure);				
}

/**
  * 函数功能: 配置ADC工作模式
  */
static void ADCx_Mode_Config(void)
{
	ADC_InitTypeDef ADC_InitStructure;	
	/* 打开ADC时钟 */
	ADC_APBxClock_FUN(ADC_CLK,ENABLE );	
	/* ADC 模式配置 */
	/* 只使用一个ADC,属于单模式 */
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;	
	/* 禁止扫描模式,多通道才要,单通道不需要 */
	ADC_InitStructure.ADC_ScanConvMode = DISABLE ; 
	/* 连续转换模式 */
	ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
	/* 不用外部触发转换,软件开启即可 */
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	/* 转换结果右对齐 */
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;	
	/* 转换通道1个 */
	ADC_InitStructure.ADC_NbrOfChannel = 1;			
	/* 初始化ADC */
	ADC_Init(ADCx, &ADC_InitStructure);
	
	/* 配置ADC时钟为PCLK2的8分频,即9MHz */
	RCC_ADCCLKConfig(RCC_PCLK2_Div8);
	
	/* 配置 ADC 通道转换顺序为1,第一个转换,采样时间为55.5个时钟周期 */
	ADC_RegularChannelConfig(ADCx, ADC_CHANNEL, 1, ADC_SampleTime_55Cycles5);
	
	/* ADC 转换结束产生中断,在中断服务程序中读取转换值 */
	ADC_ITConfig(ADCx, ADC_IT_EOC, ENABLE);
	
	/* 开启ADC ,并开始转换 */
	ADC_Cmd(ADCx, ENABLE);
	
	/* 初始化ADC 校准寄存器   */
	ADC_ResetCalibration(ADCx);
	/*等待校准寄存器初始化完成 */
	while(ADC_GetResetCalibrationStatus(ADCx));	
	/* ADC开始校准*/
	ADC_StartCalibration(ADCx);
	/*等待校准完成 */
	while(ADC_GetCalibrationStatus(ADCx));
	
	/* 由于没有采用外部触发,所以使用软件触发ADC转换  */
	ADC_SoftwareStartConvCmd(ADCx, ENABLE);
}

/**
  * 函数功能: NVIC配置:ADC中断优先级配置
  */
static void ADC_NVIC_Config(void)
{
  NVIC_InitTypeDef NVIC_InitStructure;
   /* 配置中断优先级 */  
  NVIC_InitStructure.NVIC_IRQChannel = ADCx_IRQ;
	 /* 设置抢占式优先级为0 */
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
	 /* 设置子优先级为0 */
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
	 /* 使能外部中断通道 */
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	 /* 初始化配置嵌套向量中断控制器 */
  NVIC_Init(&NVIC_InitStructure);
}

/**
  * 函数功能: ADC初始化
  */
void ADCx_Init(void)
{
	ADCx_GPIO_Config();
  ADC_NVIC_Config();
	ADCx_Mode_Config();	
}

2.bsp_adc.h

#ifndef __ADC_H__
#define	__ADC_H__

/* 包含头文件 ----------------------------------------------------------------*/
#include <stm32f10x.h>


// 注意:用作ADC采集的IO必须没有复用,否则采集电压会有影响

/********************ADC输入通道(引脚)配置**************************/
#define    ADC_APBxClock_FUN             RCC_APB2PeriphClockCmd

#define    ADCx                          ADC3
#define    ADC_CLK                       RCC_APB2Periph_ADC3
#define    ADCx_IRQ                      ADC3_IRQn
#define    ADC_IRQHandler_FUN            ADC3_IRQHandler

#define    ADC_GPIO_CLK                  RCC_APB2Periph_GPIOC  
#define    ADC_PORT                      GPIOC

#define    ADC_PIN                       GPIO_Pin_0        // 连接至板载精密可调电阻(需加跳帽)
#define    ADC_CHANNEL                   ADC_Channel_10    // 连接至板载精密可调电阻(需加跳帽)
//#define    ADC_PIN                       GPIO_Pin_1        // 连接至板载光敏电阻(需加跳帽)
//#define    ADC_CHANNEL                   ADC_Channel_11    // 连接至板载光敏电阻(需加跳帽)
                                      
/* 扩展变量 ------------------------------------------------------------------*/
/* 函数声明 ------------------------------------------------------------------*/
void ADCx_Init(void);

#endif /* __ADC_H__ */


3.main.c

/* 包含头文件 ----------------------------------------------------------------*/
#include "stm32f10x.h"

#include "stm32f10x.h"
#include "bsp/led/bsp_led.h"
#include "bsp/key/bsp_key.h"
#include "bsp/delay/delay.h"
#include "bsp/systick/bsp_SysTick.h"
#include "bsp/GeneralTIM/bsp_GeneralTIM.h" 
#include "bsp/adc/bsp_adc.h"


/* 用于保存转换计算后的电压值 */	 
float ADC_ConvertedValueLocal;   

/* 扩展变量 ------------------------------------------------------------------*/
extern __IO uint16_t ADC_ConvertedValue;
/* 私有函数原形 --------------------------------------------------------------*/
static void Delay(uint32_t time);


/**
  * 函数功能: 主函数.
  */
int main(void)
{  
  /* 调试串口初始化配置,115200-N-8-1.使能串口发送和接受 */
  USARTx_Init(); 
  
  /* ADC 初始化 */
	ADCx_Init();
  
  /* 调用格式化输出函数打印输出数据 */
  printf("----这是一个ADC单通道电压采集实验-----\n"); 
  
  /* 无限循环 */
  while (1)
  { 
    ADC_ConvertedValueLocal =(float)ADC_ConvertedValue*3.3/4096; 
	
		printf("AD转换原始值 = 0x%04X \r\n", ADC_ConvertedValue); 
		printf("计算得出电压值 = %f V \r\n",ADC_ConvertedValueLocal); 
    
    Delay(1000);
  }
}

四、实验效果

ADC电压检测

参考博客:

STM32—ADC详解_stm32最好可设置几个模拟通道-CSDN博客

结束语

本文以STM32VET6为例讲解了如何用STM32单片机中ADC进行电位检测,并发送到XCOM串口中显示,并指出其中的易坑点。希望对大家有所帮助!如果还有什么问题,欢迎评论区留言,谢谢!

  • 7
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值