From a6e5c60a9ec30574bf7c3d584046444b15f8680f Mon Sep 17 00:00:00 2001 From: zhyinch <zhyinch@gmail.com> Date: 星期四, 10 二月 2022 16:29:20 +0800 Subject: [PATCH] V1.61 修改成中断模式,测试不丢包。 --- 源码/核心板/Src/OnChipDevices/Usart.c | 153 +++++++++++++++++++++++++++++++++++++++++++++----- 1 files changed, 136 insertions(+), 17 deletions(-) diff --git "a/\346\272\220\347\240\201/\346\240\270\345\277\203\346\235\277/Src/OnChipDevices/Usart.c" "b/\346\272\220\347\240\201/\346\240\270\345\277\203\346\235\277/Src/OnChipDevices/Usart.c" index 0718116..8fe3b68 100644 --- "a/\346\272\220\347\240\201/\346\240\270\345\277\203\346\235\277/Src/OnChipDevices/Usart.c" +++ "b/\346\272\220\347\240\201/\346\240\270\345\277\203\346\235\277/Src/OnChipDevices/Usart.c" @@ -1,6 +1,7 @@ #include "Usart.h" #include <string.h> #include <stdio.h> +#include "modbus.h" //数据发送队列变量 EUART_Frame m_EUART_TxFrames[EUART_TX_FRM_SIZE]; //数据发送帧队列 @@ -10,6 +11,8 @@ //DMA数据接收缓存 uint8_t m_EUART_DMA_RXBuf[EUART_RX_BUF_SIZE]; //DMA数据接收缓存 volatile int32_t m_EUART_DMA_RXPtr = 0; //当前数据地址 +int16_t DMA_rxtemp = 0,DMA_rxtemp2 = 0; //当前数据地址缓存 +u16 datadelaycount = 0; //剩余1位数据延时等待处理 //标志变量 volatile uint8_t m_bEUARTPushingFrms = 0; //正在往发送队列存数据 volatile uint8_t m_bEUARTCheckingSend = 0; //正在确认数据发送 @@ -17,7 +20,6 @@ volatile uint8_t m_bEUARTTxEn = 0; //使能发送 void (*Usart1ParseDataCallback)(uint8_t); - void Usart1InitVariables(void) { m_EUART_TxFrm_FreeFrmLen = EUART_TX_FRM_SIZE-1; @@ -83,7 +85,7 @@ //初始化管脚 GPIO_InitStructure.GPIO_Pin = EU_RX_PIN; - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //Rx上拉输入 + GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IN_FLOATING; //Rx上拉输入 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(EU_RX_GPIO, &GPIO_InitStructure); @@ -94,7 +96,7 @@ GPIO_InitStructure.GPIO_Pin = EU_485_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //Tx推挽复用输出 GPIO_Mode_AF_PP GPIO_Init(EU_485_GPIO, &GPIO_InitStructure); - + OUT485_DISABLE; #ifdef UART_GPIO_REMAP_ENABLE GPIO_PinRemapConfig(UART_GPIO_REMAP, ENABLE); #endif @@ -121,12 +123,17 @@ UART_DMAConfiguration(); } - +extern u8 cmd_mode; +u8 overrun_flag=0; +u16 t1,t2,t3,t4; void UART_CheckReceive(void) { int32_t DMACnt = 0; int32_t MaxDataLen = EUART_RX_BUF_SIZE; - + u8 rxbuf_len = 0; + u8 buftemp1 = 0; + u8 buftemp2 = 0; + u8 i = 0; //如果正在往发送队列中添加数据,退出 if(m_bEUARTPushingFrms) return; @@ -139,20 +146,129 @@ m_bEUARTCheckingRec = 0; return; } - DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR); - while( m_EUART_DMA_RXPtr != DMACnt && MaxDataLen > 0) + + DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR);//DMACnt是DMA通道接收到数据总长度,CNDTR是dma剩余未使用数量 + + if(DMACnt!=m_EUART_DMA_RXPtr) { - Usart1ParseDataCallback(m_EUART_DMA_RXBuf[m_EUART_DMA_RXPtr]); - m_EUART_DMA_RXPtr++; - if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) + //DMA有未处理数据 + DMA_rxtemp=m_EUART_DMA_RXPtr+1; + delay_us(10); + overrun_flag=0; + if( DMA_rxtemp >= EUART_RX_BUF_SIZE ) { - m_EUART_DMA_RXPtr = 0; + DMA_rxtemp = 0; } - DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR); - MaxDataLen--; - } - m_bEUARTCheckingRec = 0; + DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR);//DMACnt是DMA通道接收到数据总长度,CNDTR是dma剩余未使用数量 + if(DMACnt!=DMA_rxtemp&&(cmd_mode==0)) + { + //DMA有超过2位数据未处理 + DMA_rxtemp = 0;//清空数组指针 + datadelaycount = 0;//等待次数清零 + buftemp1 = m_EUART_DMA_RXBuf[m_EUART_DMA_RXPtr]; + buftemp2 = m_EUART_DMA_RXBuf[m_EUART_DMA_RXPtr+1]; + if((buftemp1==0x55)&&(buftemp2==0xAA)) + { + //私有协议 + cmd_mode = 1; + } + else if(buftemp1==RS485_Addr) + { + if((buftemp2==03)||(buftemp2==06)||(buftemp2==16))//功能码正确 + { + //modubs协议 + t1++; + cmd_mode=2; + DMA_rxtemp2=m_EUART_DMA_RXPtr+7; + if( DMA_rxtemp2 >= EUART_RX_BUF_SIZE-1 ) + { + DMA_rxtemp2 -= EUART_RX_BUF_SIZE; + overrun_flag=1; + } + } + else + { + m_EUART_DMA_RXPtr+=2; + if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) + { + m_EUART_DMA_RXPtr = 0; + } + } + } + else + { + //无效码,干扰码 + if(cmd_mode==0) + { + m_EUART_DMA_RXPtr+=2; + if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) + { + m_EUART_DMA_RXPtr = 0; + } + } + } + } + else + { + //只有1位数据未处理 + if(++datadelaycount > WAITDELAYCOUNT) + { //超出等待次数 + datadelaycount = 0; + cmd_mode=0; + m_EUART_DMA_RXPtr++; + if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) + { + m_EUART_DMA_RXPtr = 0; + } + } + } + if(cmd_mode==1) + { + //私有协议 + while( m_EUART_DMA_RXPtr != DMACnt && MaxDataLen > 0) + { + Usart1ParseDataCallback(m_EUART_DMA_RXBuf[m_EUART_DMA_RXPtr]); + m_EUART_DMA_RXPtr++; + if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) + { + m_EUART_DMA_RXPtr = 0; + } + DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR); + MaxDataLen--; + } + } + else if(cmd_mode==2) + { + //modbus协议 + delay_ms(10);//等待数据接收完成 + DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR); + if((overrun_flag==0&&DMACnt>DMA_rxtemp2)||(overrun_flag==1&&DMACnt<DMA_rxtemp&&DMACnt>DMA_rxtemp2)) + { while(m_EUART_DMA_RXPtr != DMACnt && MaxDataLen > 0) + { + RS485_RX_BUFF[i] = m_EUART_DMA_RXBuf[m_EUART_DMA_RXPtr]; + m_EUART_DMA_RXPtr++; + if(++i >= EUART_RX_BUF_SIZE)i = 0; + if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) + { + m_EUART_DMA_RXPtr = 0; + } + DMACnt = EUART_RX_BUF_SIZE - (EUART_RX_DMA_CH->CNDTR); + MaxDataLen--; + } +// m_EUART_DMA_RXPtr++;//数据指向下一位 +// if( m_EUART_DMA_RXPtr >= EUART_RX_BUF_SIZE ) +// { +// m_EUART_DMA_RXPtr = 0; +// } + t2++; + RS485_Service(); + } + } + } + + m_bEUARTCheckingRec = 0; } + #ifndef EXUART_USE_TXDMA @@ -190,10 +306,12 @@ m_bEUARTCheckingSend = 0; return; } - OUT485_ENABLE; + if(!m_bEUARTTxEn) { m_bEUARTTxEn = 1; + OUT485_ENABLE; + delay_us(10); // temp32 = GPIOC->CRH; //C10复用推挽输出 // temp32 &= ~(0x00000000F<<8); // temp32 |= (0x000000009<<8); @@ -215,7 +333,7 @@ } // */ #else - /* + //检查是否数据帧需要发送 void UART_CheckSend(void) { @@ -281,6 +399,7 @@ m_EUART_TxFrm_Head = 0; m_EUART_TxFrm_FreeFrmLen--; m_bEUARTPushingFrms = 0; + UART_CheckSend(); } //将最新一帧数据Pop出队列 -- Gitblit v1.9.3