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

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

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

3天內不再提示

單片機自定義協議FIFO高效發送數據方法

jf_pJlTbmA9 ? 來源:網絡 ? 作者:網絡 ? 2023-09-28 17:32 ? 次閱讀

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

1、概述

在此之前,先來列舉一下傳統串口數據收發的不足之處:

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

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

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

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

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所示。

1666883411323131.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,("數據長度超出緩存!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;
returnsrc_len;
}
}
}
p_find_frame->find_fram_flag=0;
returnsrc_len;
}

使用例子:定義數據結構體變量:

find_frame_structslave_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_FALG0x5A
#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_structuart0_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;//設置發送標志

來源:網絡

免責聲明:本文為轉載文章,轉載此文目的在于傳遞更多信息,版權歸原作者所有。本文所用視頻、圖片、文字如涉及作品版權問題,請聯系小編進行處理

審核編輯 黃宇

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

    關注

    68

    文章

    10879

    瀏覽量

    212183
  • fifo
    +關注

    關注

    3

    文章

    389

    瀏覽量

    43742
  • 串口
    +關注

    關注

    14

    文章

    1555

    瀏覽量

    76663
收藏 人收藏

    評論

    相關推薦

    CC2541使用廣播方式發送自定義數據

    的時間為1891854us-942490us=949364us,約為1S,協議棧的定時時間會有誤差。因此成功地定時通過廣播發送出了自定義數據,實驗成功。
    發表于 04-13 15:05

    常見的自定義協議有哪些

    上次分享的《分享一個很酷的上位軟件》中,有如下協議:有位讀者朋友問數據為什么要按這樣的格式來發。其實這是個自定義協議,...
    發表于 02-23 06:52

    嵌入式開發中自定義協議的解析與組包相關案例分享

    1、嵌入式開發中自定義協議的解析與組包  在嵌入式產品開發中,經常會遇到兩個設備之間的通信、設備與服務器的通信、設備和上位的通信等,很多時候通信協議都是
    發表于 10-27 17:01

    matlab自定義函數調用的方法

    matlab自定義函數調用的方法 命令文件/函數文件+ 函數文件 - 多
    發表于 11-29 13:14 ?88次下載

    自定義fifo接口控制器

    自定義fifo接口控制器,利用sopc builder實現。
    發表于 03-22 14:09 ?1次下載

    單片機學習筆記————51單片機實現常用的自定義串口通訊協議

    單片機學習筆記————51單片機實現常用的自定義串口通訊協議
    發表于 11-23 17:06 ?37次下載
    <b class='flag-5'>單片機</b>學習筆記————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>幫助你提高!

    C#與STM32自定義通信協議

    C#與STM32自定義通信協議功能:1.可通過C#上位對多臺STM32下位進行控制2.自定義上位
    發表于 12-24 18:59 ?37次下載
    C#與STM32<b class='flag-5'>自定義</b>通信<b class='flag-5'>協議</b>

    單片機自定義串口打印程序

    單片機自定義串口打印程序#include #include void printf(const char* format
    發表于 12-27 19:19 ?10次下載
    <b class='flag-5'>單片機</b><b class='flag-5'>自定義</b>串口打印程序

    STM32 | 分享自定義協議的一些典型例子

    上次分享的《分享一個很酷的上位軟件》中,有如下協議:有位讀者朋友問數據為什么要按這樣的格式來發。其實這是個自定義協議,...
    發表于 12-29 19:27 ?13次下載
    STM32 | 分享<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>幫助你提高!

    直接使用單片機的USART發送數據

    一些對時間精度要求非常高的場合,使用printf將會帶來一系列問題,這時,如果使用單片機的USART自定義一個協議,直接發送數據到上位
    發表于 02-11 15:36 ?4次下載
    直接使用<b class='flag-5'>單片機</b>的USART<b class='flag-5'>發送</b><b class='flag-5'>數據</b>

    基于STM32單片機自定義漢字顯示使用庫文件設計源代碼

    基于STM32單片機自定義漢字顯示使用庫文件設計源代碼
    發表于 04-26 14:30 ?0次下載

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

    摘要:本文在探討傳統數據收發不足之后,介紹如何使用帶FIFO的串口來減少接收中斷次數,通過一種自定義通訊協議格式,給出幀打包方法;之后介紹一
    的頭像 發表于 05-10 09:37 ?1886次閱讀
    詳解<b class='flag-5'>單片機</b>串口<b class='flag-5'>高效</b>收發<b class='flag-5'>數據</b>的實現<b class='flag-5'>方法</b>

    基于PIC16F877A單片機自定義無線傳輸協議和短信通信協議

    電子發燒友網站提供《基于PIC16F877A單片機自定義無線傳輸協議和短信通信協議.pdf》資料免費下載
    發表于 11-08 14:47 ?1次下載
    基于PIC16F877A<b class='flag-5'>單片機</b>的<b class='flag-5'>自定義</b>無線傳輸<b class='flag-5'>協議</b>和短信通信<b class='flag-5'>協議</b>
    主站蜘蛛池模板: 久久久国产精品免费A片蜜臀| 免费看毛片网| 赤兔CHINESE最新男18GUY| 最新国产av.在线视频| 中文字幕乱码一区AV久久| 中文字幕无码他人妻味| 99re久久免费热在线视频手机| 2018高清国产一区二区三区| 5278欧美一区二区三区| 9久久免费国产精品特黄| 超碰98人人插| 国产老头与老太hd| 久久99热这里只有精品66| 蜜臀AV人妻久久无码精品麻豆| 欧美美女论坛| 亚洲a视频在线| 中国xxxxx69| jyzzjyzzz视频国产在线观看| 丰满少妇发泄14p| 国产最新进精品视频| 老头xxx| 日本粉嫩学生毛绒绒| 亚洲第一色网| SM脚奴调教丨踩踏贱奴| 国产色精品久久人妻无码看片软件 | 夜里18款禁用的免费B站动漫| 佐山爱巨大肥臀在线| 光溜溜的美女直播软件| 狠日狠干日曰射| 欧美亚洲日韩一道免费观看| 天天色天天综合网| 最新无码二区日本专区| 国产产乱码一二三区别免费| 久久精品国产亚洲AV久五月天| 日本内射精品一区二区视频| 亚洲中文久久精品AV无码| 抽插性奴中出乳精内射| 久久亚洲这里只有精品18| 我的好妈妈8高清在线观看WWW | 97超级碰碰人妻中文字幕| 国产激情视频在线播放|