【STM32标准库】DMA双缓冲模式

1.双缓冲模式简介

设置DMA_SxCR寄存器的DBM位为1可启动双缓冲传输模式,并自动激活循环模式,所以设置普通模式或者循环模式都可以。

双缓冲不应用与存储器到存储器的传输。可以应用在从存储器到外设或者外设到存储器。

双缓冲模式下, 两个存储器地址指针都有效,即DMA_SxM1AR寄存器将被激活使用。开始传输使用DMA_SxM0AR寄存器的地址指针所对应的存储区, 当这个存储区数据传输完DMA控制器会自动切换至DMA_SxM1AR寄存器的地址指针所对应的另一块存储区, 如果这一块也传输完成就再切换至DMA_SxM0AR寄存器的地址指针所对应的存储区,这样循环调用。

所以我们需要配置传输完成中断,在中断服务函数中,我们可以获取正在使用哪一个buffer,然后可以去填充另一个buffer的数据。

2.示例

#ifndef __BSP_USART_H
#define __BSP_USART_H#ifdef __cplusplus
extern "C"{#endif#include "stm32f4xx.h"
#include "stdio.h"#define LOGGER_USART              USART1
#define LOGGER_USART_BAUDRATE     115200
#define LOGGER_USART_CLK          RCC_APB2Periph_USART1
#define LOGGER_USART_IRQHandler   USART1_IRQHandler
#define LOGGER_USART_IRQ          USART1_IRQn#define USART1_TX_PIN             GPIO_Pin_9
#define USART1_TX_GPIO_Port       GPIOA
#define USART1_TX_GPIO_CLK        RCC_AHB1Periph_GPIOA
#define USART1_TX_AF              GPIO_AF_USART1
#define USART1_TX_SOURCE          GPIO_PinSource9#define USART1_RX_PIN             GPIO_Pin_10
#define USART1_RX_GPIO_Port       GPIOA
#define USART1_RX_GPIO_CLK        RCC_AHB1Periph_GPIOA
#define USART1_RX_AF              GPIO_AF_USART1
#define USART1_RX_SOURCE          GPIO_PinSource10//usart1_tx只能使用到DMA2_Stream7  Channel_4
#define USART1_TX_DMA_STREAM               DMA2_Stream7
#define USART1_TX_DMA_CHANNEL              DMA_Channel_4
#define USART1_TX_DMA_STREAM_CLK           RCC_AHB1Periph_DMA2
#define USART1_TX_DMA_IT_TCIF              DMA_IT_TCIF7
#define USART1_TX_DMA_IT_HTIF              DMA_IT_HTIF7
#define USART1_TX_DMA_STREAM_IRQn          DMA2_Stream7_IRQn
#define USART1_TX_DMA_STREAM_IRQHandler    DMA2_Stream7_IRQHandler#define TX_BUFFER_SIZE                     5
#define USART1_TX_DR_BASE                  (&USART1->DR)  //(USART1_BASE+0x04)void Init_USART(void);
void Init_USART_DMA(void);
void USART_DMA_SEND(uint8_t* data,uint32_t size);#ifdef __cplusplus
}
#endif#endif
#include "bsp_usart.h"
#include "string.h"uint8_t USART_TX_BUFFER[TX_BUFFER_SIZE]={0x00,0x02,0x04,0x06,0x08};
uint8_t USART_TX_BUFFER1[TX_BUFFER_SIZE]={0x01,0x03,0x05,0x07,0x09};void Init_USART(void)
{RCC_AHB1PeriphClockCmd(USART1_TX_GPIO_CLK|USART1_RX_GPIO_CLK,ENABLE);//使能GPIOA时钟RCC_APB2PeriphClockCmd(LOGGER_USART_CLK,ENABLE);//使能USART1时钟//USART1对应引脚复用映射GPIO_PinAFConfig(USART1_TX_GPIO_Port, USART1_TX_SOURCE,USART1_TX_AF);//PA9复用为USART1GPIO_PinAFConfig(USART1_RX_GPIO_Port, USART1_RX_SOURCE,USART1_RX_AF);//PA10复用为USART1//USART1端口配置GPIO_InitTypeDef GPIO_InitStruct;GPIO_InitStruct.GPIO_Pin=USART1_TX_PIN;GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF;//复用功能GPIO_InitStruct.GPIO_OType=GPIO_OType_PP;//推挽复用输出GPIO_InitStruct.GPIO_PuPd=GPIO_PuPd_UP;//上拉GPIO_InitStruct.GPIO_Speed=GPIO_Fast_Speed;//速度50MHzGPIO_Init(USART1_TX_GPIO_Port,&GPIO_InitStruct);//初始化PA9GPIO_InitStruct.GPIO_Pin=USART1_RX_PIN;GPIO_Init(USART1_RX_GPIO_Port,&GPIO_InitStruct);//初始化PA10//配置USART参数USART_InitTypeDef USART_Init_Struct;USART_Init_Struct.USART_BaudRate=LOGGER_USART_BAUDRATE;USART_Init_Struct.USART_HardwareFlowControl=USART_HardwareFlowControl_None;USART_Init_Struct.USART_Mode=USART_Mode_Rx|USART_Mode_Tx;USART_Init_Struct.USART_Parity=USART_Parity_No;USART_Init_Struct.USART_StopBits=USART_StopBits_1;USART_Init_Struct.USART_WordLength=USART_WordLength_8b;USART_Init(LOGGER_USART,&USART_Init_Struct);//配置中断控制器并使能USART接收中断NVIC_InitTypeDef NVIC_Init_Struct;NVIC_Init_Struct.NVIC_IRQChannel=LOGGER_USART_IRQ;NVIC_Init_Struct.NVIC_IRQChannelCmd=ENABLE;NVIC_Init_Struct.NVIC_IRQChannelPreemptionPriority=1;NVIC_Init_Struct.NVIC_IRQChannelSubPriority=0;NVIC_Init(&NVIC_Init_Struct);USART_ITConfig(LOGGER_USART,USART_IT_IDLE,ENABLE);//使能USARTUSART_Cmd(LOGGER_USART,ENABLE);//使能USART_DMAUSART_DMACmd(LOGGER_USART,USART_DMAReq_Tx|USART_DMAReq_Rx,ENABLE);
}void LOGGER_USART_IRQHandler(void)
{}//重定向c库函数printf到串口,重定向后可使用printf函数
int fputc(int ch, FILE *f)
{/* 发送一个字节数据到串口 */USART_SendData(LOGGER_USART, (uint8_t) ch);/* 等待发送完毕 */while (USART_GetFlagStatus(LOGGER_USART, USART_FLAG_TXE) == RESET);return (ch);
}void USART_DMA_SEND(uint8_t* data,uint32_t size)
{while (DMA_GetCmdStatus(USART1_TX_DMA_STREAM) != DISABLE) {}memcpy(USART_TX_BUFFER,data,size);DMA_Cmd(USART1_TX_DMA_STREAM,DISABLE);DMA_SetCurrDataCounter(USART1_TX_DMA_STREAM,size);DMA_Cmd(USART1_TX_DMA_STREAM,ENABLE);
}void Init_USART_DMA(void)
{/* 使能DMA时钟 */RCC_AHB1PeriphClockCmd(USART1_TX_DMA_STREAM_CLK, ENABLE);/* 复位初始化DMA数据流 */DMA_DeInit(USART1_TX_DMA_STREAM);/* 确保DMA数据流复位完成 */while (DMA_GetCmdStatus(USART1_TX_DMA_STREAM) != DISABLE) {}DMA_InitTypeDef  DMA_InitStructure;DMA_InitStructure.DMA_BufferSize=TX_BUFFER_SIZE;//一次DMA事务传输的数据个数DMA_InitStructure.DMA_Channel=USART1_TX_DMA_CHANNEL;DMA_InitStructure.DMA_DIR=DMA_DIR_MemoryToPeripheral;DMA_InitStructure.DMA_FIFOMode=DMA_FIFOMode_Disable;DMA_InitStructure.DMA_FIFOThreshold=DMA_FIFOThreshold_Full;DMA_InitStructure.DMA_Memory0BaseAddr= (uint32_t)USART_TX_BUFFER;DMA_InitStructure.DMA_MemoryBurst=DMA_MemoryBurst_Single;DMA_InitStructure.DMA_MemoryDataSize=DMA_MemoryDataSize_Byte;DMA_InitStructure.DMA_MemoryInc=DMA_MemoryInc_Enable;DMA_InitStructure.DMA_Mode=DMA_Mode_Normal;DMA_InitStructure.DMA_PeripheralBaseAddr=(uint32_t)USART1_TX_DR_BASE;DMA_InitStructure.DMA_PeripheralBurst=DMA_PeripheralBurst_Single;DMA_InitStructure.DMA_PeripheralDataSize=DMA_PeripheralDataSize_Byte;DMA_InitStructure.DMA_PeripheralInc=DMA_PeripheralInc_Disable;DMA_InitStructure.DMA_Priority=DMA_Priority_Low;DMA_Init(USART1_TX_DMA_STREAM,&DMA_InitStructure);//配置双缓冲DMA_DoubleBufferModeConfig(USART1_TX_DMA_STREAM,(uint32_t)USART_TX_BUFFER1,DMA_Memory_0);DMA_DoubleBufferModeCmd(USART1_TX_DMA_STREAM,ENABLE);DMA_ITConfig(USART1_TX_DMA_STREAM,DMA_IT_TC|DMA_IT_HT,ENABLE);DMA_ClearITPendingBit(USART1_TX_DMA_STREAM,USART1_TX_DMA_IT_TCIF|USART1_TX_DMA_IT_HTIF);//配置中断控制器并使能中断NVIC_InitTypeDef NVIC_InitStruct;NVIC_InitStruct.NVIC_IRQChannel=USART1_TX_DMA_STREAM_IRQn;NVIC_InitStruct.NVIC_IRQChannelCmd=ENABLE;NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority=2;NVIC_InitStruct.NVIC_IRQChannelSubPriority=0;NVIC_Init(&NVIC_InitStruct);DMA_Cmd(USART1_TX_DMA_STREAM,ENABLE);
}void USART1_TX_DMA_STREAM_IRQHandler(void)
{if(SET==DMA_GetITStatus(USART1_TX_DMA_STREAM,USART1_TX_DMA_IT_HTIF)){//half transfer complete//printf("half transfer\r\n");DMA_ClearITPendingBit(USART1_TX_DMA_STREAM,USART1_TX_DMA_IT_HTIF);}else if(SET==DMA_GetITStatus(USART1_TX_DMA_STREAM,USART1_TX_DMA_IT_TCIF)){//transfer complete//printf("transfer complete\r\n");if(0==DMA_GetCurrentMemoryTarget(USART1_TX_DMA_STREAM)){//Current memory buffer used is Memory 0for(int i=0;i<TX_BUFFER_SIZE;i++){USART_TX_BUFFER1[i]+=2;}}else{//Current memory buffer used is Memory 1for(int i=0;i<TX_BUFFER_SIZE;i++){USART_TX_BUFFER[i]+=2;}}DMA_ClearITPendingBit(USART1_TX_DMA_STREAM,USART1_TX_DMA_IT_TCIF);}
}

在这里插入图片描述

有一个函数需要注意

	//配置双缓冲DMA_DoubleBufferModeConfig(USART1_TX_DMA_STREAM,(uint32_t)USART_TX_BUFFER1,DMA_Memory_0);DMA_DoubleBufferModeCmd(USART1_TX_DMA_STREAM,ENABLE);

配置双缓冲的主要函数。

DMA_GetCurrentMemoryTarget(USART1_TX_DMA_STREAM)

该函数可以获取DMA正在使用的buffer,然后我们就可以去填充另外一个buffer了。

3.疑问

双缓冲模式应用在对DMA连续传输要求比较高的地方,不需要一次DMA buffer传输结束后有过多的操作然后进行下一次传输。但是,我们使用half transfer和transfer complete两个中断好像也可以做到,当有half transfer中断时,我们去更新buffer的前一半数据,当有transfer complete中断时,我们去更新buffer的后一半数据,然后配置循环模式。
这种方式与使用双buffer有什么区别吗?可能单buffer在传输中去修改其中的值不是一个稳妥的方式吧!

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

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

相关文章

IC认证介绍

IC认证是什么&#xff1f; IC是加拿大工业部Industry Canada的简称&#xff0c;作为政府机构&#xff0c;负责电子电器产品进入加拿大市场的认证事务。与美国的FCC相似&#xff0c;IC目前只在电磁干扰上做限制。一般规定&#xff1a;仅限制EMI&#xff0c;认证方式也与FCC相同…

领略超越王勃的AI颂扬艺术:一睹其惊艳夸赞风采

今日&#xff0c;咱也用国产AI技术&#xff0c;文心一言3.5的文字生成与可灵的图像创作&#xff0c;自动生成一篇文章&#xff0c;提示语文章末下载。 【玄武剑颂星际墨侠】 苍穹为布&#xff0c;星辰织锦&#xff0c;世间万象&#xff0c;皆入我玄武剑公众号之浩瀚画卷。此号…

Python入门 函数传递

目录 函数的多返回值 函数的多种传参形式 位置参数 关键字参数 缺省参数 不定长参数 位置传递 关键字传递 函数作为参数传递 lambda匿名函数 函数定义中 匿名函数的语法 函数的多返回值 按照返回值的顺序&#xff0c;与对应顺序的多个变量接受即可 变量之间用逗号…

SPI协议——对外部SPI操作(跨页读写)

关于W25Q32JVSSIQ的详细内容在之前的两篇文章中已经详细介绍&#xff0c;本文不做太多赘述&#xff0c;如果对芯片的了解有缺失的话&#xff0c;可以参考&#xff1a; SPI协议——对外部SPI Flash操作-CSDN博客 SPI协议——读取外部SPI Flash ID_spi flash 读取id-CSDN博客 目录…

【SpringCloud应用框架】Nacos集群配置

第八章 Spring Cloud Alibaba Nacos之集群配置 文章目录 一、Linux版NacosMySql生产环境配置具体配置&#xff1a; 二、Nacos集群配置更改Nacos启动命令配置原理具体配置测试启动总结 一、Linux版NacosMySql生产环境配置 上一篇博客中已经了解了Nacos生产环境下需要搭建集群配…

XDMA原理学习(1)——DMA技术详解

目录 一、什么是DMA&#xff1f;为什么需要DMA&#xff1f; 二、DMA分类 2.1 Block DMA 2.2 Scatter-Gather DMA 2.3 Ring buffer DMA 三、实际案例 3.1 STM32微处理器 3.1.1 Block DMA 3.1.2 Scatter-Gather DMA 3.1.3 使用场景举例&#xff1a; 3.1.4 配置与实现 …

3GPP R18 Multi-USIM 是怎么回事?(三)

这篇内容相对来说都是一些死规定,比较枯燥。主要是与MUSIM feature相关的mobility and periodic registration和service request触发过程的一些规定,两部分的内容是有部分重叠的,为保证完整性,重复部分也从24.501中摘了出来。 24.501 4.25 网络和MUSIM UE可以支持MUSIM fe…

idea控制台乱码问题解决教程

注&#xff1a;按顺序来操作&#xff0c;完成后要重启软件 方案一&#xff1a;修改Tomcat的编码设置 在Tomcat的VM options中添加了-Dfile.encodingUTF-8 方案二&#xff1a;修改IDEA的编码设置 File->Settings->Editor->File Encodings 将Global Encoding、Proj…

STM32-外部中断浅析

本篇解释了STM32中断原理 MCU为什么需要中断 中断&#xff0c;是嵌入式系统中很重要的一个功能&#xff0c;在系统运行过程中&#xff0c;当出现需要立刻处理的情况时&#xff0c;暂停当前任务&#xff0c;转而处理紧急任务&#xff0c;处理完毕后&#xff0c;恢复之前的任务…

深度学习之正则化

目标 我们学习正则化之前应该先了解我们为什么要用正则化 。正则化解决了什么问题 。我们讲正则化之前 &#xff0c;先了解一个概念–》过拟合 过拟合 数据增强 L1和L2正则化 Dropout 注意&#xff1a;Dropout是不适合用在卷积神经网络的 提前终止 一般的做法是&#xff1a;记…

B端工作台如何设计?指导思想+布局建议+大厂案例全给你

一、B端工作台设计的指导思想 要做好B端工作台设计&#xff0c;以下是一些建议和策略&#xff1a; 1. 用户研究&#xff1a;深入了解目标用户的需求、行为和工作流程。通过用户研究方法&#xff0c;如用户访谈、观察和调研&#xff0c;收集用户反馈和意见&#xff0c;了解他们…

【计算机组成原理】计算机系统概述(上)

&#xff08;一&#xff09;计算机系统简介 计算机系统有“硬件”和“软件”两大部分组成。 1&#xff09;所谓“硬件”即指见计算机的实体部分&#xff0c;它由看得见摸得着的各种电子元器件、各类光、电、机设备的事物组成&#xff0c;如主机、外设等等。 2&#xff09;所谓…

以SGET协会OSM标准首创有662引脚的OSM模组——凌华智能引领嵌入式运算市场

在可焊接的45 x 45mm尺寸上提升功率 开启嵌入式运算发展的新时代 摘要&#xff1a; 1.开放式标准模块(OSM™)&#xff0c;最大尺寸仅45 x 45mm&#xff0c;采用零开销的模块化系统简化生产&#xff0c;并提供662个引脚以增强小型化和物联网应用。 2.凌华智能提供基于NXP i.M…

快速掌握AI的最佳途径实践

科技时代&#xff0c;人工智能&#xff08;AI&#xff09;已经成为许多人希望掌握的重要技能。对于普通人来说&#xff0c;如何快速有效地学习AI仍然是一个挑战。本文将详细介绍几种快速掌握AI的途径&#xff0c;并提供具体的操作步骤和资源建议。 前言 AI的普及和应用已经深…

独立站爆款产品的选品思路及底层逻辑拆解

在这个竞争激烈的跨境电商市场&#xff0c;有一件事情比网站设计、营销策略、物流服务都更重要。那就是选品。跨境独立站选品是独立站成功的第一步&#xff0c;如果选错了产品&#xff0c;那么所有努力都可能白费。可能会面临库存积压、利润低迷、客户流失等问题。但是如果选对…

HackTheBox--BoardLight

BoardLight 测试过程 1 信息收集 NMAP端口扫描 端口扫描开放 22、80 端口 80端口测试 # 添加 boardLight.htb 到hosts文件 echo "10.10.11.11 boardLight.htb" | sudo tee -a /etc/hosts检查网页源代码&#xff0c;发现 board.htb # 添加 board.htb 到 hosts 文…

星网安全产品线成立 引领卫星互联网解决方案创新

2024年6月12日&#xff0c;盛邦安全&#xff08;688651&#xff09;成立星网安全产品线&#xff0c;这是公司宣布全面进入以场景化安全、网络空间地图和卫星互联网安全三大核心能力驱动的战略2.0时代业务落地的重要举措。 卫星互联网技术的快速发展&#xff0c;正将其塑造为全球…

Camera Raw:裁剪

Camera Raw 的裁剪 Crop面板提供了裁剪、旋转、翻转、拉直照片等功能&#xff0c;通过它们可以更精确地调整照片的视角和范围&#xff0c;以达到最佳二次构图的视觉效果。 快捷键&#xff1a;C ◆ ◆ ◆ 使用方法与技巧 1、使用预设 选择多种裁剪预设&#xff08;如 1:1、16:…

【安全设备】下一代防火墙

一、什么是防火墙 防火墙是一个网络安全产品&#xff0c;它是由软件和硬件设备组合而成&#xff0c;在内网和外网之间、专用网与公共网之间的一种保护屏障。在计算机网络的内网和外网之间构建一道相对隔离的保护屏障&#xff0c;以达到保护资料的目的。它是一种隔离技术&#…

将QComboBox下拉项中的文本居中、居右

目录 1. 需求提出 2. 解决方法 1. 需求提出 QComboBox下拉项中的文本默认是居左的&#xff0c;如下&#xff1a; 有时需要将下拉项中的文本居中、居右。如何实现&#xff1f; 2. 解决方法 首先想到的是通过样式表来解决&#xff0c;但找遍Qt Assist和网络&#xff0c;都没这…