hal库串口dma卡死_HAL库版DMA循环模式串口数据收发

本文详细介绍了如何使用STM32CubeMX的HAL库,结合DMA循环模式解决串口数据收发过程中的卡死问题。通过设置串口空闲中断和DMA传输完成中断,实现了不定长数据的接收,以及避免了等待发送完成标志导致的效率降低。此外,还提出了采用循环FIFO缓冲区管理策略,确保数据完整性和系统效率。
摘要由CSDN通过智能技术生成

在《STM32CubeMX初识与工程创建》的基础上,首先对串口进行设置,以实现通过串口对数据的收发。STM32CubeMX生成的HAL库中,提供了三类串口数据收发的接口,分别为阻塞模式,非阻塞模式和DMA模式,文本主要对DMA模式进行了分析并依据提供的接口提出了更加实用的串口数据收发方案。通过对网上资料的查找和分析,主要存在下述两个问题:

1、在数据接收过程中,采用了串口的空闲中断实现了DMA模式下不定长数据的接收,但存在的限制是单次接收的数据长度必须小于DMA缓冲区的长度,如果接收的数据长度大于DMA缓冲区的长度,就数据丢失了。

2、在数据发送过程中,DMA在发送阶段是不能再使能DMA发送的,即调用HAL库中的HAL_UART_Transmit_DMA接口后,在DMA传输数据完成之前是不能再调用该接口的,常用的方式是通过一个标志变量来确定是否可发送,但程序中若存在等待标志变量等逻辑的话,会降低程序运行的效率。

本文通过采用DMA的循环模式配合DMA传输完成中断和串口空闲中断解决DMA缓冲区长度对串口接收数据量的限制问题。采用类递归的逻辑解决串口发送数据等待发送完成标志的问题。

1 硬件配置

首先,应用STM32CubeMX对串口进行配置,

在Connectivity中勾选usart1,具体引脚根据硬件确定。重点注意的是,勾选USART1 global interrupt 使能;将收发的DMA添加上,将USART1_RX的DMA设置为Circular模式,将USART1_TX的DMA设置为Normal模式。

2 软件实现方案

2.1 初始化准备

将程序内部做驱动层和应用层的区分,串口数据收发接口的封装属于驱动层面,应用层面调用驱动层的接口实现数据的发送和获取,通过收发两个缓冲区实现数据的交互,本文中采用一组循环FIFO实现数据的缓冲。新建一组文件提供对串口收发数据的中间件接口,在头文件中定义串口中间件属性:

typedef struct

{

UART_HandleTypeDef *handle; /*HAL库提供的串口句柄*/

int16_t TransFlag; /*数据发送标志位*/

int32_t DmaSize; /*DMA缓冲区的大小*/

int32_t DamOffset; /*获取数据在DMA缓冲区的偏移量*/

uint8_t *pReadDma; /*指向接收DMA缓冲区的首地址*/

uint8_t *pWriteDma; /*指向发送DMA缓冲区的首地址*/

CFIFO ReadCFifo; /*接受数据的循环缓冲区*/

CFIFO WriteCFifo; /*发送数据的循环缓冲区*/

}MW_UART_ATTR;

上述属性值中包括了缓冲区,DMA等参数,后续实现中具体讲述各参数的用途。本文仅提供对一个串口进行配置用作演示,实际中在属性中添加了id参数实现多串口的管理,或者采用其他方式。

#define MW_TRANS_IDLE 0

#define MW_TRANS_BUSY 1

#define MW_UART_BUFFER_LEN 1024

#define MW_UART_DMA_LEN 256

static uint8_t Uart1TxBuff[MW_UART_BUF

  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在STM32 HAL库中,使用DMA实现串口数据收发也是非常简单的,只需调用相应的HAL库函数即可。以下是一个基本的示例: ```c #include "stm32f10x.h" #include "stm32f10x_hal.h" #define USART_RX_BUFFER_SIZE 256 volatile uint8_t usart_rx_buffer[USART_RX_BUFFER_SIZE]; volatile uint16_t usart_rx_write_index = 0; volatile uint16_t usart_rx_read_index = 0; UART_HandleTypeDef huart1; void usart_init(void) { GPIO_InitTypeDef GPIO_InitStruct; __HAL_RCC_GPIOA_CLK_ENABLE(); __HAL_RCC_USART1_CLK_ENABLE(); __HAL_RCC_DMA1_CLK_ENABLE(); GPIO_InitStruct.Pin = GPIO_PIN_9; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); GPIO_InitStruct.Pin = GPIO_PIN_10; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); huart1.Instance = USART1; huart1.Init.BaudRate = 115200; huart1.Init.WordLength = UART_WORDLENGTH_8B; huart1.Init.StopBits = UART_STOPBITS_1; huart1.Init.Parity = UART_PARITY_NONE; huart1.Init.Mode = UART_MODE_TX_RX; huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; HAL_UART_Init(&huart1); HAL_NVIC_SetPriority(DMA1_Channel5_IRQn, 0, 0); HAL_NVIC_EnableIRQ(DMA1_Channel5_IRQn); HAL_UART_Receive_DMA(&huart1, (uint8_t*)usart_rx_buffer, USART_RX_BUFFER_SIZE); } void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { usart_rx_write_index = USART_RX_BUFFER_SIZE - huart->hdmarx->Instance->CNDTR; while (usart_rx_read_index != usart_rx_write_index) { uint8_t data = usart_rx_buffer[usart_rx_read_index]; usart_rx_read_index = (usart_rx_read_index + 1) % USART_RX_BUFFER_SIZE; // 处理接收到的数据 // ... if (usart_rx_read_index == usart_rx_write_index) { break; } } HAL_UART_Receive_DMA(&huart1, (uint8_t*)usart_rx_buffer, USART_RX_BUFFER_SIZE); } int main(void) { usart_init(); while (1) { // 主循环中不需要处理接收到的数据 } } void DMA1_Channel5_IRQHandler(void) { HAL_DMA_IRQHandler(huart1.hdmarx); } ``` 在上述代码中,我们使用了HAL库函数 `HAL_UART_Receive_DMA` 来启动USART1的DMA接收,同时使用了HAL库函数 `HAL_UART_RxCpltCallback` 来处理接收到的数据。需要注意的是,我们需要在 `HAL_UART_RxCpltCallback` 中再次调用 `HAL_UART_Receive_DMA` 来启动下一轮DMA接收。 此外,在HAL库中,我们也需要在主函数中调用 `HAL_NVIC_EnableIRQ` 函数来使能DMA中断,并且需要在中断处理函数中调用 `HAL_DMA_IRQHandler` 函数来处理DMA中断。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值