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