0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于宏高效實(shí)現(xiàn)環(huán)形緩沖區(qū)教程

strongerHuang ? 來(lái)源:小麥大叔 ? 作者:菜刀和小麥 ? 2021-09-02 09:24 ? 次閱讀

來(lái)源 | 小麥大叔

循環(huán)緩沖區(qū)是嵌入式軟件工程師在日常開發(fā)過(guò)程中的關(guān)鍵組件。

多年來(lái),互聯(lián)網(wǎng)上出現(xiàn)了許多不同的循環(huán)緩沖區(qū)實(shí)現(xiàn)和示例。我非常喜歡這個(gè)模塊,可以GitHub上找到這個(gè)開源的 CBUF.h 模塊。

地址:https://github.com/barraq/BRBrain/blob/master/firmware/CBUF.h

CBUF.h 模塊使用宏實(shí)現(xiàn)循環(huán)緩沖區(qū),具體源碼如下所示;

#if !defined( CBUF_H )#define CBUF_H /**《 Include Guard *//* ---- Include Files ---------------------------------------------------- *//* ---- Constants and Types ---------------------------------------------- *//**

* Initializes the circular buffer for use.

*/

#define CBUF_Init( cbuf ) cbuf.m_getIdx = cbuf.m_putIdx = 0/**

* Returns the number of elements which are currently contained in the

* circular buffer.

*/#define CBUF_Len( cbuf ) ((typeof( cbuf.m_putIdx ))(( cbuf.m_putIdx ) - ( cbuf.m_getIdx )))/**

* Appends an element to the end of the circular buffer

*/#define CBUF_Push( cbuf, elem ) (cbuf.m_entry)[ cbuf.m_putIdx++ & (( cbuf##_SIZE ) - 1 )] = (elem)/**

* Retrieves an element from the beginning of the circular buffer

*/#define CBUF_Pop( cbuf ) (cbuf.m_entry)[ cbuf.m_getIdx++ & (( cbuf##_SIZE ) - 1 )]/**

* Retrieves the i‘th element from the beginning of the circular buffer

*/#define CBUF_Get( cbuf, idx ) (cbuf.m_entry)[( cbuf.m_getIdx + idx ) & (( cbuf##_SIZE ) - 1 )]/**

* Retrieves the i’th element from the end of the circular buffer

*/#define CBUF_GetEnd( cbuf, idx ) (cbuf.m_entry)[( cbuf.m_putIdx - idx - 1 ) & (( cbuf##_SIZE ) - 1 )]/**

* Determines if the circular buffer is empty

*/#define CBUF_IsEmpty( cbuf ) ( CBUF_Len( cbuf ) == 0 )/**

* Determines if the circular buffer is full.

*/#define CBUF_IsFull( cbuf ) ( CBUF_Len( cbuf ) == ( cbuf##_SIZE ))/**

* Determines if the circular buffer is currenly overflowed or underflowed.

*/#define CBUF_Error( cbuf ) ( CBUF_Len( cbuf ) 》 cbuf##_SIZE )#if defined( __cplusplus )template 《 class IndexType, unsigned Size, class EntryType 》

class CBUF

{public

CBUF()

{

m_getIdx = m_putIdx = 0;

}

IndexType Len() const { return m_putIdx - m_getIdx; }

bool IsEmpty() const { return Len() == 0; }

bool IsFull() const { return Len() == Size; }

bool Error() const { return Len() 》 Size; }

void Push( EntryType val )

{

m_entry[ m_putIdx++ & ( Size - 1 )] = val;

}

EntryType Pop()

{

return m_entry[ m_getIdx++ & ( Size - 1 )];

}

private:

volatile IndexType m_getIdx;

volatile IndexType m_putIdx;

EntryType m_entry[ Size ];

};

#endif // __cplusplus/* ---- Variable Externs ------------------------------------------------- *//* ---- Function Prototypes ---------------------------------------------- *//** @} */#endif // CBUF_H

現(xiàn)在一般我不喜歡以這種方式使用宏,但實(shí)現(xiàn)已被證明是快速、高效且工作相對(duì)良好的,這是很難爭(zhēng)論的。

循環(huán)緩沖區(qū)的設(shè)置非常簡(jiǎn)單。首先,需要定義循環(huán)緩沖區(qū)的大小。這是通過(guò)定義宏 myQ_SIZE 來(lái)完成的,同時(shí)記住緩沖區(qū)大小需要是 2 的冪。

然后通過(guò)創(chuàng)建一個(gè) myQ 類型的變量來(lái)聲明循環(huán)緩沖區(qū)。例如,如果 myQ_SIZE 定義為 64 字節(jié),則可以定義 UART 的發(fā)送和接收緩沖區(qū),如下面的圖 1 所示。

1b3b1556-0b4a-11ec-8fb8-12bb97331649.png

圖 1 – 定義循環(huán)緩沖區(qū)

在此示例中,myQ 被定義為靜態(tài)以限制緩沖區(qū)的范圍并聲明為易失性,因?yàn)樗鼈冊(cè)谥袛鄡?nèi)被修改。定義循環(huán)緩沖區(qū)只是第一步。為了分配緩沖區(qū),必須將這些變量傳遞給 CBUF_INIT 宏,如下圖 2 所示。

1b476e82-0b4a-11ec-8fb8-12bb97331649.png
圖 2 – 緩沖區(qū)初始化

除了這個(gè)初始設(shè)置之外,緩沖區(qū)相當(dāng)簡(jiǎn)單且易于使用。例如,可以使用 CBUF_PUSH 將通過(guò)串行接口接收 UART接收的字符推送到循環(huán)緩沖區(qū),如圖 3 所示。

1b506712-0b4a-11ec-8fb8-12bb97331649.png

圖 3 – 推入緩沖區(qū)

開發(fā)人員不僅希望將數(shù)據(jù)推送到循環(huán)緩沖區(qū)上,還希望從緩沖區(qū)彈出或獲取數(shù)據(jù)??吹竭@一點(diǎn)的一個(gè)簡(jiǎn)單示例是需要獲取字符并通過(guò) UART 傳輸?shù)拇邪l(fā)送器。圖 4 中可以看到一個(gè)示例傳輸函數(shù)。

1b7135fa-0b4a-11ec-8fb8-12bb97331649.png

圖 4 – 從緩沖區(qū)彈出數(shù)據(jù)

在健壯的應(yīng)用程序中,還應(yīng)檢查循環(huán)緩沖區(qū)長(zhǎng)度和溢出狀態(tài)。CBUF 模塊確實(shí)提供了能夠檢查這些重要指標(biāo)的宏。

要記住的一個(gè)重要問(wèn)題是,如果需要對(duì) CBUF 本身進(jìn)行任何調(diào)試,這是不可能的。無(wú)法為宏設(shè)置斷點(diǎn),因此如果出現(xiàn)問(wèn)題,則需要對(duì)模塊進(jìn)行功能化以逐步執(zhí)行和調(diào)試。

多年來(lái)使用這個(gè)模塊雖然我沒(méi)有發(fā)現(xiàn)任何問(wèn)題。循環(huán)緩沖區(qū)是在嵌入式系統(tǒng)中與串行設(shè)備通信的一個(gè)重要方面。

循環(huán)緩沖區(qū)也很好理解,應(yīng)該創(chuàng)建它們以便它們可以模塊化并從一個(gè)應(yīng)用程序到下一個(gè)應(yīng)用程序重復(fù)使用。

到目前為止,CBUF 模塊已被證明是這樣一個(gè)模塊,所以在這里,我強(qiáng)烈推薦一下這個(gè)模塊。

免責(zé)聲明:本文來(lái)源網(wǎng)絡(luò),版權(quán)歸原作者所有。如涉及作品版權(quán)問(wèn)題,請(qǐng)與我聯(lián)系刪除。

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 緩沖
    +關(guān)注

    關(guān)注

    0

    文章

    53

    瀏覽量

    17839
  • uart
    +關(guān)注

    關(guān)注

    22

    文章

    1242

    瀏覽量

    101536
  • 嵌入式軟件
    +關(guān)注

    關(guān)注

    4

    文章

    240

    瀏覽量

    26672
  • GitHub
    +關(guān)注

    關(guān)注

    3

    文章

    473

    瀏覽量

    16501

原文標(biāo)題:用宏高效實(shí)現(xiàn)環(huán)形緩沖區(qū)

文章出處:【微信號(hào):strongerHuang,微信公眾號(hào):strongerHuang】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    基于C語(yǔ)言實(shí)現(xiàn)環(huán)形緩沖區(qū)/循環(huán)隊(duì)列

    這里分享一個(gè)自己用純C實(shí)現(xiàn)環(huán)形緩沖區(qū)。
    的頭像 發(fā)表于 04-11 10:39 ?3356次閱讀
    基于C語(yǔ)言<b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>環(huán)形</b><b class='flag-5'>緩沖區(qū)</b>/循環(huán)隊(duì)列

    STM32進(jìn)階之串口環(huán)形緩沖區(qū)實(shí)現(xiàn)

    完了數(shù)據(jù),‘0’地址空間的數(shù)據(jù)進(jìn)行釋放掉,列隊(duì)頭指向下一個(gè)可以處理數(shù)據(jù)的地址‘1’。從而實(shí)現(xiàn)整個(gè)環(huán)形緩沖區(qū)的數(shù)據(jù)讀寫??磮D,隊(duì)列頭就是指向已經(jīng)存儲(chǔ)的數(shù)據(jù),并且這個(gè)數(shù)據(jù)是待處理的。下一個(gè)CPU處理的數(shù)據(jù)
    發(fā)表于 06-08 14:03

    MCU進(jìn)階之串口環(huán)形緩沖區(qū)實(shí)現(xiàn)

    是列隊(duì)頭的數(shù)據(jù),處理完了數(shù)據(jù),‘0’地址空間的數(shù)據(jù)進(jìn)行釋放掉,列隊(duì)頭指向下一個(gè)可以處理數(shù)據(jù)的地址‘1’。從而實(shí)現(xiàn)整個(gè)環(huán)形緩沖區(qū)的數(shù)據(jù)讀寫。看圖,隊(duì)列頭就是指向已經(jīng)存儲(chǔ)的數(shù)據(jù),并且這個(gè)數(shù)據(jù)是待處理的。下一個(gè)
    發(fā)表于 08-17 13:11

    環(huán)形緩沖區(qū)的設(shè)計(jì)分享!

    去訪問(wèn)該緩沖區(qū)的最后一個(gè)內(nèi)存位置的的后一位置時(shí)回到環(huán)形緩沖區(qū)的起點(diǎn)。類似一個(gè)環(huán)一樣。這樣形容就很好理解了,當(dāng)然有辦法實(shí)現(xiàn)了。我在這里采用了2種方式實(shí)
    發(fā)表于 10-28 23:29

    環(huán)形緩沖區(qū)簡(jiǎn)介

    STM32串口數(shù)據(jù)接收 --環(huán)形緩沖區(qū)環(huán)形緩沖區(qū)簡(jiǎn)介??在單片機(jī)中串口通信是我們使用最頻繁的,使用串口通信就會(huì)用到串口的數(shù)據(jù)接收與發(fā)送,環(huán)形
    發(fā)表于 08-17 06:56

    怎么實(shí)現(xiàn)串口環(huán)形緩沖區(qū)

    怎么實(shí)現(xiàn)串口環(huán)形緩沖區(qū)?
    發(fā)表于 12-06 06:01

    請(qǐng)問(wèn)串口的DMA接收緩沖區(qū)是不是環(huán)形緩沖區(qū)

    大家好!請(qǐng)問(wèn)串口的DMA接收緩沖區(qū)是不是環(huán)形緩沖區(qū)?通過(guò)閱讀串口部分的代碼,我了解到這樣幾點(diǎn):1、串口的DMA接收時(shí)循環(huán)接收,當(dāng)緩沖區(qū)滿了會(huì)重新從頭開始覆蓋掉之前的數(shù)據(jù),和
    發(fā)表于 08-30 14:27

    環(huán)形緩沖區(qū)讀寫操作的分析與實(shí)現(xiàn)

    環(huán)形緩沖區(qū)是嵌入式系統(tǒng)中一種重要的常用數(shù)據(jù)結(jié)構(gòu)。在多任務(wù)環(huán)境下實(shí)現(xiàn)時(shí),如果有多個(gè)讀寫任務(wù),一般需要用信號(hào)量來(lái)保護(hù)多個(gè)任務(wù)共享的環(huán)形緩沖區(qū)。但
    發(fā)表于 04-15 11:35 ?40次下載

    環(huán)形緩沖區(qū)實(shí)現(xiàn)原理

    在通信程序中,經(jīng)常使用環(huán)形緩沖區(qū)作為數(shù)據(jù)結(jié)構(gòu)來(lái)存放通信中發(fā)送和接收的數(shù)據(jù)。環(huán)形緩沖區(qū)是一個(gè)先進(jìn)先出的循環(huán)緩沖區(qū),可以向通信程序提供對(duì)
    的頭像 發(fā)表于 03-22 10:03 ?7564次閱讀
    <b class='flag-5'>環(huán)形</b><b class='flag-5'>緩沖區(qū)</b>的<b class='flag-5'>實(shí)現(xiàn)</b>原理

    緩沖區(qū)是啥意思 STM32串口數(shù)據(jù)接收之環(huán)形緩沖區(qū)

    緩沖區(qū)顧名思義是緩沖數(shù)據(jù)用的。實(shí)現(xiàn)緩沖區(qū)最簡(jiǎn)單的辦法時(shí),定義多個(gè)數(shù)組,接收一包數(shù)據(jù)到數(shù)組A,就把接收數(shù)據(jù)的地址換成數(shù)組B,每個(gè)數(shù)據(jù)有個(gè)標(biāo)記字節(jié)用于表示這個(gè)數(shù)組是否收到數(shù)據(jù),收到數(shù)據(jù)是否
    的頭像 發(fā)表于 07-22 15:33 ?1.1w次閱讀

    STM32串口數(shù)據(jù)接收 --環(huán)形緩沖區(qū)

    STM32串口數(shù)據(jù)接收 --環(huán)形緩沖區(qū)環(huán)形緩沖區(qū)簡(jiǎn)介??在單片機(jī)中串口通信是我們使用最頻繁的,使用串口通信就會(huì)用到串口的數(shù)據(jù)接收與發(fā)送,環(huán)形
    發(fā)表于 12-28 19:24 ?31次下載
    STM32串口數(shù)據(jù)接收 --<b class='flag-5'>環(huán)形</b><b class='flag-5'>緩沖區(qū)</b>

    環(huán)形緩沖區(qū)實(shí)現(xiàn)思路

    單片機(jī)程序開發(fā)一般都會(huì)用到UART串口通信,通過(guò)通信來(lái)實(shí)現(xiàn)上位機(jī)和單片機(jī)程序的數(shù)據(jù)交互。通信中為了實(shí)現(xiàn)正常的收發(fā),一般都會(huì)有對(duì)應(yīng)的發(fā)送和接收緩存來(lái)暫存通信數(shù)據(jù)。這里使用環(huán)形緩沖區(qū)的方式
    的頭像 發(fā)表于 01-17 15:07 ?1665次閱讀

    STM32進(jìn)階之串口環(huán)形緩沖區(qū)實(shí)現(xiàn)

    STM32進(jìn)階之串口環(huán)形緩沖區(qū)實(shí)現(xiàn)
    的頭像 發(fā)表于 09-19 09:20 ?2438次閱讀
    STM32進(jìn)階之串口<b class='flag-5'>環(huán)形</b><b class='flag-5'>緩沖區(qū)</b><b class='flag-5'>實(shí)現(xiàn)</b>

    C++環(huán)形緩沖區(qū)設(shè)計(jì)與實(shí)現(xiàn)

    的存儲(chǔ)空間。環(huán)形緩沖區(qū)的特點(diǎn)是其終點(diǎn)和起點(diǎn)是相連的,形成一個(gè)環(huán)狀結(jié)構(gòu)。這種數(shù)據(jù)結(jié)構(gòu)在處理流數(shù)據(jù)和實(shí)現(xiàn)數(shù)據(jù)緩存等場(chǎng)景中具有廣泛的應(yīng)用。 環(huán)形緩沖區(qū)
    的頭像 發(fā)表于 11-09 11:21 ?2166次閱讀
    C++<b class='flag-5'>環(huán)形</b><b class='flag-5'>緩沖區(qū)</b>設(shè)計(jì)與<b class='flag-5'>實(shí)現(xiàn)</b>

    單片機(jī)中的幾種環(huán)形緩沖區(qū)的分析和實(shí)現(xiàn)

    單片機(jī)中的幾種環(huán)形緩沖區(qū)的分析和實(shí)現(xiàn)一、簡(jiǎn)介環(huán)形緩沖區(qū)(RingBuffer)是一種高效的使用內(nèi)
    的頭像 發(fā)表于 08-14 08:39 ?1006次閱讀
    單片機(jī)中的幾種<b class='flag-5'>環(huán)形</b><b class='flag-5'>緩沖區(qū)</b>的分析和<b class='flag-5'>實(shí)現(xiàn)</b>