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

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

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

詳解單片機串口高效收發(fā)數(shù)據(jù)的實現(xiàn)方法

Dp1040 ? 來源:玩轉(zhuǎn)嵌入式 ? 2023-05-10 09:37 ? 次閱讀

摘要:本文在探討傳統(tǒng)數(shù)據(jù)收發(fā)不足之后,介紹如何使用帶FIFO的串口來減少接收中斷次數(shù),通過一種自定義通訊協(xié)議格式,給出幀打包方法;之后介紹一種特殊的串口數(shù)據(jù)發(fā)送方法,可在避免使用串口發(fā)送中斷的情況下,提高系統(tǒng)的響應(yīng)速度。

1. 簡介

串口由于使用簡單,價格低廉,配合RS485芯片可以實現(xiàn)長距離、抗干擾能力強的局域網(wǎng)絡(luò)而被廣泛使用。隨著產(chǎn)品功能的增多,需要處理的任務(wù)也越來越復(fù)雜,系統(tǒng)任務(wù)也越來越需要及時響應(yīng)。

絕大多數(shù)的現(xiàn)代單片機ARM7、Cortex-M3)串口都帶有一定數(shù)量的硬件FIFO,本文將介紹如何使用硬件FIFO來減少接收中斷次數(shù),提高發(fā)送效率。在此之前,先來列舉一下傳統(tǒng)串口數(shù)據(jù)收發(fā)的不足之處:

(1)每接收一個字節(jié)數(shù)據(jù),產(chǎn)生一次接收中斷。不能有效的利用串口硬件FIFO,減少中斷次數(shù)。

(2)應(yīng)答數(shù)據(jù)采用等待發(fā)送的方法。由于串行數(shù)據(jù)傳輸?shù)臅r間遠(yuǎn)遠(yuǎn)跟不上CPU的處理時間,等待串口發(fā)送完當(dāng)前字節(jié)再發(fā)送下一字節(jié)會造成CPU資源浪費,不利于系統(tǒng)整體響應(yīng)(在1200bps下,發(fā)送一字節(jié)大約需要10ms,如果一次發(fā)送幾十個字節(jié)數(shù)據(jù),CPU會長時間處于等待狀態(tài))。

(3)應(yīng)答數(shù)據(jù)采用中斷發(fā)送。增加一個中斷源,增加系統(tǒng)的中斷次數(shù),這會影響系統(tǒng)整體穩(wěn)定性(從可靠性角度考慮,中斷事件應(yīng)越少越好)。

(4)針對上述的不足之處,將結(jié)合一個常用自定義通訊協(xié)議,提供一個完整的解決方案。

2. 串口FIFO

串口FIFO可以理解為串口專用的緩存,該緩存采用先進(jìn)先出方式。數(shù)據(jù)接收FIFO和數(shù)據(jù)發(fā)送FIFO通常是獨立的兩個硬件。

串口接收的數(shù)據(jù),先放入接收FIFO中,當(dāng)FIFO中的數(shù)據(jù)達(dá)到觸發(fā)值(通常觸發(fā)值為1、2、4、8、14字節(jié))或者FIFO中的數(shù)據(jù)雖然沒有達(dá)到設(shè)定值但是一段時間(通常為3.5個字符傳輸時間)沒有再接收到數(shù)據(jù),則通知CPU產(chǎn)生接收中斷;發(fā)送的數(shù)據(jù)要先寫入發(fā)送FIFO,只要發(fā)送FIFO未空,硬件會自動發(fā)送FIFO中的數(shù)據(jù)。

寫入發(fā)送FIFO的字節(jié)個數(shù)受FIFO最大深度影響,通常一次寫入最多允許16字節(jié)。上述列舉的數(shù)據(jù)跟具體的硬件有關(guān),CPU類型不同,特性也不盡相同,使用前應(yīng)參考相應(yīng)的數(shù)據(jù)手冊。

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

FIFO可以緩存串口接收到的數(shù)據(jù),因此我們可以利用FIFO來減少中斷次數(shù)。以NXPlpc1778芯片為例,接收FIFO的觸發(fā)級別可以設(shè)置為1、2、4、8、14字節(jié),推薦使用8字節(jié)或者14字節(jié),這也是PC串口接收FIFO的默認(rèn)值。

這樣,當(dāng)接收到大量數(shù)據(jù)時,每8個字節(jié)或者14個字節(jié)才會產(chǎn)生一次中斷(最后一次接收除外),相比接收一個字節(jié)即產(chǎn)生一個中斷,這種方法串口接收中斷次數(shù)大大減少。

將接收FIFO設(shè)置為8或者14字節(jié)也十分簡單,還是以lpc1778為例,只需要設(shè)置UART FIFO控制寄存器UnFCR即可。

接收的數(shù)據(jù)要符合通訊協(xié)議規(guī)定,數(shù)據(jù)與協(xié)議是密不可分的。通常我們需要將接收到的數(shù)據(jù)根據(jù)協(xié)議打包成一幀,然后交由上層處理。下面介紹一個自定義的協(xié)議幀格式,并給出一個通用打包成幀的方法。

自定義協(xié)議格式如圖3-1所示。

66a4e872-eecf-11ed-90ce-dac502259ad0.png

  • 幀首:通常是3~5個0xFF或者0xEE

  • 地址號:要進(jìn)行通訊的設(shè)備的地址編號,1字節(jié)

  • 命令號:對應(yīng)不同的功能,1字節(jié)

  • 長度:數(shù)據(jù)區(qū)域的字節(jié)個數(shù),1字節(jié)

  • 數(shù)據(jù):與具體的命令號有關(guān),數(shù)據(jù)區(qū)長度可以為0,整個幀的長度不應(yīng)超過256字節(jié)

  • 校驗:異或和校驗(1字節(jié))或者CRC16校驗(2字節(jié)),本例使用CRC16校驗

下面介紹如何將接收到的數(shù)據(jù)按照圖3-1所示的格式打包成一幀。

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


		

typedefstruct { uint8_t*dst_buf;//指向接收緩存 uint8_tsfd;//幀首標(biāo)志,為0xFF或者0xEE uint8_tsfd_flag;//找到幀首,一般是3~5個FF或EE uint8_tsfd_count;//幀首的個數(shù),一般3~5個 uint8_treceived_len;//已經(jīng)接收的字節(jié)數(shù) uint8_tfind_fram_flag;//找到完整幀后,置1 uint8_tframe_len;//本幀數(shù)據(jù)總長度,這個區(qū)域是可選的 }find_frame_struct;

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


		

/** *@brief初始化尋找?guī)臄?shù)據(jù)結(jié)構(gòu) *@paramp_fine_frame:指向打包幀數(shù)據(jù)結(jié)構(gòu)體變量 *@paramdst_buf:指向幀緩沖區(qū) *@paramsfd:幀首標(biāo)志,一般為0xFF或者0xEE */ voidinit_find_frame_struct(find_frame_struct*p_find_frame,uint8_t*dst_buf,uint8_tsfd) { 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ù)個數(shù) *@paramp_find_frame:指向打包幀數(shù)據(jù)結(jié)構(gòu)體變量 *@paramsrc_buf:指向串口接收的原始數(shù)據(jù) *@paramdata_len:src_buf本次串口接收到的原始數(shù)據(jù)個數(shù) *@paramsum_len:幀緩存的最大長度 *@return本次處理的數(shù)據(jù)個數(shù) */ uint32_tfind_one_frame(find_frame_struct*p_find_frame,constuint8_t*src_buf,uint32_tdata_len,uint32_tsum_len) { uint32_tsrc_len=0; while(data_len--) { if(p_find_frame->sfd_flag==0) {//沒有找到起始幀首 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 {//是否是"長度"字節(jié)?Y->獲取這幀的數(shù)據(jù)長度 if(7==p_find_frame->received_len) { p_find_frame->frame_len=src_buf[src_len]+5+1+1+1+2;//幀首+地址號+命令號+數(shù)據(jù)長度+校驗 if(p_find_frame->frame_len>=sum_len) {//這里處理方法根據(jù)具體應(yīng)用不一定相同 MY_DEBUGF(SLAVE_DEBUG,("數(shù)據(jù)長度超出緩存! ")); 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; returnsrc_len; } } } p_find_frame->find_fram_flag=0; returnsrc_len; }

使用例子:

定義數(shù)據(jù)結(jié)構(gòu)體變量:


		

find_frame_struct slave_find_frame_srt;

定義接收數(shù)據(jù)緩沖區(qū):


		

#defineSLAVE_REC_DATA_LEN128 uint8_tslave_rec_buf[SLAVE_REC_DATA_LEN];

在串口初始化中調(diào)用結(jié)構(gòu)體變量初始化函數(shù):


		

init_find_frame_struct(&slave_find_frame_srt,slave_rec_buf,0xEE);

在串口接收中斷中調(diào)用數(shù)據(jù)打包函數(shù):


		

find_one_frame(&slave_find_frame_srt,tmp_rec_buf,data_len,SLAVE_REC_DATA_LEN);

其中,rec_buf是串口接收臨時緩沖區(qū),data_len是本次接收的數(shù)據(jù)長度。

4. 數(shù)據(jù)發(fā)送

前文提到,傳統(tǒng)的等待發(fā)送方式會浪費CPU資源,而中斷發(fā)送方式雖然不會造成CPU資源浪費,但又增加了一個中斷源。在我們的使用中發(fā)現(xiàn),定時器中斷是幾乎每個應(yīng)用都會使用的,我們可以利用定時器中斷以及硬件FIFO來進(jìn)行數(shù)據(jù)發(fā)送,通過合理設(shè)計后,這樣的發(fā)送方法即不會造成CPU資源浪費,也不會多增加中斷源和中斷事件。

需要提前說明的是,這個方法并不是對所有應(yīng)用都合適,對于那些沒有開定時器中斷的應(yīng)用本方法當(dāng)然是不支持的,另外如果定時器中斷間隔較長而通訊波特率又特別高的話,本方法也不太適用。

公司目前使用的通訊波特率一般比較?。?200bps、2400bps),在這些波特率下,定時器間隔為10ms以下(含10ms)就能滿足。如果定時器間隔為1ms以下(含1ms),是可以使用115200bps的。

本方法主要思想是:定時器中斷觸發(fā)后,判斷是否有數(shù)據(jù)要發(fā)送,如果有數(shù)據(jù)要發(fā)送并且滿足發(fā)送條件,則將數(shù)據(jù)放入發(fā)送FIFO中,對于lpc1778來說,一次最多可以放16字節(jié)數(shù)據(jù)。之后硬件會自動啟動發(fā)送,無需CPU參與。

下面介紹如何使用定時器發(fā)送數(shù)據(jù),硬件載體為RS485。因為發(fā)送需要操作串口寄存器以及RS485方向控制引腳,需跟硬件密切相關(guān),以下代碼使用的硬件為lpc1778,但思想是通用的。

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


		

/*串口幀發(fā)送結(jié)構(gòu)體*/ typedefstruct { uint16_tsend_sum_len;//要發(fā)送的幀數(shù)據(jù)長度 uint8_tsend_cur_len;//當(dāng)前已經(jīng)發(fā)送的數(shù)據(jù)長度 uint8_tsend_flag;//是否發(fā)送標(biāo)志 uint8_t*send_data;//指向要發(fā)送的數(shù)據(jù)緩沖區(qū) }uart_send_struct;

4.2 定時處理函數(shù)


		

/** *@brief定時發(fā)送函數(shù),在定時器中斷中調(diào)用,不使用發(fā)送中斷的情況下減少發(fā)送等待 *@paramUARTx:指向硬件串口寄存器基地址 *@paramp:指向串口幀發(fā)送結(jié)構(gòu)體變量 */ #defineFARME_SEND_FALG 0x5A #defineSEND_DATA_NUM12 staticvoiduart_send_com(LPC_UART_TypeDef*UARTx,uart_send_struct*p) { uint32_ti; uint32_ttmp32; if(UARTx->LSR&(0x01<<6))//發(fā)送為空 { if(p->send_flag==FARME_SEND_FALG) { RS485ClrDE;//置485為發(fā)送狀態(tài) tmp32=p->send_sum_len-p->send_cur_len; if(tmp32>SEND_DATA_NUM)//向發(fā)送FIFO填充字節(jié)數(shù)據(jù) { for(i=0;iTHR=p->send_data[p->send_cur_len++]; } } else { for(i=0;iTHR=p->send_data[p->send_cur_len++]; } p->send_flag=0; } } else { RS485SetDE; } } }

其中,RS485ClrDE為宏定義,設(shè)置RS485為發(fā)送模式;RS485SetDE也為宏定義,設(shè)置RS485為接收模式。

使用例子:

定義數(shù)據(jù)結(jié)構(gòu)體變量:


		

uart_send_struct uart0_send_str;

定義發(fā)送緩沖區(qū):


		

uint8_tuart0_send_buf[UART0_SEND_LEN];

根據(jù)使用的硬件串口,對定時處理函數(shù)做二次封裝:


		

voiduart0_send_data(void) { uart_send_com(LPC_UART0,&uart0_send_str); }

將封裝函數(shù)uart0_send_data();放入定時器中斷處理函數(shù)中;

在需要發(fā)送數(shù)據(jù)的地方,設(shè)置串口幀發(fā)送結(jié)構(gòu)體變量:


		

uart0_send_str.send_sum_len=data_len;//data_len為要發(fā)送的數(shù)據(jù)長度 uart0_send_str.send_cur_len=0;//固定為0 uart0_send_str.send_data=uart0_send_buf;//綁定發(fā)送緩沖區(qū) uart0_send_str.send_flag=FARME_SEND_FALG;//設(shè)置發(fā)送標(biāo)志

5. 總結(jié)

本文主要討論了一種高效的串口數(shù)據(jù)收發(fā)方法,并給出了具體的代碼實現(xiàn)。在當(dāng)前處理器任務(wù)不斷增加的情況下,提供了一個占用資源少,可提高系統(tǒng)整體性能的新的思路。

審核編輯:湯梓紅


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

    關(guān)注

    6050

    文章

    44701

    瀏覽量

    641290
  • cpu
    cpu
    +關(guān)注

    關(guān)注

    68

    文章

    10951

    瀏覽量

    213947
  • RS485
    +關(guān)注

    關(guān)注

    39

    文章

    1175

    瀏覽量

    82898
  • fifo
    +關(guān)注

    關(guān)注

    3

    文章

    392

    瀏覽量

    44105
  • 串口
    +關(guān)注

    關(guān)注

    14

    文章

    1570

    瀏覽量

    77700

原文標(biāo)題:詳解:單片機串口高效收發(fā)數(shù)據(jù)的實現(xiàn)方法

文章出處:【微信號:玩點嵌入式,微信公眾號:玩點嵌入式】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    STM32單片機串口接收數(shù)據(jù)方法

    串口作為單片機開發(fā)的一個常用的外設(shè),應(yīng)用范圍非常廣。大部分時候,串口需要接收處理的數(shù)據(jù)長度是不定的。那么怎么才能判斷一幀數(shù)據(jù)是否結(jié)束呢,今天
    發(fā)表于 09-21 14:39 ?1.2w次閱讀

    51單片機串口檢測程序

    本內(nèi)容提供了51單片機串口檢測程序,詳解列出了本程序
    發(fā)表于 05-09 11:43 ?6986次閱讀

    基于STC51單片機串口收發(fā)的源代碼

    基于 STC51單片機串口收發(fā)的源代碼
    發(fā)表于 11-16 19:14 ?56次下載

    DSP與單片機串口通信的設(shè)計與實現(xiàn)

    DSP與單片機串口通信的設(shè)計與實現(xiàn)
    發(fā)表于 10-20 10:11 ?5次下載
    DSP與<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>通信的設(shè)計與<b class='flag-5'>實現(xiàn)</b>

    基于51單片機的UART串口通信

    基于51單片機的UART串口通信詳解
    發(fā)表于 11-21 10:14 ?6.4w次閱讀
    基于51<b class='flag-5'>單片機</b>的UART<b class='flag-5'>串口</b>通信

    詳細(xì)圖文剖析STM32單片機串口一鍵下載電路與操作方法

    在此介紹STM32單片機串口一鍵下載電路與操作方法詳解。
    的頭像 發(fā)表于 12-31 23:07 ?4.7w次閱讀
    詳細(xì)圖文剖析STM32<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>一鍵下載電路與操作<b class='flag-5'>方法</b>

    使用單片機進(jìn)行串口收發(fā)字符的數(shù)據(jù)類型是怎么樣的詳細(xì)資料說明

    今天在用51單片機進(jìn)行串口收發(fā)數(shù)據(jù)的時候遇到了這樣一個問題,上位單片機的字符
    發(fā)表于 08-30 17:28 ?2次下載
    使用<b class='flag-5'>單片機</b>進(jìn)行<b class='flag-5'>串口</b><b class='flag-5'>收發(fā)</b>字符的<b class='flag-5'>數(shù)據(jù)</b>類型是怎么樣的詳細(xì)資料說明

    使用單片機實現(xiàn)收發(fā)短信的設(shè)計資料說明

    借助系統(tǒng)模型,闡明GSM模塊收發(fā)短信的基本概念以及串口控制SMS的基本原理。詳細(xì)介紹單片機控制GSM模塊工作的軟件實現(xiàn)過程,對怎樣用單片機
    的頭像 發(fā)表于 07-18 11:08 ?4826次閱讀

    使用單片機實現(xiàn)串口通信的資料詳細(xì)說明

    串口通訊對單片機而言意義重大,不但可以實現(xiàn)單片機數(shù)據(jù)傳輸?shù)诫娔X端,而且也能實現(xiàn)電腦對
    的頭像 發(fā)表于 08-09 14:13 ?9987次閱讀
    使用<b class='flag-5'>單片機</b><b class='flag-5'>實現(xiàn)</b><b class='flag-5'>串口</b>通信的資料詳細(xì)說明

    如何在51單片機實現(xiàn)串口收發(fā)命令

    串口通信在電子行業(yè)中應(yīng)用較廣,通過上位發(fā)送命令,實現(xiàn)各種功能的控制及數(shù)據(jù)的反饋。本文就是在51 單片機
    發(fā)表于 11-26 17:02 ?12次下載

    51單片機串口通訊詳解

    串口,作為單片機程序開發(fā)中最常用、最方便,也是應(yīng)用最廣泛的程序調(diào)試方法;無論是作為調(diào)試工具,打印出調(diào)試信息,還是對功能模塊進(jìn)行通信,串口是每個單片機
    發(fā)表于 11-11 17:06 ?15次下載
    51<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>通訊<b class='flag-5'>詳解</b>

    串口通信詳解(51單片機

    串口傳輸,實現(xiàn)單片機與PC的雙機通信。串口控制(51單片機
    發(fā)表于 11-12 10:06 ?81次下載
    <b class='flag-5'>串口</b>通信<b class='flag-5'>詳解</b>(51<b class='flag-5'>單片機</b>)

    單片機學(xué)習(xí)筆記————51單片機實現(xiàn)串口收發(fā)

    單片機學(xué)習(xí)筆記————51單片機實現(xiàn)串口收發(fā)
    發(fā)表于 11-23 17:06 ?42次下載
    <b class='flag-5'>單片機</b>學(xué)習(xí)筆記————51<b class='flag-5'>單片機</b><b class='flag-5'>實現(xiàn)</b>從<b class='flag-5'>機</b>的<b class='flag-5'>串口</b><b class='flag-5'>收發(fā)</b>

    C++ 單片機-上位-串口收發(fā)-COM收發(fā)

    C++ 單片機-上位-串口收發(fā)-COM收發(fā)屬性值平臺Win10IDEVS2019語言C++功能:讀取COM口
    發(fā)表于 11-23 17:36 ?13次下載
    C++ <b class='flag-5'>單片機</b>-上位<b class='flag-5'>機</b>-<b class='flag-5'>串口</b><b class='flag-5'>收發(fā)</b>-COM<b class='flag-5'>收發(fā)</b>

    51單片機串口配置方法

    串口,作為單片機程序開發(fā)中最常用、最方便,也是應(yīng)用最廣泛的程序調(diào)試方法;無論是作為調(diào)試工具,打印出調(diào)試信息,還是對功能模塊進(jìn)行通信,串口是每個單片機
    的頭像 發(fā)表于 04-14 14:58 ?5454次閱讀
    51<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>配置<b class='flag-5'>方法</b>