背景:RS485是最常用的工業現場通訊手段,它的傳輸字節采用了異步串口UART的規范。在通常的工控應用中,需要傳輸由多個字節組成的數據幀,而RS485并沒有對數據幀有任何規范,需要應用程序自己做數據幀的鑒別。
本文介紹在ESM6800、ESM7000和ESM8000主板上,利用iMX6/7/8串口的9bit RS485模式,實現RS485通訊網絡的整幀數據收發的功能。該功能可大大簡化應用程序接收線程的復雜性,提高RS485通訊的效率。
整幀數據擁有固定的數據長度,由地址和數據構成,地址為一個字節,其余都為數據字節,如下圖:
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 |
ESM6800E | ttyS1—ttyS6 | 適合作為Master |
ESM6800L | ttyS1—ttyS6 | 適合作為Slave |
ESM6802 | ttyS1—ttyS4 | 適合作為Master |
ESM7000 | ttyS1—ttyS6 | 適合作為Master |
如果要使用9bit RS485模式,需要在程序中進行使能,使能后串口就會進入到該模式中,在發送數據的時候,可以支持兩種方式,一種是發送地址字節,另一種是發送數據字節。在接收數據字節的時候,分為master和slave兩種模式,這兩種模式都需要先接收地址字節,才能夠接收數據字節,如果沒有接收到地址字節,會自動將數據字節自動全部濾掉。他們的區別在于master模式下,只要接收到地址字節,就會將這之后的數據字節全部接收,并交給應用程序處理。而在slave模式下,需要先設置設備地址,只有接收到的地址字節和設備地址相同時,才會開始接收數據字節。
master模式下,接收數據示意圖:
Slave模式下,接收數據示意圖:
采用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));
主板實際輸出的波形如下:
感興趣的客戶可以和英創的工程師聯系,索取完整的測試工程。
成都英創信息技術有限公司 028-8618 0660