FIFO在嵌入式應(yīng)用的非常廣泛,可以說(shuō)有數(shù)據(jù)收發(fā)的地方,基本就有FIFO的存在,今天給大家分享一款基于C語(yǔ)言實(shí)現(xiàn)的FIFO模塊:xqueue.
1. 為什么需要FIFO
FIFO 是First-In First-Out的縮寫(xiě),它是一個(gè)具有先入先出特點(diǎn)的緩沖區(qū)。
可以理解成一個(gè)大的水池,水對(duì)應(yīng)數(shù)據(jù),注水速度對(duì)應(yīng)數(shù)據(jù)輸入的頻率,放水速度對(duì)應(yīng)數(shù)據(jù)處理的速度,當(dāng)注水速度和放水速度相同時(shí),我們不需要使用水池來(lái)緩沖,但是當(dāng)注水速度大于放水速度,或者注水速度突然變大時(shí)(突發(fā)),為了保證水池不溢出(數(shù)據(jù)不丟失),就需要水池(緩沖區(qū))來(lái)處理這種突發(fā)情況,并設(shè)置合理大小的水池空間(FIFO的深度)。
或者為了降低CPU負(fù)擔(dān),提高數(shù)據(jù)處理效率,可以在積累到一定的數(shù)據(jù)量之后,再一次性處理。
在FPGA中,F(xiàn)IFO一般是使用RAM存儲(chǔ)器作為緩沖區(qū),可以分為同步FIFO或異步FIO,一般用于數(shù)據(jù)緩沖,或者不同時(shí)鐘域之間的數(shù)據(jù)傳遞。
在單片機(jī)中,一般是基于一維數(shù)組和結(jié)構(gòu)體實(shí)現(xiàn)的循環(huán)隊(duì)列(Queue),或者叫環(huán)形隊(duì)列。
FIFO的使用,既可以保證數(shù)據(jù)的完整性,還可以讓數(shù)據(jù)被及時(shí)的處理。
本文介紹,基于C語(yǔ)言的循環(huán)隊(duì)列緩沖區(qū)原理、設(shè)計(jì)與實(shí)現(xiàn)。
2. FIFO的存取順序
定義一個(gè)一維數(shù)組當(dāng)作存儲(chǔ)區(qū),數(shù)組長(zhǎng)度為6,再定義兩個(gè)讀寫(xiě)指針變量。
初始化時(shí),F(xiàn)IFO為空,讀寫(xiě)指針相等,并都置為0。
寫(xiě)入一個(gè)數(shù)據(jù)1之后,寫(xiě)指針遞增,讀指針不變:
再寫(xiě)兩個(gè)數(shù)據(jù)2和3,寫(xiě)指針遞增,讀指針不變:
寫(xiě)了三個(gè)數(shù)據(jù)之后,我們讀出一個(gè)數(shù)據(jù)1,寫(xiě)指針不變,讀指針遞增:
讀出一個(gè)數(shù)據(jù)2,再寫(xiě)兩個(gè)數(shù)據(jù)4和5,讀寫(xiě)指針變化:
再寫(xiě)一個(gè)數(shù)據(jù)6,此時(shí)超過(guò)數(shù)組長(zhǎng)度,但是數(shù)組頭部還有空間,所以寫(xiě)指針回到數(shù)組起始地址0:
再寫(xiě)一個(gè)數(shù)據(jù)7,此時(shí)判斷FIFO滿:
可能會(huì)有朋友疑惑,不是還有一個(gè)空位置可以存放數(shù)據(jù)嗎?
如果再存入一個(gè)數(shù)據(jù)之后,讀寫(xiě)指針相等,此時(shí)可以判斷是滿狀態(tài)嗎?
顯然是不能,因?yàn)楫?dāng)FIFO為空時(shí),也是讀寫(xiě)指針相等,所以這種情況就無(wú)法判斷滿和空。
這里就涉及到FIFO設(shè)計(jì)中,最重要的滿和空的判斷條件,需要遵循FIFO讀寫(xiě)的兩個(gè)規(guī)則:
FIFO為空時(shí),不能執(zhí)行讀操作
FIFO為滿時(shí),不能執(zhí)行寫(xiě)操作
為了避免這種情況發(fā)生,我們空出一個(gè)元素位置,寫(xiě)指針指向的位置永遠(yuǎn)為空,這樣就會(huì)有兩種滿的情況:
rd < wr
rd > wr
對(duì)于第一種情況,當(dāng)(wr + 1) % FIFO_SIZE == rd時(shí),可以認(rèn)為FIFO滿,F(xiàn)IFO_SIZE是指數(shù)組長(zhǎng)度;
對(duì)于第二種情況,當(dāng)wr + 1 == rd時(shí),可以認(rèn)為FIFO滿。
以上兩種情況可以合并為一種,即(wr + 1) % FIFO_SIZE == rd時(shí),判斷FIFO滿。
所以這種判斷方式,會(huì)犧牲一個(gè)存儲(chǔ)位置,實(shí)際可以存儲(chǔ)的元素個(gè)數(shù)為FIFO_SIZE-1。
同理,獲取當(dāng)前FIFO內(nèi)元素的個(gè)數(shù),也可以分為兩種情況:
當(dāng)wr > rd時(shí), count = wr - rd
當(dāng)wr < rd時(shí),count = wr + FIFO_SIZE - rd
3. FIFO的代碼實(shí)現(xiàn)
根據(jù)以上FIFO存取邏輯,我們可以使用一維數(shù)組來(lái)構(gòu)造一個(gè)環(huán)形緩沖區(qū),讀寫(xiě)地址循環(huán)遞增,分別實(shí)現(xiàn)FIFO初始化、讀寫(xiě)操作、判斷空滿、獲取元素個(gè)數(shù)等函數(shù),并封裝成模塊。
xqueue.h
xqueue.c文件
實(shí)際應(yīng)用:
運(yùn)行結(jié)果:
循環(huán)隊(duì)列元素的數(shù)據(jù)類型,可以根據(jù)需要指定,也可以是結(jié)構(gòu)體類型。
審核編輯:劉清
-
嵌入式
+關(guān)注
關(guān)注
5086文章
19142瀏覽量
306036 -
存儲(chǔ)器
+關(guān)注
關(guān)注
38文章
7509瀏覽量
163975 -
RAM
+關(guān)注
關(guān)注
8文章
1369瀏覽量
114763 -
C語(yǔ)言
+關(guān)注
關(guān)注
180文章
7608瀏覽量
137080 -
FIFO芯片
+關(guān)注
關(guān)注
0文章
10瀏覽量
8832
原文標(biāo)題:分享一款基于C語(yǔ)言實(shí)現(xiàn)的FIFO模塊
文章出處:【微信號(hào):strongerHuang,微信公眾號(hào):strongerHuang】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論