|
|
|
@ -18,6 +18,10 @@
|
|
|
|
|
|
|
|
|
|
#include "WiresX.h"
|
|
|
|
|
#include "YSFPayload.h"
|
|
|
|
|
#include "YSFFICH.h"
|
|
|
|
|
#include "Sync.h"
|
|
|
|
|
#include "CRC.h"
|
|
|
|
|
#include "Log.h"
|
|
|
|
|
|
|
|
|
|
#include <cstdio>
|
|
|
|
|
#include <cassert>
|
|
|
|
@ -26,35 +30,69 @@ const unsigned char CALL_DX[] = {0x5DU, 0x71U};
|
|
|
|
|
const unsigned char CALL_CONNECT[] = {0x5DU, 0x41U};
|
|
|
|
|
const unsigned char CALL_ALL[] = {0x5DU, 0x66U};
|
|
|
|
|
|
|
|
|
|
const unsigned char DEFAULT_FICH[] = {0x20U, 0x00U, 0x15U, 0x00U};
|
|
|
|
|
|
|
|
|
|
const unsigned char NET_HEADER[] = "YSFDGATEWAY ALL ";
|
|
|
|
|
|
|
|
|
|
CWiresX::CWiresX(CNetwork* network) :
|
|
|
|
|
m_network(network),
|
|
|
|
|
m_reflector()
|
|
|
|
|
m_reflector(),
|
|
|
|
|
m_timer(1000U, 0U, 100U + 750U),
|
|
|
|
|
m_csd1(NULL)
|
|
|
|
|
{
|
|
|
|
|
assert(network != NULL);
|
|
|
|
|
|
|
|
|
|
m_csd1 = new unsigned char[20U];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CWiresX::~CWiresX()
|
|
|
|
|
{
|
|
|
|
|
delete[] m_csd1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WX_STATUS CWiresX::process(const unsigned char* data, unsigned char fi, unsigned char dt, unsigned char fn)
|
|
|
|
|
{
|
|
|
|
|
if (fi != YSF_FI_COMMUNICATIONS || dt != YSF_DT_DATA_FR_MODE || fn != 1U)
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
assert(data != NULL);
|
|
|
|
|
|
|
|
|
|
unsigned char buffer[20U];
|
|
|
|
|
if (dt != YSF_DT_DATA_FR_MODE)
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
|
|
|
|
|
CYSFPayload payload;
|
|
|
|
|
bool valid = payload.readDataFRModeData2(data, buffer);
|
|
|
|
|
if (!valid)
|
|
|
|
|
|
|
|
|
|
if (fi == YSF_FI_HEADER) {
|
|
|
|
|
payload.readDataFRModeData1(data, m_csd1);
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (::memcmp(buffer + 1U, CALL_DX, 2U) == 0)
|
|
|
|
|
processDX();
|
|
|
|
|
else if (::memcmp(buffer + 1U, CALL_ALL, 2U) == 0)
|
|
|
|
|
processAll();
|
|
|
|
|
else if (::memcmp(buffer + 1U, CALL_CONNECT, 2U) == 0)
|
|
|
|
|
return processConnect();
|
|
|
|
|
if (fi == YSF_FI_COMMUNICATIONS && fn == 0U) {
|
|
|
|
|
if (::memcmp(m_csd1, " ", 20U) == 0)
|
|
|
|
|
payload.readDataFRModeData1(data, m_csd1);
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fi == YSF_FI_TERMINATOR) {
|
|
|
|
|
if (::memcmp(m_csd1, " ", 20U) == 0)
|
|
|
|
|
payload.readDataFRModeData1(data, m_csd1);
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (fi == YSF_FI_COMMUNICATIONS && fn == 1U) {
|
|
|
|
|
unsigned char buffer[20U];
|
|
|
|
|
bool valid = payload.readDataFRModeData2(data, buffer);
|
|
|
|
|
if (!valid) {
|
|
|
|
|
::memset(m_csd1, ' ', 20U);
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (::memcmp(buffer + 1U, CALL_DX, 2U) == 0)
|
|
|
|
|
processDX();
|
|
|
|
|
else if (::memcmp(buffer + 1U, CALL_ALL, 2U) == 0)
|
|
|
|
|
processAll();
|
|
|
|
|
else if (::memcmp(buffer + 1U, CALL_CONNECT, 2U) == 0)
|
|
|
|
|
return processConnect();
|
|
|
|
|
else
|
|
|
|
|
::memset(m_csd1, ' ', 20U);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return WXS_NONE;
|
|
|
|
|
}
|
|
|
|
@ -66,7 +104,9 @@ std::string CWiresX::getReflector() const
|
|
|
|
|
|
|
|
|
|
void CWiresX::processDX()
|
|
|
|
|
{
|
|
|
|
|
::LogDebug("Received DX from %10.10s", m_csd1 + 10U);
|
|
|
|
|
|
|
|
|
|
m_timer.start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CWiresX::processAll()
|
|
|
|
@ -81,5 +121,110 @@ WX_STATUS CWiresX::processConnect()
|
|
|
|
|
|
|
|
|
|
void CWiresX::clock(unsigned int ms)
|
|
|
|
|
{
|
|
|
|
|
m_timer.clock(ms);
|
|
|
|
|
if (m_timer.isRunning() && m_timer.hasExpired()) {
|
|
|
|
|
LogDebug("Send reply");
|
|
|
|
|
|
|
|
|
|
unsigned char data[150U];
|
|
|
|
|
::memset(data, 0x00U, 150U);
|
|
|
|
|
::memset(data, ' ', 128U);
|
|
|
|
|
|
|
|
|
|
data[0U] = 0x03U; // XXX Serial no
|
|
|
|
|
data[1U] = 0x5DU;
|
|
|
|
|
data[2U] = 0x51U;
|
|
|
|
|
data[3U] = 0x5FU; // Start of data marker
|
|
|
|
|
data[4U] = 0x26U; // Repeater type
|
|
|
|
|
|
|
|
|
|
data[127U] = 0x03U; // End of data marker
|
|
|
|
|
data[128U] = CCRC::addCRC(data, 128U);
|
|
|
|
|
|
|
|
|
|
createReply(data, 140U);
|
|
|
|
|
|
|
|
|
|
m_timer.stop();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CWiresX::createReply(const unsigned char* data, unsigned int length)
|
|
|
|
|
{
|
|
|
|
|
assert(data != NULL);
|
|
|
|
|
assert(length > 0U);
|
|
|
|
|
|
|
|
|
|
// Write the header
|
|
|
|
|
unsigned char buffer[200U];
|
|
|
|
|
::memcpy(buffer, NET_HEADER, 34U);
|
|
|
|
|
buffer[34U] = 0x00U;
|
|
|
|
|
|
|
|
|
|
CSync::add(buffer + 35U);
|
|
|
|
|
|
|
|
|
|
CYSFFICH fich;
|
|
|
|
|
fich.load(DEFAULT_FICH);
|
|
|
|
|
fich.setFI(YSF_FI_HEADER);
|
|
|
|
|
fich.encode(buffer + 35U);
|
|
|
|
|
|
|
|
|
|
CYSFPayload payload;
|
|
|
|
|
payload.writeDataFRModeData1(m_csd1, buffer + 35U);
|
|
|
|
|
// payload.writeDataFRModeData2(" ", buffer + 35U);
|
|
|
|
|
|
|
|
|
|
m_network->write(buffer);
|
|
|
|
|
|
|
|
|
|
unsigned char bt = length / 260U;
|
|
|
|
|
fich.setBT(bt);
|
|
|
|
|
|
|
|
|
|
unsigned char fn = 0U;
|
|
|
|
|
unsigned char bn = 0U;
|
|
|
|
|
|
|
|
|
|
unsigned int offset = 0U;
|
|
|
|
|
while (offset < length) {
|
|
|
|
|
switch (fn) {
|
|
|
|
|
case 0U: {
|
|
|
|
|
unsigned int len = length - offset;
|
|
|
|
|
if (len > 220U) fich.setFT(7U);
|
|
|
|
|
else if (len > 180U) fich.setFT(6U);
|
|
|
|
|
else if (len > 140U) fich.setFT(5U);
|
|
|
|
|
else if (len > 100U) fich.setFT(4U);
|
|
|
|
|
else if (len > 60U) fich.setFT(3U);
|
|
|
|
|
else if (len > 20U) fich.setFT(2U);
|
|
|
|
|
else fich.setFT(1U);
|
|
|
|
|
payload.writeDataFRModeData1(m_csd1, buffer + 35U);
|
|
|
|
|
// payload.writeDataFRModeData2(" ", buffer + 35U);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 1U:
|
|
|
|
|
// payload.writeDataFRModeData1(" ", buffer + 35U);
|
|
|
|
|
payload.writeDataFRModeData2(data + offset, buffer + 35U);
|
|
|
|
|
offset += 20U;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
payload.writeDataFRModeData1(data + offset, buffer + 35U);
|
|
|
|
|
offset += 20U;
|
|
|
|
|
payload.writeDataFRModeData2(data + offset, buffer + 35U);
|
|
|
|
|
offset += 20U;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fich.setFN(fn);
|
|
|
|
|
fich.setBN(bn);
|
|
|
|
|
fich.setFI(YSF_FI_COMMUNICATIONS);
|
|
|
|
|
fich.encode(buffer + 35U);
|
|
|
|
|
|
|
|
|
|
m_network->write(buffer);
|
|
|
|
|
|
|
|
|
|
fn++;
|
|
|
|
|
if (fn >= 8U) {
|
|
|
|
|
fn = 0U;
|
|
|
|
|
bn++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Write the trailer
|
|
|
|
|
buffer[34U] = 0x01U;
|
|
|
|
|
|
|
|
|
|
fich.load(DEFAULT_FICH);
|
|
|
|
|
fich.setFI(YSF_FI_TERMINATOR);
|
|
|
|
|
fich.encode(buffer + 35U);
|
|
|
|
|
|
|
|
|
|
payload.writeDataFRModeData1(m_csd1, buffer + 35U);
|
|
|
|
|
// payload.writeDataFRModeData2(" ", buffer + 35U);
|
|
|
|
|
|
|
|
|
|
m_network->write(buffer);
|
|
|
|
|
}
|
|
|
|
|