STM32单片机芯片与内部33 ADC 单通道连续DMA

目录

一、ADC DMA配置——标准库

1、ADC配置

2、DMA配置

二、ADC DMA配置——HAL库

1、ADC配置

2、DMA配置

三、用户侧

1、DMA开关

(1)、标准库

(2)、HAL库

2、DMA乒乓

(1)、标准库

(2)、HAL库


上文提到了当转换速度较高的时候需要由DMA进行搬运。

一、ADC DMA配置——标准库

1、ADC配置

        可以看到ADC配置几乎不用变。

	// 使能ADC DMA 请求ADC_DMACmd(ADCx, ENABLE);

2、DMA配置

        最重要的是源地址、目的地址、传输大小。如下配置为将每次ADC的数据从DR源地址搬运到ADC_ConvertedValue变量,因为只有一个大小长度,因此设定为1,大小设置为两个字节。

__IO uint16_t ADC_ConvertedValue;DMA_InitTypeDef DMA_InitStructure;// 打开DMA时钟RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);// 打开ADC时钟ADC_APBxClock_FUN ( ADC_CLK, ENABLE );// 复位DMA控制器DMA_DeInit(ADC_DMA_CHANNEL);// 配置 DMA 初始化结构体// 外设基址为:ADC 数据寄存器地址DMA_InitStructure.DMA_PeripheralBaseAddr = ( uint32_t ) ( & ( ADCx->DR ) );// 存储器地址,实际上就是一个内部SRAM的变量DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADC_ConvertedValue;// 数据源来自外设DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;// 缓冲区大小为1,缓冲区的大小应该等于存储器的大小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 传输通道优先级为高,当使用一个DMA通道时,优先级设置不影响DMA_InitStructure.DMA_Priority = DMA_Priority_High;// 禁止存储器到存储器模式,因为是从外设到存储器DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;// 初始化DMADMA_Init(ADC_DMA_CHANNEL, &DMA_InitStructure);// 使能 DMA 通道DMA_Cmd(ADC_DMA_CHANNEL , ENABLE);

        有人会好奇,DR寄存器32位,为什么定义的16位,不会丢失什么嘛,前面数据手册介绍过。高16位用于双ADC,单ADC仅用到低16位,且一般右端对齐,则为低12位有效。

二、ADC DMA配置——HAL库

1、ADC配置

        同样不修改配置。

    HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t*)&ADC_ConvertedValue, 1);

2、DMA配置

        可以看到源地址和目的地址和长度最重要的被放在了HAL_ADC_Start_DMA中,这样可以实现不修改初始化的情况下,进行地址、长度的修改。

DMA_HandleTypeDef hdma_adcx;
__IO uint16_t ADC_ConvertedValue;// 开启DMA时钟RHEOSTAT_ADC_DMA_CLK_ENABLE();// 数据传输通道hdma_adcx.Instance = RHEOSTAT_ADC_DMA_STREAM;hdma_adcx.Init.Direction=DMA_PERIPH_TO_MEMORY;;            //存储器到外设hdma_adcx.Init.PeriphInc=DMA_PINC_DISABLE;                 //外设非增量模式hdma_adcx.Init.MemInc=DMA_MINC_ENABLE;                     //存储器增量模式 hdma_adcx.Init.PeriphDataAlignment=DMA_PDATAALIGN_HALFWORD;//外设数据长度:16位hdma_adcx.Init.MemDataAlignment=DMA_MDATAALIGN_HALFWORD;   //存储器数据长度:16位hdma_adcx.Init.Mode= DMA_CIRCULAR;                         //外设普通模式hdma_adcx.Init.Priority=DMA_PDATAALIGN_HALFWORD;               //中等优先级//初始化DMA流,流相当于一个大的管道,管道里面有很多通道HAL_DMA_Init(&hdma_adcx); __HAL_LINKDMA( &ADC_Handle,DMA_Handle,hdma_adcx);

三、用户侧

还是前面的问题,如果需要处理1000个点,该怎么办?

        如果不需要前1000和后1000连续,则可以进行DMA的开关或ADC的开关,如果要求连续则开启DMA乒乓切换。

1、DMA开关

        说明不需要DMA的连续转换,而是传输1000个点则停止,处理后再进行一次DMA传输。

(1)、标准库

        需要修改为单次的缓冲区大小、单次传输模式。

define max_size 1000
__IO uint16_t ADC_ConvertedValue[max_size ];// 存储器地址,实际上就是一个内部SRAM的数组DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADC_ConvertedValue;// 缓冲区大小为1,缓冲区的大小应该等于存储器的大小DMA_InitStructure.DMA_BufferSize = max_size ;// 单次传输模式DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;	// 存储器地址,实际上就是一个内部SRAM的变量DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue;// 存储器地址递增DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; 

        DMA中断赋值flag,主函数用户检测到flag=1说明单次1000个点的数据好了,就可以进行处理,然后再次开启DMA和ADC。

        DMA进入中断,关闭ADC转换并将数据转换结果置1。当然也可以直接在主函数检测DMA的传输完成TCIF。

// DMA 完成后产生中断,停止 DMA,用户处理数据
void DMA1_Channel1_IRQHandler(void)
{if (DMA_GetITStatus(DMA1_IT_TC1))  // 检查 DMA 传输完成中断{// 清除 DMA 中断标志DMA_ClearITPendingBit(DMA1_IT_TC1);// 关闭 ADC 转换ADC_SoftwareStartConvCmd(ADCx, DISABLE);flag=1;}
}

        主函数检测到flag置位1后,说明可以进行数据处理,处理完成后,就可以重启DMA和ADC转换了。

if(flag==1)
{data_process();flag=0;// 重新启动 DMADMA_Cmd(ADC_DMA_CHANNEL, ENABLE);// 重新启动 ADCADC_SoftwareStartConvCmd(ADCx, ENABLE);
}

(2)、HAL库

        需要修改为单次的缓冲区大小、单次传输模式。

define max_size 1000
__IO uint16_t ADC_ConvertedValue[max_size ];hdma_adcx.Init.MemInc=DMA_MINC_ENABLE;                     //存储器增量模式 hdma_adcx.Init.Mode= DMA_NORMAL;                         //外设普通模式HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t*)&ADC_ConvertedValue, max_size );

有StartDMA,自然也有StopDMA,在中断服务函数直接执行即可。

HAL_ADC_Stop_DMA(&ADC_Handle)

2、DMA乒乓

        每次中断后修改目的地址,并开启新的中断,只需要修改中断服务函数即可。

(1)、标准库

__IO uint16_t ADC_ConvertedValue[2][max_size];  // 定义两个缓冲区,双缓冲区// 存储器地址,实际上就是一个内部SRAM的变量DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue[currentBuffer];
// DMA 完成后产生中断,停止 DMA,用户处理数据
void DMA1_Channel1_IRQHandler(void)
{if (DMA_GetITStatus(DMA1_IT_TC1))  // 检查 DMA 传输完成中断{// 清除中断标志DMA_ClearITPendingBit(DMA1_IT_TC1);// 切换到下一个缓冲区currentBuffer = (currentBuffer + 1) % 2;  // 切换到另一个缓冲区// 重新配置DMA传输目标地址DMA_Init(ADC_DMA_CHANNEL, &DMA_InitStructure);DMA_Cmd(ADC_DMA_CHANNEL, ENABLE); // 重新启动DMA}
}

        在主函数中判断,如果currentBuffer为1,说明当前在向第二部分写入此时可以处理第一部分,如果为0,则说明在向第一部分写入此时可以处理第二部分。

(2)、HAL库

        可以看到,得力于源地址和目的地址和长度最重要的被放在了HAL_ADC_Start_DMA中,可以很方便实现。

// DMA 完成后产生中断,停止 DMA,用户处理数据
void DMA1_Channel1_IRQHandler(void)
{if (DMA_GetITStatus(DMA1_IT_TC1))  // 检查 DMA 传输完成中断{// 清除中断标志DMA_ClearITPendingBit(DMA1_IT_TC1);// 切换到下一个缓冲区currentBuffer = (currentBuffer + 1) % 2;  // 切换到另一个缓冲区HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t)ADC_ConvertedValue[currentBuffer], max_size );}
}

        在主函数中判断,如果currentBuffer为1,说明当前在向第二部分写入此时可以处理第一部分,如果为0,则说明在向第一部分写入此时可以处理第二部分。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/890526.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vulnhub靶场——Log4j2

第一步:搭建靶场环境 #开启环境 cd vulhub/log4j/CVE-2021-44228 docker-compose up -d 来到网站首页 第二步:搭建一个dnslog平台上获取我们注入的效果 第三步:发现 /solr/admin/cores?action 这里有个参数可以传 我们可以看到留下了访问记录并且前面的参数被执行后给我们回…

leetcode 05 回文字符串

leetcode 05 回文字符串 1. 描述 给你一个字符串,找到里面最长的回文字符串 2. 事例 示例 1: 输入:s "babad" 输出:"bab" 解释:"aba" 同样是符合题意的答案。示例 2: …

使用idea创建JDK8的SpringBoot项目

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 使用idea创建JDK8的SpringBoot项目 前言我们经常在创建新的springboot项目,默认使用的是spring.io进行创建,但是它总是只会提供高版本的创建方式&…

U9多组织销退业务流程的总结

多组织销退业务流程,它的运行模式也是奇葩,确实不好理解其中的道理。用户实践中更是障碍重重,束手无策。左也不是右也不是,无路可走。 不能理解透彻造成的吧,所以做一个总结。 既有退货,必有出货单。从出货…

cudnn版本gpu架构

nvcc --help 可以看 --gpu-architecture 写到的支持的架构 NVIDIA 的 GPU 架构是按代次发布的,以下是这些架构的对应说明: NVIDIA Hopper: 这是 NVIDIA 于 2022 年推出的架构之一,面向高性能计算(HPC)和人工智能&…

Prompt格式到底有多重要?它竟然这样影响LLM函数调用能力(附提示词模版)

函数调用能力的关键地位 在当前大语言模型(LLM)的应用生态中,函数调用能力(Function Calling)已经成为一项不可或缺的核心能力。它使LLM能够通过调用外部API获取实时信息、操作第三方服务,从而将模型的语言…

Mono里运行C#脚本3—mono_jit_init

前面已经介绍了配置参数的读取,这样就可以把一些特殊的配置读取进来,完成了用户配置阶段的参数,接着下来就需要进行大工程的建造了。 为什么这样说呢,因为需要解释并执行C#编译的受托管的代码,相当于就是建立一个C#代码运行的虚拟机,而这个虚拟机还是很复杂的,不但要支…

有了chatgpt4o,普通人还需要学代码吗?

或许AI到达“终极智能”时,普通人就不用学代码了。不过现阶段或很长的一段时间内这还是不可能的。目前AI编程还是以辅助编程为主,普通人可以借助AI实现一些简单的编程。 其实这个问题可以扩大到AI编程的出现对编程行业或程序员职业影响的问题。就这个问…

CE8.【C++ Cont】练习题组6

目录 1.矩阵转置 题目描述 输入格式 输出格式 输入输出样例 错误代码 提交结果 正确代码 提交结果 2.图像相似度 题目描述 输入格式 输出格式 输入输出样例 代码 提交结果 3. 计算矩阵边缘元素之和 题目描述 输入格式 输出格式 输入输出样例 说明/提示 …

混合开发环境---使用编程AI辅助开发Qt

文章目录 [toc]1、说明2、演示视频 1、说明 新时代的浪潮早就已经来临,上不了船的人终将被抛弃,合理使用AI辅助开发、提升效率是大趋势 注意:不要被AI奴隶 合理使用AI辅助编程,十倍提升效率。 大部分的编程AI都有vs code插件&…

编译笔记:vs 中 正在从以下位置***加载符号 C# 中捕获C/C++抛出的异常

加载符号 解决方法: 进入VS—工具—选项----调试----符号,看右边有个“Microsoft符号服务器”,将前面的勾去掉,(可能还有删除下面的那个缓存)。 参考 C# 中捕获C/C抛出的异常 在需要捕捉破坏性异常的函数…

【Rust自学】6.1. 定义枚举

喜欢的话别忘了点赞、收藏加关注哦,对接下来的教程有兴趣的可以关注专栏。谢谢喵!(・ω・) 6.1.1. 什么是枚举 枚举允许我们列举所有可能的值来定义一个类型。这与其他编程语言中的枚举类似,但 Rust 的枚举更加灵活和强…

QT笔记- QClipboard剪切板对QByteArray数据的复制与粘贴

复制 // 存储在剪切板 QByteArray data; QClipboard * clipboard QGuiApplication::clipboard(); // 获取系统剪贴板对象 QMimeData * mimeData new QMimeData; // 注意, 剪切板会接管对象的释放 QString customMimeType "Test"; // 设置数据标识, 粘贴时将根据…

哪个网页版思维导图好用?这5款高效软件不容错过!

眼下虽然每个人的电脑硬盘越来越大,但很多人还是保留着“能不装软件就不装”的理念,在选择办公软件时,会更倾向于选用推出了网页版的软件,这对思维导图来说也不例外。 那具体到思维导图网页版,有哪些软件值得推荐&…

【双指针算法】--复写零

文章目录 1. 题目2. 题目解析3. 代码 1. 题目 在线oj 给你一个长度固定的整数数组 arr ,请你将该数组中出现的每个零都复写一遍,并将其余的元素向右平移。 注意:请不要在超过该数组长度的位置写入元素。请对输入的数组 就地 进行上述修改&a…

石岩基督教福音堂

周末娃,娃的阿婆和我一起去石岩基督教福音堂做礼拜。 这是我第一次进入石岩的教堂。教堂很高。应该有3,4层楼高。 这周末做礼拜的人很多一楼人满了,阿婆去二楼做礼拜,娃和我去三楼的儿童室。很多家长楼下做礼拜,小朋友被安排三楼…

通过Hydra 配置文件实例化Lightning类

通过 Hydra 的配置文件灵活控制 LightningModule、LightningDataModule、Callback 和 Trainer 的实例化过程。 Lightning 与 Hydra 的结合 1. 配置文件结构 一个典型的配置文件目录可能如下: configs/ ├── callbacks/ │ ├── early_stopping.yaml │ ├── che…

Flutter 异步编程简述

1、isolate 机制 1.1 基本使用 Dart 是基于单线程模型的语言。但是在开发当中我们经常会进行耗时操作比如网络请求,这种耗时操作会堵塞我们的代码。因此 Dart 也有并发机制 —— isolate。APP 的启动入口main函数就是一个类似 Android 主线程的一个主 isolate。与…

从 $PGDATA 到文件组:深入解析 PostgreSQL 与 SQL Server 的存储策略

在数据库领域,数据存储和管理的效率与可靠性是决定系统性能、可扩展性和易于管理的关键因素。PostgreSQL 和 SQL Server 在数据存储方面采取了略有不同的方式。 PostgreSQL 中一个数据库管理员经常遇到的关键概念是 $PGDATA 文件夹。在这里,我们将探讨 $PGDATA 文件夹是什么…

24.12.25 AOP

前置通知环绕通知后置通知最终通知异常通知 API类似,只是生效的时机不一样,并且,不能保证,各个通知的顺序 try {//前置通知before();//环绕通知,内部是执行的方法around(proxy,method,args,methodProxy);//后置通知a…