/** ****************************************************************************** * @file : KLink.c * @brief : K-Link Protocol program body ****************************************************************************** */ #include "KLink.h" #include "functions.h" #include "string.h" #include "PLCFunctions.h" #include "stm32f0xx_hal.h" unsigned char KLPacketBuf1[256]; unsigned char KLPacketBuf2[256]; unsigned char KLBufferIn[16]={0}; unsigned char KLBufferOut[16]={0}; unsigned char nKLAddr=1; unsigned char nKLSeq=0; unKLStat nKLStatus ={0}; int KLThisuS=0; int KLRecvTimeuS=0; unsigned char KLBCC(void * pData, int nSize) { unsigned char k; k=0; for (int i=0;iRplyStSgn=KLSignReply; p1->DstAddr=dst; p1->nStatus=Status; p1->nRplyCMD=nCmd; p1->nSize1=DataLen; memcpy(p1->Datas,pData,DataLen); p1->Datas[DataLen]=BCC(p1,sizeof(stKLRplyPktHdr)+DataLen-1); PacketLenth=sizeof(stKLRplyPktHdr)+DataLen; switch (nCmd) { case KLCmdNone: PacketLenth=0; break; case KLCmdPing: // case KLCmdPingReply: //p1->data[DataLen+1]=KLEndSign; //PacketLenth=sizeof(stKLPacket)+DataLen+1; break; case KLCmdRead: // case KLCmdReadReply: //p1->data[DataLen+1]=KLEndSign; //PacketLenth=sizeof(stPacket)+DataLen+1; break; case KLCmdWrite: // case KLCmdWriteReply: break; case KLCmdErrRply: break; default: break; } return PacketLenth; } int KLCheckPacket(void * pBuf,int nLen1) { pKLReqPktHdr p1 = (pKLReqPktHdr) pBuf; if (p1->ReqStSgn != KLSignStart) { return 1; } int DataLen=0; //p1->LoadLen; if (DataLen>KLMaxPacketLength) { return 2; } if (nLen1data[DataLen+1] != EndSign) // { // ChnStats[nCurPollId].NoEndErr++; // Uart2Stat.LengthErr++; // return -2; // } unsigned char thisBCC=KLBCC(p1,nLen1-1); if (thisBCC != ((uchar *)pBuf)[nLen1-1]) {//BCC Error; return 4; } return 0; } inline void SetAddrBit(unsigned short * pW, unsigned char bitAddr) { (*pW)|=bitMasks[bitAddr&0xf]; } inline void ResetBit(unsigned short * pW, unsigned char bitAddr) { (*pW)&=~bitMasks[bitAddr&0xf]; } static inline void SetBitValue(unsigned short * pW, unsigned char bitAddr, unsigned char Value) { if (Value) { SetAddrBit(pW, bitAddr);} else {ResetBit(pW, bitAddr);} } int KLParseReqPacket(void * pBuf,int Len1) { pKLReqPktHdr p1 = (pKLReqPktHdr)pBuf; int nDstAddr=p1->DstAddr; KLRecvTimeuS=KLThisuS; int nDataType=p1->nType1; int nAddr=0; int DataLen=0; //p1->LoadLen; //int nSrcAddr=p1->SrcAddr; nKLStatus.nSEQ = ((pKLStat)(&(p1->Stat)))->nSEQ;; void * pData=0; pKLRplyPktHdr p2=(pKLRplyPktHdr)KLPacketBuf2; char rData[4]; int PacketLen=0; KLRecvTimeuS=KLThisuS; if (nDstAddr==nKLAddr || nDstAddr==0xff) { switch (p1->nCMD) { case KLCmdNone: break; case KLCmdPing: DataLen=nDataType; // KLBufferIn[0]=p1->Params[0]; // p1->Params[0]=KLBufferOut[0]; PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,KLCmdPing,DataLen,p1->Params); SendPacket1(p2,PacketLen); break; // case KLCmdPingReply: // break; case KLCmdInfo: //if (nDataType == KLDataTypeDT) DataLen= 4; pData=&KMem.nEventCount; PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,p1->nCMD,DataLen,pData); SendPacket1(p2,PacketLen); break; case KLCmdRead: case KLCmdReadDataByte: nAddr=p1->Params[0]+ (p1->Params[1]<<8); DataLen= p1->Params[2]; if (nDataType ==KLDataTypeDT) { pData=KMem.DT+nAddr; } else if (nDataType == KLDataTypeSDT) { pData=KMem.SDT+nAddr; } else if (nDataType == KLDataTypeWX) { pData=KMem.WX+nAddr; } else if (nDataType == KLDataTypeWY) { pData=KMem.WY+nAddr; } else if (nDataType == KLDataTypeWR) { pData=KMem.WR+nAddr; } else if (nDataType == KLDataTypeWLX) { pData=KMem.WLX+nAddr; } else if (nDataType == KLDataTypeWLY) { pData=KMem.WLY+nAddr; } else if (nDataType == KLDataTypeSV) { pData=KMem.SDT+nAddr; } else if (nDataType == KLDataTypeEV) { pData=KMem.SDT+nAddr; } else if (nDataType == KLDataTypeTest) { pData=KMem.SDT+nAddr; } else if (nDataType == KLDataSysCfg) { pData = (unsigned short *)&KMSysCfg + nAddr;} else if (nDataType == KLDataTypeFlash) { pData = (unsigned short *)FLASH_BASE + nAddr;} else { pData=KLBufferOut+nAddr; } PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,p1->nCMD,DataLen,pData); SendPacket1(p2,PacketLen); break; // case KLCmdReadReply: // break; case KLCmdWrite: case KLCmdWriteDataByte: //memcpy(DispBuf,p1->data,DataLen); nAddr=p1->Params[0]+ (p1->Params[1]<<8); DataLen= p1->Params[2]; if (nDataType ==KLDataTypeDT) { pData=KMem.DT+nAddr; } else if (nDataType == KLDataTypeSDT) { pData=KMem.SDT+nAddr; } else if (nDataType == KLDataTypeWX) { pData=KMem.WX+nAddr; } else if (nDataType == KLDataTypeWY) { pData=KMem.WY+nAddr; } else if (nDataType == KLDataTypeWR) { pData=KMem.WR+nAddr; } else if (nDataType == KLDataTypeWLX) { pData=KMem.WLX+nAddr; } else if (nDataType == KLDataTypeWLY) { pData=KMem.WLY+nAddr; } else if (nDataType == KLDataTypeSV) { pData=KMem.SDT+nAddr; DataLen=0;} else if (nDataType == KLDataTypeEV) { pData=KMem.SDT+nAddr; DataLen=0;} else if (nDataType == KLDataTypeTest) { pData=KMem.SDT+nAddr; DataLen=0;} else if (nDataType == KLDataSysCfg) { pData = (unsigned short *)&KMSysCfg + nAddr;} else if (nDataType == KLDataTypeFlash) { pData = (unsigned short *)FLASH_BASE + nAddr;} else { pData=KLBufferOut+nAddr; DataLen=0; } memcpy(pData,p1->Params+4,DataLen); PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,p1->nCMD,0,0); SendPacket1(p2,PacketLen); break; // case KLCmdWriteReply: // break; case KLCmdRead1Bit: nAddr=p1->Params[0]+ (p1->Params[1]<<8); DataLen = 1; if (nDataType == KLCoilTypeX) { rData[0] = ((KMem.WX[nAddr>>4]&bitMasks[nAddr&0x0f])>0);} else if (nDataType == KLCoilTypeY) { rData[0] = ((KMem.WY[nAddr>>4]&bitMasks[nAddr&0x0f])>0);} else if (nDataType == KLCoilTypeR) { rData[0] = ((KMem.WR[nAddr>>4]&bitMasks[nAddr&0x0f])>0);} else if (nDataType == KLCoilTypeLX) { rData[0] = ((KMem.WLX[nAddr>>4]&bitMasks[nAddr&0x0f])>0);} else if (nDataType == KLCoilTypeLY) { rData[0] = ((KMem.WLY[nAddr>>4]&bitMasks[nAddr&0x0f])>0);} else if (nDataType == KLCoilTypeT) { rData[0] = PLCMem.Timers[nAddr].bTon;} else if (nDataType == KLCoilTypeC) { rData[0] = PLCMem.Timers[nAddr].bTon;} else if (nDataType == KLCoilTypeLR) { rData[0] = 0;} else {rData[0]=0;} PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,KLCmdRead1Bit,DataLen,rData); SendPacket1(p2,PacketLen); break; case KLCmdWrite1Bit: nAddr=p1->Params[0]+ (p1->Params[1]<<8); if (nDataType == KLCoilTypeX) { SetBitValue( &KMem.WX[nAddr>>4],nAddr&0x0f,p1->Params[2]);} else if (nDataType == KLCoilTypeY) { SetBitValue( &KMem.WY[nAddr>>4],nAddr&0x0f,p1->Params[2]);} else if (nDataType == KLCoilTypeR) { SetBitValue( &KMem.WR[nAddr>>4],nAddr&0x0f,p1->Params[2]);} else if (nDataType == KLCoilTypeLX) {SetBitValue( &KMem.WLX[nAddr>>4],nAddr&0x0f,p1->Params[2]);} else if (nDataType == KLCoilTypeLY) {SetBitValue( &KMem.WLY[nAddr>>4],nAddr&0x0f,p1->Params[2]);} else if (nDataType == KLCoilTypeT) { PLCMem.Timers[nAddr].bTon = p1->Params[2];} else if (nDataType == KLCoilTypeC) { PLCMem.Timers[nAddr].bTon = p1->Params[2];} else if (nDataType == KLCoilTypeLR) { ;} else {rData[0]=0;} DataLen=0; PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,KLCmdWrite1Bit,DataLen,rData); SendPacket1(p2,PacketLen); break; case KLCmdReadBits: break; case KLCmdWriteBits: break; case KLCmdSaveSysCfg: WriteSysCfgToFlash(&KMSysCfg); PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,KLCmdSaveSysCfg,0,0); SendPacket1(p2,PacketLen); break; case KLCmdSaveRunStat: SaveRunStat(&KMRunStat); PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,KLCmdSaveRunStat,0,0); SendPacket1(p2,PacketLen); break; case KLCmdGetEventLogCount: DataLen= 4; pData=&KMem.nEventCount; PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,p1->nCMD,DataLen,pData); SendPacket1(p2,PacketLen); break; case KLCmdGetEventLog: nAddr=p1->Params[0]+ (p1->Params[1]<<8); DataLen= p1->Params[2] * sizeof(stEventLog); pData=GetEventLogAddr(nAddr); PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,p1->nCMD,DataLen,pData); SendPacket1(p2,PacketLen); break; case KLCmdClearEventLog: ClearEventLog(); PacketLen=KLMakeRplyPacket(p2,nDstAddr,nKLStatus.StatByte,KLCmdClearEventLog,0,0); SendPacket1(p2,PacketLen); break; default: DataLen=1; rData[0]=KL_UNKNOWN; PacketLen=KLMakeRplyPacket(p2,nKLAddr,nKLStatus.StatByte,KLCmdErrRply,DataLen,rData); SendPacket1(p2,PacketLen); break; } } return 0; } int KLParsePacket(void * pBuf,int Len1) { pKLReqPktHdr p1 = (pKLReqPktHdr)pBuf; KLThisuS=GetuS(); pKLRplyPktHdr p2=(pKLRplyPktHdr)KLPacketBuf2; int PacketLen=Len1; int Result=KLCheckPacket(p1,Len1); if (Result != KL_OK) { int DataLen=4; char rData[4]; rData[0]=Result; PacketLen=KLMakeRplyPacket(p2,nKLAddr,nKLStatus.StatByte,KLCmdErrRply,DataLen,rData); // PacketLen=KLMakePacket(p2,0,nAddr,cmdKLPing,p1->nSEQ,DataLen,rData); SendPacket1(p2,PacketLen); return Result; } // if (bMaster) Result=ParseMasterPacket(p1,Len1); // memcpy(KLPacketBuf2,pBuf,Len1); // SendPacket1(p2,PacketLen); Result=KLParseReqPacket(p1,Len1); //int len1=p1->PacketLen; // if (p1->DstAddr!=255&&p1->DstAddr!=2) return -3; // pPacket p2=(pPacket)PacketBuf2; // Uart2Stat.OKPacket++; return Result; }