You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

208 lines
5.9 KiB

// Copyright © 2015 Jean-Luc Deltombe (LX3JL). All rights reserved.
// urfd -- The universal reflector
// Copyright © 2021 Thomas A. Early N7TAE
//
// 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 3 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, see <https://www.gnu.org/licenses/>.
#include "Main.h"
#include "Packet.h"
// default constructor
CPacket::CPacket()
{
m_uiStreamId = 0;
m_uiDstarPacketId = 0;
m_uiDmrPacketId = 0;
m_uiDmrPacketSubid = 0;
m_uiYsfPacketId = 0;
m_uiYsfPacketSubId = 0;
m_uiYsfPacketFrameId = 0;
m_uiM17FrameNumber = 0;
m_cModule = ' ';
m_eOrigin = EOrigin::local;
m_eCodecIn = ECodecType::none;
m_bLastPacket = false;
};
// for the network
unsigned int CPacket::GetNetworkSize()
{
return 20u;
}
CPacket::CPacket(const CBuffer &buf)
{
if (buf.size() > 19)
{
auto data = buf.data();
m_eCodecIn = (ECodecType)data[4];
m_eOrigin = (EOrigin)data[5];
m_bLastPacket = data[6] ? true : false;
m_cModule = data[7];
m_uiStreamId = data[8]*0x100u + data[9];
m_uiM17FrameNumber = data[10]*0x1000000u + data[11]*0x10000u + data[12]*0x100 + data[13];
m_uiDstarPacketId = data[14];
m_uiDmrPacketId = data[15];
m_uiDmrPacketSubid = data[16];
m_uiYsfPacketId = data[17];
m_uiYsfPacketSubId = data[18];
m_uiYsfPacketFrameId = data[19];
}
else
std::cerr << "CPacket initialization failed because the buffer is too small!" << std::endl;
}
void CPacket::EncodeInterlinkPacket(const char *magic, CBuffer &buf) const
{
buf.Set(magic);
buf.resize(20);
auto data = buf.data();
data[4] = (uint8_t)m_eCodecIn;
data[5] = (uint8_t)m_eOrigin;
data[6] = m_bLastPacket ? 1 : 0;
data[7] = m_cModule;
data[8] = m_uiStreamId / 0x100u;
data[9] = m_uiStreamId % 0x100u;
data[10] = (m_uiM17FrameNumber / 0x1000000u) % 0x100u;
data[11] = (m_uiM17FrameNumber / 0x10000u) % 0x100u;
data[12] = (m_uiM17FrameNumber / 0x100u) % 0x100u;
data[13] = m_uiM17FrameNumber % 100u;
data[14] = m_uiDstarPacketId;
data[15] = m_uiDmrPacketId;
data[16] = m_uiDmrPacketSubid;
data[17] = m_uiYsfPacketId;
data[18] = m_uiYsfPacketSubId;
data[19] = m_uiYsfPacketFrameId;
}
// dstar contstructor
CPacket::CPacket(uint16_t sid, uint8_t dstarpid)
{
m_uiStreamId = sid;
m_uiDstarPacketId = dstarpid;
m_uiDmrPacketId = 0xFF;
m_uiDmrPacketSubid = 0xFF;
m_uiYsfPacketId = 0xFF;
m_uiYsfPacketSubId = 0xFF;
m_uiYsfPacketFrameId = 0xFF;
m_uiM17FrameNumber = 0xFFFFFFFFU;
m_cModule = ' ';
m_eOrigin = EOrigin::local;
m_eCodecIn = ECodecType::dstar;
m_bLastPacket = (0x40U == (dstarpid & 0x40U));
};
// dmr constructor
CPacket::CPacket(uint16_t sid, uint8_t dmrpid, uint8_t dmrspid, bool lastpacket)
{
m_uiStreamId = sid;
m_uiDmrPacketId = dmrpid;
m_uiDmrPacketSubid = dmrspid;
m_uiDstarPacketId = 0xFF;
m_uiYsfPacketId = 0xFF;
m_uiYsfPacketSubId = 0xFF;
m_uiYsfPacketFrameId = 0xFF;
m_uiM17FrameNumber = 0xFFFFFFFFU;
m_cModule = ' ';
m_eOrigin = EOrigin::local;
m_eCodecIn = ECodecType::dmr;
m_bLastPacket = lastpacket;
};
// ysf constructor
CPacket::CPacket(uint16_t sid, uint8_t ysfpid, uint8_t ysfsubpid, uint8_t ysffrid, bool lastpacket)
{
m_uiStreamId = sid;
m_uiYsfPacketId = ysfpid;
m_uiYsfPacketSubId = ysfsubpid;
m_uiYsfPacketFrameId = ysffrid;
m_uiDstarPacketId = 0xFF;
m_uiDmrPacketId = 0xFF;
m_uiDmrPacketSubid = 0xFF;
m_uiM17FrameNumber = 0xFFFFFFFFU;
m_cModule = ' ';
m_eOrigin = EOrigin::local;
m_eCodecIn = ECodecType::dmr;
m_bLastPacket = lastpacket;
}
// bm constructor
CPacket::CPacket(uint16_t sid, uint8_t dstarpid, uint8_t dmrpid, uint8_t dmrsubpid, uint8_t ysfpid, uint8_t ysfsubpid, uint8_t ysffrid, ECodecType codecIn, bool lastpacket)
{
m_uiStreamId = sid;
m_uiDstarPacketId = dstarpid;
m_uiDmrPacketId = dmrpid;
m_uiDmrPacketSubid = dmrsubpid;
m_uiYsfPacketId = ysfpid;
m_uiYsfPacketSubId = ysfsubpid;
m_uiYsfPacketFrameId = ysffrid;
m_uiM17FrameNumber = 0xFFFFFFFFU;
m_cModule = ' ';
m_eOrigin = EOrigin::local;
m_eCodecIn = codecIn;
m_bLastPacket = lastpacket;
}
// m17 constructor
CPacket::CPacket(const CM17Packet &m17) : CPacket()
{
m_uiStreamId = m17.GetStreamId();
m_uiDstarPacketId = 0xFF;
m_uiDmrPacketId = 0xFF;
m_uiDmrPacketSubid = 0xFF;
m_uiYsfPacketId = 0xFF;
m_uiYsfPacketSubId = 0xFF;
m_uiYsfPacketFrameId = 0xFF;
m_eCodecIn = (0x6U == (0x6U & m17.GetFrameType())) ? ECodecType::c2_1600 : ECodecType::c2_3200;
m_uiM17FrameNumber = 0xFFFFU & m17.GetFrameNumber();
m_bLastPacket = m17.IsLastPacket();
}
////////////////////////////////////////////////////////////////////////////////////////
// pid conversion
void CPacket::UpdatePids(const uint32_t pid)
{
// called while pushing this packet in a stream queue
// so now packet sequence number is known and undefined pids can be updated
// this is needed as dtsar & dmr pids are different and cannot be
// derived from each other
// dstar pid needs update ?
if ( m_uiDstarPacketId == 0xFF )
{
m_uiDstarPacketId = (pid % 21);
}
// dmr pids need update ?
if ( m_uiDmrPacketId == 0xFF )
{
m_uiDmrPacketId = ((pid / 3) % 6);
m_uiDmrPacketSubid = ((pid % 3) + 1);
}
// ysf pids need update ?
if ( m_uiYsfPacketId == 0xFF )
{
m_uiYsfPacketId = ((pid / 5) % 8);
m_uiYsfPacketSubId = pid % 5;
m_uiYsfPacketFrameId = ((pid / 5) & 0x7FU) << 1;
}
// m17 needs update?
if (m_uiM17FrameNumber == 0xFFFFFFFFU)
{
// frames are every 20 milliseconds, so the m17 data will come every 40 milliseconds
m_uiM17FrameNumber = (pid / 2) % 0x8000U;
}
}

Powered by TurnKey Linux.