老司機(jī)分享一個(gè)從機(jī)通信查詢(xún)方式,適用于嵌入式以及上位機(jī)
查詢(xún)通信包是否結(jié)束,有很多的方式,看過(guò)很多,都不夠簡(jiǎn)潔明了,我分享一個(gè)我用了幾年的查詢(xún)方式(大家如果有什么好的方式可以與我分享),可以給新手做通信的一些啟發(fā),便于移植,同時(shí)簡(jiǎn)單明了:對(duì)于使用了操作系統(tǒng),直接在線程中查詢(xún),裸機(jī)就直接使用定時(shí)器進(jìn)行查詢(xún)。
比如使用了ucos ii時(shí),我一般會(huì)建立一個(gè)線程用來(lái)通信查詢(xún)(串口使用DMA,完全無(wú)需中斷干預(yù),高效)
while(1) { cnt?=?UARTx_GetRxCnt(RS485_UART_CH1); //獲取接收數(shù)據(jù)長(zhǎng)度 OSTimeDlyHMSM(0,0,0,50); //延時(shí),等待數(shù)據(jù)結(jié)束 if((cnt?!=?0)?&&?(cnt?==?UARTx_GetRxCnt(RS485_UART_CH1))) //收到數(shù)據(jù)了,并且2次查詢(xún)到的數(shù)據(jù)長(zhǎng)度一樣,判斷為幀結(jié)束 { //收到數(shù)據(jù)包后的處理 if(cnt>3) { if(CONFIG_HostHandle(UartBuff,?cnt,?MODBUS_SendData)==FALSE) //先判斷是否為上位機(jī)通信 { if(MODBUS_SLAVE_Handle(&SlaveHandle,?UartBuff,?cnt)==TRUE) //MODBUS?從機(jī)通信處理 { if(LastWriteRegCnt?!=?SlaveHandle.WriteRegCnt) { LastWriteRegCnt?=?SlaveHandle.WriteRegCnt; MODBUS_InputReg[7]?=?MODBUS_HoldReg[0]; //調(diào)試模式 } } } } UARTx_ClearRxCnt(RS485_UART_CH1); //清除接收的數(shù)據(jù) } else //2次查詢(xún)的數(shù)據(jù)長(zhǎng)度不一樣,延時(shí),等待數(shù)據(jù)結(jié)束 { OSTimeDlyHMSM(0,0,0,50); } }
上位機(jī)等待通信包結(jié)束
上位機(jī)一般作為主機(jī),使用了類(lèi)似的方法查詢(xún)數(shù)據(jù)結(jié)束
DWORD?cnt?=?0; DWORD?TimeOut?=?500?/?50; //超時(shí)為800ms //等待數(shù)據(jù)返回 do { cnt?=?this->pUART->MYUART_GetRxCnt(this->mUartHandle); //獲取接收到的數(shù)據(jù)長(zhǎng)度 Sleep(50); //延時(shí)10ms if?(cnt?==?this->pUART->MYUART_GetRxCnt(this->mUartHandle)) //完成接收數(shù)據(jù)了,退出等待 { TimeOut--; if?((cnt?>?0)?&&?(TimeOut?!=?0)) { if?(cnt?>?30) { Sleep(200); //收完后再等待200ms防止CH340這類(lèi)串口分包導(dǎo)致數(shù)據(jù)丟失,串口波特率不一樣時(shí)等待的實(shí)際會(huì)不一樣,大數(shù)據(jù)包等待的時(shí)間會(huì)更長(zhǎng) } Sleep(20); //收完后再等待20ms防止PL2303這類(lèi)串口分包導(dǎo)致數(shù)據(jù)丟失 TimeOut?=?1; //數(shù)據(jù)接收完畢,退出 } } }?while?(TimeOut);
//裸機(jī)下的操作
//使用了一個(gè)定時(shí)器,產(chǎn)生50ms中斷進(jìn)行輪詢(xún)
void?TIM6_IRQHandler(void) { static?u32?cnt; //一定要用靜態(tài) if(TIM6->SR&BIT0)//溢出中斷 { TIM6->SR?=?0; //清除中斷標(biāo)志位? TIM6->CR1?&=?~BIT0; //關(guān)閉定時(shí)器6 if((MODEBUS_GetDataCnt(ModeBusHandle.UartCh)?==?cnt)?&&?(cnt?>?0)) //數(shù)據(jù)長(zhǎng)度不為0,并且2次查詢(xún)的一樣,則認(rèn)為幀結(jié)束 { //uart_printf("收到數(shù)據(jù):%drn",cnt); if((cnt?>?3)?&&?(MODEBUS_SLAVE_Handle(&ModeBusHandle,?cnt)?==?TRUE))//MODBUS從機(jī)通信處理 { } MODEBUS_ClearRxCnt(ModeBusHandle.UartCh); //清除接收 } cnt?=?MODEBUS_GetDataCnt(ModeBusHandle.UartCh); //獲取接收數(shù)據(jù)長(zhǎng)度,同下次進(jìn)行對(duì)比 ? TIM6->CR1?|=?BIT0;?//開(kāi)啟定時(shí)器6} } }