使用环形缓冲区ringbuffer实现串口数据接收_uart ringbuffer-程序员宅基地

技术标签: STM32  串口  队列  数据结构  

1. ringbuffer简单介绍

环形缓冲区(ringbuffer),实际上就是一种队列数据结构,只不过它不是线性队列,而是环形队列。

关于环形缓冲区(ringbuffer)的详细介绍,网上一搜一大把,这里不重复介绍了,我这里直接上代码。

详细介绍可以参考下面链接里面的介绍:

2. ringbuffer的代码实现

实现环形缓冲区的形式有使用数组的,也可以使用链表。我这里为了实现简单,就用数组作为 ringbuffer 的内存来实现。

在实现 ringbuffer 时,要有两个指针,读指针和写指针。每当向 ringbuffer 中写入一个数据时,写指针加1;同理从 ringbuffer 中读取一个数据时,读指针加1。

对于 ringbuffer 的读写操作,我们有几个重点问题需要考虑:

  • 读写指针移动到 ringbuffer 的最大长度之后,如何返回首位置?

    对于 ringbuffer 的读写指针位置的计算,精髓就在于对读写指针进行取模运算。即当读写指针移动一个位置时,然后对 ringbuffer 的大小进行取模运算,这样当读写指针移动到最末尾时,取模运算的结果就是 0,即返回的 ringbuffer 的首位置了。代码表示如下:

    write_index     : 当前写位置
    read_index      : 当前读位置
    ringbuffer_size : ringbuffer 缓冲区的大小
    
    /* 读写指针每移动一个位置,都对 ringbuffer 大小进行取模运算 */
    write_index = (write_index + 1) % ringbuffer_size	
    read_index = (read_index + 1) % ringbuffer_size
    
  • 如何判断 ringbuffer 为空?

    读写指针的位置相等时,说明 ringbuffer 为空。

    write_index == read_index
    
  • 如何判断 ringbuffer 为满?

    当写指针的下一个位置等于读指针的位置时,那么 ringbuffer 为满。

    (write_index + 1) % ringbuffer_size == read_index
    

2.1 ringbuffer数据结构定义

ringbuffer 的数据结构封装如下,主要成员有读写指针,还有指向用户提供 buffer 的指针和 buffer 的大小。

其中,读写指针的这两个成员,很可能会因为外部一些原因(比如串口中断)造成读写位置的变化,而这个变化编译器很可能不知道,所以为了防止编译器优化而加上 volatile 关键字修饰。

typedef struct _ringbuffer_t
{
    
	volatile unsigned int read_index;           /* 当前读位置 */
	volatile unsigned int write_index;          /* 当前写位置 */  
	unsigned int buffer_size;					/* ringbuffer大小 */
    unsigned char *buffer_ptr;  				/* 指向ringbuffer */    
} ringbuffer_t;

2.2 ringbuffer初始化

/*
 * 函数作用 : 初始化ringbuffer结构体(句柄)
 * 参数  rb   : 指向ringbuffer句柄
 * 参数  pool : 指向ringbuffer缓冲区,用户调用时一般提供一个数组
 * 参数  size : 缓冲区的大小 
 * 返回值 : 无
 */
void ringbuffer_init(ringbuffer_t *rb, unsigned char *pool, unsigned int size)
{
    
    /* initialize read and write index */
    rb->read_index = 0;
    rb->write_index = 0;

    /* set buffer pool and size */
    rb->buffer_ptr = pool;
    rb->buffer_size = size;
}

主要是初始化 ringbuffer_t 结构体成员。用户需要提供一个定义好的数组变量,传递到这个初始化函数中,从而使得 buffer_ptr 这个指向具体 buffer 的成员指向用户提供的一个数组。

2.3 ringbuffer写数据

前面已经介绍了,读写指针移动运算的精髓就在于,对 ringbuffer 的大小进行取模运算。

另外,当写指针的下一个位置与当前读位置相等时,说明 ringbuffer 已经满了,这个时候就不再继续向环形缓冲区写入数据了。

代码实现如下:

/*
 * 函数作用 : 向目标缓冲区写入一个字节数据
 * 参数  ch : 要写入ringbuffer的数据
 * 参数  rb : 指向ringbuffer句柄
 * 返回值   : 写入成功返回0,失败返回-1
 */
int ringbuffer_write(unsigned char ch, ringbuffer_t *rb)
{
    
    if (rb->read_index == ((rb->write_index + 1) % rb->buffer_size))
    {
    
        return -1;
    }
    else
    {
    
        rb->buffer_ptr[rb->write_index] = ch;
        rb->write_index = (rb->write_index + 1) % rb->buffer_size;
        return 0;
    }
}

2.4 ringbuffer读数据

当读写指针相等时,ringbuffer 为空。具体代码实现如下:

/*
 * 函数作用 : 向目标缓冲区读取一个字节数据
 * 参数  ch : 把读取到的数据保存到ch所指向的内存
 * 参数  rb : 指向ringbuffer句柄
 * 返回值   : 读取成功返回0,失败返回-1
 */
int ringbuffer_read(unsigned char *ch, ringbuffer_t *rb)
{
    
    if (rb->read_index == rb->write_index)
    {
    
        return -1;
    }
    else
    {
    
        *ch = rb->buffer_ptr[rb->read_index];
        rb->read_index = (rb->read_index + 1) % rb->buffer_size;
        return 0;
    }
}

3. 在串口中使用ringbuffer

3.1 为什么需要ringbuffer接收串口数据

串口中断接收数据时,每接收到一个字节数据就会触发一次中断,然后我们再把这一字节的数据交给上一层的程序进行处理。很多时候,如果我们接收到一个字节数据就处理一下,太过于频繁。有时也可能因为数据量太大,或者接收数据太快,而上层代码来不及处理数据,等到下一次接收的数据来到时,很可能会覆盖掉没来得及处理的数据。这是就会出现丢包的现象。

为了防止丢包,我们可以在中断中暂时先把接收到的数据放到一个缓冲区里面,等到CPU去处理时,一次性就把所有的数据都取出来进行处理。而对于这种对数据的读和写的过程,使用环形缓冲区是非常适合的。

3.2 初始化串口和ringbuffer

使用串口接收数据,先对串口进行初始化,以及对 ringbuffer 进行初始化。

static unsigned char uart_rx_buffer[16];  // 环形缓冲区所指向的数组
static ringbuffer_t uart_rx_ringbuffer;   // 环形缓冲区句柄
UART_HandleTypeDef huart1;

void MX_USART1_UART_Init(void)
{
    
    /* 初始化ringbuffer,使得ringbuffer指向用户提供的数组 */
    ringbuffer_init(&uart_rx_ringbuffer, uart_rx_buffer, sizeof(uart_rx_buffer));
	
    /* 串口参数配置 */
    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;
    
    /* 串口引脚初始化 */
    if (HAL_UART_Init(&huart1) != HAL_OK)
    {
    
        Error_Handler();
    }
    
    /* 串口中断配置 */
    HAL_NVIC_SetPriority(USART1_IRQn, 4, 0);		// 中断优先级配置
    HAL_NVIC_EnableIRQ(USART1_IRQn);				// 使能串口1中断
    __HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE);	// 使能串口1接收中断
}

3.3 串口中断接收数据

在串口中断中,把接收到的数据保存到我们刚刚定义的 ringbuffer 中。

void USART1_IRQHandler(void)
{
    
    int ch = -1;

    if ((__HAL_UART_GET_FLAG(&(huart1), UART_FLAG_RXNE) != RESET) &&
        (__HAL_UART_GET_IT_SOURCE(&(huart1), UART_IT_RXNE) != RESET))
    {
    
        while (1)
        {
    
            ch = -1;
            if (__HAL_UART_GET_FLAG(&(huart1), UART_FLAG_RXNE) != RESET)
            {
    
                ch =  huart1.Instance->DR & 0xff;
            }
            if (ch == -1)
            {
    
                break;
            }
            /* 中断接收到的数据,存入 ringbuffer */
            ringbuffer_write(ch, &uart_rx_ringbuffer);
        }
    }
}

4. 测试结果

4.1 测试是否丢包

使用串口助手,每隔 10ms 自动发送一次数据,而在 main 函数故意延时20ms再去把 ringbuffer 的数据读出来在发送到串口助手上。

我们前面的代码定义的 ringbuffer 的大小是 16 字节,在串口助手中,当我们每隔 10ms 发送 5 个字节时,main 函数延时 20ms 再接收。这时可以发现是没有出现丢包的现象的,实验结果如下:

在这里插入图片描述

4.2 补充测试

但是,如果我们一次性发送的数据大于 ringbuffer 的大小(16字节)时,那么就会出现丢包的现象了。

在这里插入图片描述

另外,如果每 10ms 一次性发送 10 个字节,由于 main 函数延时了 20ms 才去处理数据,如果长期堆积下去的话,这时也会造成数据丢包的现象。

所以说,如果我们一次性要接收的数据量太大,或者说处理的速度太慢,为了防止数据丢包现象,最好自己评估把 ringbuffer 的大小设置大一点。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/luobeihai/article/details/126548369

智能推荐

settext 下划线_Android TextView 添加下划线的几种方式-程序员宅基地

文章浏览阅读748次。总结起来大概有5种做法:将要处理的文字写到一个资源文件,如string.xml(使用html用法格式化)当文字中出现URL、E-mail、电话号码等的时候,可以将TextView的android:autoLink属性设置为相应的的值,如果是所有的类型都出来就是**android:autoLink="all",当然也可以在java代码里 做,textView01.setAutoLinkMask(Li..._qaction::settext 无法添加下划线

TableStore时序数据存储 - 架构篇_tablestore 时间类型处理-程序员宅基地

文章浏览阅读6.3k次,点赞2次,收藏10次。摘要: 背景 随着近几年物联网的发展,时序数据迎来了一个不小的爆发。从DB-Engines上近两年的数据库类型增长趋势来看,时序数据库的增长是非常迅猛的。在去年我花了比较长的时间去了解了一些开源时序数据库,写了一个系列的文章(综述、HBase系、Cassandra系、InfluxDB、Prometheus),感兴趣的可以浏览。背景随着近几年物联网的发展,时序数据迎来了一个不小的爆发。从DB..._tablestore 时间类型处理

Ubuntu20.04下成功运行VINS-mono_uabntu20.04安装vins-mono-程序员宅基地

文章浏览阅读5.7k次,点赞8次,收藏49次。可以编译成功但是运行时段错误查找原因应该是ROS noetic版本中自带的OpenCV4和VINS-mono中需要使用的OpenCV3冲突的问题。为了便于查找问题,我只先编译feature_tracker包。解决思路历程:o想着把OpenCV4相关的库移除掉,但是发现编译feature_tracker的时候仍然会关联到Opencv4的库,查找原因是因为cv_bridge是依赖opencv4的,这样导致同时使用了opencv3和opencv4,因此运行出现段错误。oo进一步想着(1)把vins-mon_uabntu20.04安装vins-mono

TMS320C6748_EMIF时钟配置_tms 6748-程序员宅基地

文章浏览阅读3.6k次,点赞3次,收藏12次。创龙TL6748开发板中,EMIFA模块使用默认的PLL0_SYSCLK3时钟,使用AISgen for D800K008工具加载C6748配置文件C6748AISgen_456M_config(Configuration files,在TL_TMS6748/images文件夹下),由图可以看到DIV3等于4,注意这里的DIV3就是实际的分频值(x),而不是写入相应PLL寄存器的值(x-1)。_tms 6748

eigen稀疏矩阵拼接(基于块操作的二维拼接)的思考-程序员宅基地

文章浏览阅读5.9k次,点赞4次,收藏13次。转载请说明出处:eigen稀疏矩阵拼接(块操作)eigen稀疏矩阵拼接(块操作)关于稀疏矩阵的块操作:参考官方链接 However, for performance reasons, writing to a sub-sparse-matrix is much more limited, and currently only contiguous sets of columns..._稀疏矩阵拼接

基于Capon和信号子空间的变形算法实现波束形成附matlab代码-程序员宅基地

文章浏览阅读946次,点赞19次,收藏19次。波束形成是天线阵列信号处理中的一项关键技术,它通过对来自不同方向的信号进行加权求和,来增强特定方向的信号并抑制其他方向的干扰。本文介绍了两种基于 Capon 和信号子空间的变形算法,即最小方差无失真响应 (MVDR) 算法和最小范数算法,用于实现波束形成。这些算法通过优化波束形成权重向量,来最小化波束形成输出的方差或范数,从而提高波束形成性能。引言波束形成在雷达、声纳、通信和医学成像等众多应用中至关重要。它可以增强目标信号,抑制干扰和噪声,提高系统性能。

随便推点

Ubuntu好用的软件推荐_ubuntu开发推荐软件-程序员宅基地

文章浏览阅读3.4w次。转自:http://www.linuxidc.com/Linux/2017-07/145335.htm使用Ubuntu开发已经有些时间了。写下这篇文章,希望记录下这一年的小小总结。使用Linux开发有很多坑,同时也有很多有趣的东西,可以编写一些自动化脚本,添加定时器,例如下班定时关机等自动化脚本,同时对于服务器不太了解的朋友,建议也可以拿台Linux来实践下,同时Ubuntu在Androi_ubuntu开发推荐软件

Nginx反向代理获取客户端真实IP_nginx获取到的是交换机的ip-程序员宅基地

文章浏览阅读2.2k次。一,问题 nginx反向代理后,在应用中取得的ip都是反向代理服务器的ip,取得的域名也是反向代理配置的url的域名,解决该问题,需要在nginx反向代理配置中添加一些配置信息,目的将客户端的真实ip和域名传递到应用程序中。二,解决 Nginx服务器增加转发配置 proxy_set_header Host $host;_nginx获取到的是交换机的ip

Wireshark TCP数据包跟踪 还原图片 WinHex应用_wireshark抓包还原图片-程序员宅基地

文章浏览阅读1.4k次。Wireshark TCP数据包跟踪 还原图片 WinHex简单应用 _wireshark抓包还原图片

Win8蓝屏(WHEA_UNCORRECTABLE_ERROR)-程序员宅基地

文章浏览阅读1.5k次。Win8下安装VS2012时,蓝屏,报错WHEA_UNCORRECTABLE_ERROR(P.S.新的BSOD挺有创意":("),Google之,发现[via]需要BIOS中禁用Intel C-State,有严重Bug的嫌疑哦原因有空再看看..._win8.1 whea_uncorrectable_error蓝屏代码

案例课1——科大讯飞_科大讯飞培训案例-程序员宅基地

文章浏览阅读919次,点赞21次,收藏22次。科大讯飞是一家专业从事智能语音及语音技术研究、软件及芯片产品开发、语音信息服务的软件企业,语音技术实现了人机语音交互,使人与机器之间沟通变得像人与人沟通一样简单。语音技术主要包括语音合成和语音识别两项关键技术。此外,语音技术还包括语音编码、音色转换、口语评测、语音消噪和增强等技术,有着广阔的应用。_科大讯飞培训案例

perl下载与安装教程【工具使用】-程序员宅基地

文章浏览阅读4.7k次。Perl是一个高阶程式语言,由 Larry Wall和其他许多人所写,融合了许多语言的特性。它主要是由无所不在的 C语言,其次由 sed、awk,UNIX shell 和至少十数种其他的工具和语言所演化而来。Perl对 process、档案,和文字有很强的处理、变换能力,ActivePerl是一个perl脚本解释器。其包含了包括有 Perl for Win32、Perl for ISAPI、PerlScript、Perl。_perl下载

推荐文章

热门文章

相关标签