关于APM32F407配置串口DMA收发没有数据的问题记录

2024-05-29 18:12

本文主要是介绍关于APM32F407配置串口DMA收发没有数据的问题记录,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一.问题环境

​ 整活了一套APM32F407的板子,用了APM32F4xx_SDK_V1.4的标准外设库,正在搭建移植底层BSP框架串口部分,BSP底层配置逻辑是从STM32F407移植过来的。DMA发送时才使能通道及配置外设地址及缓存大小。

​ 串口1DMA配置过程如下:


static USART_DMAConfig_t USART_ComDmaTx(DMA_ChannelType_t channel,DMA_StreamType_t stream,uint32_t peripheralBaseAddr,uint8_t IRQChannel,uint32_t dmaFlag)
{USART_DMAConfig_t dmaConfig = {0};dmaConfig.channel = channel;dmaConfig.stream = stream;dmaConfig.dma.channel = channel;dmaConfig.dma.peripheralBaseAddr = peripheralBaseAddr;			//DMA外设地址:USART数据寄存器地址dmaConfig.dma.memoryBaseAddr = (uint32_t)0;									//内存地址,使用时再配置dmaConfig.dma.dir = DMA_DIR_MEMORYTOPERIPHERAL;									//外设地址为目的地址dmaConfig.dma.bufferSize = (uint32_t)0;										//传输时缓冲区大小,使用时再配置dmaConfig.dma.peripheralInc = DMA_PERIPHERAL_INC_DISABLE;					//外设地址固定,不递增dmaConfig.dma.memoryInc = DMA_MEMORY_INC_ENABLE;								//内存地址递增dmaConfig.dma.peripheralDataSize = DMA_PERIPHERAL_DATA_SIZE_BYTE;				//外设数据格式为字节dmaConfig.dma.memoryDataSize = DMA_MEMORY_DATA_SIZE_BYTE;						//内存数据格式为字节dmaConfig.dma.loopMode = DMA_MODE_NORMAL;										//工作在正常模式,不循环dmaConfig.dma.priority = DMA_PRIORITY_HIGH;									//DMA传输优先级为高(VeryHigh/High/Medium/Low)dmaConfig.dma.fifoMode = DMA_FIFOMODE_DISABLE;										//禁能DMA的两个Memory中变量相互访问dmaConfig.dma.peripheralBurst = DMA_PERIPHERALBURST_SINGLE;						dmaConfig.dma.fifoThreshold = DMA_FIFOTHRESHOLD_QUARTER;dmaConfig.dma.memoryBurst = DMA_MEMORYBURST_SINGLE;dmaConfig.nvic.NVIC_IRQChannel = IRQChannel;dmaConfig.nvic.NVIC_IRQChannelPreemptionPriority = 6;							//抢占优先级dmaConfig.nvic.NVIC_IRQChannelSubPriority = 0;									//响应优先级dmaConfig.nvic.NVIC_IRQChannelCmd = DISABLE;									//使能中断dmaConfig.dmaFlag = dmaFlag;return dmaConfig;
}static USART_DMAConfig_t USART_Com1DmaTx(void)
{DMA_ChannelType_t channel;DMA_StreamType_t stream;uint32_t peripheralBaseAddr;uint8_t IRQChannel;uint32_t dmaFlag;channel = DMA_CHANNEL_4;stream = DMA2_Stream7;peripheralBaseAddr = (uint32_t)(&(USART1->DATA));IRQChannel = DMA2_STR7_IRQn;dmaFlag = DMA_INT_TCIFLG7;return USART_ComDmaTx(channel,stream,peripheralBaseAddr,IRQChannel,dmaFlag);
};static void USART_DmaTxConfig(USART_t USART)
{NVIC_InitType_t		NVIC_InitStructure;/* 使能时钟 */if ((uint32_t)USART->config.dmaTx.stream > (uint32_t)DMA2){GPIO_RcmAHB1PeriphClockCmd(RCM_AHB1_PERIPH_DMA2, ENABLE);		//开启DMA时钟,用于USART发射}else{GPIO_RcmAHB1PeriphClockCmd(RCM_AHB1_PERIPH_DMA1, ENABLE);		//开启DMA时钟,用于USART发射}NVIC_InitStructure = USART->config.dmaTx.nvic;DMA_Disable(USART->config.dmaTx.stream);									//先禁止DMA通道,若之前有DMA传输,则会终止 ,必须先关闭DMA通道才能配置DMA_Reset(USART->config.dmaTx.stream);														//复位DMA1通道2的初始化DMA_Config(USART->config.dmaTx.stream,&USART->config.dmaTx.dma);		//DMA初始化if (NVIC_InitStructure.NVIC_IRQChannelCmd)  {  NVIC_EnableIRQRequest((IRQn_Type)NVIC_InitStructure.NVIC_IRQChannel,NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority,NVIC_InitStructure.NVIC_IRQChannelSubPriority);									//使能中断}  DMA_ClearStatusFlag(USART->config.dmaTx.stream,USART->config.dmaTx.dmaFlag);	//清除 全部标志DMA_EnableInterrupt(USART->config.dmaTx.stream,DMA_INT_TCIFLG);			//使能DMA通道4传输完成中断DMA_Disable(USART->config.dmaTx.stream);								//更新内存地址和传输大小之后再使能}void USART_TypeInit(USART_t USART)
{USART_Reset(USART->usart);USART_DisableDMA(USART->usart,USART_DMA_TX);USART_DisableDMA(USART->usart,USART_DMA_RX);USART_Construct(USART);USART_RccConfig(USART);USART_GpioConfig(USART);USART_Disable(USART->usart);USART_InitConfig(USART);USART_NvicConfig(USART);if (USART->config.isDmaTxEnable){USART_DmaTxConfig(USART);USART_EnableDMA(USART->usart, USART_DMA_TX);}if (USART->config.isDmaRxEnable){USART_DmaRxConfig(USART);USART_EnableDMA(USART->usart, USART_DMA_RX);}
}static void USART_SendMessage(USART_t USART, uint8_t *sendBuf, uint32_t len)
{if (USART->config.isRs485Enable){USART_Rs485TxEnable(USART);}if (USART->config.isDmaTxEnable){DMA_Disable(USART->config.dmaTx.stream);			//先禁止DMA通道,若之前有DMA传输,则会终止 ,必须先关闭DMA通道才能配置while (DMA_ReadCmdStatus(USART->config.dmaTx.stream) != DISABLE);		//确保DMA可以被设置//=======直接操作寄存器更新内存地址和传输大小-----------------------------------------------------------USART->config.dmaTx.stream->M0ADDR = (uint32_t)(sendBuf);	//更新内存地址USART->config.dmaTx.stream->NDATA = len;	//更新传输时缓冲区大小DMA_ClearStatusFlag(USART->config.dmaTx.stream,USART->config.dmaTx.dmaFlag);//清除Channel2全部标志,主要是清除传输完成标志DMA_Enable(USART->config.dmaTx.stream);//等待DMA发送结束while(DMA_ReadIntFlag(USART->config.dmaTx.stream,USART->config.dmaTx.dmaFlag) == RESET);//清除标志DMA_ClearStatusFlag(USART->config.dmaTx.stream,USART->config.dmaTx.dmaFlag);if (USART->config.isRs485Enable){USART_Rs485TxDisable(USART);}USART->send->flag = true;}
}

二.问题现象

​ 串口DMA接收和中断接收都没有问题,发送中断也没有问题。但发送配置为DMA就不能发送数据。所有串口端口配置都有这个问题,确定同样的板子刷入STM32F407的的程序也能正常使用,唯独使用APM的外设库就不行。

三.原因分析

​ 当同时使用APM32F4的DMA接收和发送时,虽然同时配置了USART_EnableDMA(USART->usart, USART_DMA_TX)和USART_EnableDMA(USART->usart, USART_DMA_RX),但USART_EnableDMA这个函数你看他是怎么实现的:

/*!* @brief     Enables the USART DMA interface** @param     usart:   Select the USART or the UART peripheral** @param     dmaReq:  Specifies the DMA request*                     This parameter can be one of the following values:*                     @arg USART_DMA_TX    : USART DMA receive request*                     @arg USART_DMA_RX    : USART DMA transmit request*                     @arg USART_DMA_TX_RX : USART DMA transmit/receive request** @retval    None** @note      The usart can be USART1, USART2, USART3, UART4, UART5, USART6, UART7 and UART8*/
void USART_EnableDMA(USART_T* usart, USART_DMA_T dmaReq)
{usart->CTRL3_B.DMARXEN = dmaReq & 0x01;usart->CTRL3_B.DMATXEN = dmaReq >> 1;
}

是不是很明显的看出了问题所在,原来usart->CTRL3_B.DMARXEN = dmaReq & 0x01;这一句用的是直接赋值操作,而不是我们熟悉的按位或,这样当配置了USART_EnableDMA(USART->usart, USART_DMA_RX)后,原来的USART_EnableDMA(USART->usart, USART_DMA_TX);就被重置了,所以只能接收不能发送。

四.问题解决

发现问题后,对代码初始化部分进行改进,经过测试,正常收发。代码如下:

	if (USART->config.isDmaTxEnable && USART->config.isDmaRxEnable){USART_DmaTxConfig(USART);USART_DmaRxConfig(USART);USART_EnableDMA(USART->usart, USART_DMA_TX_RX);}else if (USART->config.isDmaTxEnable){USART_DmaTxConfig(USART);USART_EnableDMA(USART->usart, USART_DMA_TX);}else if (USART->config.isDmaRxEnable){USART_DmaRxConfig(USART);USART_EnableDMA(USART->usart, USART_DMA_RX);}

这篇关于关于APM32F407配置串口DMA收发没有数据的问题记录的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

PHP轻松处理千万行数据的方法详解

《PHP轻松处理千万行数据的方法详解》说到处理大数据集,PHP通常不是第一个想到的语言,但如果你曾经需要处理数百万行数据而不让服务器崩溃或内存耗尽,你就会知道PHP用对了工具有多强大,下面小编就... 目录问题的本质php 中的数据流处理:为什么必不可少生成器:内存高效的迭代方式流量控制:避免系统过载一次性

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

mybatis映射器配置小结

《mybatis映射器配置小结》本文详解MyBatis映射器配置,重点讲解字段映射的三种解决方案(别名、自动驼峰映射、resultMap),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定... 目录select中字段的映射问题使用SQL语句中的别名功能使用mapUnderscoreToCame

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

Vue3绑定props默认值问题

《Vue3绑定props默认值问题》使用Vue3的defineProps配合TypeScript的interface定义props类型,并通过withDefaults设置默认值,使组件能安全访问传入的... 目录前言步骤步骤1:使用 defineProps 定义 Props步骤2:设置默认值总结前言使用T

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Vite 打包目录结构自定义配置小结

《Vite打包目录结构自定义配置小结》在Vite工程开发中,默认打包后的dist目录资源常集中在asset目录下,不利于资源管理,本文基于Rollup配置原理,本文就来介绍一下通过Vite配置自定义... 目录一、实现原理二、具体配置步骤1. 基础配置文件2. 配置说明(1)js 资源分离(2)非 JS 资