吴忠躺衫网络科技有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

詳解單片機串口高效收發數據的實現方法

Dp1040 ? 來源:玩轉嵌入式 ? 2023-05-10 09:37 ? 次閱讀

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

1. 簡介

串口由于使用簡單,價格低廉,配合RS485芯片可以實現長距離、抗干擾能力強的局域網絡而被廣泛使用。隨著產品功能的增多,需要處理的任務也越來越復雜,系統任務也越來越需要及時響應。

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

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

(2)應答數據采用等待發送的方法。由于串行數據傳輸的時間遠遠跟不上CPU的處理時間,等待串口發送完當前字節再發送下一字節會造成CPU資源浪費,不利于系統整體響應(在1200bps下,發送一字節大約需要10ms,如果一次發送幾十個字節數據,CPU會長時間處于等待狀態)。

(3)應答數據采用中斷發送。增加一個中斷源,增加系統的中斷次數,這會影響系統整體穩定性(從可靠性角度考慮,中斷事件應越少越好)。

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

2. 串口FIFO

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

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

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

3. 數據接收與打包

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

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

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

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

自定義協議格式如圖3-1所示。

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

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

  • 地址號:要進行通訊的設備的地址編號,1字節

  • 命令號:對應不同的功能,1字節

  • 長度:數據區域的字節個數,1字節

  • 數據:與具體的命令號有關,數據區長度可以為0,整個幀的長度不應超過256字節

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

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

3.1 定義數據結構


		

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

3.2 初始化數據結構,一般放在串口初始化中


		

/** *@brief初始化尋找幀的數據結構 *@paramp_fine_frame:指向打包幀數據結構體變量 *@paramdst_buf:指向幀緩沖區 *@paramsfd:幀首標志,一般為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 數據打包程序


		

/** *@brief尋找一幀數據返回處理的數據個數 *@paramp_find_frame:指向打包幀數據結構體變量 *@paramsrc_buf:指向串口接收的原始數據 *@paramdata_len:src_buf本次串口接收到的原始數據個數 *@paramsum_len:幀緩存的最大長度 *@return本次處理的數據個數 */ 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 {//是否是"長度"字節?Y->獲取這幀的數據長度 if(7==p_find_frame->received_len) { p_find_frame->frame_len=src_buf[src_len]+5+1+1+1+2;//幀首+地址號+命令號+數據長度+校驗 if(p_find_frame->frame_len>=sum_len) {//這里處理方法根據具體應用不一定相同 MY_DEBUGF(SLAVE_DEBUG,("數據長度超出緩存! ")); 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; }

使用例子:

定義數據結構體變量:


		

find_frame_struct slave_find_frame_srt;

定義接收數據緩沖區:


		

#defineSLAVE_REC_DATA_LEN128 uint8_tslave_rec_buf[SLAVE_REC_DATA_LEN];

在串口初始化中調用結構體變量初始化函數:


		

init_find_frame_struct(&slave_find_frame_srt,slave_rec_buf,0xEE);

在串口接收中斷中調用數據打包函數:


		

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

其中,rec_buf是串口接收臨時緩沖區,data_len是本次接收的數據長度。

4. 數據發送

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

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

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

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

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

4.1 定義數據結構


		

/*串口幀發送結構體*/ typedefstruct { uint16_tsend_sum_len;//要發送的幀數據長度 uint8_tsend_cur_len;//當前已經發送的數據長度 uint8_tsend_flag;//是否發送標志 uint8_t*send_data;//指向要發送的數據緩沖區 }uart_send_struct;

4.2 定時處理函數


		

/** *@brief定時發送函數,在定時器中斷中調用,不使用發送中斷的情況下減少發送等待 *@paramUARTx:指向硬件串口寄存器基地址 *@paramp:指向串口幀發送結構體變量 */ #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))//發送為空 { if(p->send_flag==FARME_SEND_FALG) { RS485ClrDE;//置485為發送狀態 tmp32=p->send_sum_len-p->send_cur_len; if(tmp32>SEND_DATA_NUM)//向發送FIFO填充字節數據 { 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為宏定義,設置RS485為發送模式;RS485SetDE也為宏定義,設置RS485為接收模式。

使用例子:

定義數據結構體變量:


		

uart_send_struct uart0_send_str;

定義發送緩沖區:


		

uint8_tuart0_send_buf[UART0_SEND_LEN];

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


		

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

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

在需要發送數據的地方,設置串口幀發送結構體變量:


		

uart0_send_str.send_sum_len=data_len;//data_len為要發送的數據長度 uart0_send_str.send_cur_len=0;//固定為0 uart0_send_str.send_data=uart0_send_buf;//綁定發送緩沖區 uart0_send_str.send_flag=FARME_SEND_FALG;//設置發送標志

5. 總結

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

審核編輯:湯梓紅


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 單片機
    +關注

    關注

    6043

    文章

    44622

    瀏覽量

    638550
  • cpu
    cpu
    +關注

    關注

    68

    文章

    10902

    瀏覽量

    213014
  • RS485
    +關注

    關注

    39

    文章

    1165

    瀏覽量

    82594
  • fifo
    +關注

    關注

    3

    文章

    389

    瀏覽量

    43857
  • 串口
    +關注

    關注

    14

    文章

    1557

    瀏覽量

    77041

原文標題:詳解:單片機串口高效收發數據的實現方法

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

收藏 人收藏

    評論

    相關推薦

    STM32單片機串口接收數據方法

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

    51單片機串口檢測程序

    本內容提供了51單片機串口檢測程序,詳解列出了本程序
    發表于 05-09 11:43 ?6967次閱讀

    基于STC51單片機串口收發的源代碼

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

    DSP與單片機串口通信的設計與實現

    DSP與單片機串口通信的設計與實現
    發表于 10-20 10:11 ?5次下載
    DSP與<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>通信的設計與<b class='flag-5'>實現</b>

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

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

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

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

    使用單片機進行串口收發字符的數據類型是怎么樣的詳細資料說明

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

    使用單片機實現收發短信的設計資料說明

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

    使用單片機實現串口通信的資料詳細說明

    串口通訊對單片機而言意義重大,不但可以實現單片機數據傳輸到電腦端,而且也能實現電腦對
    的頭像 發表于 08-09 14:13 ?9907次閱讀
    使用<b class='flag-5'>單片機</b><b class='flag-5'>實現</b><b class='flag-5'>串口</b>通信的資料詳細說明

    如何在51單片機實現串口收發命令

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

    51單片機串口通訊詳解

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

    串口通信詳解(51單片機

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

    單片機學習筆記————51單片機實現串口收發

    單片機學習筆記————51單片機實現串口收發
    發表于 11-23 17:06 ?42次下載
    <b class='flag-5'>單片機</b>學習筆記————51<b class='flag-5'>單片機</b><b class='flag-5'>實現</b>從<b class='flag-5'>機</b>的<b class='flag-5'>串口</b><b class='flag-5'>收發</b>

    C++ 單片機-上位-串口收發-COM收發

    C++ 單片機-上位-串口收發-COM收發屬性值平臺Win10IDEVS2019語言C++功能:讀取COM口
    發表于 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'>收發</b>-COM<b class='flag-5'>收發</b>

    51單片機串口配置方法

    串口,作為單片機程序開發中最常用、最方便,也是應用最廣泛的程序調試方法;無論是作為調試工具,打印出調試信息,還是對功能模塊進行通信,串口是每個單片機
    的頭像 發表于 04-14 14:58 ?5325次閱讀
    51<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>配置<b class='flag-5'>方法</b>
    百家乐官网2号干扰| 百家乐官网出千技巧| 神话百家乐官网的玩法技巧和规则| 五张百家乐的玩法技巧和规则| 百家乐官网游戏机高手| 百家乐下载免费软件| 通渭县| 百家乐室系统软件| 百家乐官网娱乐城网站| 百家乐二游戏机| 阳新县| LV百家乐娱乐城| 网络百家乐官网游赌博| 大发888开户xa11| 百家乐破解视频| 网络百家乐官网最安全| 百家乐视| 利都百家乐官网国际赌场娱乐网规则| 大发888熊之舞怎么玩| 百家乐最常见的路子| 百家乐官网视频网络游戏| 大发888开户即送58| 金博士百家乐官网的玩法技巧和规则 | 试玩百家乐官网1000| 香港六合彩彩色图库| 金海岸百家乐娱乐城| 百家乐官网百家乐官网视频游戏世界| 大发888客户端的软件| 百佬汇百家乐官网的玩法技巧和规则 | 三公百家乐官网玩法| 滕州市| 大发888投注明升网址| 网上百家乐怎么赌能赢钱| 娱乐百家乐官网可以代理吗| 呼图壁县| 娱乐城送体验金38元| 大发888备用网址大发娱乐城| 哪家百家乐最好| 百家乐是否能赢| 百家乐论坛代理合作| 百家乐注册就送|