• <code id="lnncl"></code>
    <th id="lnncl"><sup id="lnncl"><acronym id="lnncl"></acronym></sup></th>
        <tr id="lnncl"><option id="lnncl"></option></tr>
      1. 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


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

        国语自产拍在线视频中文,国产精品亚洲专区无码,色综合欧美在线视频区,秋霞高清视频在线直播,亚洲成AⅤ人在线观看,无遮爆乳喷汁无遮掩动漫 日韩av在线观看,无码AV一道日韩在线观看,va欧美国产在线视频,国产精品zipaitoupai,欧美A级在线现免费观看,精品亚洲国产成人,亚洲欧美偷国产日韩 97色伦图片97综合影院,无码一卡二卡三卡四卡,强乱中文字幕在线播放,日本道av无码无卡三区,亚洲AV欧美AV天堂,一本道久久综合久久爱 曰本女人牲交全视频播放,天堂Av无码Av日韩Av,老子影院午夜伦不卡,特黄A级毛片,免费 人妻 无码 不卡中文...,亚洲日韩国产有码观看 曰本女人牲交全视频播放,天堂Av无码Av日韩Av,老子影院午夜伦不卡,特黄A级毛片,免费 人妻 无码 不卡中文...,亚洲日韩国产有码观看 偷拍亚洲制服另类无码专区,AV无码无在线观看,欧美高清视频手机在在线,手机在线拍揄自揄视频,亚洲AV无码不卡无码,野狼AV午夜福利电影 欧美成人精品视频在线不卡,国产又色又爽又黄刺激免费观看,成年无码AV片在线观看蜜芽,黄页网址大全免费观看,亚洲中文字幕无码久久2017,人人澡 人人澡 人人看 2828婷婷五月,无码专区国产精品视频,性欧美牲交在线视频,国产a片在线播放 国产毛片,日本一本免费一区二区三区免,2021在线精品自偷自拍无码
        欧美肥老太牲交大片,无码中字出轨中文人妻中文中,97色中色,久久WWW免费人成_看片,天影视色香欲综合视频,中文字日本熟妇色在线观看 亚洲中文字幕无码一区在线,国产一级aa无码大片293,亚洲中文字幕人成乱码,中文字幕人妻系列人妻有码,亚洲熟女少妇乱图片区,97色在色在线播放免费 亚洲 欧美 卡通 另类 小说,精品国自产拍天天拍,亚洲一日韩欧美中文字幕在线,国内精品自在自线视频,欧美人与禽交片欧美,色AV永久无码AV影院 久久亚洲中文字幕不卡一二区,av老司机亚洲精品天堂,国产中文字幕乱码免费,一一本之道高清视频在线观看,亚洲国产天堂久久综合,国产野外无码理论片在线观看 黑粗硬大欧美在线视频,无码中文精品视视在线观看,色橹橹欧美在线观看视频高清,美国人与动性XXX杂交,美国黄色片,久热精品香蕉在线播放 97午夜理论电影影院,强乱中文字幕在线播放不卡,高潮搐痉挛潮喷AV,日本乱偷人妻中文字幕,中文无码字幕在线观看,狠狠爱天天综合色欲网 馬与人黃色毛片一部,少妇人妻偷人精品免费视频,欧美va天堂在线电影,精品国产高清自在线看,一本之道不卡av无码,国产极品久久久久极品 欧美亚洲亚洲日韩在线,亚洲精品国偷拍自产电影,日本一区二区三区不卡免费,中国大陆国产高清AⅤ毛片免费视频,怡红院在线AⅤ男人的天堂 不戴套玩新婚人妻,国产亚洲精品AA片在线观看,中文字幕不卡乱偷在线观看,第一福利在线永久视频网站,天天爽夜夜爽人人爽,香蕉视频在线精品视频 亚洲色欲色欲高清无码,欧美三级不卡在线观线看,天天躁夜夜躁狠狠躁2020,中文人妻av大区中文不卡,有码中文丝袜在线视频,日本高清乱理伦片中文字幕