<wbr id="ht9fj"><ins id="ht9fj"></ins></wbr>

        RS485網絡的整幀數據收發

         2020-7-28     作者:黃志超    

          背景:RS485是最常用的工業現場通訊手段,它的傳輸字節采用了異步串口UART的規范。在通常的工控應用中,需要傳輸由多個字節組成的數據幀,而RS485并沒有對數據幀有任何規范,需要應用程序自己做數據幀的鑒別。


          本文介紹在ESM6800、ESM7000和ESM8000主板上,利用iMX6/7/8串口的9bit RS485模式,實現RS485通訊網絡的整幀數據收發的功能。該功能可大大簡化應用程序接收線程的復雜性,提高RS485通訊的效率。


          整幀數據擁有固定的數據長度,由地址和數據構成,地址為一個字節,其余都為數據字節,如下圖:


        RS485網絡的整幀數據收發.png


          9bit RS485模式使用了串口固定校驗位的功能,定義了地址字節和數據字節,地址字節是指固定校驗位始終為1的字節。而數據字節則是指固定校驗位始終為0的字節。同時9bit RS485模式實現了一些硬件過濾的功能,在接收的時候,必須要先接收到地址字節才會開始接收數據字節,否則硬件會將收到的數據字節全部過濾掉,通過這種方式降低了設備的負載。所以9bit RS485模式顧名思義,通常使用在RS485模式上面,因為RS485可以作為總線掛接多個設備,在多路設備通訊的情況下通過這種校驗方式可以有效的降低設備負載和軟件的復雜程度。


          英創工控主板中,能夠支持9bit RS485模式的主板和串口如下表,其中ES6801/ES6801L和ESM6800L這三款核心板能夠滿足低成本的需求,可以考慮作為RS485網絡中的Slave端:


        主板型號支持9bit RS485模式的串口備注
        ES6801(L)ttyS1—ttyS6適合作為Slave
        ESM6800(H)ttyS1—ttyS5適合作為Master
        ESM6800EttyS1—ttyS6適合作為Master
        ESM6800LttyS1—ttyS6適合作為Slave
        ESM6802ttyS1—ttyS4適合作為Master
        ESM7000ttyS1—ttyS6適合作為Master


          如果要使用9bit RS485模式,需要在程序中進行使能,使能后串口就會進入到該模式中,在發送數據的時候,可以支持兩種方式,一種是發送地址字節,另一種是發送數據字節。在接收數據字節的時候,分為master和slave兩種模式,這兩種模式都需要先接收地址字節,才能夠接收數據字節,如果沒有接收到地址字節,會自動將數據字節自動全部濾掉。他們的區別在于master模式下,只要接收到地址字節,就會將這之后的數據字節全部接收,并交給應用程序處理。而在slave模式下,需要先設置設備地址,只有接收到的地址字節和設備地址相同時,才會開始接收數據字節。


          master模式下,接收數據示意圖:


        RS485網絡的整幀數據收發-2.png


          Slave模式下,接收數據示意圖:


        RS485網絡的整幀數據收發-3.png


          采用9bit RS485模式,有兩個優點,第一點是不需要判斷是否接收到地址字節,因為串口要在接收到地址字節(校驗位為1)后,才會接收數據字節,特別是在slave模式下,只有當地址字節和設置的設備地址相等時,才會接收數據。第二點是不需要切換校驗方式,當串口啟用了9bit RS485模式,就可以正常接收所有地址字節和數據字節了,只有在發送地址字節和數據字節的時候需要切換不同的設置,可以減少軟件上的操作。


          英創公司在提供的例程Step2_serialtest中封裝的串口類CSerial的基礎上派生出一個專用于9bit RS485的類CRS485,在這個類中我們增加使能9bit RS485模式的函數,讓客戶可以直接調用來實現相關功能。


        /**
         *    派生用于9bit RS485的類
         *
        **/
        class CRS485 : public CSerial
        {
        private:
               //串口模式、設備地址和接收超時時間
               int serial_mode;
               int serial_addr;
         
        public:
               //接收數據緩存和長度
               char frame[100];
               int  frame_len;
         
               /**
                *    派生類的構造函數
                *
                *    在構造函數中初始化變量,以及設置9-bit RS485模式下的串口是處于master還是slave模式
                *
                *    參數說明:
                *    mode:值為0對應master模式,值為1對應slave模式
                *    addr:設備地址,大小為8bit,當且僅當mode為1是有效。
                *
               **/
               CRS485(int mode, int addr);
         
               /**
                *    發送9bit RS485整包數據
                *
                *    函數會將地址字節和數據字節填寫,并設置為相應的模式一并發送
                *
                *    參數說明:
                *    addr:設備地址,大小為8bit,填入發送數據的地址字節中
                *    Buf:發送的數據字節
                *    len:發送數據字節的長度
                *
                *    返回值說明:
                *    len:成功
                *    -1:失敗
                *
               **/
               int send_rs485_frame(char addr, char *Buf, int len);
         
               /**
                *    接收9bit RS485整包數據
                *
                *    函數會阻塞接收指定長度的數據,可以設置超時時間,如果超過超時時間沒有接收到指定長度的數據,則返回-1
                *
                *    參數說明:
                *    Buf:接收的數據字節
                *    len:發送數據字節的長度
                *    timeout:超時時間,單位毫秒。如果在超時時間內沒有收到指定長度的數據,則返回-1。值為0則不阻塞,讀取不到數據立即返回。值為-1則沒有超時時間,如果接受不到指定長度數據會一直等待
                *
                *    返回值說明:
                *    成功則返回接收到的數據長度
                *    -1:超時
                *
               **/
               int recv_rs485_frame(char *Buf, int len, int timeout);
         
               /**
                *    繼承自CSerial類的接收處理函數
                *
                *    在CSerial類的接收線程中會調用這個函數,可以在函數中調用recv_rs485_frame()函數,并處理接收到的數據字節
                *     
                *
               **/
               int PackagePro();
        };


          在類實例化的時候,代入參數就可以決定串口是處于master模式還是slave模式,如果是出于slave模式可以一起代入需要設定的設備地址:


        //master模式
        class CRS485  m_Serial(0, 0);
         
        //slave模式,設備地址為0x55
        class CRS485  m_Serial(1, 0x55);


          接收處理的時候,數據的長度通過宏DATA_LEN定義,客戶可以在PackagePro()函數中可以定義超時時間,然后調用recv_rs485_frame()函數來接收整包數據,recv_rs485_frame()函數會阻塞,直至收到指定長度的數據,或者到達超時時間才會返回。接收到整包數據后,就可以開始進行數據的處理,在接收線程調中循環調用PackagePro函數:


        #define DATA_LEN 10                    // 數據長度
         
        // 接收串口數據處理函數
        int CRS485::PackagePro()
        {
               int i1, timeout;
         
               //設置超時時間,單位毫秒
               timeout = 500;
         
               //調用接收函數來獲取指定長度的整包數據
               i1 = recv_rs485_frame(DatBuf, m_DatLen, timeout);
         
               //接收到整包數據,調用處理程序,這里只是簡單的打印
               if(i1 != -1)
               {
                      printf("frame addr = 0x%x\n", frame[0]);
                      printf("frame data = ");
                      for(i1=1; i1<DATA_LEN; i1++) {
                             printf("0x%x ", frame[i1]);
                      }
                      printf("\n");
         
                      //處理完數據,清除各個變量,重新設置串口以等待下一包數據
                      memset(frame, 0, 100);
                      frame_len = 0;
               }
               else
                      printf("time out!\n");
         
               return i1;
        }


          在線程中的處理,循環調用接收處理函數即可,因為recv_rs485_frame()函數會阻塞,直至收到指定長度的數據,或者到達超時時間才會返回:


        int CSerial::ReceiveThreadFunc(void* lparam)
        {
               CSerial *pSer = (CSerial*)lparam;
         
               //定義讀事件集合
               fd_set fdRead;
               int ret;
               struct timeval     aTime;
         
               while( 1 )
               {
                      //接收處理函數
                      pSer->PackagePro( pSer->DatBuf, pSer->m_DatLen);
         
               }
         
               printf( "ReceiveThreadFunc finished\n");
               pthread_exit( NULL );
               return 0;
        }


          串口在發送的時候,比較簡單,直接調用send_rs485_frame()函數,填入需要發送的地址和數據即可,使用下面的代碼來測試:


        char        addr = 0x55;
        char    Buf[2];
         
        Buf[0] = 0x55;
        Buf[1] = 0xaa;
         
        //發送地址字節和數據字節
        m_Serial.send_rs485_frame(addr, Buf, sizeof(Buf));


          主板實際輸出的波形如下:


        RS485網絡的整幀數據收發.png


          感興趣的客戶可以和英創的工程師聯系,索取完整的測試工程。

        日韩欧美亚洲每日更新在线,草草线在线禁18成年在线,全黄一级A片免费看,一本大道香蕉九九99在线视频,日本亚洲AV综合网图片,亚洲老汉色Av影院在线 色天使久久,AV喷水高潮喷水在线观看,久久婷婷五月综合色d啪,欧美一级旡码高清在线↘,高清性色生活片视频在线观看,国产欧美日韩一区二区赛车 日本视频一区在线播放,天天躁夜夜躁狠狠夜夜澡,性色av,成年性午夜免费视频网站,国产人成午夜免电影费观看,思思99思思久久最新精品三级 免费国产凹凸在线视频,国产无套护士在线观看,精品国产高清自在线看,国产高清狼人香蕉在线,亚洲一级毛片无码专区,国产一级aa无码大片293 思思99在热线女精品视频,色狠狠色偷偷色噜噜色,奇米影色777四色在线,亚洲欧洲日韩淙合久久,无码高潮尖叫av,奇米影视第四色 首页 成人在线免费视频,大香线蕉伊人精品超碰,白嫩少妇喷水正在播放,免费岛国在线播放x片,在线观看91精品国产,久久婷香五月综合色啪 久久久噜噜噜久久熟女色,欧美老汉色老汉首页a亚洲,亚洲国产欧洲综合997久久,日本无码高清中文字幕视频,亚洲аv电影天堂网无码,国产在线观看香蕉视频 国产v综合v亚洲欧美大,777米奇影院狠狠色六月,亚洲国产欧美在线看片,亚欧乱色熟女一区二区三区,手机看片日本在线观看视频,中文字幕在线视频一字幕 色悠久久久久综合网国产,欧美成人午夜免费全部完,免费看黄色三级片,av老司机午夜福利片免费观看,无码中文有码中文人妻中文,欧美 亚洲 国产 日韩 综合aⅴ
        亚洲18禁速播影院,久久精品人人槡人妻人人玩,国产在线精选免费视频,免费一本色道久久一区,福利一区二区微拍视频,综合亚洲伊人午夜网 日本中文字幕有码在线视频,欧美刺激性视频黄页,2020国产在线视精品在亚洲,日韩AV在线观看A片,午夜性刺激在线看免费,久久综合久久自在自线精品自 色综合欧美五月俺也去,凹凸视频分类老司机视频分类,加勒比色老久久综合网,免费特黄一级欧美大片久久网,男女性色大片免费网站,国产思思99re99在线观看 国产亚洲欧美综合在线区yw,天天综合网视频网站,免费不卡国产福利在线观看,国产欧美亚洲精品a,久久亚洲中文字幕不卡一二区,奇米影视7777久久精品 日韩欧美亚洲每日更新在线,草草线在线禁18成年在线,全黄一级A片免费看,一本大道香蕉九九99在线视频,日本亚洲AV综合网图片,亚洲老汉色Av影院在线 成人视频手机免费播放,久久色综合在,亚洲一区二区经典在线播放,日韩中文字幕精品乱码,狠狠色噜狠狠狠狠的777米奇,日韩A片R级无码中文 亚洲国产日韩欧美综合另类bd,色天天综合色天天久久婷婷,91国产自拍视频,奇米四色狠狠中文字幕,亚洲不卡av不卡一区二区,奇米影视777狠狠色 色综合伊人色综合网站,小草在线观看免费播放,国产精品视频二区不卡,无码中文在线二区免费,777米奇色狠狠俺去啦,久久国产美女精品久久 成人在线免费视频,大香线蕉伊人精品超碰,白嫩少妇喷水正在播放,免费岛国在线播放x片,在线观看91精品国产,久久婷香五月综合色啪 欧美日韩在线亚洲综合国产人,亚洲男人AV天堂男人社区,老色鬼综合网在线观看,欧美一级日韩一级亚洲一级,高清性色生活片在线视频在线播放