色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

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

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

3天內不再提示

單片機串口發送數據很慢?這種方法幫助你提高!

jf_pJlTbmA9 ? 來源:嵌入式資訊精選 ? 作者:嵌入式資訊精選 ? 2023-11-02 17:43 ? 次閱讀

本文介紹如何使用帶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所示。

poYBAGIB2MSACyc5AAAVsBQEc38093.png

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

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

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

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

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

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

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

3.1 定義數據結構

typedef struct 
{  
    uint8_t * dst_buf;                  //指向接收緩存  
    uint8_t sfd;                        //幀首標志,為0xFF或者0xEE  
    uint8_t sfd_flag;                   //找到幀首,一般是3~5個FF或EE  
    uint8_t sfd_count;                  //幀首的個數,一般3~5個  
    uint8_t received_len;               //已經接收的字節數  
    uint8_t find_fram_flag;             //找到完整幀后,置1  
    uint8_t frame_len;                  //本幀數據總長度,這個區域是可選的  
}find_frame_struct;

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

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

/** 
* @brief    尋找一幀數據  返回處理的數據個數 
* @param    p_find_frame:指向打包幀數據結構體變量 
* @param    src_buf:指向串口接收的原始數據 
* @param    data_len:src_buf本次串口接收到的原始數據個數 
* @param    sum_len:幀緩存的最大長度 
* @return   本次處理的數據個數 
*/  
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)                        
        {   //沒有找到起始幀首  
            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,("數據長度超出緩存!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;  
} 

使用例子:

定義數據結構體變量:

find_frame_struct slave_find_frame_srt;

定義接收數據緩沖區:

#define SLAVE_REC_DATA_LEN  128
uint8_t slave_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資源浪費,也不會多增加中斷源和中斷事件。

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

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

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

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

4.1 定義數據結構

/*串口幀發送結構體*/  
typedef struct 
{  
    uint16_t send_sum_len;          //要發送的幀數據長度  
    uint8_t  send_cur_len;          //當前已經發送的數據長度  
    uint8_t  send_flag;             //是否發送標志  
    uint8_t * send_data;            //指向要發送的數據緩沖區  
}uart_send_struct;  

4.2 定時處理函數

/** 
* @brief    定時發送函數,在定時器中斷中調用,不使用發送中斷的情況下減少發送等待 
* @param    UARTx:指向硬件串口寄存器基地址 
* @param    p:指向串口幀發送結構體變量 
*/  
#define FARME_SEND_FALG 0x5A          
#define SEND_DATA_NUM   12  
static void uart_send_com(LPC_UART_TypeDef *UARTx,uart_send_struct *p)  
{  
    uint32_t i;  
    uint32_t tmp32;  
      
    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_t uart0_send_buf[UART0_SEND_LEN];

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

void uart0_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. 總結

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

來源:嵌入式資訊精選(作者:張巧龍)


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

    關注

    6037

    文章

    44561

    瀏覽量

    635599
  • fifo
    +關注

    關注

    3

    文章

    388

    瀏覽量

    43692
  • 串口
    +關注

    關注

    14

    文章

    1555

    瀏覽量

    76555
收藏 人收藏

    評論

    相關推薦

    什么是單片機串口通信?

    串口通信是單片機一個重要的部分,單片機和PC,單片機單片機之間的通信大都用串口
    發表于 07-25 07:36 ?6.1w次閱讀
    什么是<b class='flag-5'>單片機</b>的<b class='flag-5'>串口</b>通信?

    單片機入門匯編之單片機接收計算機串口發送數據程序免費下載

    單片機入門匯編之單片機接收計算機串口發送數據程序免費下載。
    發表于 03-07 14:54 ?14次下載
    <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>程序免費下載

    單片機入門匯編之單片機通過串口向計算機發送數據程序免費下載

    本文檔的主要內容詳細介紹的是單片機入門匯編之單片機通過串口向計算機發送數據程序免費下載。
    發表于 03-07 14:54 ?8次下載
    <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>程序免費下載

    單片機串口接收和發送數據的程序免費下載

    本文檔的主要內容詳細介紹的是進行單片機串口接收和發送數據的程序免費下載。
    發表于 05-16 17:15 ?4次下載
    <b class='flag-5'>單片機</b>的<b class='flag-5'>串口</b>接收和<b class='flag-5'>發送</b><b class='flag-5'>數據</b>的程序免費下載

    使用單片機串口發送數據幀的程序免費下載

    本文檔的主要內容詳細介紹的是使用單片機串口發送數據幀的程序免費下載。
    發表于 08-15 17:32 ?7次下載
    使用<b class='flag-5'>單片機</b>的<b class='flag-5'>串口</b><b class='flag-5'>發送</b><b class='flag-5'>數據</b>幀的程序免費下載

    如何使用51單片機進行串口通信的發送與接收

    中斷處于開放狀態,單片機都會進入串口中斷處理程序。在中斷程序中,要區分出來究竟是發送引起的中斷,還是接收引起的中斷,然后分別進行處理??吹竭^一些書籍和文章,在串口收、發
    發表于 07-08 17:41 ?18次下載
    如何使用51<b class='flag-5'>單片機</b>進行<b class='flag-5'>串口</b>通信的<b class='flag-5'>發送</b>與接收

    單片機串口LED顯示電路的資料和程序說明

    單片機來連接led顯示器的設計是經常的事情,常用的有兩種方法一是接并口顯示,這種方法占用了大量的單片機端口資源,利用8279等芯片可以實現動態顯示,程序也容易寫,在某
    的頭像 發表于 08-23 11:18 ?4291次閱讀
    <b class='flag-5'>單片機</b><b class='flag-5'>串口</b>LED顯示電路的資料和程序說明

    單片機串口發送16進制、ASCII

    單片機串口發送16進制、ASCII單片機串口是,你給的是什么格式他就發送什么格式,在使用
    發表于 11-17 10:36 ?26次下載
    <b class='flag-5'>單片機</b><b class='flag-5'>串口</b><b class='flag-5'>發送</b>16進制、ASCII

    單片機——串口通信(從串口接收多位數據保存到數組,發送多位數據串口

    單片機串口通信中,接收多位數據到數組,發送多位數據的代碼// 下面的代碼用于:單片機
    發表于 11-17 11:06 ?45次下載
    <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>多位<b class='flag-5'>數據</b>到<b class='flag-5'>串口</b>)

    51單片機串口通信(自動發送

    51單片機串口通信(自動發送)51單片機串口通信,本次實驗程序是實現字符的自動發送,該程序可以
    發表于 11-23 17:06 ?13次下載
    51<b class='flag-5'>單片機</b><b class='flag-5'>串口</b>通信(自動<b class='flag-5'>發送</b>)

    單片機串口發送數據很慢?這種方法幫助你提高

    大家好,我是張巧龍,本文介紹如何使用帶FIFO的串口來減少接收中斷次數,通過一種自定義通訊協議格式,給出幀打包方法;之后介紹一種特殊的串口數據發送
    發表于 12-02 14:36 ?0次下載
    <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>?<b class='flag-5'>這種方法</b><b class='flag-5'>幫助你</b><b class='flag-5'>提高</b>!

    單片機串口數據處理(1)——串口中斷發送數據

    實時性在嵌入式開發中的非常重要,優化MCU串口傳輸處理方式可以提高嵌入式系統的實時性。在互聯網上學習并親自實驗(基于STM32單片機)后,我將分兩次介紹優化MCU串口收發
    發表于 12-08 11:36 ?2次下載
    <b class='flag-5'>單片機</b><b class='flag-5'>串口</b><b class='flag-5'>數據</b>處理(1)——<b class='flag-5'>串口</b>中斷<b class='flag-5'>發送</b><b class='flag-5'>數據</b>

    單片機串口發送數據很慢?這種方法幫助你提高!

    本文介紹如何使用帶FIFO的串口來減少接收中斷次數,通過一種自定義通訊協議格式,給出幀打包方法;之后介紹一種特殊的串口數據發送
    發表于 02-08 15:17 ?1次下載
    <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>?<b class='flag-5'>這種方法</b><b class='flag-5'>幫助你</b><b class='flag-5'>提高</b>!

    基于單片機串口發送數據點陣屏滾動顯示仿真程序

    基于單片機串口發送數據點陣屏滾動顯示仿真設計
    發表于 05-22 15:27 ?0次下載

    單片機串口通信的接收與發送

    的原理。串口通信是通過發送和接收兩根線來實現的,分別為發送線(Tx)和接收線(Rx)。當單片機發送數據
    的頭像 發表于 12-20 14:03 ?3731次閱讀
    主站蜘蛛池模板: 99国产这里只有精品视频| 4399亚洲AV无码V无码网站| 亚洲男人97色综合久久久| 成人国产三级在线播放| 麻花传媒MD0044视频| 伊人影院网| 精子网久久国产精品| 亚洲91av| 国产在线高清视频| 亚洲 色 欧美 爱 视频 日韩| 第一次破女视频出血视频| 青青伊人久久| 扒开粉嫩的小缝末成年小美女| 麻豆第一区MV免费观看网站| 中国成人在线视频| 亚洲精品中文字幕在线| 原神美女被超污app| 精品高潮呻吟99AV无码视频| 亚洲成年人影院| 国产乱人视频在线观看| 午夜黄视频| 国产中文字幕免费观看| 亚洲精品一线二线三线无人区| 狠狠色香婷婷久久亚洲精品| 亚洲免费综合色视频| 久久re这里精品在线视频7| 曰批视频免费40分钟不要钱 | 国产大片51精品免费观看| 丝袜美女被艹| 国产在线精品一区二区网站免费| 亚洲精品久久久久AV无码林星阑| 久久草这在线观看免费| 97午夜理论片影院在线播放| 秋霞av伦理片在线观看| 国产精品久久自在自2021| 亚洲视频网站欧美视频网站| 美女漏bb| 国产精品VIDEOS麻豆TUBE| 一区二区乱子伦在线播放| 欧美亚洲日韩国产在线在线| 国产精品色无码AV在线观看|