STM32L4 串口通信(DMA+空闲中断方式)

2023-12-28 10:48

本文主要是介绍STM32L4 串口通信(DMA+空闲中断方式),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

芯片:STM32L452RE

 通过CubeMx生成工程文件,利用HAL库实现串口通信(DMA+空闲中断)任意长度的数据接收,

该程序不同于其它博客的写法,不用在主函数判断空闲中断再调用串口DMA接收函数,且解决第一次接收不到数据或数据不完整的情况。。

1.在STM32CubeMX里配置所需功能

1.1 时钟系统

 建议选择MSI作为时钟输入源,HSI反应有Bug(没有去尝试过,暂时省略)

 

1.2 设置串口中断和DMA

这里不做详细介绍,网上有大量教程可以参考

1.3 生成工程文件(keil5)

不熟悉的同学可以参考其它博客,有写得很详细的。

2. 程序编写

不用在主函数里调用DMA接收函数,且解决第一次接收不到数据或数据不完整的情况(不用环形缓冲区)。

2.1设计思路

首先开启串口中断接收函数,允许接收中断 HAL_UART_Receive_IT(&huart1, (uint8_t *)&Buftemp, 1);

当接收到第一个字节时产生中断(非空闲中断),此时的中断会调用HAL_UART_RxCpltCallback(),

在该函数里开启空闲中断和DMA接收,由于第一个中断已向缓冲区写入一个字节,此时的DMA接收区需要调整(若不设置,第一次接收的数据会丢失第一个字节);

在stm32l4xx_it.c里,编写中断服务程序。如果有空闲中断,编写空闲中断处理函数(清除标志位,停止DMA传输),再开启DMA接收

main.c

   主函数

unsigned char data[MAX_RCV_LEN];/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while (1){/* USER CODE END WHILE *//* USER CODE BEGIN 3 */printf("****start*****\n");HAL_Delay(1000);	GetRcvData(&huart1 ,data, sizeof(data));if(strlen(data)){printf("recv: %s \n",data);HAL_Delay(1000);}}/* USER CODE END 3 */

usart.h

/* Define to prevent recursive inclusion -------------------------------------*/
#ifndef __usart_H
#define __usart_H
#ifdef __cplusplusextern "C" {
#endif/* Includes ------------------------------------------------------------------*/
#include "stm32l4xx_hal.h"
#include "main.h"/* USER CODE BEGIN Includes */
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
/* USER CODE END Includes */extern UART_HandleTypeDef huart1;
extern UART_HandleTypeDef huart2;
extern UART_HandleTypeDef huart3;/* USER CODE BEGIN Private defines */
#define MAX_RCV_LEN 1024
extern uint8_t USART1RECV[MAX_RCV_LEN];	 //串口1
/* USER CODE END Private defines */extern void _Error_Handler(char *, int);void MX_USART1_UART_Init(void);
void MX_USART2_UART_Init(void);
void MX_USART3_UART_Init(void);/* USER CODE BEGIN Prototypes */
/* 清空*/
void USART_Clear(UART_HandleTypeDef *huart);extern void USART_IDLECallBack(void);uint16_t GetRcvNum(UART_HandleTypeDef *huart);
extern void GetRcvData(UART_HandleTypeDef *huart, uint8_t *buf, uint16_t rcv_len);
void USART_Write(UART_HandleTypeDef *huart, uint8_t *Data, uint16_t len);
/* USER CODE END Prototypes */#ifdef __cplusplus
}
#endif
#endif /*__ usart_H */

usart.c

/* USER CODE BEGIN 0 */
uint8_t Buftemp;uint16_t usart1_recv_len;
uint8_t USART1RECV[MAX_RCV_LEN];
/* USER CODE END 0 */

 在串口1的初始化程序里,开启串口接收中断

  HAL_UART_Receive_IT(&huart1, (uint8_t *)&Buftemp, 1);

void MX_USART1_UART_Init(void)
{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;huart1.Init.OverSampling = UART_OVERSAMPLING_16;huart1.Init.OneBitSampling = UART_ONE_BIT_SAMPLE_DISABLE;huart1.AdvancedInit.AdvFeatureInit = UART_ADVFEATURE_NO_INIT;if (HAL_UART_Init(&huart1) != HAL_OK){_Error_Handler(__FILE__, __LINE__);}HAL_UART_Receive_IT(&huart1, (uint8_t *)&Buftemp, 1);//使能第一次中断
}

 由第一次中断调用 HAL_UART_RxCpltCallback()函数,再开启空闲中断和DMA接收

/* USER CODE BEGIN 1 */
void USART_Clear(UART_HandleTypeDef *huart)
{if(huart->Instance == USART1){usart1_recv_len = 0;memset(USART1RECV, 0x0, sizeof(USART1RECV));}else if(huart->Instance == USART2){}
}void USART_IDLECallBack(void)
{unsigned int temp;__HAL_UART_CLEAR_IDLEFLAG(&huart1);//清除标志位temp = USART1->RDR;  //清除状态寄存器RDRtemp = temp;HAL_UART_DMAStop(&huart1); //	
}
void GetRcvData(UART_HandleTypeDef *huart, uint8_t *buf, uint16_t rcv_len)
{ if(huart->Instance == USART1){	if(buf){memcpy(buf,USART1RECV, rcv_len);}USART_Clear(&huart1);}else if(huart->Instance == USART2){}
}void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{UNUSED(huart);if(huart->Instance == USART1){//只有第一次中断会调用if(usart1_recv_len==0){USART1RECV[usart1_recv_len++]=Buftemp;//第一次中断的数据被写入USART1RECV[0]处__HAL_UART_CLEAR_IDLEFLAG(&huart1);//清除标志位__HAL_UART_ENABLE_IT(&huart1,UART_IT_IDLE);//使能空闲中断HAL_UART_Receive_DMA(&huart1, USART1RECV+1, MAX_RCV_LEN);	//设置第一次DMA接收缓冲区,}}
}

stm32l4xx_it.c 

判断是否为空闲中断,如果是,调用空闲中断处理函数USART_IDLECallBack(),如果需要处理串口收到的数据可以再该函数里进行;;

最后,开启DMA接收(不用调整接收缓冲区)

/**
* @brief This function handles USART1 global interrupt.
*/
void USART1_IRQHandler(void)
{/* USER CODE BEGIN USART1_IRQn 0 *//* USER CODE END USART1_IRQn 0 */HAL_UART_IRQHandler(&huart1);/* USER CODE BEGIN USART1_IRQn 1 */if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET){		USART_IDLECallBack();	HAL_UART_Receive_DMA(&huart1, USART1RECV, MAX_RCV_LEN);	}/* USER CODE END USART1_IRQn 1 */
}


  3 运行结果 

完美的情况

失败的情况

第一次的数据丢失第一个字节,后面的都正常

相关资源已上传

STM32L4系列 串口通信 空闲中断+DMA 实现任意长度的数据接收
包含两种方式 
1:不用在主函数调用DMA接收函数(推荐,非常实用)
2:许多教程的写法,需要在主函数调用判断空闲中断状态再调用DMA接收函数

https://download.csdn.net/download/sinat_37853238/10935731

这篇关于STM32L4 串口通信(DMA+空闲中断方式)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

内核启动时减少log的方式

内核引导选项 内核引导选项大体上可以分为两类:一类与设备无关、另一类与设备有关。与设备有关的引导选项多如牛毛,需要你自己阅读内核中的相应驱动程序源码以获取其能够接受的引导选项。比如,如果你想知道可以向 AHA1542 SCSI 驱动程序传递哪些引导选项,那么就查看 drivers/scsi/aha1542.c 文件,一般在前面 100 行注释里就可以找到所接受的引导选项说明。大多数选项是通过"_

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

深入理解RxJava:响应式编程的现代方式

在当今的软件开发世界中,异步编程和事件驱动的架构变得越来越重要。RxJava,作为响应式编程(Reactive Programming)的一个流行库,为Java和Android开发者提供了一种强大的方式来处理异步任务和事件流。本文将深入探讨RxJava的核心概念、优势以及如何在实际项目中应用它。 文章目录 💯 什么是RxJava?💯 响应式编程的优势💯 RxJava的核心概念

【即时通讯】轮询方式实现

技术栈 LayUI、jQuery实现前端效果。django4.2、django-ninja实现后端接口。 代码仓 - 后端 代码仓 - 前端 实现功能 首次访问页面并发送消息时需要设置昵称发送内容为空时要提示用户不能发送空消息前端定时获取消息,然后展示在页面上。 效果展示 首次发送需要设置昵称 发送消息与消息展示 提示用户不能发送空消息 后端接口 发送消息 DB = []@ro

脏页的标记方式详解

脏页的标记方式 一、引言 在数据库系统中,脏页是指那些被修改过但还未写入磁盘的数据页。为了有效地管理这些脏页并确保数据的一致性,数据库需要对脏页进行标记。了解脏页的标记方式对于理解数据库的内部工作机制和优化性能至关重要。 二、脏页产生的过程 当数据库中的数据被修改时,这些修改首先会在内存中的缓冲池(Buffer Pool)中进行。例如,执行一条 UPDATE 语句修改了某一行数据,对应的缓

vue2 组件通信

props + emits props:用于接收父组件传递给子组件的数据。可以定义期望从父组件接收的数据结构和类型。‘子组件不可更改该数据’emits:用于定义组件可以向父组件发出的事件。这允许父组件监听子组件的事件并作出响应。(比如数据更新) props检查属性 属性名类型描述默认值typeFunction指定 prop 应该是什么类型,如 String, Number, Boolean,

Java 多线程的基本方式

Java 多线程的基本方式 基础实现两种方式: 通过实现Callable 接口方式(可得到返回值):