串口持续地接收不定长、不定时的数据,把每一帧数据缓存下来且灵活地利用内存空间,下面提供一种方式供参考。原理是利用串口空闲中断和DMA,每当对方发来一帧完整的数据后,串口接收开始空闲,触发中断,在中断处理中新建一个接收队列节点,把DMA缓存的数据copy到接收队列里。当需要的时候就从接收队列里提出数据。定期清理队列防止堆空间溢出。

话不多说,上代码。

定义数据结构:

/*USART接收队列*/

typedef struct _USART_REC_Queue

{

    u16 index;                          //序号

    char *buf;                          //链接的字符串

    struct _USART_REC_Queue*  next;     //链接到下一个节点

}USART_REC_Queue;

声明全局变量:

#define USART3_REC_len 320                       //单次最大接收数

extern u8 USART3_REC_buf[USART3_REC_len];        //用于DMA的临时数据中转

extern u16 USART3_REC_counter;                   //接收计数器

extern USART_REC_Queue* USART3_REC_Queue_head;   //接收队列固定头节点

extern USART_REC_Queue* USART3_REC_Queue_tail;   //始终指向最后一个节点

准备阶段:

在启动汇编文件里,把堆空间改大,防止接收一点点数据就内存溢出。

Heap_Size       EQU     0x00004000    //默认200字节,改大

实例化全局变量:

u8 USART3_REC_buf[] = {}; 

u16 USART3_REC_counter = ;

USART_REC_Queue* USART3_REC_Queue_head = NULL;

USART_REC_Queue* USART3_REC_Queue_tail = NULL;

初始化各个硬件,使能了串口接收空闲中断,串口接收DMA,为接收队列头节点分配内存空间:

void USART3_Init(u32 BaudRate)

{

    //初始化参数结构体

    GPIO_InitTypeDef   GPIO_InitStruct;       //IO

    USART_InitTypeDef  USART_InitStruct;      //串口

    NVIC_InitTypeDef   NVIC_InitStruct;       //中断控制

    DMA_InitTypeDef    DMA_InitStruct;        //DMA

    /*全局指针初始化*/

    USART3_REC_Queue_head = USART_REC_Queue_Creat();  //构建串口3接收队列头节点

    USART3_REC_Queue_tail = USART3_REC_Queue_head;    //构建串口3接收队列尾节点

    //RCC使能

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);     //IO时钟

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE);    //串口3时钟

    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);        //DMA时钟

    //PB11 USART1_TXD

    GPIO_InitStruct.GPIO_Pin = GPIO_Pin_10;

    GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽输出

    GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOB, &GPIO_InitStruct);  

    //PB10 USART1_RXD

    GPIO_InitStruct.GPIO_Pin = GPIO_Pin_11;

    GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入

    //GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOB, &GPIO_InitStruct);

    //内嵌向量中断控制器初始化

    NVIC_InitStruct.NVIC_IRQChannel = USART3_IRQn;

    NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = ;//抢占优先级1

    NVIC_InitStruct.NVIC_IRQChannelSubPriority = ;//子优先级1

    NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;//使能IRQ通道

    NVIC_Init(&NVIC_InitStruct);

    //USART初始化

    USART_InitStruct.USART_BaudRate = BaudRate;//波特率 一般9600

    USART_InitStruct.USART_WordLength = USART_WordLength_8b;//字节数据格式8位

    USART_InitStruct.USART_StopBits = USART_StopBits_1;//一个停止位

    USART_InitStruct.USART_Parity = USART_Parity_No;//无奇偶字节校验

    USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件流控制

    USART_InitStruct.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;//收发模式

    USART_Init(USART3, &USART_InitStruct);//初始化USART

    //USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);//使能接收中断

    USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);//使能总线空闲中断

    USART_Cmd(USART3, ENABLE);//使能串口 

    DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)(&USART3->DR); //读取哪一个寄存器

    DMA_InitStruct.DMA_MemoryBaseAddr = (u32)(&USART3_REC_buf);      //读取到的数据的存放地址

    DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;                  //指定外设为源地址

    DMA_InitStruct.DMA_BufferSize = USART3_REC_len;                  //数据存放区大小

    DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;    //外设寄存器地址是否偏移

    DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;             //数据存放地址是否偏移

    DMA_InitStruct.DMA_PeripheralDataSize = DMA_MemoryDataSize_Byte; //外设数据宽度8位

    DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;     //定义存储器数据宽度8位

    DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;                       //正常操作模式

    DMA_InitStruct.DMA_Priority = DMA_Priority_High;                 //通道优先级

    DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;                        //是否开启存储器到存储器模式

    DMA_Init(DMA1_Channel3, &DMA_InitStruct);                        //写入设置到DMA1通道  

    DMA_Cmd(DMA1_Channel3, ENABLE);                                  //使能DMA1通道 

    USART_DMACmd(USART3, USART_DMAReq_Rx, ENABLE);                   //注意不要忘了使能串口的DMA功能

}

串口中断处理(核心):

void USART3_IRQHandler(void)

{ 

    if(USART_GetITStatus(USART3, USART_IT_IDLE) != RESET)

    {

        char *buf_new;                  //新字符串

        USART_REC_Queue* queue_new;     //新队列节点               

        u16 len;

        USART3->DR;                                       //读取数据。注意:这句必须要,否则不能够清除中断标志位

        USART_ClearITPendingBit(USART3, USART_IT_IDLE);                  //清中断

        DMA_Cmd(DMA1_Channel3, DISABLE);                                 //关闭DMA1通道3

        len = USART3_REC_len - DMA_GetCurrDataCounter(DMA1_Channel3);    //计算接收长度

        buf_new = (char *)malloc((len+) * sizeof(char));                //为新字符串分配内存,预留空间添加序号

        //if(buf_new == NULL)  GPIO_SetBits(LedPort, Led1);              //内存不够的提示

        queue_new = USART_REC_Queue_Creat();                             //为新队列节点分配内存

        //if(queue_new == NULL)  GPIO_SetBits(LedPort, Led2);            //内存不够的提示

        USART3_REC_counter ++;                                           //计数器加1

        queue_new->index = USART3_REC_counter;                           //新节点的序号

        sprintf(buf_new, "#%d:%s", USART3_REC_counter, USART3_REC_buf);  //复制缓存到新字符串并添加序号

        queue_new->buf = buf_new;                                        //新队列节点链接新字符串

        USART3_REC_Queue_tail->next = queue_new;                         //接收队列尾节点链接新的节点

        USART3_REC_Queue_tail = queue_new;                               //更新尾节点

        DMA1_Channel3->CNDTR = USART3_REC_len;                           //重置DMA1通道3缓存计数器

        DMA_Cmd(DMA1_Channel3, ENABLE);                                  //重开DMA1通道3

    }

}

创建与销毁接收队列节点:

USART_REC_Queue* USART_REC_Queue_Creat(void)

{

    USART_REC_Queue* p_temp = (USART_REC_Queue*)malloc(sizeof(USART_REC_Queue));

    if(p_temp == NULL)   return NULL;   

    memset(p_temp, , sizeof(USART_REC_Queue));

    //p_temp->next = NULL;

    return p_temp; 

}

void USART_REC_Queue_Delete(USART_TypeDef* USARTx)

{

    if(USARTx == USART3)

    {

        USART_REC_Queue* temp;

        temp = USART3_REC_Queue_head->next;

        if(temp == NULL || temp->next == NULL)   return;

        free(temp->buf);

        USART3_REC_Queue_head->next = temp->next;

        free(temp);

    } 

}

主函数里每隔5秒刷新显示接收队列的数据,并清理。

while()

{                        

    if(tim3_flag == )
{
tim3_flag = ; USART_REC_Queue_display(USART3); USART_REC_Queue_Delete(USART3);
} }

在OLED屏上显示接收队列里的数据:

void USART_REC_Queue_display(USART_TypeDef* USARTx)

{

    if(USARTx == USART3)

    {

        USART_REC_Queue* temp = USART3_REC_Queue_head->next;

        if(temp == NULL)   return;                                //队列还没生成则返回

        OLED_Clear();                                             //清屏

        OLED_ShowString(, , USART3_REC_Queue_head->next->buf);  //显示字符串

    }

}

Stm32使用串口空闲中断,基于队列来接收不定长、不定时数据的更多相关文章

  1. STM32串口接收不定长数据原理与源程序(转)

    今天说一下STM32单片机的接收不定长度字节数据的方法.由于STM32单片机带IDLE中断,所以利用这个中断,可以接收不定长字节的数据,由于STM32属于ARM单片机,所以这篇文章的方法也适合其他的A ...

  2. 串口配合DMA接收不定长数据(空闲中断+DMA接收)-(转载)

    1.空闲中断和别的接收完成(一个字节)中断,发送完成(发送寄存器控)中断的一样是串口中断: 2.空闲中断是接收到一个数据以后,接收停顿超过一字节时间  认为桢收完,总线空闲中断是在检测到在接收数据后, ...

  3. 串口1配合DMA接收不定长数据(空闲中断+DMA接收)

    1.空闲中断和别的接收完成(一个字节)中断,发送完成(发送寄存器控)中断的一样是串口中断: 2.空闲中断是接收到一个数据以后,接收停顿超过一字节时间  认为桢收完,总线空闲中断是在检测到在接收数据后, ...

  4. STM32 HAL库使用中断实现串口接收不定长数据

    以前用DMA实现接收不定长数据,DMA的方法接收串口助手的数据,全部没问题,不过如果接收模块返回的数据,而这些数据如果包含回车换行的话就会停止接收,例如接收:AT\r\nOK\r\n,就只能接收到AT ...

  5. STM32使用串口1配合DMA接收不定长数据,减轻CPU载荷

    STM32使用串口1配合DMA接收不定长数据,减轻CPU载荷 http://www.openedv.com/thread-63849-1-1.html 实现思路:采 用STM32F103的串口1,并配 ...

  6. STM32之串口DMA接收不定长数据

    STM32之串口DMA接收不定长数据 引言 在使用stm32或者其他单片机的时候,会经常使用到串口通讯,那么如何有效地接收数据呢?假如这段数据是不定长的有如何高效接收呢? 同学A:数据来了就会进入串口 ...

  7. 2-关于单片机通信数据传输(中断接收,大小端,IEEE754浮点型格式,共用体,空闲中断,环形队列)

    上一篇链接 http://www.cnblogs.com/yangfengwu/p/8628219.html 先说明一点这种方式,不光对于单片机类的,,对于上位机接收数据同样适用----不骗人的,自己 ...

  8. STM32 ~ USART接收不定长数据

    IDLE中断什么时候发生? IDLE就是串口收到一帧数据后,发生的中断.什么是一帧数据呢?比如说给单片机一次发来1个字节,或者一次发来8个字节,这些一次发来的数据,就称为一帧数据,也可以叫做一包数据. ...

  9. STM32串口空闲中断

    串口初始化 #include "usart5.h" vu16 UART5_RX_STA=0; char UART5_RX_BUF[UART5_REC_LEN]; u8 UART5_ ...

随机推荐

  1. 【02】对象的Getter and Setter

    java和C#非常相似,它们大部分的语法是一样的,但尽管如此,也有一些地方是不同的. 为了更好地学习java或C#,有必要分清它们两者到底在哪里不同. 我们这次要来探讨对象的Getter and Se ...

  2. 全网最通俗易懂的Kafka入门!

    前言 只有光头才能变强. 文本已收录至我的GitHub仓库,欢迎Star:https://github.com/ZhongFuCheng3y/3y 在这篇之前已经写过两篇基础文章了,强烈建议先去阅读: ...

  3. 【Android - 控件】之MD - FloatingActionButton的使用

    FloatingActionButton(FAB) 是 Android 5.0 新特性——Material Design 中的一个控件,是一种悬浮的按钮. FloatingActionButton 是 ...

  4. 源码安装php7.2

    `# 安装依赖包 yum install -y gcc gcc-c++ make zlib zlib-devel pcre pcre-devel \ libjpeg libjpeg-devel lib ...

  5. 文件上传之靶场upload-labs (1-10)

    第一关 sj 绕过 源码如下: lasIndexOf是返回函数最后一次出现的地方(从右到左) substring是用来截取函数的 indexOf是返回 表示从.出现的地方开始截取并判断是否在允许的字符 ...

  6. C#中的Stopwatch类简单使用

    Stopwatch实例可以度量一个间隔的运行时间, 或度量多个间隔内所用时间的总和. 命名空间System.Diagnostics. 简单使用 using System; using System.D ...

  7. Spring Boot 整合 Druid

    Spring Boot 整合 Druid 概述 Druid 是阿里巴巴开源平台上的一个项目,整个项目由数据库连接池.插件框架和 SQL 解析器组成.该项目主要是为了扩展 JDBC 的一些限制,可以让程 ...

  8. Chrome快捷键吐血整理

    Chrom是平时开发过程中最常用到的浏览器,使用快捷键操作Chrome能提高我们的使用效率,而且可以脱离鼠标进行操作.本篇博客就对平时我们常用的Chrome快捷键做一个整理总结.大家拿走不谢,哈哈~~ ...

  9. play-with-docker搭配ffsend完成文件上传及下载(解决从docker hub拉取镜像慢问题)

    由于众所周知的原因,大家有的时候pull docker hub上的镜像是很困难的,下载到99%就这么不动了也是很正常的事情 这个时候以下步骤是100%可以解决问题的: 1.找一台国外的服务器安装doc ...

  10. [TimLinux] CPU 常见架构介绍

    1. 简介 系统性能依赖硬件架构,CPU架构决定了硬件的布局.常见的CPU架构:SMP, NUMA, MPP. 2. SMP(对称多处理器) SMP:Symmetric Multiprocessing ...