stm32的rxne和idle中断_STM32使用串口IDLE中断的两种接收不定长数据的方式

本文介绍了两种利用STM32串口IDLE中断接收不定长数据的方法。方法1借助DMA在后台接收数据,数据帧接收完毕后产生中断,通过DMA_GetCurrDataCounter()计算数据长度。方法2直接使用RXNE和IDLE中断,根据中断次数判断数据帧结束。这两种方法适用于MODBUS、GPS数据解析等场景。
摘要由CSDN通过智能技术生成

现在有很多数据处理都要用到不定长数据,而单片机串口的RXNE中断一次只能接收一个字节的数据,没有缓冲区,无法接收一帧多个数据,现提供两种利用串口IDLE空闲中断的方式接收一帧数据,方法如下:

方法1:实现思路:采用STM32F103的串口1,并配置成空闲中断IDLE模式且使能DMA接收,并同时设置接收缓冲区和初始化DMA。那么初始化完成之后,当外部给单片机发送数据的时候,假设这帧数据长度是200个字节,那么在单片机接收到一个字节的时候并不会产生串口中断,而是DMA在后台把数据默默地搬运到你指定的缓冲区里面。当整帧数据发送完毕之后串口才会产生一次中断,此时可以利用DMA_GetCurrDataCounter();函数计算出本次的数据接受长度,从而进行数据处理。

应用对象:适用于各种串口相关的通信协议,如:MODBUS,PPI ;还有类似于GPS数据接收解析,串口WIFI的数据接收等,都是很好的应用对象。

关键代码分析:

usart.H

#ifndef __USART_H

#define __USART_H

#include "stdio.h"

#include "sys.h"

#define DMA_Rec_Len 200      //定义一个长度为200个字节的数据缓冲区。(建议定义的长度比你可能接收到的最长单帧数据长度长!)

void uart_init(u32 bound);

void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx);

#endif

usart.C

//初始化IO 串口1

//bound:波特率

void uart_init(u32 bound)

{

//GPIO端口设置

GPIO_InitTypeDef GPIO_InitStructure;

USART_InitTypeDef USART_InitStructure;

NVIC_InitTypeDef NVIC_InitStructure;

DMA_InitTypeDef DMA_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA,ENABLE); //使能USART1,GPIOA时钟

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输

RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);//使能USART2时钟

USART_DeInit(USART1);  //复位串口1

//USART1_TX   PA.9

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32中,您可以使用IDLE中断RXNE中断来实现接收不定数据。下面是一个简单的示例代码,演示了如何使用这两个中断接收串口数据: ```c #include "stm32f4xx.h" #define BUFFER_SIZE 256 uint8_t rxBuffer[BUFFER_SIZE]; uint8_t rxIndex = 0; uint8_t dataReceived = 0; void USART2_IRQHandler(void) { if (USART_GetITStatus(USART2, USART_IT_IDLE) != RESET) { // IDLE中断触发,表示接收完整一帧数据 USART_ReceiveData(USART2); // 清除IDLE标志位 dataReceived = 1; // 设置数据接收完成标志 } if (USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) { // RXNE中断触发,表示接收到一个字节的数据 rxBuffer[rxIndex++] = USART_ReceiveData(USART2); if (rxIndex >= BUFFER_SIZE) { // 缓冲区溢出,做相应处理 rxIndex = 0; } } } int main(void) { // 初始化串口和GPIO等 // 使能IDLE中断RXNE中断 USART_ITConfig(USART2, USART_IT_IDLE | USART_IT_RXNE, ENABLE); NVIC_EnableIRQ(USART2_IRQn); while (1) { if (dataReceived) { // 处理接收到的数据,例如打印到串口或进行其他操作 for (int i = 0; i < rxIndex; i++) { printf("%c", rxBuffer[i]); } // 清除接收缓冲区和相关标志位 memset(rxBuffer, 0, sizeof(rxBuffer)); rxIndex = 0; dataReceived = 0; } } } ``` 在这个示例中,我们使用USART2作为接收串口,并使能了IDLE中断RXNE中断。当接收数据时,会触发RXNE中断,将接收到的数据存储在缓冲区中。当接收完整一帧数据时,会触发IDLE中断,通过设置标志位来表示数据接收完成。在主循环中,我们检查数据接收完成标志,如果为真,则处理接收到的数据,并清除相关的缓冲区和标志位。 请根据实际需求修改代码中的串口、GPIO等初始化部分,并根据需要进行数据处理。希望对您有帮助!如果您还有其他问题,请继续提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值