首页 > 其他分享 >【STM32】ADC模拟数字转换-规则组单通道

【STM32】ADC模拟数字转换-规则组单通道

时间:2024-08-10 23:27:21浏览次数:18  
标签:转换 STM32 InitStructure ADC ADC1 GPIO 通道 单通道

本篇博客重点在于标准库函数的理解与使用,搭建一个框架便于快速开发 

目录

 ADC简介

ADC时钟配置

引脚模拟输入模式

规则组通道选择

ADC初始化 

工作模式

数据对齐

 触发转换方式

连续与单次转换模式

扫描模式

组内的通道个数

ADC初始化框架

ADC上电

ADC校验

 获取转换数据

​​​​​​​​​​​​​​ ADC规则组单通道框架

 AD.h

 AD.c


 ADC简介

  • ADC(Analog-Digital Converter)模拟-数字转换器
  • ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁
  • 12位逐次逼近型ADC,1us转换时间
  • 输入电压范围:0~3.3V,
  • 转换结果范围:0~4095 ,18个输入通道,可测量16个外部和2个内部信号源
  • 规则组和注入组两个转换单元
  • 模拟看门狗自动监测输入电压范围

STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道

ADC时钟配置

就是配置上图的RCC时钟部分,为ADC提供时钟(CLOCKk)信号

已知ADC都在APB2总线(如图)

再由RCC时钟树,需要使能APB2外设的ADC时钟。ADCCLK最大为14MHz,可选6分频或8分频,分别提供最大12MHz和9MHz的时钟

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);//ADC1外设时钟使能
RCC_ADCCLKConfig(RCC_PCLK2_Div6);//选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz

引脚模拟输入模式

 内部通道

温度传感器和通道ADC1_IN16相连接

内部参照电压VREFINT和ADC1_IN17相连接

注意: 温度传感器和VREFINT只能出现在主ADC1中

外部通道

选择模拟输入模式,外部的模拟信号直接作为ADCx的INy(y=0~15)

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

GPIO初始化

STM32F103C8T6的PA0默认复用为ADC1_IN0

GPIO的其它参数的理解可以阅读下方博客,这里不再赘述。

【STM32】GPIO和AFIO标准库使用框架-CSDN博客

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;  //ADC1的通道0
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

规则组通道选择

在任意多个通道上以任意顺序进行的一系列转换构成成组转换。

先学习规则组通道转换,1个组里可以有0~16个通道的任意排列组合

例如,可以如下顺序完成转换

构成转换的规则组

转换的序号通道号(ADCxINy)
1通道3
2通道8

3

通道2
4通道 2
5通道0
6通道2
7通道2
8通道15

 每一个规则组里的通道可以单独配置组里的序号,所以单独配置几个序号就调用几个通道配置函数

参数解释

ADCxx 可以是 选择哪一个ADC 外设
ADC_Channel被设置的 ADC 通道
Rank规则组采样顺序
ADC_SampleTime指定 ADC 通道的采样时间值

ADC转换需要时间,在这里可以设置每个序号对应的通道以不同的时间采样

 

//配置放在规则组序列1里的通道ADC1_IN0及采样时间55.5个ADC采样周期
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);

ADC初始化 

工作模式

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;

数据对齐

转换结果为12位,转换完成后存放在16位数据寄存器中,需要选择数据左对齐还是右对齐

只有一个数据寄存器,转换后的结果会被覆盖,要及时取走

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐

 触发转换方式

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;

连续与单次转换模式

连续转换模式

 连续模式使能条件下, 连续模式开启,只需要触发一次,就可不停的转换,这时,不需要查看转换完成标志位,直接读取数据寄存器即可得到数据。

单次转换模式

执行一次转换后ADC停止,需要重新发出触发信号,再次启动转换。

ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//单次转换模式

来自参考手册:

 ● 规则组由多达16个转换组成。规则通道和它们的转换顺序在ADC_SQRx寄存器中选择。规 则组中转换的总数应写入ADC_SQR1寄存器的L[3:0]位中。

● 注入组由多达4个转换组成。注入通道和它们的转换顺序在ADC_JSQR寄存器中选择。注入 组里的转换总数目应写入ADC_JSQR寄存器的L[1:0]位中。

如果ADC_SQRx或ADC_JSQR寄存器在转换期间被更改,当前的转换被清除,一个新的启动脉冲将发送到ADC以转换新选择的组。

规则组序列一有1个通道,并且配置为连续转换模式时,不建议更改序列1位置的通道号。不建议这时通过更改序列1的通道号来实现多个ADCx通道转换

扫描模式

非扫描模式只转换组内的序列一位置的通道,扫描模式会从序列一的通道依次转换到指定的序列号的通道,组内最后一个转换完成置EOC标志位

ADC_InitStructure.ADC_ScanConvMode = DISABLE;

组内的通道个数

非扫描模式时,通道数为1,仅在扫描模式下,才需要指定大于1的数

ADC_InitStructure.ADC_NbrOfChannel = 1;

ADC初始化框架

	ADC_InitTypeDef ADC_InitStructure;
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;
	ADC_InitStructure.ADC_NbrOfChannel = 1;
    ADC_Init(ADC1, &ADC_InitStructure);

单个ADC框图 

 

ADC上电

ADC_Cmd(ADC1, ENABLE);//使能ADC1,ADC开始运行

ADC校验

ADC有一个内置自校准模式。校准可大幅减小因内部电容器组的变化而造成的准精度误差。在 校准期间,在每个电容器上都会计算出一个误差修正码(数字值),这个码用于消除在随后的转换 中每个电容器上产生的误差。 通过设置ADC_CR2寄存器的CAL位启动校准。一旦校准结束,CAL位被硬件复位,可以开始正 常转换。建议在上电时执行一次ADC校准。校准阶段结束后,校准码储存在ADC_DR中。

注意:

1 建议在每次上电后执行一次校准。

2 启动校准前,ADC必须处于关电状态(ADON=’0’)超过至少两个ADC时钟周期

直接调用即可 

	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
    while (ADC_GetCalibrationStatus(ADC1) == SET);

 获取转换数据

uint16_t AD_GetValue(uint8_t ADC_Channel)
{
	ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5);//在每次转换前,根据函数形参灵活更改规则组的通道1
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);//软件触发AD转换一次
	while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);//等待EOC标志位,即等待AD转换结束
	return ADC_GetConversionValue(ADC1);//读数据寄存器,得到AD转换的结果
}

​​​​​​​​​​​​​​ ADC规则组单通道框架

 AD.h

#ifndef __AD_H
#define __AD_H

void AD_Init(void);
uint16_t AD_GetValue(uint8_t ADC_Channel);

#endif

 AD.c

#include "stm32f10x.h"                  // Device header

void AD_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6); //选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);

	/*
      不在此处配置规则组序列,而是在每次AD转换前配置,这样可以灵活更改AD转换的通道
      如果要设置序列1为固定的通道在这里调用一次就可,AD_GetValue()中的通道配置应删除	
    */

    //ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);

	ADC_InitTypeDef ADC_InitStructure;
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//模式,选择独立模式,即单独使用ADC1
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//数据右对齐
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//不用硬件触发,软件触发
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//连续转换,失能,每转换一次规则组序列后停止
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;扫描模式,失能,只转换规则组的序列1这一个位置
	ADC_InitStructure.ADC_NbrOfChannel = 1; //通道数,为1,仅在扫描模式下,才需要指定大于1的数,在非扫描模式下,只能是1
	ADC_Init(ADC1, &ADC_InitStructure);
	
	ADC_Cmd(ADC1, ENABLE);//使能ADC1,ADC开始运行
	
	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1) == SET);
}


/**
  * 函    数:获取AD转换的值
  * 参    数:ADC_Channel 指定AD转换的通道,范围:ADC_Channel_x,其中x可以是0/1/2
  * 返 回 值:AD转换的值,范围:0~4095
  */
uint16_t AD_GetValue(uint8_t ADC_Channel)
{
	ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5);//在每次转换前,根据函数形参灵活更改规则组的通道1
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);//软件触发AD转换一次
	while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);//等待EOC标志位,即等待AD转换结束
	return ADC_GetConversionValue(ADC1);//读数据寄存器,得到AD转换的结果
}

标签:转换,STM32,InitStructure,ADC,ADC1,GPIO,通道,单通道
From: https://blog.csdn.net/m0_75090944/article/details/141063594

相关文章

  • 从零开始制作一个简易数字示波器,超详细看完必懂 举例基于stm32f103标准库
    为了实现使用STM32f103制作一个简易数值示波器,大体流程如下:1-模拟信号(输入)2-ADC(模数转换)(适当采样率和分辨率配置)3-DMA(直接存储器访问)(DMA搬运,减少CPU负担)4-内存缓冲区(数据转入缓冲区)5-FFT处理(频域数据)(FFT处理(可选,用于频域显示))6-显示处理(像素转换)(数据映射为......
  • 1410、proteus仿真-STM32单片机血压体温血氧心率检测阈值报警加远程监控设计(程序+pro
    毕设帮助、开题指导、技术解答(有偿)见文未 目录方案选择单片机的选择一、设计功能二、proteus仿真图三、程序源码资料包括:需要完整的资料可以点击下面的名片加下我,找我要资源压缩包的百度网盘下载地址及提取码。方案选择单片机的选择方案一:AT89C51是美国ATMEL公......
  • STM32通过ESP8266连接机智云APP(第二讲,烧录固件并测试)
       今天与大家分享STM32通过ESP8266连接机智云APP,并完成数据的上传下发第二讲。带大家一起烧录机智云固件到ESP8266并且将其通过机智云串口助手与APP进行测试。     另外新创建了公众号“宝藏Code园”,本系列相关代码将在公众号里面开源。需要本项目相关工程代码......
  • STM32 虚拟串口发不出数据
    在某个项目上,使用到了USB虚拟串口功能,芯片是STM32F105,在芯片刚复位时功能是正常,但是多发几次数据后,就发不出去了,一直返回USBD_BUSY,在下图中的344行返回USBD_BUSY解决办法是在这个函数内增加一些代码,等待数据发送完成之后再退出,或者超时退出,超时时间可以自己调整,根据自己的使......
  • 基于STM32设计的智能家居系统(采用OneNet云平台)
    鱼弦:公众号【红尘灯塔】,CSDN博客专家、内容合伙人、新星导师、全栈领域优质创作者、51CTO(Top红人+专家博主)、github开源爱好者(go-zero源码二次开发、游戏后端架构https://github.com/Peakchen)基于STM32设计的智能家居系统(采用OneNet云平台)1.介绍基于STM32微控制器和......
  • 从STM32CubeMX导入项目到Embedded Studio。(原文题目:Import projects from STM32CubeM
    原文链接https://wiki.segger.com/Import_projects_from_STM32CubeMX_to_Embedded_Studio原文来自于SEGGER的wiki,题目是ImportprojectsfromSTM32CubeMXtoEmbeddedStudio原文最后编辑于2022/2/21.摘要:CubeMX生成项目,导入到EmbeddedStudio,并添加必要的文件软件:STM32C......
  • 【STM32】IO口取反 | 寄存器方式 | 异或运算符 | 原理
    目录STM32IO口取反|寄存器方式|异或运算符|原理1.引言2.GPIO基础知识2.1GPIO概述2.2STM32的GPIO架构2.3GPIO寄存器简介3.GPIO引脚取反原理3.1寄存器操作实现取反3.2异或运算符的应用4.示例代码4.1基础示例:LED闪烁4.2应用实例:继电器控制5.GPIO引脚......
  • STM32之SPI
    ADI对SPI的介绍1.硬件结构普通并联结构(使用片选信号CS或者NSS选通芯片)菊花链结构(部分芯片支持)2.通讯方式(四种)3.通信速率......
  • 七、1 ADC模数转换器介绍+有关知识点
    目录1、介绍(1)ADC,模拟信号转换为数字信号(2)DAC和PWM,数字信号转换为模拟信号(3)ADC的两个关键参数(4)(5)(6)(7)(8)2、逐次逼近型ADC(1)型号为ADC0809的普通ADC(2)STM32的ADC1)2)触发ADC开始转换的信号3)4)3、ADC通道与引脚的对应关系4、转换模式(4种)5、触发控制6、数据对齐7、转......
  • STM32CubleMX创建FreeRtos工程教程,图文教程
        前言:STM32CubeMX是一个开发工具,它已经将FreeRTOS这个实时操作系统(RTOS)集成到其工具中。换句话说,通过STM32CubeMX,可以非常方便地为STM32微控制器生成配置代码,其中包括对FreeRTOS的支持。    而本篇就是使用STM32CubleMX,生成支持FreeRtos的图文教程......