STM32中AD采样的三种方法分析
在进行STM32F中AD采样的学习中,我们知道AD采样的方法有多种,按照逻辑程序处理有三种方式,一种是查询模式,一种是中断处理模式,一种是DMA模式。三种方法按照处理复杂方法DMA模式处理模式效率最高,其次是中断处理模式,最差是查询模式,相信很多学者在学习AD采样程序时,很多例程采用DMA模式,在这里我针对三种程序进行分别分析。
1、AD采样查询模式
在AD采样查询模式中,我们需要注意的是IO口的初始化配置,这里我采用PA2作为模拟采集的引脚(AIN2)和串口3作为打印输出。
具体如下:建立一个USART3.C和USART3.H文件,其程序为:
#include "usart3.h"
#include "stdarg.h"
u8 SendBuff[SENDBUFF_SIZE];
void USART3_Config(void)
{
//定义结构体
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
//开启外部时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO,
ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE );
// USART3 GPIO config
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOAtiNG;
GPIO_Init(GPIOB, &GPIO_InitStructure);
//USART3 mode config
USART_InitStructure.USART_BaudRate = 115200;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl =
USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
USART_Init(USART3, &USART_InitStructure);
USART_Cmd(USART3, ENABLE);
}
其次建立一个ADC.C和一个ADC.H文件,其中ADC.C中程序为:
void ADC1_Init(void)
{
ADC1_GPIO_Config();
ADC1_Mode_Config();
}
static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
//开启外部时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 |
RCC_APB2Periph_GPIOA,ENABLE);
//配置PA2引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
//配置为模拟输入
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
//调用库函数
GPIO_Init(GPIOA, &GPIO_InitStructure);
}
static void ADC1_Mode_Config(void)
{
//ADC1_ configuration
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
//独立ADC模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
//禁止扫描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
//开启连续转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
//不使用外部触发转换
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //采集数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //要转换的通道数目1
ADC_Init(ADC1,&ADC_InitStructure);
//配置ADC时钟,为PCLK2的8分频,即9Mhz
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
//配置ADC1的通道2位55.5个采集周期
ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 1,
ADC_SampleTime_55Cycles5);
ADC_Cmd(ADC1,ENABLE);
//复位校准寄存器
ADC_ResetCalibration(ADC1);
//等待校准寄存器复位完成
while(ADC_GetResetCalibrationStatus(ADC1));
//ADC校准
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
//由于没有使用外部触发,所以使用软件触发ADC转换
ADC_SoftwareStartConvCmd(ADC1,ENABLE);
}
然后在主函数main中其程序代码如下:
int main(void)
{
USART3_Config();
ADC1_Init();
printf("输入ADC值");
while(1)
{
ADC_ConvertedValue = ADC_GetConversionValue(ADC1);
ADC_ConvertedValueLocal =(float)ADC_ConvertedValue/4096*3.3;
//读取ADC转换的值
printf("\r\n the current AD value = 0x%04X \r\n",ADC_ConvertedValue);
printf("\r\n the current AD value = %f V
\r\n",ADC_ConvertedValueLocal);
Delay(0xFFFFEE);
}
}
这样采用查询的方法即可以采集ADC的电压值,一个值为16进制转换的值,一个是转换计算的值。说明一下:ADC_ConvertedValue =
ADC_GetConversionValue(ADC1);
一定要放在while中,只有这样,采集的ADC电压值才是实时采集的电压值。放在while外面,则采集的电压值为第一次的电压值,且读取的电压值不会变化。对于4096的值来源在于ADC采集的数值是12位ADC,即是2的12次方。
2、中断查询ADC程序
对于中断查询采集ADC程序主要是在ADC.C和main函数中有差别。具体ADC.C程序为:
void ADC1_Init(void)
{
ADC1_GPIO_Config();
ADC1_Mode_Config();
ADC_NVIC_Config();
}
static void ADC_NVIC_Config(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
NVIC_InitStructure.NVIC_IRQChannel = ADC1_2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
//开启外部时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 |
RCC_APB2Periph_GPIOA,ENABLE);
//配置PA2引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
//配置为模拟输入
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
//调用库函数
GPIO_Init(GPIOA, &GPIO_InitStructure);
}
static void ADC1_Mode_Config(void)
{
//ADC1_ configuration
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
//独立ADC模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
//禁止扫描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
//开启连续转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
//不使用外部触发转换
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //采集数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //要转换的通道数目1
ADC_Init(ADC1,&ADC_InitStructure);
//配置ADC时钟,为PCLK2的8分频,即9Mhz
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
//配置ADC1的通道2位55.5个采集周期
ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 1,
ADC_SampleTime_55Cycles5);
ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE); //开启ADC采集中断
ADC_Cmd(ADC1,ENABLE);
//复位校准寄存器
ADC_ResetCalibration(ADC1);
//等待校准寄存器复位完成
while(ADC_GetResetCalibrationStatus(ADC1));
//ADC校准
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
//由于没有使用外部触发,所以使用软件触发ADC转换
ADC_SoftwareStartConvCmd(ADC1,ENABLE);
}
对于main函数如下:
int main(void)
{
USART3_Config();
ADC1_Init();
printf("输入ADC值");
while(1)
{
ADC_ConvertedValueLocal =(float)ADC_ConvertedValue/4096*3.3;
//读取ADC转换的值
printf("\r\n the current AD value = 0x%04X \r\n",ADC_ConvertedValue);
printf("\r\n the current AD value = %f V
\r\n",ADC_ConvertedValueLocal);
Delay(0xFFFFEE);
}
}
void ADC_IRQHandler(void)
{
IF (ADC_GetITStatus(ADC1, ADC_IT_EOC) == SET)
{
ADC_ConvertedValue = ADC_GetConversionValue(ADC1);
}
ADC_ClearITPendingBit(ADC1, ADC_IT_EOC);
}
在引入void ADC_IRQHandler(void)这个中断服务函数之前,一定要进行
#define ADC_IRQHandler ADC1_2_IRQHandler
否则中断无法执行,无法进行ADC采集。
3、DMA模式的ADC采集程序
采用这种方式的ADC采集程序,其在ADC.C程序为:
void ADC1_Init(void)
{
ADC1_GPIO_Config();
ADC1_Mode_Config();
}
static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
//开启外部时钟
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 |
RCC_APB2Periph_GPIOA,ENABLE);
//配置PA2引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
//配置为模拟输入
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
//调用库函数
GPIO_Init(GPIOA, &GPIO_InitStructure);
}
static void ADC1_Mode_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = 1;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;
DMA_InitStructure.DMA_PeripheralDataSize= DMA_PeripheralDataSize_HalfWord;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1,&DMA_InitStructure);
DMA_Cmd (DMA1_Channel1,ENABLE);
//ADC1_ configuration
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
//独立ADC模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
//禁止扫描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
//开启连续转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
//不使用外部触发转换
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //采集数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //要转换的通道数目1
ADC_Init(ADC1,&ADC_InitStructure);
//配置ADC时钟,为PCLK2的8分频,即9Mhz
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
//配置ADC1的通道2位55.5个采集周期
ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 1,
ADC_SampleTime_55Cycles5);
ADC_DMACmd(ADC1,ENABLE);
ADC_Cmd(ADC1,ENABLE);
//复位校准寄存器
ADC_ResetCalibration(ADC1);
//等待校准寄存器复位完成
while(ADC_GetResetCalibrationStatus(ADC1));
//ADC校准
ADC_StartCalibration(ADC1);
while(ADC_GetCalibrationStatus(ADC1));
//由于没有使用外部触发,所以使用软件触发ADC转换
ADC_SoftwareStartConvCmd(ADC1,ENABLE);
}
在这里需要对ADC1_DR_Address地址值进行定义,具体定义可以在ADC.H文件中,表现为:#define ADC1_DR_Address
((u32)0x40012400+0x4c)
在main中函数为:
int main(void)
{
USART3_Config();
ADC1_Init();
printf("输入ADC值");
while(1)
{
ADC_ConvertedValueLocal =(float)ADC_ConvertedValue/4096*3.3;
//读取ADC转换的值
printf("\r\n the current AD value = 0x%04X \r\n",ADC_ConvertedValue);
printf("\r\n the current AD value = %f V
\r\n",ADC_ConvertedValueLocal);
Delay(0xFFFFEE);
}
}
通过实际测试,三种程序处理方式得到的结果都是一样,这表明三种方式是可行的。不过后续在具体功能程序设计时,建议采用中断查询或者DMA模式。
AD的资料暂时没有我就给搞一些pcb以及DMA和中断的资料供大家在学习过程中参考吧
老司机倾囊相授-PCB大牛修炼秘籍
http://www.makeru.com.cn/live/3472_1296.html?s=45051
PADS-PCB原图绘制
http://www.makeru.com.cn/live/4006_1430.html?s=45051
(DMA专题讲解)
http://www.makeru.com.cn/live/1392_1048.html?s=45051
stm32 如何用DMA搬运数据
http://www.makeru.com.cn/live/detail/1484.html?s=45051
(STM32中断系统)
http://www.makeru.com.cn/live/1392_1124.html?s=45051
STM32中AD采样的三种方法分析的更多相关文章
- Android中全屏 取消标题栏,TabHost中设置NoTitleBar的三种方法(转)
Android中全屏 取消标题栏,TabHost中设置NoTitleBar的三种方法http://www.cnblogs.com/zdz8207/archive/2013/02/27/android- ...
- Openerp 中打开 URL 的三种 方法
来自:http://shine-it.net/index.php/topic,8013.0.html 最近总结了,Openerp 中打开 URL 的三种 方法: 一.在form view 添加 < ...
- mysql 中添加索引的三种方法
原文:http://www.andyqian.com/2016/04/06/database/mysqleindex/ 在mysql中有多种索引,有普通索引,全文索引,唯一索引,多列索引,小伙伴们可以 ...
- jQuery中detach&&remove&&empty三种方法的区别
jQuery中empty&&remove&&detach三种方法的区别 empty():移除指定元素内部的所有内容,但不包括它本身 remove():移除指定元素内部的 ...
- LwIP协议栈开发嵌入式网络的三种方法分析
LwIP协议栈开发嵌入式网络的三种方法分析 摘要 轻量级的TCP/IP协议栈LwIP,提供了三种应用程序设计方法,且很容易被移植到多任务的操作系统中.本文结合μC/OS-II这一实时操作系统,以 ...
- mfc 在VC的两个对话框类中传递参数的三种方法
弄了好久,今天终于把在VC中的对话框类之间传递参数的问题解决了,很开心,记录如下: 1. 我所建立的工程是一个基于MFC对话框的应用程序,一共有三个对话框,第一个对话框为主对话框,所对应的类为CTMD ...
- cocos2dx中创建动画的三种方法
1.最最原始的方法,先创建动画帧,再创建动画打包(animation),再创建动画(animate) 第一步: 创建动画帧:CCSpriteFrame,依赖于原始的资源图片(xx.png,xx.jpg ...
- vue后台管理项目中菜单栏切换的三种方法
第一种方法:vue嵌套路由(二) <el-menu :default-active="defaultActive" style="min-height: 100%; ...
- URL转Drawable之 Android中获取网络图片的三种方法
转载自: http://doinone.iteye.com/blog/1074283 Android中获取网络图片是一件耗时的操作,如果直接获取有可能会出现应用程序无响应(ANR:Applicatio ...
随机推荐
- JAVA反序列化漏洞基础原理
JAVA反序列化漏洞基础原理 1.1 什么是序列化和反序列化? Java序列化是指把Java对象转换为字节序列的过程: Java反序列化是指把字节序列恢复为Java对象的过程: 1.2 为什么要序列化 ...
- MySQL查询之内连接,外连接查询场景的区别与不同
前言 我在写sql查询的时候,用的最多的就是where条件查询,这种查询也叫内连查询inner join,当然还有外连查询outer join,左外连接,右外连接查询,常用在多对多关系中,那他们区别和 ...
- Python中with...as...的用法详解
简介 with是从Python2.5引入的一个新的语法,它是一种上下文管理协议,目的在于从流程图中把 try,except 和finally 关键字和资源分配释放相关代码统统去掉,简化try-.exc ...
- Jmeter系列(23)- 常用逻辑控制器(2) | 事务控制器Transaction Controller
事务控制器(Transaction Controller) 作用 选择一些请求,作为事务,放在该控制器下 比如:我有三个请求,注册.登录.下单.这三个请求其实就是一个下单完成过程,可以作为一个下单事务 ...
- 搞不定 NodeJS 内存泄漏?先从了解垃圾回收开始
通常来说,内存管理有两种方式,一种是手动管理,一种是自动管理. 手动管理需要开发者自己管理内存,什么时候申请内存空间,什么时候释放都需要小心处理,否则容易形成内存泄漏和指针乱飞的局面.C 语言开发是典 ...
- 测试验收标准checklist
需求实现功能清单 功能实现目的 需求改造功能清单 关联功能清单 关联系统 端到端全流程场景 业务联系性场景 业务全流程场景 上下需求关联规则 业务角度在流程中关注项 财报.评级 授信方案 反洗钱 面向 ...
- P6775-[NOI2020]制作菜品【贪心,dp】
正题 题目链接:https://www.luogu.com.cn/problem/P6775 题目大意 \(n\)种原材料,第\(i\)个有\(d_i\)个,\(m\)道菜品都需要\(k\)个原料而且 ...
- P5437-[XR-2]约定【拉格朗日差值,数学期望】
正题 题目链接:https://www.luogu.com.cn/problem/P5437 题目大意 \(n\)个点的完全图,连接\(i,j\)的边权值为\((i+j)^k\).随机选出一个生成树, ...
- Spring系列之Redis的两种集成方式
在工作中,我们用到分布式缓存的时候,第一选择就是Redis,今天介绍一下SpringBoot如何集成Redis的,分别使用Jedis和Spring-data-redis两种方式. 一.使用Jedis方 ...
- WPF实现聚光灯效果
WPF开发者QQ群: 340500857 | 微信群 -> 进入公众号主页 加入组织 前言 效果仿照 CSS聚光灯效果 实现思路: 1. 设置底部Canvas背景色 #222222 . 2. ...