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.
dvmhost/nxdn/data/Layer3.cpp

223 lines
6.0 KiB

/**
* Digital Voice Modem - Host Software
* GPLv2 Open Source. Use is subject to license terms.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* @package DVM / Host Software
*
*/
//
// Based on code from the MMDVMHost project. (https://github.com/g4klx/MMDVMHost)
// Licensed under the GPLv2 License (https://opensource.org/licenses/GPL-2.0)
//
/*
* Copyright (C) 2018 by Jonathan Naylor G4KLX
* Copyright (C) 2022 by Bryan Biedenkapp N2PLL
*
* 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.
*/
#include "nxdn/NXDNDefines.h"
#include "nxdn/data/Layer3.h"
#include "Utils.h"
using namespace nxdn;
using namespace nxdn::data;
#include <cstdio>
#include <cassert>
#include <cstring>
// ---------------------------------------------------------------------------
// Public Class Members
// ---------------------------------------------------------------------------
/// <summary>
/// Initializes a copy instance of the Layer3 class.
/// </summary>
/// <param name="data"></param>
Layer3::Layer3(const Layer3& data) :
m_verbose(false),
m_messageType(MESSAGE_TYPE_IDLE),
m_srcId(0U),
m_dstId(0U),
m_group(true),
m_dataBlocks(0U),
m_data(NULL)
{
m_data = new uint8_t[22U];
::memcpy(m_data, data.m_data, 22U);
m_messageType = m_data[0U] & 0x3FU; // Message Type
m_group = (m_data[2U] & 0x80U) != 0x80U;
m_srcId = (m_data[3U] << 8) | m_data[4U]; // Source Radio Address
m_dstId = (m_data[5U] << 8) | m_data[6U]; // Target Radio Address
m_dataBlocks = (m_data[8U] & 0x0FU) + 1U; // Data Blocks
}
/// <summary>
/// Initializes a new instance of the Layer3 class.
/// </summary>
Layer3::Layer3() :
m_verbose(false),
m_messageType(MESSAGE_TYPE_IDLE),
m_srcId(0U),
m_dstId(0U),
m_group(true),
m_dataBlocks(0U),
m_data(NULL)
{
m_data = new uint8_t[22U];
::memset(m_data, 0x00U, 22U);
}
/// <summary>
/// Finalizes a instance of Layer3 class.
/// </summary>
Layer3::~Layer3()
{
delete[] m_data;
}
/// <summary>
/// Equals operator.
/// </summary>
/// <param name="data"></param>
/// <returns></returns>
Layer3& Layer3::operator=(const Layer3& data)
{
if (&data != this) {
::memcpy(m_data, data.m_data, 22U);
m_verbose = data.m_verbose;
m_messageType = data.m_messageType;
m_group = data.m_group;
m_srcId = data.m_srcId;
m_dstId = data.m_dstId;
m_dataBlocks = m_dataBlocks;
}
return *this;
}
/// <summary>
/// Decode layer 3 data.
/// </summary>
/// <param name="data"></param>
/// <returns>True, if SACCH was decoded, otherwise false.</returns>
void Layer3::decode(const uint8_t* data, uint32_t length, uint32_t offset)
{
assert(data != NULL);
for (uint32_t i = 0U; i < length; i++, offset++) {
bool b = READ_BIT(data, i);
WRITE_BIT(m_data, offset, b);
}
if (m_verbose) {
Utils::dump(2U, "Decoded Layer 3 Data", m_data, 22U);
}
m_messageType = m_data[0U] & 0x3FU; // Message Type
m_group = (m_data[2U] & 0x80U) != 0x80U;
m_srcId = (m_data[3U] << 8) | m_data[4U]; // Source Radio Address
m_dstId = (m_data[5U] << 8) | m_data[6U]; // Target Radio Address
m_dataBlocks = (m_data[8U] & 0x0FU) + 1U; // Data Blocks
}
/// <summary>
/// Encode layer 3 data.
/// </summary>
/// <param name="data"></param>
void Layer3::encode(uint8_t* data, uint32_t length, uint32_t offset)
{
assert(data != NULL);
m_data[0U] = m_messageType & 0x3FU; // Message Type
m_data[2U] = (m_group ? 0x80U : 0x00U);
m_data[3U] = (m_srcId >> 8U) & 0xFFU; // Source Radio Address
m_data[4U] = (m_srcId >> 0U) & 0xFFU; // ...
m_data[5U] = (m_dstId >> 8U) & 0xFFU; // Target Radio Address
m_data[6U] = (m_dstId >> 0U) & 0xFFU; // ...
m_data[8U] = m_dataBlocks & 0x0FU; // Data Blocks
for (uint32_t i = 0U; i < length; i++, offset++) {
bool b = READ_BIT(m_data, offset);
WRITE_BIT(data, i, b);
}
if (m_verbose) {
Utils::dump(2U, "Encoded Layer 3 Data", data, length);
}
}
/// <summary>
///
/// </summary>
void Layer3::reset()
{
::memset(m_data, 0x00U, 22U);
m_messageType = MESSAGE_TYPE_IDLE;
m_srcId = 0U;
m_dstId = 0U;
m_group = true;
m_dataBlocks = 0U;
}
/// <summary>
/// Gets the raw layer 3 data.
/// </summary>
/// <param name="data"></param>
void Layer3::getData(uint8_t* data) const
{
::memcpy(data, m_data, 22U);
}
/// <summary>
/// Sets the raw layer 3 data.
/// </summary>
/// <param name="data"></param>
/// <param name="length"></param>
void Layer3::setData(const uint8_t* data, uint32_t length)
{
::memset(m_data, 0x00U, 22U);
::memcpy(m_data, data, length);
m_messageType = m_data[0U] & 0x3FU;
m_group = (m_data[2U] & 0x80U) != 0x80U;
m_srcId = (m_data[3U] << 8) | m_data[4U];
m_dstId = (m_data[5U] << 8) | m_data[6U];
m_dataBlocks = (m_data[8U] & 0x0FU) + 1U;
}

Powered by TurnKey Linux.