當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > 大魚(yú)機(jī)器人
[導(dǎo)讀]大家好,我是張巧龍,本文介紹如何使用帶FIFO的串口來(lái)減少接收中斷次數(shù),通過(guò)一種自定義通訊協(xié)議格式,給出幀打包方法;之后介紹一種特殊的串口數(shù)據(jù)發(fā)送方法,可在避免使用串口發(fā)送中斷的情況下,提高系統(tǒng)的響應(yīng)速度。一種可1.簡(jiǎn)介串口由于使用簡(jiǎn)單,價(jià)格低廉,配合RS485芯片可以實(shí)現(xiàn)長(zhǎng)距離...

大家好,我是張巧龍,本文介紹如何使用帶FIFO的串口來(lái)減少接收中斷次數(shù),通過(guò)一種自定義通訊協(xié)議格式,給出幀打包方法;之后介紹一種特殊的串口數(shù)據(jù)發(fā)送方法,可在避免使用串口發(fā)送中斷的情況下,提高系統(tǒng)的響應(yīng)速度。一種可

1. 簡(jiǎn)介

串口由于使用簡(jiǎn)單,價(jià)格低廉,配合RS485芯片可以實(shí)現(xiàn)長(zhǎng)距離、抗干擾能力強(qiáng)的局域網(wǎng)絡(luò)而被廣泛使用。隨著產(chǎn)品功能的增多,需要處理的任務(wù)也越來(lái)越復(fù)雜,系統(tǒng)任務(wù)也越來(lái)越需要及時(shí)響應(yīng)。絕大多數(shù)的現(xiàn)代單片機(jī)(ARM7、Cortex-M3)串口都帶有一定數(shù)量的硬件FIFO,本文將介紹如何使用硬件FIFO來(lái)減少接收中斷次數(shù),提高發(fā)送效率。在此之前,先來(lái)列舉一下傳統(tǒng)串口數(shù)據(jù)收發(fā)的不足之處:(1)每接收一個(gè)字節(jié)數(shù)據(jù),產(chǎn)生一次接收中斷。不能有效的利用串口硬件FIFO,減少中斷次數(shù)。(2)應(yīng)答數(shù)據(jù)采用等待發(fā)送的方法。由于串行數(shù)據(jù)傳輸?shù)臅r(shí)間遠(yuǎn)遠(yuǎn)跟不上CPU的處理時(shí)間,等待串口發(fā)送完當(dāng)前字節(jié)再發(fā)送下一字節(jié)會(huì)造成CPU資源浪費(fèi),不利于系統(tǒng)整體響應(yīng)(在1200bps下,發(fā)送一字節(jié)大約需要10ms,如果一次發(fā)送幾十個(gè)字節(jié)數(shù)據(jù),CPU會(huì)長(zhǎng)時(shí)間處于等待狀態(tài))。(3)應(yīng)答數(shù)據(jù)采用中斷發(fā)送。增加一個(gè)中斷源,增加系統(tǒng)的中斷次數(shù),這會(huì)影響系統(tǒng)整體穩(wěn)定性(從可靠性角度考慮,中斷事件應(yīng)越少越好)。(4)針對(duì)上述的不足之處,將結(jié)合一個(gè)常用自定義通訊協(xié)議,提供一個(gè)完整的解決方案。

2.串口FIFO

串口FIFO可以理解為串口專用的緩存,該緩存采用先進(jìn)先出方式。數(shù)據(jù)接收FIFO和數(shù)據(jù)發(fā)送FIFO通常是獨(dú)立的兩個(gè)硬件。串口接收的數(shù)據(jù),先放入接收FIFO中,當(dāng)FIFO中的數(shù)據(jù)達(dá)到觸發(fā)值(通常觸發(fā)值為1、2、4、8、14字節(jié))或者FIFO中的數(shù)據(jù)雖然沒(méi)有達(dá)到設(shè)定值但是一段時(shí)間(通常為3.5個(gè)字符傳輸時(shí)間)沒(méi)有再接收到數(shù)據(jù),則通知CPU產(chǎn)生接收中斷;發(fā)送的數(shù)據(jù)要先寫(xiě)入發(fā)送FIFO,只要發(fā)送FIFO未空,硬件會(huì)自動(dòng)發(fā)送FIFO中的數(shù)據(jù)。寫(xiě)入發(fā)送FIFO的字節(jié)個(gè)數(shù)受FIFO最大深度影響,通常一次寫(xiě)入最多允許16字節(jié)。上述列舉的數(shù)據(jù)跟具體的硬件有關(guān),CPU類型不同,特性也不盡相同,使用前應(yīng)參考相應(yīng)的數(shù)據(jù)手冊(cè)。

3.數(shù)據(jù)接收與打包

FIFO可以緩存串口接收到的數(shù)據(jù),因此我們可以利用FIFO來(lái)減少中斷次數(shù)。以NXP的lpc1778芯片為例,接收FIFO的觸發(fā)級(jí)別可以設(shè)置為1、2、4、8、14字節(jié),推薦使用8字節(jié)或者14字節(jié),這也是PC串口接收FIFO的默認(rèn)值。這樣,當(dāng)接收到大量數(shù)據(jù)時(shí),每8個(gè)字節(jié)或者14個(gè)字節(jié)才會(huì)產(chǎn)生一次中斷(最后一次接收除外),相比接收一個(gè)字節(jié)即產(chǎn)生一個(gè)中斷,這種方法串口接收中斷次數(shù)大大減少。將接收FIFO設(shè)置為8或者14字節(jié)也十分簡(jiǎn)單,還是以lpc1778為例,只需要設(shè)置UART FIFO控制寄存器UnFCR即可。接收的數(shù)據(jù)要符合通訊協(xié)議規(guī)定,數(shù)據(jù)與協(xié)議是密不可分的。通常我們需要將接收到的數(shù)據(jù)根據(jù)協(xié)議打包成一幀,然后交由上層處理。下面介紹一個(gè)自定義的協(xié)議幀格式,并給出一個(gè)通用打包成幀的方法。自定義協(xié)議格式如圖3-1所示。
  • 幀首:通常是3~5個(gè)0xFF或者0xEE
  • 地址號(hào):要進(jìn)行通訊的設(shè)備的地址編號(hào),1字節(jié)
  • 命令號(hào):對(duì)應(yīng)不同的功能,1字節(jié)
  • 長(zhǎng)度:數(shù)據(jù)區(qū)域的字節(jié)個(gè)數(shù),1字節(jié)
  • 數(shù)據(jù):與具體的命令號(hào)有關(guān),數(shù)據(jù)區(qū)長(zhǎng)度可以為0,整個(gè)幀的長(zhǎng)度不應(yīng)超過(guò)256字節(jié)
  • 校驗(yàn):異或和校驗(yàn)(1字節(jié))或者CRC16校驗(yàn)(2字節(jié)),本例使用CRC16校驗(yàn)
下面介紹如何將接收到的數(shù)據(jù)按照?qǐng)D3-1所示的格式打包成一幀。

3.1 定義數(shù)據(jù)結(jié)構(gòu)

typedef?struct?
{
??
????uint8_t?*?dst_buf;??????????????????//指向接收緩存??
????uint8_t?sfd;????????????????????????//幀首標(biāo)志,為0xFF或者0xEE??
????uint8_t?sfd_flag;???????????????????//找到幀首,一般是3~5個(gè)FF或EE??
????uint8_t?sfd_count;??????????????????//幀首的個(gè)數(shù),一般3~5個(gè)??
????uint8_t?received_len;???????????????//已經(jīng)接收的字節(jié)數(shù)??
????uint8_t?find_fram_flag;?????????????//找到完整幀后,置1??
????uint8_t?frame_len;??????????????????//本幀數(shù)據(jù)總長(zhǎng)度,這個(gè)區(qū)域是可選的??
}find_frame_struct;

3.2 初始化數(shù)據(jù)結(jié)構(gòu),一般放在串口初始化中

/**?
*?@brief????初始化尋找?guī)臄?shù)據(jù)結(jié)構(gòu)?
*?@param????p_fine_frame:指向打包幀數(shù)據(jù)結(jié)構(gòu)體變量?
*?@param????dst_buf:指向幀緩沖區(qū)?
*?@param????sfd:幀首標(biāo)志,一般為0xFF或者0xEE?
*/
??
void?init_find_frame_struct(find_frame_struct?*?p_find_frame,uint8_t?*dst_buf,uint8_t?sfd)??
{??
????p_find_frame->dst_buf=dst_buf;??
????p_find_frame->sfd=sfd;??
????p_find_frame->find_fram_flag=0;??
????p_find_frame->frame_len=10;???????
????p_find_frame->received_len=0;??
????p_find_frame->sfd_count=0;??
????p_find_frame->sfd_flag=0;??
}?

3.3 數(shù)據(jù)打包程序

/**?
*?@brief????尋找一幀數(shù)據(jù)??返回處理的數(shù)據(jù)個(gè)數(shù)?
*?@param????p_find_frame:指向打包幀數(shù)據(jù)結(jié)構(gòu)體變量?
*?@param????src_buf:指向串口接收的原始數(shù)據(jù)?
*?@param????data_len:src_buf本次串口接收到的原始數(shù)據(jù)個(gè)數(shù)?
*?@param????sum_len:幀緩存的最大長(zhǎng)度?
*?@return???本次處理的數(shù)據(jù)個(gè)數(shù)?
*/
??
uint32_t?find_one_frame(find_frame_struct?*?p_find_frame,const?uint8_t?*?src_buf,uint32_t?data_len,uint32_t?sum_len)??
{??
????uint32_t?src_len=0;??
????while(data_len--)??
????{??
????????if(p_find_frame?->sfd_flag==0)????????????????????????
????????{???//沒(méi)有找到起始幀首??
????????????if(src_buf[src_len ]==p_find_frame?->sfd)??
????????????{??
????????????????p_find_frame?->dst_buf[p_find_frame?->received_len ]=p_find_frame?->sfd;??
????????????????if( p_find_frame?->sfd_count==5)??????????
????????????????{??
????????????????????p_find_frame?->sfd_flag=1;??
????????????????????p_find_frame?->sfd_count=0;??
????????????????????p_find_frame?->frame_len=10;??
????????????????}??
????????????}??
????????????else??
????????????{??
????????????????p_find_frame?->sfd_count=0;???
????????????????p_find_frame?->received_len=0;???
????????????}??
????????}??
????????else???
????????{???//是否是"長(zhǎng)度"字節(jié)??Y->獲取這幀的數(shù)據(jù)長(zhǎng)度??
????????????if(7==p_find_frame?->received_len)????????????????
????????????{??
????????????????p_find_frame->frame_len=src_buf[src_len] 5 1 1 1 2;?//幀首 地址號(hào) 命令號(hào) 數(shù)據(jù)長(zhǎng)度 校驗(yàn)???????
????????????????if(p_find_frame->frame_len>=sum_len)??
????????????????{???//這里處理方法根據(jù)具體應(yīng)用不一定相同??
????????????????????MY_DEBUGF(SLAVE_DEBUG,("數(shù)據(jù)長(zhǎng)度超出緩存!\n"));??
????????????????????p_find_frame->frame_len=?sum_len;???????
????????????????}??
????????????}??
??????????????
????????????p_find_frame?->dst_buf[p_find_frame->received_len ]=src_buf[src_len ];????????????????
????????????if(p_find_frame?->received_len==p_find_frame?->frame_len)??????????????????
????????????{??
????????????????p_find_frame?->received_len=0;??????????????//一幀完成????
????????????????p_find_frame?->sfd_flag=0;??
????????????????p_find_frame?->find_fram_flag=1;???????????????????
????????????????return?src_len;??
????????????}??
????????}??
????}??
????p_find_frame?->find_fram_flag=0;??
????return?src_len;??
}?

使用例子:定義數(shù)據(jù)結(jié)構(gòu)體變量:find_frame_struct?slave_find_frame_srt;
定義接收數(shù)據(jù)緩沖區(qū):#define?SLAVE_REC_DATA_LEN??128
uint8_t?slave_rec_buf[SLAVE_REC_DATA_LEN];

在串口初始化中調(diào)用結(jié)構(gòu)體變量初始化函數(shù):init_find_frame_struct(
本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉