嵌入式框架|环形缓冲区的实现

嵌入式大杂烩 2023-11-06 20:26

星标公众号」,一起进步!

来源:https://github.com/54zorb/Zorb-Framework

Zorb Framework是一个基于面向对象的思想来搭建一个轻量级的嵌入式框架。

本次分享的是Zorb Framework的环形缓冲区功能zf_buffer。

环形缓冲区主要应用在字节数据流传输上,如串口、网口的收发都可以通过环形缓冲区进行缓存。

例如我要通过串口发送命令“LED ON”来控制开发板的led灯亮起来,但开发板串口接收不是一次把“LED ON”同时接收,而是一个字节一个字节地接收,因此需要使用缓冲区来缓存数据,然后解析器来解析缓冲区的数据。

环形缓冲区设计

我们先来看看要实现的缓冲区长什么样子,提供什么功能,这样方便我们设计。

初步要提供的功能如下:

1、要有可以缓存数据的空间

2、可以知道总空间的大小

3、可以知道已用空间的数量

4、可以压入数据

5、可以弹出数据

6、我也可以在不弹出数据的情况下,读到特定长度的数据

因此,初步设计的数据结构如下:

/* 环形缓冲区数据结构 */
typedef struct _RingBuffer
{

    bool IsExternBuffer;  /* 是否外部缓冲区,是则销毁时不释放 */
    uint8_t *pBuf;        /* 缓冲区指针 */
    uint32_t Head;        /* 缓冲区头地址 */
    uint32_t Trail;       /* 缓冲区尾地址 */
    uint32_t Size;        /* 缓冲区大小 */
    uint32_t Count;       /* 数据字节数 */

    /* 缓冲器是否已满 */
    bool (*IsFull)(struct _RingBuffer * const pRb);

    /* 缓冲器是否空 */
    bool (*IsEmpty)(struct _RingBuffer * const pRb);

    /* 压入一个字节 */
    bool (*SaveByte)(struct _RingBuffer * const pRb, uint8_t byte);

    /* 取出一个字节 */
    bool (*GetByte)(struct _RingBuffer * const pRb, uint8_t *pByte);

    /* 读取缓冲器已使用字节个数 */
    uint32_t (*GetCount)(struct _RingBuffer * const pRb);

    /* 读取n个字节(n超过最大数据数时全部读出) */
    bool (*ReadBytes)(struct _RingBuffer * const pRb, uint8_t *pArray,
        uint32_t n);

    /* 丢弃n个字节(n超过最大数据数时全部丢弃) */
    bool (*DropBytes)(struct _RingBuffer * const pRb, uint32_t n);

    /* 清空缓冲器 */
    bool (*Clear)(struct _RingBuffer * const pRb);

    /* 释放缓冲器(不释放外部创建的缓冲区) */
    bool (*Dispose)(struct _RingBuffer * const pRb);
} RingBuffer;

其实按实际需要,可能远不止上面提到的6种情况,例如我可以丢弃特定数量的字节数据,也可以直接清空掉缓冲区数据,甚至可以设想提供动态缓冲区的功能,也就是说可以释放缓冲器自己。

创建环形缓冲器

/******************************************************************************
 * 描述  :创建环形缓冲器(内部分配空间,size=0表示使用外部数据)
 * 参数  :(out)-ppRb  环形缓冲器结构体指针的指针
 *         (in)-size   缓冲器大小
 *         (out)-ppBuf 缓冲器空间指针的指针
 * 返回  :无
******************************************************************************/

bool RB_create(RingBuffer **ppRb, uint32_t size)
{
    RingBuffer *pRb;
    uint8_t *pBuf;
    
    ZF_ASSERT(ppRb != (RingBuffer **)0)
    
    pRb = (RingBuffer *)ZF_MALLOC(sizeof(RingBuffer));
    if (pRb == NULL)
    {
        ZF_DEBUG(LOG_E, "malloc ringbuffer space error\r\n");
        return false;
    }
    
    pRb->Head = (uint32_t)0;
    pRb->Trail = (uint32_t)0;
    pRb->Count = 0;
    
    if (size > 0)
    {
        pBuf = (void *)ZF_MALLOC(size);
        if (pBuf == NULL)
        {
            ZF_DEBUG(LOG_E, "malloc ringbuffer buffer space error\r\n");
            return false;
        }
        
        pRb->pBuf = pBuf;
        pRb->Size = size;
        pRb->IsExternBuffer = false;
    }
    else
    {
        pRb->pBuf = NULL;
        pRb->Size = 0;
        pRb->IsExternBuffer = true;
    }
    
    /* 初始化方法 */
    pRb->IsFull = RB_isFull;
    pRb->IsEmpty = RB_isEmpty;
    pRb->SaveByte = RB_saveByte;
    pRb->SaveRange = RB_saveRange;
    pRb->GetByte = RB_getByte;
    pRb->GetCount = RB_getCount;
    pRb->ReadBytes = RB_readBytes;
    pRb->DropBytes = RB_dropBytes;
    pRb->Clear = RB_clear;
    pRb->Dispose = RB_dispose;
    
    /* 输出 */
    *ppRb = pRb;
    
    return true;
}

释放缓冲器

/******************************************************************************
 * 描述  :释放缓冲器(不释放外部创建的缓冲区)
 * 参数  :(in)-pRb     环形缓冲器结构体指针
 * 返回  :-true     成功
 *         -false    失败
******************************************************************************/

bool RB_dispose(RingBuffer * const pRb)
{
    ZF_ASSERT(pRb != (RingBuffer *)0)
    
    /* 外部创建的缓冲区不释放 */
    if (!pRb->IsExternBuffer)
    {
        ZF_FREE(pRb->pBuf);
    }
    
    ZF_FREE(pRb);
    
    return true;
}

压入一个字节

/******************************************************************************
 * 描述  :压入一个字节
 * 参数  :(in)-pRb  环形缓冲器结构体指针
 *         (in)-byte 要压入的字节
 * 返回  :-true     成功
 *         -false    失败
******************************************************************************/

bool RB_saveByte(RingBuffer * const pRb, uint8_t byte)
{
    bool res = false;
    
    ZF_ASSERT(pRb != (RingBuffer *)0)
    ZF_ASSERT(pRb->pBuf != (uint8_t *)0)
    
    if (!RB_isFull(pRb))
    {
        pRb->pBuf[pRb->Trail++] = byte;
        pRb->Trail %= pRb->Size;
        pRb->Count++;
        
        res = true;
    }
    
    return res;
}

取出一个字节

/******************************************************************************
 * 描述  :取出一个字节
 * 参数  :(in)-pRb    环形缓冲器结构体指针
 *         (out)-pByte 存放取出字节的地址
 * 返回  :-true       成功
 *         -false      失败
******************************************************************************/

bool RB_getByte(RingBuffer * const pRb, uint8_t *pByte)
{
    bool res = false;
    
    ZF_ASSERT(pRb != (RingBuffer *)0)
    ZF_ASSERT(pRb->pBuf != (uint8_t *)0)
    ZF_ASSERT(pByte != (uint8_t *)0)
    
    if (!RB_isEmpty(pRb))
    {
        *pByte = pRb->pBuf[pRb->Head++];
        pRb->Head %= pRb->Size;
        pRb->Count--;
        
        res = true;
    }
    
    return res;
}

环形缓冲区结果测试

/**
  *****************************************************************************
  * @file    app_buffer.c
  * @author  Zorb
  * @version V1.0.0
  * @date    2018-06-28
  * @brief   环形缓冲区测试的实现
  *****************************************************************************
  * @history
  *
  * 1. Date:2018-06-28
  *    Author:Zorb
  *    Modification:建立文件
  *
  *****************************************************************************
  */


#include "app_buffer.h"
#include "zf_includes.h"

/* 环形缓冲区指针 */
RingBuffer *rb;

/******************************************************************************
 * 描述  :任务初始化
 * 参数  :无
 * 返回  :无
******************************************************************************/

void App_Buffer_init(void)
{
    /* 创建500字节的缓冲区 */
    RB_create(&rb, 500);
}

/******************************************************************************
 * 描述  :任务程序
 * 参数  :无
 * 返回  :无
******************************************************************************/

void App_Buffer_process(void)
{
    uint32_t i;
    uint8_t buf[11];
    uint8_t byte;

    ZF_DEBUG(LOG_D, "rb count before adding data is %d\r\n", rb->Count);

    /* 填充10个字节数据(0-9) */
    for (i = 0; i < 10; i++)
    {
        rb->SaveByte(rb, i);
    }

    ZF_DEBUG(LOG_D, "rb count after adding data is %d\r\n", rb->Count);

    /* 读出数据看是否正确 */
    rb->ReadBytes(rb, buf, 10);

    ZF_DEBUG(LOG_D, "rb data is ");

    for (i = 0; i < 10; i++)
    {
        ZF_DEBUG(LOG_D, "%d ", buf[i]);
    }

    ZF_DEBUG(LOG_D, "\r\n\r\n");

    /* 弹出数据 */
    for (i = 0; i < 10; i++)
    {
        rb->GetByte(rb, &byte);

        ZF_DEBUG(LOG_D, "byte %d is %d\r\n", i, byte);
        ZF_DEBUG(LOG_D, "rb count is %d\r\n", rb->Count);
    }

    while(1);
}

/******************************** END OF FILE ********************************/

结果:

本文来源网络,免费传达知识,版权归原作者所有。如涉及作品版权问题,请联系我进行删除。

往期推荐

《嵌入式Linux驱动大全》

如何高效阅读嵌入式项目代码?


在公众号聊天界面回复1024,可获取嵌入式资源

嵌入式大杂烩 专注于嵌入式技术,包括但不限于C/C++、嵌入式、物联网、Linux等编程学习笔记,同时,内包含大量的学习资源。欢迎关注,一同交流学习,共同进步!
评论 (0)
  • 曾几何时,汽车之家可是汽车资讯平台领域响当当的“扛把子”。2005 年成立之初,它就像一位贴心的汽车小助手,一下子就抓住了大家的心。它不仅吸引了海量用户,更是成为汽车厂商和经销商眼中的“香饽饽”,广告投放、合作推广不断,营收和利润一路高歌猛进,2013年成功在纽交所上市,风光无限。2021年更是在香港二次上市,达到了发展的巅峰,当年3月15日上市首日,港股股价一度高达184.6港元,市值可观。然而,如今的汽车之家却陷入了困境,业务下滑明显。业务增长瓶颈从近年来汽车之家公布的财报数据来看,情况不容
    用户1742991715177 2025-04-07 21:48 113浏览
  •   物质扩散与污染物监测系统软件:多领域环境守护的智能中枢   北京华盛恒辉物质扩散与污染物监测系统软件,作为一款融合了物质扩散模拟、污染物监测、数据分析以及可视化等多元功能的综合性工具,致力于为环境科学、公共安全、工业生产等诸多领域给予强有力的技术支撑。接下来,将从功能特性、应用场景、技术实现途径、未来发展趋势等多个维度对这类软件展开详尽介绍。   应用案例   目前,已有多个物质扩散与污染物监测系统在实际应用中取得了显著成效。例如,北京华盛恒辉和北京五木恒润物质扩散与污染物监测系统。这
    华盛恒辉l58ll334744 2025-04-09 14:54 77浏览
  •   卫星图像智能测绘系统:地理空间数据处理的创新引擎   卫星图像智能测绘系统作为融合卫星遥感、地理信息系统(GIS)、人工智能(AI)以及大数据分析等前沿技术的综合性平台,致力于达成高精度、高效率的地理空间数据采集、处理与应用目标。借助自动化、智能化的技术路径,该系统为国土资源管理、城市规划、灾害监测、环境保护等诸多领域输送关键数据支撑。   应用案例   目前,已有多个卫星图像智能测绘系统在实际应用中取得了显著成效。例如,北京华盛恒辉北京五木恒润卫星图像智能测绘系统。这些成功案例为卫星
    华盛恒辉l58ll334744 2025-04-08 16:19 75浏览
  • 文/Leon编辑/侯煜‍就在小米SU7因高速交通事故、智驾性能受到质疑的时候,另一家中国领先的智驾解决方案供应商华为,低调地进行了一场重大人事变动。(详情见:雷军熬过黑夜,寄望小米SU7成为及时雨)4月4日上午,有网友发现余承东的职务发生了变化,华为官网、其个人微博认证信息为“常务董事,终端BG董事长”,不再包括“智能汽车解决方案BU董事长”。余承东的确不再兼任华为车BU董事长,但并非完全脱离华为的汽车业务,而是聚焦鸿蒙智行。据悉,华为方面寻求将车BU独立出去,但鸿蒙智行仍留在华为终端BG部门。
    华尔街科技眼 2025-04-09 15:28 61浏览
  •   物质扩散与污染物监测系统:环境守护的关键拼图   一、物质扩散原理剖析   物质扩散,本质上是物质在浓度梯度、温度梯度或者压力梯度等驱动力的作用下,从高浓度区域向低浓度区域迁移的过程。在环境科学范畴,物质扩散作为污染物在大气、水体以及土壤中迁移的关键机制,对污染物的分布态势、浓度动态变化以及环境风险程度有着直接且重大的影响。   应用案例   目前,已有多个物质扩散与污染物监测系统在实际应用中取得了显著成效。例如,北京华盛恒辉和北京五木恒润物质扩散与污染物监测系统。这些成功案例为物质
    华盛恒辉l58ll334744 2025-04-09 11:24 46浏览
  •   卫星图像智能测绘系统全面解析   一、系统概述   卫星图像智能测绘系统是基于卫星遥感技术、图像处理算法与人工智能(AI)技术的综合应用平台,旨在实现高精度、高效率的地理空间数据获取、处理与分析。该系统通过融合多源卫星数据(如光学、雷达、高光谱等),结合AI驱动的智能算法,实现自动化、智能化的测绘流程,广泛应用于城市规划、自然资源调查、灾害监测等领域。   应用案例   目前,已有多个卫星图像智能测绘系统在实际应用中取得了显著成效。例如,北京华盛恒辉和北京五木恒润卫星图像智能测绘系统
    华盛恒辉l58ll334744 2025-04-08 15:04 81浏览
  •     在研究Corona现象时发现:临界电压与介电材料表面的清洁程度有关。表面越清洁的介电材料,临界电压越高;表面污染物越多的地方,越容易“爬电”。关于Corona现象,另见基础理论第007篇。    这里说的“污染物”,定义为——可能影响介电强度或表面电阻率的固体、液体或气体(电离气体)的任何情况。    IEC 60664-1 (对应GB/T 16935.1-2023) 定义了 Pollution Degree,中文术语是“污染等
    电子知识打边炉 2025-04-07 22:06 103浏览
  • ## DL/T645-2007* 帧格式:* 帧起始字符:68H* 地址域:A0 A1 A2 A3 A4 A5* 帧起始字符:68H* 控制码:1字节* 主站:* 13H:请求读电能表通信地址* 11H:请求读电能表数据* 1CH:请求跳闸、合闸* 从站:* 91H:正常应答读电能表* 9CH:正常应答跳闸、合闸* 数据域长度:1字节* 数据域:DI0 DI1 DI2 DI3* 发送方:每字节+33H* 接收方:每字节-33H* 数据标识:* 电能量* 最大需量及发生时间* 变量* 事件记录*
    四毛打印店 2025-04-09 10:53 40浏览
  • HDMI从2.1版本开始采用FRL传输模式,和2.0及之前的版本不同。两者在物理层信号上有所区别,这就需要在一些2.1版本的电路设计上增加匹配电路,使得2.1版本的电路能够向下兼容2.0及之前版本。2.1版本的信号特性下面截取自2.1版本规范定义,可以看到2.1版本支持直流耦合和交流耦合,其共模电压和AVCC相关,信号摆幅在400mV-1200mV2.0及之前版本的信号特性HDMI2.0及之前版本采用TMDS信号物理层,其结构和参数如下:兼容设计根据以上规范定义,可以看出TMDS信号的共模电压范
    durid 2025-04-08 19:01 154浏览
  • 在万物互联时代,智能化安防需求持续升级,传统报警系统已难以满足实时性、可靠性与安全性并重的要求。WT2003H-16S低功耗语音芯片方案,以4G实时音频传输、超低功耗设计、端云加密交互为核心,重新定义智能报警设备的性能边界,为家庭、工业、公共安防等领域提供高效、稳定的安全守护。一、技术内核:五大核心突破,构建全场景安防基座1. 双模音频传输,灵活应对复杂场景实时音频流传输:内置高灵敏度MIC,支持环境音实时采集,通过4G模块直接上传至云端服务器,响应速度低至毫秒级,适用于火灾警报、紧急呼救等需即
    广州唯创电子 2025-04-08 08:59 143浏览
  • 文/郭楚妤编辑/cc孙聪颖‍伴随贸易全球化的持续深入,跨境电商迎来蓬勃发展期,物流行业 “出海” 成为不可阻挡的必然趋势。加之国内快递市场渐趋饱和,存量竞争愈发激烈。在此背景下,国内头部快递企业为突破发展瓶颈,寻求新的增长曲线,纷纷将战略目光投向海外市场。2024 年,堪称中国物流企业出海进程中的关键节点,众多企业纷纷扬帆起航,开启海外拓展之旅。然而,在一片向好的行业发展表象下,部分跨境物流企业的经营状况却不容乐观。它们受困于激烈的市场竞争、不断攀升的运营成本,以及复杂的国际物流环境,陷入了微利
    华尔街科技眼 2025-04-09 15:15 65浏览
  •     根据 IEC术语,瞬态过电压是指持续时间几个毫秒及以下的过高电压,通常是以高阻尼(快速衰减)形式出现,波形可以是振荡的,也可以是非振荡的。    瞬态过电压的成因和机理,IEC 60664-1给出了以下四种:    1. 自然放电,最典型的例子是雷击,感应到电力线路上,并通过电网配电系统传输,抵达用户端;        2. 电网中非特定感性负载通断。例如热处理工厂、机加工工厂对
    电子知识打边炉 2025-04-07 22:59 142浏览
  •   工业自动化领域电磁兼容与接地系统深度剖析   一、电磁兼容(EMC)基础认知   定义及关键意义   电磁兼容性(EMC),指的是设备或者系统在既定的电磁环境里,不但能按预期功能正常运转,而且不会对周边其他设备或系统造成难以承受的电磁干扰。在工业自动化不断发展的当下,大功率电机、变频器等设备被大量应用,现场总线、工业网络等技术也日益普及,致使工业自动化系统所处的电磁环境变得愈发复杂,电磁兼容(EMC)问题也越发严峻。   ​电磁兼容三大核心要素   屏蔽:屏蔽旨在切断电磁波的传播路
    北京华盛恒辉软件开发 2025-04-07 22:55 230浏览
  • 在人工智能技术飞速发展的今天,语音交互正以颠覆性的方式重塑我们的生活体验。WTK6900系列语音识别芯片凭借其离线高性能、抗噪远场识别、毫秒级响应的核心优势,为智能家居领域注入全新活力。以智能风扇为起点,我们开启一场“解放双手”的科技革命,让每一缕凉风都随“声”而至。一、核心技术:精准识别,无惧环境挑战自适应降噪,听懂你的每一句话WTK6900系列芯片搭载前沿信号处理技术,通过自适应降噪算法,可智能过滤环境噪声干扰。无论是家中电视声、户外虫鸣声,还是厨房烹饪的嘈杂声,芯片均能精准提取有效指令,识
    广州唯创电子 2025-04-08 08:40 182浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦