2017-05-14 19:14:09 +02:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2017 by Jonathan Naylor G4KLX
|
|
|
|
|
*
|
|
|
|
|
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
*/
|
|
|
|
|
|
2017-08-02 20:30:39 +02:00
|
|
|
#include "DMRDataHeader.h"
|
|
|
|
|
#include "DMRFullLC.h"
|
|
|
|
|
#include "DMRCSBK.h"
|
2017-05-14 19:14:09 +02:00
|
|
|
#include "Rewrite.h"
|
2017-08-03 21:49:44 +02:00
|
|
|
#include "DMREMB.h"
|
2017-05-14 19:14:09 +02:00
|
|
|
|
2017-08-02 20:30:39 +02:00
|
|
|
#include <cstdio>
|
|
|
|
|
|
|
|
|
|
CRewrite::CRewrite() :
|
|
|
|
|
m_lc(),
|
|
|
|
|
m_embeddedLC(),
|
|
|
|
|
m_data(NULL),
|
|
|
|
|
m_writeNum(0U),
|
2017-08-03 21:49:44 +02:00
|
|
|
m_readNum(0U),
|
|
|
|
|
m_lastN(0U)
|
2017-08-02 20:30:39 +02:00
|
|
|
{
|
|
|
|
|
m_data = new CDMREmbeddedData[2U];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CRewrite::~CRewrite()
|
2017-05-14 19:14:09 +02:00
|
|
|
{
|
2017-08-02 20:30:39 +02:00
|
|
|
delete[] m_data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::processMessage(CDMRData& data)
|
|
|
|
|
{
|
|
|
|
|
unsigned char dataType = data.getDataType();
|
|
|
|
|
|
|
|
|
|
switch (dataType) {
|
|
|
|
|
case DT_VOICE_LC_HEADER:
|
|
|
|
|
case DT_TERMINATOR_WITH_LC:
|
|
|
|
|
processHeader(data, dataType);
|
|
|
|
|
break;
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
case DT_VOICE_SYNC:
|
|
|
|
|
processVoiceSync(data);
|
|
|
|
|
break;
|
|
|
|
|
|
2017-08-02 20:30:39 +02:00
|
|
|
case DT_VOICE:
|
|
|
|
|
processVoice(data);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DT_CSBK:
|
|
|
|
|
processCSBK(data);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DT_DATA_HEADER:
|
|
|
|
|
processDataHeader(data);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DT_RATE_12_DATA:
|
|
|
|
|
case DT_RATE_34_DATA:
|
|
|
|
|
case DT_RATE_1_DATA:
|
2017-08-03 21:49:44 +02:00
|
|
|
processData(data);
|
2017-08-02 20:30:39 +02:00
|
|
|
break;
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
case DT_VOICE_PI_HEADER:
|
2017-08-02 20:30:39 +02:00
|
|
|
default:
|
|
|
|
|
// Not sure what to do
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::setLC(FLCO flco, unsigned int srcId, unsigned int dstId)
|
|
|
|
|
{
|
|
|
|
|
if (flco == m_lc.getFLCO() && srcId == m_lc.getSrcId() && dstId == m_lc.getDstId())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
m_lc.setFLCO(flco);
|
|
|
|
|
m_lc.setSrcId(srcId);
|
|
|
|
|
m_lc.setDstId(dstId);
|
|
|
|
|
|
|
|
|
|
m_embeddedLC.setLC(m_lc);
|
|
|
|
|
|
|
|
|
|
m_readNum = 0U;
|
|
|
|
|
m_writeNum = 0U;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
void CRewrite::processEmbeddedData(unsigned char* data, unsigned char n)
|
2017-08-02 20:30:39 +02:00
|
|
|
{
|
2017-08-03 21:49:44 +02:00
|
|
|
CDMREMB emb;
|
|
|
|
|
emb.putData(data);
|
2017-08-02 20:30:39 +02:00
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
unsigned char lcss = emb.getLCSS();
|
2017-08-02 20:30:39 +02:00
|
|
|
|
|
|
|
|
m_data[m_writeNum].addData(data, lcss);
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
if (m_readNum == 0U && m_writeNum == 0U) {
|
|
|
|
|
lcss = m_embeddedLC.getData(data, n);
|
|
|
|
|
emb.setLCSS(lcss);
|
|
|
|
|
emb.getData(data);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-08-02 20:30:39 +02:00
|
|
|
|
|
|
|
|
CDMRLC* lc = m_data[m_readNum].getLC();
|
2017-08-03 21:49:44 +02:00
|
|
|
if (lc == NULL) {
|
|
|
|
|
lcss = m_embeddedLC.getData(data, n);
|
|
|
|
|
emb.setLCSS(lcss);
|
|
|
|
|
emb.getData(data);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-08-02 20:30:39 +02:00
|
|
|
|
|
|
|
|
FLCO flco = lc->getFLCO();
|
|
|
|
|
|
|
|
|
|
delete lc;
|
|
|
|
|
|
|
|
|
|
// Replace any identity embedded data with the new one
|
|
|
|
|
if (flco == FLCO_GROUP || flco == FLCO_USER_USER)
|
2017-08-03 21:49:44 +02:00
|
|
|
lcss = m_embeddedLC.getData(data, n);
|
2017-08-02 20:30:39 +02:00
|
|
|
else
|
2017-08-03 21:49:44 +02:00
|
|
|
lcss = m_data[m_readNum].getData(data, n);
|
|
|
|
|
|
|
|
|
|
emb.setLCSS(lcss);
|
|
|
|
|
emb.getData(data);
|
2017-08-02 20:30:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::swap()
|
|
|
|
|
{
|
2017-08-03 21:49:44 +02:00
|
|
|
// If we get a voice sync straight after a voice header (or another voice sync)
|
|
|
|
|
if (m_lastN == 0U)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-08-02 20:30:39 +02:00
|
|
|
if (m_readNum == 0U && m_writeNum == 0U) {
|
|
|
|
|
m_writeNum = 1U;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (m_readNum == 0U)
|
|
|
|
|
m_readNum = 1U;
|
|
|
|
|
else
|
|
|
|
|
m_readNum = 0U;
|
|
|
|
|
|
|
|
|
|
if (m_writeNum == 0U)
|
|
|
|
|
m_writeNum = 1U;
|
|
|
|
|
else
|
|
|
|
|
m_writeNum = 0U;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::processHeader(CDMRData& data, unsigned char dataType)
|
|
|
|
|
{
|
|
|
|
|
setLC(data.getFLCO(), data.getSrcId(), data.getDstId());
|
|
|
|
|
|
|
|
|
|
unsigned char buffer[DMR_FRAME_LENGTH_BYTES];
|
|
|
|
|
data.getData(buffer);
|
|
|
|
|
|
|
|
|
|
CDMRFullLC fullLC;
|
|
|
|
|
fullLC.encode(m_lc, buffer, dataType);
|
|
|
|
|
|
|
|
|
|
data.setData(buffer);
|
2017-08-03 21:49:44 +02:00
|
|
|
|
|
|
|
|
m_lastN = 0U;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::processVoiceSync(CDMRData& data)
|
|
|
|
|
{
|
|
|
|
|
swap();
|
|
|
|
|
|
|
|
|
|
m_lastN = 0U;
|
2017-08-02 20:30:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::processVoice(CDMRData& data)
|
|
|
|
|
{
|
|
|
|
|
setLC(data.getFLCO(), data.getSrcId(), data.getDstId());
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
unsigned char n = data.getN();
|
|
|
|
|
|
|
|
|
|
// In case we missed a voice sync packet
|
|
|
|
|
if (n < m_lastN)
|
|
|
|
|
swap();
|
|
|
|
|
|
2017-08-02 20:30:39 +02:00
|
|
|
unsigned char buffer[DMR_FRAME_LENGTH_BYTES];
|
|
|
|
|
data.getData(buffer);
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
processEmbeddedData(buffer, n);
|
2017-08-02 20:30:39 +02:00
|
|
|
|
|
|
|
|
data.setData(buffer);
|
2017-08-03 21:49:44 +02:00
|
|
|
|
|
|
|
|
m_lastN = n;
|
2017-08-02 20:30:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CRewrite::processDataHeader(CDMRData& data)
|
|
|
|
|
{
|
|
|
|
|
unsigned char buffer[DMR_FRAME_LENGTH_BYTES];
|
|
|
|
|
data.getData(buffer);
|
|
|
|
|
|
|
|
|
|
CDMRDataHeader dataHeader;
|
|
|
|
|
bool ret = dataHeader.put(buffer);
|
|
|
|
|
if (!ret)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
dataHeader.setGI(data.getFLCO() == FLCO_GROUP);
|
|
|
|
|
dataHeader.setSrcId(data.getSrcId());
|
|
|
|
|
dataHeader.setDstId(data.getDstId());
|
|
|
|
|
|
|
|
|
|
dataHeader.get(buffer);
|
|
|
|
|
|
|
|
|
|
data.setData(buffer);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-03 21:49:44 +02:00
|
|
|
void CRewrite::processData(CDMRData& data)
|
|
|
|
|
{
|
|
|
|
|
// Nothing to do
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-02 20:30:39 +02:00
|
|
|
void CRewrite::processCSBK(CDMRData& data)
|
|
|
|
|
{
|
|
|
|
|
unsigned char buffer[DMR_FRAME_LENGTH_BYTES];
|
|
|
|
|
data.getData(buffer);
|
|
|
|
|
|
|
|
|
|
CDMRCSBK csbk;
|
|
|
|
|
bool ret = csbk.put(buffer);
|
|
|
|
|
if (!ret)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
csbk.setGI(data.getFLCO() == FLCO_GROUP);
|
|
|
|
|
csbk.setSrcId(data.getSrcId());
|
|
|
|
|
csbk.setDstId(data.getDstId());
|
|
|
|
|
|
|
|
|
|
csbk.get(buffer);
|
|
|
|
|
|
|
|
|
|
data.setData(buffer);
|
2017-05-14 19:14:09 +02:00
|
|
|
}
|