激情综合丁香-激情综合六月-激情综合婷婷亚洲图片-激情综合图区-激情综合网五月

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


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

主站蜘蛛池模板: 亚洲欧美综合网| 国产精品久久国产精品99 gif| 一区二区三区杨幂在线观看| 日本aaaa级| 亚洲国产精品va在线观看麻豆| 欧美一级毛片欧美一级无片| 国产超级碰碰在线公开视频| 黄视频在线播放| 成人 亚洲| 国产羞羞视频| 久久香蕉国产线看观看精品蕉| 婷婷综合国产激情在线| 亚洲综合久久久久久888| 一级毛片不卡免费看老司机| 丰满的日本护士xxx| 中文字幕亚洲不卡在线亚瑟| 在线观看免费播放网址成人| 男人黄女人色视频在线观看| 国内外成人在线视频| 国产码欧美日韩高清综合一区 | 国产精品女主播自在线拍 | 伊人精品国产| 一级毛片一级毛片一级毛片| 久热在线视频| 日韩一区二三区无| 中文字幕乱码二三区免费| 在线一区播放| 网红主播大秀一区二区| 免费一级美国片在线观看| 欧美一级毛片黄| 欧美三级在线播放| 欧美日韩成人午夜免费| 欧美精品片| 欧美成人h版整片合集| 高清配种视频xxxxx| 欧美日韩亚洲国产精品| 手机看片在线精品观看| 亚洲高清一区二区三区| 亚洲国产精品婷婷久久久久| 亚洲精品99久久久久久欧美版| 亚洲精品国产一区二区图片欧美 |