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

news2025/1/18 20:28:12

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寄存器

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

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

​ 示例中用到了串口2DMA0通道1(串口2的DMA接收)和通道3(串口2的DMA发送),串口2的TxPB10RxPB11

  • 串口接收数据缓冲区

    #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();
            }
    		
            // 其他处理
            ... ...
        }
    }
    

公众号 | FunIO
微信搜一搜 “funio”,发现更多精彩内容。
个人博客 | blog.boringhex.top

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

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

相关文章

给第一行单元格赋值 + WPS JS获取工作表的总行数 + WPS JS获取工作表的总行数

戳我&#xff0c;了解更多相关办公的小技巧 给第一行单元格赋值 1、在计算机中有一种ASCII编码&#xff0c;其中A在计算机中的表示的数字是65&#xff0c;a的ascii码是97&#xff0c;b的ascii码是98。 2、从A1到F1可以看到第一个字母在变化&#xff0c;第2个数字始终是1&#x…

过电流保护原理

过电流保护是指当电流超过预定最⼤值时&#xff0c;使保护装置动作的⼀种保护⽅式。当流过被保护原件中的电流超过预先整定 的某个数值时&#xff0c;保护装置启动&#xff0c;并⽤时限保证动作的选择性&#xff0c;使断路器跳闸或给出报警信号。 过电流保护主要包括短路保护和…

北京通信展的精华内容,都在这里!(上篇)

友情提醒&#xff1a;本文图片较多&#xff0c;请大家注意手机流量。 大家好&#xff0c;我是小枣君。 昨天&#xff0c;中国国际信息通信展览会&#xff08;PT展&#xff09;在北京正式落幕了。 小枣君全程参加了这场行业盛会。按照惯例&#xff0c;我来给大家汇报一下现场的情…

ChatGPT Prompt 提示词设计技巧必知必会

本文内容整理自图灵社区直播《朱立成&#xff1a;ChatGPT Prompt提示词技巧必知必会》。 朱立成&#xff0c;图灵社区《ChatGPT即学即用》视频课程作者&#xff0c;软件工程师&#xff0c;对新事物充满好奇&#xff0c;关注ChatGPT应用。2001年毕业于浙江大学&#xff0c;从事软…

硬件设计电源系列文章-电路电源设计流程

文章目录 概要整体架构流程技术名词解释技术细节小结 概要 本文主要介绍硬件单板电源设计流程。 整体架构流程 提示&#xff1a;这里可以添加技术整体架构 主要分三部&#xff1a; 电源需求&#xff1a; 根据硬件总体方案&#xff0c;确定各芯片所需的供电电压&#xff1…

go-Context详解

Context详解 简介 官网 context go package context-blog Context是一个很特殊的接口&#xff0c;在go里面主要承担的责任是在边界&#xff08;方法&#xff0c;线程等&#xff09;传递上下文&#xff0c;这些上下文包括 取消信号超时时间特殊的参数 需要有几个注意点 …

Tcl常用语法备忘录-字符串篇

TCL语言中的string命令用于对字符串进行操作&#xff0c;常用的有以下几种用法&#xff1a; string length 语法&#xff1a;string length string 参数说明&#xff1a;string为要计算长度的字符串。 示例&#xff1a; set str "Hello TCL" puts [string lengt…

盘点一个Jupyter显示的细节问题

点击上方“Python爬虫与数据挖掘”&#xff0c;进行关注 回复“书籍”即可获赠Python从入门到进阶共10本电子书 今 日 鸡 汤 弦弦掩抑声声思&#xff0c;似诉平生不得志。 大家好&#xff0c;我是皮皮。 一、前言 前几天在Python白银群【小王子】问了一个Python基础的问题&…

北京通信展的精华内容,都在这里!(下篇)

█ 中国卫通 看上去像一笼汤圆&#xff0c;其实都是天线振子&#xff1a; 紧密围绕刚发射的中星26&#xff1a; 中国卫通的高通量卫星覆盖计划&#xff1a; 针对航空的卫星通信场景&#xff1a; 针对海运的场景&#xff1a; █ 中国电科 中国信科的东西&#xff0c;很多都是应急…

Python潮流周刊#6:Python 3.12 有我贡献的代码!

△点击上方“Python猫”关注 &#xff0c;回复“1”领取电子书 你好&#xff0c;我是猫哥。这里记录每周值得分享的 Python 及通用技术内容&#xff0c;部分为英文&#xff0c;已在小标题注明。&#xff08;标题取自其中一则分享&#xff0c;不代表全部内容都是该主题&#xff…

这是铁哥们

晚上在看读者给我的留言&#xff0c;有一篇文章里面挂了 300 多条留言&#xff0c;因为微信公众号一篇文章只能放出 100 条留言&#xff0c;剩余的就只能存在后台了。 然后我看到有读者的留言还被扔到了垃圾箱。扔垃圾箱这个事情应该是微信的一个默认设置&#xff0c;但是这样励…

【夜深人静算法介绍 | 第一篇】KMP算法

目录 前言&#xff1a; KMP算法简介&#xff1a; 引入概念&#xff1a; 前缀后缀 前缀表&#xff1a; 简单例子&#xff1a; 暴力遍历&#xff1a; KMP算法&#xff1a;​ KMP算法难点&#xff1a; 总结&#xff1a; 前言&#xff1a; 本篇我们将详细的从理论层面介绍一…

Java库Lombok常用注解使用

Lombok已经是很多Java项目最常用的库之一了&#xff0c;我也一直在用&#xff0c;但是仅限于Data、XxxConstructer、Slf4j之类的注解&#xff0c;没有看过其它的注解。 直到前段时间看到别人的代码&#xff0c;使用了一个SneakyThrows注解&#xff0c;搜索了一下&#xff0c;才…

华为OD机试真题 JavaScript 实现【数字涂色】【2022Q4 100分】,附详细解题思路

一、题目描述 疫情过后&#xff0c;希望小学终于又重新开学了&#xff0c;三年二班开学第一天的任务是将后面的黑板报重新制作。 黑板上已经写上了N个正整数&#xff0c;同学们需要给这每个数分别上一种颜色。 为了让黑板报既美观又有学习意义&#xff0c;老师要求同种颜色的…

粘包和半包的解决

粘包产生 public class HelloWordServer {static final Logger log LoggerFactory.getLogger(HelloWordServer.class);public static void main(String[] args) {NioEventLoopGroup boss new NioEventLoopGroup(1);NioEventLoopGroup worker new NioEventLoopGroup();try {…

Java实现微信公众号直接发送参数二维码给用户

文章目录 前言一、参数二维码的作用二、功能实现1. 生成带参数二维码2. 上传二维码图片3. 发送带参数二维码给用户 总结 前言 公众号开发近些年是一个比较热门的方向&#xff0c;今天为大家讲解的是用Java如何实现自动生成二维码图片&#xff0c;有如何把这个和用户信息单独绑…

Python3数据分析与挖掘建模(12)复合分析-相关分析与实现示例

1. 相关分析 1.1 概述 相关分析是一种统计分析方法&#xff0c;用于研究两个或多个变量之间的关系和相互影响程度。它帮助我们了解变量之间的线性关系、趋势和相关程度。 在相关分析中&#xff0c;常用的指标是相关系数&#xff0c;用于衡量两个变量之间的相关程度。最常见的…

linux 定时任务

可以用非root用户创建定时任务 Linux crontab 是用来定期执行程序的命令。 当安装完成操作系统之后&#xff0c;默认便会启动此任务调度命令。 crond 命令每分钟会定期检查是否有要执行的工作&#xff0c;如果有要执行的工作便会自动执行该工作。 注意&#xff1a;新创建的 cro…

怎么查询电脑的登录记录及密码更改情况?

源头是办公室公用的电脑莫名其妙打不开了&#xff0c;问别人也都不知道密码是多少 因为本来就没设密码啊&#xff01;&#xff08;躺倒&#xff09; 甚至已经想好了如果是50万想攻破电脑&#xff0c;被po抓住要怎么花这笔钱了 是我想太多 当然最后也没解决&#xff0c;莫名…

27 getcwd 的调试

前言 同样是一个 很常用的 glibc 库函数 不管是 用户业务代码 还是 很多类库的代码, 基本上都会用到 获取当前路径 不过 我们这里是从 具体的实现 来看一下 测试用例 就是简单的使用了一下 getcwd rootubuntu:~/Desktop/linux/HelloWorld# cat Test04Getcwd.c #inc…