STM32同时测量4路模拟电压,用DMA自动搬运到数组内,不用软件干预。

2024-03-21 05:52

本文主要是介绍STM32同时测量4路模拟电压,用DMA自动搬运到数组内,不用软件干预。,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

下午我做的实验室同时测量4路模拟信号,测量的结果通过ADC开启DMA触发,叫数据转运小帮手自动把数据搬走,放到数组内,就省去了,各种检测啊,恢复标志位啊,等等的麻烦操作,直接去读数组的值就好了,数据已经被更新了。是不是很方便呢?好了,先看看实验的结果:

这样看起来是不是和上篇文章的结果差不多呢?

再来看看主函数中的操作吧,就很简单了:

这样看是不是就非常简单了啊,只是调用一下初始化函数,就把需要读的4路模拟电压数据放到了一个数组中了,接下来就来看看这个初始化到底都干了些什么工作吧?我先来简单的总结一下初始化顺序:代码中可能有点不一样,那是没有总结的结果。

1:/*开启时钟(ADC1,GPIOA,DMA1)*/

2:/*设置ADC时钟*/

3:/*GPIO初始化*/(模拟输入模式,A0,A1,A2,A3四个端口)

4:/*规则组通道配置*/(序列1放通道0,序列2放通道1,序列3放通道2,序列4放通道3)

5:/*ADC初始化*/(独立模式,数据右对齐,不使用外部触发,连续转换,扫描模式,通道数4)

6:/*DMA初始化*/(外设地址:ADC的DR也就是电压值的寄存器

                                        数据宽度:16位的半字

                                        地址自增:外设不自增,内存地址(目标地址)自增

                                        传输方向:外设为源

                                        转运次数:4(一次16位)

                                        DMA_Mode模式:循环模式

                                        DMA_M2M存储器到存储器:失能,由ADC外设触发转运

                                        优先级:中等)

7:    /*DMA使能*/ DMA_Cmd()

8:      /*ADC1触发DMA1使能*/  ADC_DMACmd

9:     /*ADC使能*/ADC_Cmd()

10:/*ADC校准*/  复位校准   开始校准

11:/*ADC触发*/    ADC_SoftwareStartConvCmd

好了总共总结起来就是上面的11步就能实现STM32自动把4路模拟信号的值搬运到一个数组中,完成在主函数中只调用数组就能知道结果的目标。下面开始展示我写的程序了,感觉有点乱,就不整理了:

MyADC.c文件:

#include "stm32f10x.h"                  // Device headeruint16_t AD_Value[4];       //###########################void MyADC_Init(void){//开始RCC时钟,包括ADC和GPIO的时钟,ADCCLK的分频器也需要配置一下。RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);   //开启ADC1的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);  //开启GPIOA的时钟RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);RCC_ADCCLKConfig(RCC_PCLK2_Div6);      // ADC的时钟选择6分频,也就是72M/6=12M//配置GPIO,把需要用的GPIO配置成模拟输入的模式GPIO_InitTypeDef GPIO_InitStruct;    //GPIO初始化的结构体GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AIN;  // 模式为模拟输入GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;       // 初始化端口0GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;  //端口频率50MGPIO_Init(GPIOA, &GPIO_InitStruct);           // GPIOA初始化//配置多路开关,把左边的通道接入规则组列表里ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_28Cycles5); //ADC1规则配置(ADC1,通道1,列表1,转换时间28.5个时钟(12M))ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_28Cycles5);  //#############################ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_28Cycles5);  //###########################ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_28Cycles5);  //XXXXXXXXXXXXXXXX//配置ADC转换器,用结构体配置,一大块的参数。ADC_InitTypeDef ADC_InitStruct;                    //ADC初始化结构体ADC_InitStruct.ADC_ContinuousConvMode = ENABLE;            //连续转换模式:开启  ####################ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;            // 数据对齐:右对齐ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;   //ADC中断触发:空,也就是软件触发ADC_InitStruct.ADC_Mode = ADC_Mode_Independent;               // ADC模式:独立模式ADC_InitStruct.ADC_NbrOfChannel = 4;                   //规则组中的通道数:4   #######################################3ADC_InitStruct.ADC_ScanConvMode = ENABLE;             //扫描模式:开启 ######################################ADC_Init(ADC1, &ADC_InitStruct);                    //ADC初始化//************************************************//调用DMA_Init,初始化各个参数(包括外设和存储器的起始地址,数据宽度,地址是否自增,方向,传输计数器,是否需要自动重装,选择触发源,通道优先级)DMA_InitTypeDef DMA_InitStruct;DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;        //######################################DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;   //数据宽度,按半字的宽度(16位)搬运  #######################DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;    //不启用地址自增   ########################// 以上是外设站点(数据来源)的起始地址、数据宽度、是否自增。DMA_InitStruct.DMA_MemoryBaseAddr = (uint32_t)AD_Value;  //###############################DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;  //数据宽度,按半字的宽度粘贴  #######################DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;       //启用地址++自增//以上3条是存储器(目的地)的起始地址、数据宽度、是否自增。DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;    //传输方向DMA_InitStruct.DMA_BufferSize = 4;  //缓存区大小,其实就是传输计数器 传输4次半个字(16位)  ##########################DMA_InitStruct.DMA_Mode = DMA_Mode_Circular;   //传输模式,其实就是是否启用自动重装    自动重装  ###############DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;   //选择是硬件触发还是软件触发    硬件触发  ######################DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;   // 优先级    选择中等优先级DMA_Init(DMA1_Channel1, &DMA_InitStruct);  //第一个参数选择了是哪个DMA到哪个DMA通道,第二个参数结构体//调用DMA_CMD,通道使能(要在对应的外设调用XXX_DMACmd开启一下触发信号的输出,如果需要DMA的中断,就调用DMA_ITConfig,开启中断输出,再在NVIC里,配置中断通道,最后写中断函数就行了)DMA_Cmd(DMA1_Channel1, ENABLE);           //###################  失能改使能  ADC_DMACmd(ADC1, ENABLE);      //开启ADC1的DMA请求   ##################################//************************************************//打开ADC_CMD()开启ADC。ADC_Cmd(ADC1, ENABLE);                        // ADC开启//校准ADCADC_ResetCalibration(ADC1);                        // 复位校准ADCwhile(ADC_GetResetCalibrationStatus(ADC1) == SET);    // 等待校准标志位置0ADC_StartCalibration(ADC1);                         // 开始复位校准ADCwhile(ADC_GetCalibrationStatus(ADC1) == SET);          // 等待开始校准结束标志位置0ADC_SoftwareStartConvCmd(ADC1, ENABLE);  //软件触发
}

下面是MyADC.h文件:

#ifndef __MYADC_H
#define __MYADC_Hextern uint16_t AD_Value[4]; void MyADC_Init(void);#endif

下面是主函数main.c文件:

#include "stm32f10x.h"                  // Device header
#include "OLED.h"
#include "MyADC.h"
#include "Delay.h"int main(void)
{OLED_Init();       //oled  屏幕初始化MyADC_Init();       //ADC初始化OLED_ShowString(1,1,"Val_1:");OLED_ShowString(2,1,"Val_2:");OLED_ShowString(3,1,"Val_3:");OLED_ShowString(4,1,"Val_4:");while(1){OLED_ShowNum(1,7, AD_Value[0], 4);      //显示val的数值,这个数值的范围为0-4095OLED_ShowNum(2,7, AD_Value[1], 4);OLED_ShowNum(3,7, AD_Value[2], 4);OLED_ShowNum(4,7, AD_Value[3], 4);Delay_ms(200);}
}

好了,通过上面的一顿操作猛如虎,编译下载后就能看到自己想要的结果了,想要屏幕上的字不闪的那么快主函数中的循环内就加大点延时,不在乎就小点或是没有,我为了拍照拍全就加了200毫秒的延时,就能拍全了。

这篇关于STM32同时测量4路模拟电压,用DMA自动搬运到数组内,不用软件干预。的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/831926

相关文章

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

基于Redis自动过期的流处理暂停机制

《基于Redis自动过期的流处理暂停机制》基于Redis自动过期的流处理暂停机制是一种高效、可靠且易于实现的解决方案,防止延时过大的数据影响实时处理自动恢复处理,以避免积压的数据影响实时性,下面就来详... 目录核心思路代码实现1. 初始化Redis连接和键前缀2. 接收数据时检查暂停状态3. 检测到延时过

Java中数组与栈和堆之间的关系说明

《Java中数组与栈和堆之间的关系说明》文章讲解了Java数组的初始化方式、内存存储机制、引用传递特性及遍历、排序、拷贝技巧,强调引用数据类型方法调用时形参可能修改实参,但需注意引用指向单一对象的特性... 目录Java中数组与栈和堆的关系遍历数组接下来是一些编程小技巧总结Java中数组与栈和堆的关系关于

SpringBoot实现RSA+AES自动接口解密的实战指南

《SpringBoot实现RSA+AES自动接口解密的实战指南》在当今数据泄露频发的网络环境中,接口安全已成为开发者不可忽视的核心议题,RSA+AES混合加密方案因其安全性高、性能优越而被广泛采用,本... 目录一、项目依赖与环境准备1.1 Maven依赖配置1.2 密钥生成与配置二、加密工具类实现2.1

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

SQL Server跟踪自动统计信息更新实战指南

《SQLServer跟踪自动统计信息更新实战指南》本文详解SQLServer自动统计信息更新的跟踪方法,推荐使用扩展事件实时捕获更新操作及详细信息,同时结合系统视图快速检查统计信息状态,重点强调修... 目录SQL Server 如何跟踪自动统计信息更新:深入解析与实战指南 核心跟踪方法1️⃣ 利用系统目录

python运用requests模拟浏览器发送请求过程

《python运用requests模拟浏览器发送请求过程》模拟浏览器请求可选用requests处理静态内容,selenium应对动态页面,playwright支持高级自动化,设置代理和超时参数,根据需... 目录使用requests库模拟浏览器请求使用selenium自动化浏览器操作使用playwright

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

MyBatis-Plus 自动赋值实体字段最佳实践指南

《MyBatis-Plus自动赋值实体字段最佳实践指南》MyBatis-Plus通过@TableField注解与填充策略,实现时间戳、用户信息、逻辑删除等字段的自动填充,减少手动赋值,提升开发效率与... 目录1. MyBATis-Plus 自动赋值概述1.1 适用场景1.2 自动填充的原理1.3 填充策略