/******************************************************************************\ * Copyright (c) 2004-2006 * * Author(s): * Volker Fischer * Protocol message definition MAIN FRAME ---------- +------------+------------+------------------+--------------+-------------+ | 2 bytes ID | 1 byte cnt | 2 bytes length n | n bytes data | 2 bytes CRC | +------------+------------+------------------+--------------+-------------+ - message ID defined by the defines PROTMESSID_x - cnt: counter which is increment for each message and wraps around at 255 - length n in bytes of the data - actual data, dependent on message type - 16 bits CRC, calculating over the entire message, is transmitted inverted Generator polynom: G_16(x) = x^16 + x^12 + x^5 + 1, initial state: all ones MESSAGES -------- - Jitter buffer size: PROTMESSID_JITT_BUF_SIZE +--------------------------+ | 2 bytes number of blocks | +--------------------------+ * ****************************************************************************** * * This program is free software; you can redistribute it and/or modify it under * the terms of the GNU General Public License as published by the Free Software * Foundation; either version 2 of the License, or (at your option) any later * version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License along with * this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * \******************************************************************************/ #include "protocol.h" /* Implementation *************************************************************/ bool CClientProtocol::ParseMessage ( const CVector& vecbyData, const int iNumBytes ) { /* return code: true -> ok; false -> error */ int iRecCounter, iRecID; CVector vecData; // convert unsigned char in uint8_t, TODO convert all buffers in uint8_t CVector vecbyDataConv ( vecbyData.Size () ); for ( int i = 0; i < vecbyData.Size (); i++ ) { vecbyDataConv[i] = static_cast ( vecbyData[i] ); } if ( ParseMessageFrame ( vecbyDataConv, iRecCounter, iRecID, vecData ) ) { switch ( iRecID ) { case PROTMESSID_ACKN: // TODO implement acknowledge code -> do implementation in CProtocol since // it can be used in the server protocol, too break; } return true; // everything was ok } else { return false; // return error code } } void CClientProtocol::CreateJitBufMes ( const int iJitBufSize ) { CVector vecData ( 2 ); unsigned int iPos = 0; // build data vector PutValOnStream ( vecData, iPos, static_cast ( iJitBufSize ), 2 ); // build complete message GenMessageFrame ( vecMessage, iCounter, PROTMESSID_JITT_BUF_SIZE, vecData ); } /******************************************************************************\ * Message generation (parsing) * \******************************************************************************/ bool CProtocol::ParseMessageFrame ( const CVector& vecIn, int& iCnt, int& iID, CVector& vecData ) { /* return code: true -> ok; false -> error */ int iLenBy, i; unsigned int iCurPos; // query length of input vector const int iVecInLenByte = vecIn.Size(); // vector must be at least "MESS_LEN_WITHOUT_DATA_BYTE" bytes long if ( iVecInLenByte < MESS_LEN_WITHOUT_DATA_BYTE ) { return false; // return error code } // decode header ----- iCurPos = 0; // start from beginning /* 2 bytes ID */ iID = static_cast ( GetValFromStream ( vecIn, iCurPos, 2 ) ); /* 1 byte cnt */ iCnt = static_cast ( GetValFromStream ( vecIn, iCurPos, 1 ) ); /* 2 bytes length */ iLenBy = static_cast ( GetValFromStream ( vecIn, iCurPos, 2 ) ); // make sure the length is correct if ( iLenBy != iVecInLenByte - MESS_LEN_WITHOUT_DATA_BYTE ) { return false; // return error code } // now check CRC ----- CCRC CRCObj; const int iLenCRCCalc = MESS_HEADER_LENGTH_BYTE + iLenBy; iCurPos = 0; // start from beginning for ( i = 0; i < iLenCRCCalc; i++ ) { CRCObj.AddByte ( static_cast ( GetValFromStream ( vecIn, iCurPos, 1 ) ) ); } if ( CRCObj.GetCRC () != GetValFromStream ( vecIn, iCurPos, 2 ) ) { return false; // return error code } // decode data ----- iCurPos = MESS_HEADER_LENGTH_BYTE; // start from beginning of data for ( i = 0; i < iLenBy; i++ ) { vecData[i] = static_cast ( GetValFromStream ( vecIn, iCurPos, 1 ) ); } return true; // everything was ok } uint32_t CProtocol::GetValFromStream ( const CVector& vecIn, unsigned int& iPos, const unsigned int iNumOfBytes ) { /* note: iPos is automatically incremented in this function */ // 4 bytes maximum since we return uint32 ASSERT ( ( iNumOfBytes > 0 ) && ( iNumOfBytes <= 4 ) ); ASSERT ( vecIn.Size() >= iPos + iNumOfBytes ); uint32_t iRet = 0; for ( int i = 0; i < iNumOfBytes; i++ ) { iRet |= vecIn[iPos] << ( i * 8 /* size of byte */ ); iPos++; } return iRet; } void CProtocol::GenMessageFrame ( CVector& vecOut, const int iCnt, const int iID, const CVector& vecData ) { int i; // query length of data vector const int iDataLenByte = vecData.Size(); // total length of message = 7 + "iDataLenByte" // 2 byte ID + 1 byte cnt + 2 byte length + n bytes data + 2 bytes CRC const int iTotLenByte = 7 + iDataLenByte; // init message vector vecOut.Init( iTotLenByte ); // encode header ----- unsigned int iCurPos = 0; // init position pointer /* 2 bytes ID */ PutValOnStream ( vecOut, iCurPos, static_cast ( iID ), 2 ); /* 1 byte cnt */ PutValOnStream ( vecOut, iCurPos, static_cast ( iCnt ), 1 ); /* 2 bytes length */ PutValOnStream ( vecOut, iCurPos, static_cast ( iDataLenByte ), 2 ); // encode data ----- for ( i = 0; i < iDataLenByte; i++ ) { PutValOnStream ( vecOut, iCurPos, static_cast ( vecData[i] ), 1 ); } // encode CRC ----- CCRC CRCObj; iCurPos = 0; // start from beginning const int iLenCRCCalc = MESS_HEADER_LENGTH_BYTE + iDataLenByte; for ( i = 0; i < iLenCRCCalc; i++ ) { CRCObj.AddByte ( static_cast ( GetValFromStream ( vecOut, iCurPos, 1 ) ) ); } PutValOnStream ( vecOut, iCurPos, static_cast ( CRCObj.GetCRC () ), 2 ); } void CProtocol::PutValOnStream ( CVector& vecIn, unsigned int& iPos, const uint32_t iVal, const unsigned int iNumOfBytes ) { /* note: iPos is automatically incremented in this function */ // 4 bytes maximum since we use uint32 ASSERT ( ( iNumOfBytes > 0 ) && ( iNumOfBytes <= 4 ) ); ASSERT ( vecIn.Size() >= iPos + iNumOfBytes ); for ( int i = 0; i < iNumOfBytes; i++ ) { vecIn[iPos] = ( iVal >> ( i * 8 /* size of byte */ ) ) & 255 /* 11111111 */; iPos++; } }