GD32F4单片机实现接收超时中断+DMA实现串口的不定长接收和DMA发送

发布者:美好的人生最新更新时间:2024-12-11 来源: elecfans关键字:单片机  DMA  串口  不定长接收  DMA发送 手机看文章 扫描二维码
随时随地手机看文章

1、通常的实现方式介绍

  1. 环形缓冲区+定时器超时中断的方式

    • 设备任务比较繁重时,使用中断接收可能会丢失数据。尤其是在长时间关闭中断或者串口中断优先级不高时

    • 频繁进出中断。在使用 RTOS 的系统中,每收到一个数据就会进行一次任务到中断的切换和中断到任务的切换

    • 环形缓冲区可以接收多帧数据

    • 数据帧超时间隔可以设置

    • 优点

    • 缺点

  2. 使用串口接收空闲中断+DMA 的方式

  • 空闲中断的时间对于同一个波特率来说是固定的,但某些时候 1 个字节的接收时间太短,不能作为数据帧接收完成的标志

  • 不会频繁在任务和中断之间切换,效率会更高

  • 一般不会丢失数据

  • 优点

  • 缺点

2、接收超时中断的相关内容

GD32F4 系列的单片机串口除了空闲中断外,还有可配置时间的接收超时中断(STM32F4 系列没有此中断、STM32L4 系列有),使能配置在USART_CTL3寄存器的 RTIE ,如下图

图片

USART_CTL3寄存器

接收超时标志在USART_STAT1寄存器的 RTF ,如下图

图片

USART_STAT1寄存器

超时时间在USART_RT寄存器中**RT[23:0]**配置,如下图

图片

USART_RT寄存器


其中RT为24位,单位是波特率的位时间,即 bps。举个例子,如果串口的参数配置位 8-N-1(一个开始位、8 个数据位、没有奇偶校验位、一个停止位),即一个字节的传输需要 10 个波特率的比特位,RT 设置为 100,则表示 10(100/10)个字节的传输超时时间。


3、接收超时中断+DMA 实现

示例中用到了 串口 2 、DMA0的 通道 1 (串口 2 的 DMA 接收)和 通道 3 (串口 2 的 DMA 发送),串口 2 的Tx为 PB10 、Rx为 PB11 。


串口接收数据缓冲区

#define BLE_UART USART2      ///< 串口2

#define RX_SERIAL_BUF_SIZE 256    ///< 串口2的接收缓冲区大小

static char recv_buf[RX_SERIAL_BUF_SIZE]; ///< receive buffer

static uint8_t uart2_rx_state = 0;   ///< 串口接收完成标志。1表示接收完成

static uint8_t uart2_tx_state = 0;   ///< 串口DMA发送完成标志。1表示发送完成

static uint16_t uart2_rx_len = 0;   ///< 串口实际接收的数据长度

串口中断处理函数

/**

  * @brief uart2的中断处理函数

  *  只关心接收超时中断

  *

  * @retval void

  *

  * @note

  */

void USART2_IRQHandler(void)

{

 /* UART接收超时中断 */

 if ((usart_interrupt_flag_get(BLE_UART, USART_INT_FLAG_RT) != RESET) &&

         (usart_flag_get(BLE_UART, USART_FLAG_RT) != RESET))

 {

  /* disable DMA and reconfigure */

  dma_channel_disable(DMA0, DMA_CH1); //关闭DMA,在没有读取该接收帧数据之前禁止DMA再接收数据

  dma_flag_clear(DMA0, DMA_CH1, DMA_FLAG_FTF);  // 清除DMA传输完成标志位


  /* Clear receiver timeout flag */

//  usart_flag_clear(BLE_UART, USART_FLAG_RT);

  usart_interrupt_flag_clear(BLE_UART,USART_INT_FLAG_RT);

  usart_data_receive(BLE_UART); /* 清除接收完成标志位 */


  // 设置接收的数据长度

  uart2_rx_len = get_uart2_dma_recv_data_size();


  /* 接收超时后,说明一帧数据接收完毕,置接收完成标志 */

  uart2_rx_state = 1;

 }

}

DMA0_Channel1 传输完成中断(用于串口的接收完成),正常情况下,此中断不会发生。

/**

  * @brief DMA0_Channel1传输完成中断

  *  用于串口DMA接收

  *

  * @retval void

  *

  * @note 因用到了串口的接收超时中断方式,正常情况下,串口的DMA接收完成不会发生

  */

void DMA0_Channel1_IRQHandler(void)

{

 if(dma_interrupt_flag_get(DMA0, DMA_CH1, DMA_INT_FLAG_FTF))

 {

     dma_interrupt_flag_clear(DMA0, DMA_CH1, DMA_INT_FLAG_FTF);

 // uart2_rx_state = 1;

  dma_channel_disable(DMA0, DMA_CH1);  // 关闭DMA接收传输

 }

}

DMA0_Channel3 传输完成中断(用于串口的发送完成)

/**

  * @brief DMA0_Channel3传输完成中断

  *  用于BLE模块的串口DMA发送

  *

  * @retval void

  *

  * @note

  */

void DMA0_Channel3_IRQHandler(void)

{

 if(dma_interrupt_flag_get(DMA0, DMA_CH3, DMA_INT_FLAG_FTF))

 {

     dma_interrupt_flag_clear(DMA0, DMA_CH3, DMA_INT_FLAG_FTF);

  uart2_tx_state = 1;

  dma_channel_disable(DMA0, DMA_CH3);  // 关闭DMA发送传输

    }

}

获取 uart2 串口 DMA 接收的数据长度

/**

  * @brief 获取uart2串口DMA接收的数据长度.

  *

  * @retval void

  *

  * @note

  */

static unsigned int get_uart2_dma_recv_data_size(void)

{

    /*

    dma_transfer_number_get(DMA_CH2);是获取当前指针计数值,

    用内存缓冲区大小 - 此计数值 = 接收到的数据长度(这里单位为字节)。

    需要说明下在读取数据长度的时候需要先把接收DMA关闭,读取完了或者是数据处理完了在打开接收DMA,防止在处理的过程中有数据到来而出错。

    */

    return (RT_SERIAL_RB_BUFSZ - (dma_transfer_number_get(DMA0, DMA_CH1)));

}

uart2 串口初始化

/**

  * @brief uart2串口初始化.

  *   串口接收通过DMA+接收超时中断实现,设置的超时时间为100个bps

  *

  * @param baudrate 串口波特率

  *

  * @retval void

  *

  * @note

  */

static void uart2_init(uint32_t baudrate)

{

 /*uart dma rx and tx set*/

 dma_single_data_parameter_struct dma_init_uart;


 /***************************** 配置uart2的gpio *****************************/

 /* enable GPIO clock */

 rcu_periph_clock_enable(RCU_GPIOB);

 /* connect port to USARTx_Tx */

 gpio_af_set(GPIOB, GPIO_AF_7, GPIO_PIN_10);

 /* connect port to USARTx_Rx */

 gpio_af_set(GPIOB, GPIO_AF_7, GPIO_PIN_11);

 /* configure USART Tx as alternate function push-pull */

 gpio_mode_set(GPIOB, GPIO_MODE_AF, GPIO_PUPD_PULLUP, GPIO_PIN_10);

 gpio_output_options_set(GPIOB, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_10);

 /* configure USART Rx as alternate function push-pull */

 gpio_mode_set(GPIOB, GPIO_MODE_AF, GPIO_PUPD_PULLUP, GPIO_PIN_11);

 gpio_output_options_set(GPIOB, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_11);


 /***************************** 配置uart2的参数 *****************************/

 /* enable USART clock */

 rcu_periph_clock_enable(RCU_USART2);

 /* USART configure */

 usart_deinit(BLE_UART);

 usart_oversample_config(BLE_UART, USART_OVSMOD_8);

 usart_baudrate_set(BLE_UART, baudrate); // 波特率

 usart_parity_config(BLE_UART, USART_PM_NONE); // 校验位:NONE

 usart_word_length_set(BLE_UART, USART_WL_8BIT); // 数据位:8

 usart_stop_bit_set(BLE_UART, USART_STB_1BIT); // 停止位:1

 usart_receive_config(BLE_UART, USART_RECEIVE_ENABLE); // 打开串口接收功能

 usart_transmit_config(BLE_UART, USART_TRANSMIT_ENABLE); // 打开串口发送功能

 // 接收超时设置,100个波特率的比特位

 usart_receiver_timeout_threshold_config(BLE_UART, 100);

 usart_interrupt_enable(BLE_UART, USART_INT_RT);

 usart_receiver_timeout_enable(BLE_UART);

 /* USART interrupt configuration */

 nvic_irq_enable(USART2_IRQn, 0, 1);

 usart_enable(BLE_UART);

 usart_dma_receive_config(BLE_UART, USART_DENR_ENABLE); // 使能DMA接收功能

 usart_dma_transmit_config(BLE_UART, USART_DENT_ENABLE); // 使能DMA发送功能


 /***************************** 配置uart2的DMA接收 ****************************/

 /* enable DMA0 */

 rcu_periph_clock_enable(RCU_DMA0);

 /* deinitialize DMA channel */

 dma_deinit(DMA0, DMA_CH1);

 dma_init_uart.direction = DMA_PERIPH_TO_MEMORY;

 dma_init_uart.memory0_addr = (uint32_t)(recv_buf); // 存储器地址

 dma_init_uart.memory_inc = DMA_MEMORY_INCREASE_ENABLE;

 dma_init_uart.periph_memory_width = DMA_PERIPH_WIDTH_8BIT;

 dma_init_uart.number = sizeof(recv_buf);

 dma_init_uart.periph_addr = (uint32_t)&USART_DATA(BLE_UART);

 dma_init_uart.periph_inc = DMA_PERIPH_INCREASE_DISABLE;

 dma_init_uart.priority = DMA_PRIORITY_ULTRA_HIGH;

 dma_init_uart.circular_mode = DMA_CIRCULAR_MODE_DISABLE;

 dma_single_data_mode_init(DMA0, DMA_CH1, &dma_init_uart);

 dma_channel_subperipheral_select(DMA0, DMA_CH1, DMA_SUBPERI4);


    uart2_rx_state = 0;

    uart2_rx_len = 0;


 //使能通道

 dma_channel_enable(DMA0, DMA_CH1);


 /***************************** 配置uart2的DMA发送 ***************************/

 /* deinitialize DMA channel */

 dma_deinit(DMA0, DMA_CH3);

 dma_init_uart.direction = DMA_MEMORY_TO_PERIPH;

 dma_init_uart.memory0_addr = RT_NULL;  // 内存基地址

 dma_init_uart.number = 0;  // len个数据

 dma_single_data_mode_init(DMA0, DMA_CH3, &dma_init_uart);

 dma_channel_subperipheral_select(DMA0, DMA_CH3, DMA_SUBPERI4);


// nvic_irq_enable(DMA0_Channel3_IRQn, 0, 2);


 uart2_tx_state = 0;


 return;

}

重新配置 uart2 串口的 DMA 接收通道

/**

  * @brief 重新配置uart2串口的DMA接收通道

  *

  * @retval void

  *

  * @note

  */

static void uart2_dma_rx_refcg(void)

{

 /* disable DMA and reconfigure */

 dma_channel_disable(DMA0, DMA_CH1); //关闭DMA,在没有读取该接收帧数据之前禁止DMA再接收数据

// DMA_INTC0(DMA0) |= DMA_FLAG_ADD(DMA_CHINTF_RESET_VALUE, DMA_CH1);


 dma_memory_address_config(DMA0, DMA_CH1, DMA_MEMORY_0, (uint32_t)(recv_buf)); // 存储器地址

 dma_transfer_number_config(DMA0, DMA_CH1, sizeof(recv_buf););


    uart2_rx_state = 0;

 uart2_rx_len = 0;


 // 使能通道

 dma_channel_enable(DMA0, DMA_CH1);

}

DMA 串口发送

串口发送使用 DMA 方式时,直接调用 uart2_sendData_DMA 函数即可。等待发送完成时,可以通过等待(uart2_tx_state == 1)实现,或者等待 DMA0 通道 3 的 DMA_FLAG_FTF 置位实现,或者使用 RTOS 的信号量实现

/**

  * @brief BLE模块的DMA串口发送.

  *

  *

  * @param data 发送数据缓冲区地址

  * @param len 发送数据长度

  *

  * @retval void

  *

  * @note

  */

void uart2_sendData_DMA(uint8_t *data, uint32_t len)

{

 /* disable DMA and reconfigure */

 dma_channel_disable(DMA0, DMA_CH3);

 dma_flag_clear(DMA0, DMA_CH3, DMA_FLAG_FTF);  // 清除DMA传输完成标志位


 dma_memory_address_config(DMA0, DMA_CH3, DMA_MEMORY_0, (uint32_t)(data)); // 存储器地址

 dma_transfer_number_config(DMA0, DMA_CH3, len);


// /* enable DMA0 channel3 transfer complete interrupt */

// dma_interrupt_enable(DMA0, DMA_CH3, DMA_CHXCTL_FTFIE);

// uart2_tx_state = 0;

 dma_channel_enable(DMA0, DMA_CH3);  // 使能DMA传输


 // 等待传输完成

 while(dma_flag_get(DMA0, DMA_CH3, DMA_FLAG_FTF) == RESET)

 {}


// // 等待传输完成

// while(uart2_tx_state == 0);


// printf('uart2_sendData_DMA complete');

}

接收处理任务

/**

  * @brief  main函数

  * @param  argc

  * @param  argv

  * @note   等待接收数据完成,然后做相应的处理

  * @retval None

  */

int main(char argc, char *argv[])

{

 uart2_init(115200);


    while (1)

    {

        // 接收完成一帧数据

        if(uart2_rx_state == 1)

        {

            // 数据处理

            ... ...


            // 处理完成后,重新启动串口的DMA接收

            uart2_dma_rx_refcg();

        }


        // 其他处理

        ... ...

    }

}


关键字:单片机  DMA  串口  不定长接收  DMA发送 引用地址:GD32F4单片机实现接收超时中断+DMA实现串口的不定长接收和DMA发送

上一篇:【GD32F470紫藤派开发板使用手册】第二讲 GPIO-按键查询实验
下一篇:【兆易创新GD32VF103R-START开发板试用体验】步进电机驱动

推荐阅读最新更新时间:2026-03-19 20:30

GD32F4单片机实现接收超时中断+DMA实现串口的不定长接收DMA发送
1、通常的实现方式介绍 环形缓冲区+定时器超时中断的方式 设备任务比较繁重时,使用中断接收可能会丢失数据。尤其是在长时间关闭中断或者串口中断优先级不高时 频繁进出中断。在使用 RTOS 的系统中,每收到一个数据就会进行一次任务到中断的切换和中断到任务的切换 环形缓冲区可以接收多帧数据 数据帧超时间隔可以设置 优点 缺点 使用串口接收空闲中断+DMA 的方式 空闲中断的时间对于同一个波特率来说是固定的,但某些时候 1 个字节的接收时间太短,不能作为数据帧接收完成的标志 不会频繁在任务和中断之间切换,效率会更高 一般不会丢失数据 优点 缺点 2、接收超时中断的相关内容 GD32F4 系列的单片机串口除
[单片机]
<font color='red'>GD32F4</font><font color='red'>单片机</font>实现<font color='red'>接收</font><font color='red'>超时</font><font color='red'>中断</font>+<font color='red'>DMA</font>实现<font color='red'>串口</font>的不<font color='red'>定长</font><font color='red'>接收</font>和<font color='red'>DMA</font><font color='red'>发送</font>
#C51串口通讯2-#一串数据#定时中断实现超时接收
一.场景 实现一串非固定长度的数据接收,并返回对应数据(数据解析的基础框架) 二.编程实现 1.设计思想 借助T0定时器,不断的计数+1 接收到一帧数据(1Byte)后,串口中断服务函数将定时器T0计数清0(类似喂狗),并创建一个计数标志 一帧数据(1Byte)长度约为1.04ms 。当串口中断数据接收完毕后,短时间无有效数据接收并进入中断服务函数,此时T0计数器不被清0,不断累加 持续检测到大于固定时间时,认为此刻一串数据已传输完毕。 固定时间间隔一般设置3-5倍的一帧数据长度(1.04ms) 2.代码设计 主函数暂时处理为返回接收到的字符串 UART中断服务函数处理: 接收到一个字节,打开T0计数软件标志,清一次计数器
[单片机]
#C51<font color='red'>串口</font>通讯2-#一串数据#定时<font color='red'>中断</font>实现<font color='red'>超时</font><font color='red'>接收</font>
STM32的串口采用DMA方式发送数据测试
环境: 主机:WIN7 开发环境:MDK4.23 MCU:STM32F103CBT6 源代码: 配置: //---------------------串口功能配置--------------------- //打开串口对应的外设时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE); //启动DMA时钟 RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //DMA发送中断设置 NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_IRQn;
[单片机]
USART 串口 DMA 发送接收
串口DMA发送: 发送数据的流程: 前台程序中有数据要发送,则需要做如下几件事 1. 在数据发送缓冲区内放好要发送的数据,说明:此数据缓冲区的首地址必须要在DMA初始化的时候写入到DMA配置中去。 2. 将数据缓冲区内要发送的数据字节数赋值给发送DMA通道,(串口发送DMA和串口接收DAM不是同一个DMA通道) 3. 开启DMA,一旦开启,则DMA开始发送数据,说明一下:在KEIL调试好的时候,DMA和调试是不同步的,即不管Keil 是什么状态,DMA总是发送数据。 4. 等待发送完成标志位,即下面的终端服务函数中的第3点设置的标志位。或者根据自己的实际情况来定,是否要一直等待这个标志位,也可以通过状态机的
[单片机]
stm32串口HAL库的DMA发送问题
本文使用stm32f411ret的串口1的DMA方式发送数据,刚开始调试的时候发现串口只能发送一次数据,之后就把系统hang住了。通过网上搜资料和不断尝试,发现问题是中断回调函数没有写的原因。 使用HAL库的DMA,需要同时实现DMA中断回调函数和串口中断回调函数。 void DMA2_Stream7_IRQHandler(void) { HAL_DMA_IRQHandler(Uart1Handle.hdmatx); } void USART1_IRQHandler(void) { HAL_NVIC_ClearPendingIRQ(USART1_IRQn); HAL_UART_IRQHandler(&Uart1
[单片机]
STM32CubeMX学习笔记6:按键控制DMA串口发送
MCU:STM32F103ZET6 IDE: MDK-ARM V5 +STM32CubeMX5.0.0 串口调试助手:SSCOM3.2 功能描述:通过KEY_UP按键控制DMA串口1数据的传送。 需要配置DMA,串口USART1,使能按键中断,LED提示灯。 一. 在 Pinout&Configuration---System Core中: 1. 首先设置时钟RCC的HSE(外部高速时钟)为晶振模式:Crystal/ceramic Resonator 2. 设置系统SYS的Debug为Serial Wire: 3. 设置GPIO中的LED管脚。 在MCU管脚图中找到PC0和PC1管脚(查原理图对应
[单片机]
STM32CubeMX学习笔记6:按键控制<font color='red'>DMA</font><font color='red'>串口</font><font color='red'>发送</font>
STM32F103之DMA实验,内存通过DMA串口1发送数据
#include dma.h ////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////// DMA_InitTypeDef DMA_InitStructure; u16 DMA1_MEM_LEN;//保存DMA每次数据传送的长度 //DMA1的各通道配置 //这里的传输形式是固定的,这点要根据不同的情况来修改 //从存储器- 外设模式
[单片机]
串口使用DMA发送数据时的数据覆盖问题
在STM32等单片机中,为了提高程序的执行效率,在使用串口时,经常会使用DMA的方式来进行收发数据,这样,CPU只需要把数据放在发送缓存区即可离开。 但是,由于在串口发送中使用的是引用调用,即数据传递时是将要发送的指针进行传递,而非将数据复制了一份。这样的好处是方便快捷,但同时带来的一个后果时,当该数据还未发送完时,下一串数据就不能放入该发送缓冲区,否则,就会发生数据覆盖问题,即后一串数据覆盖前边还未发完的数据。 一般发生这种数据覆盖问题,都是在连续发送两串以上数据的时候,第一次的还没发完,后边的就会将前一次的数据覆盖。 若是在每次发送之前进行判断上次数据是否发完,如果没有发完就等待的话,就会降低CPU的效率;或者,如
[单片机]
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved