linbpq/ARDOP.c

6372 lines
143 KiB
C
Raw Blame History

/*
Copyright 2001-2022 John Wiseman G8BPQ
This file is part of LinBPQ/BPQ32.
LinBPQ/BPQ32 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.
LinBPQ/BPQ32 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 LinBPQ/BPQ32. If not, see http://www.gnu.org/licenses
*/
//
// Interface to allow G8BPQ switch to use ARDOP Virtual TNC
#define _CRT_SECURE_NO_DEPRECATE
#include <stdio.h>
#include <time.h>
#ifdef WIN32
//#include <Psapi.h>
#else
// For serial over i2c support
#ifdef MACBPQ
#define NOI2C
#endif
#ifdef FREEBSD
#define NOI2C
#endif
#ifndef NOI2C
#include "i2c-dev.h"
#endif
#endif
#include "CHeaders.h"
int (WINAPI FAR *GetModuleFileNameExPtr)();
int (WINAPI FAR *EnumProcessesPtr)();
#define SD_RECEIVE 0x00
#define SD_SEND 0x01
#define SD_BOTH 0x02
#define APMaxStreams 10 // First is used for ARDOP, even though ARDOP uses channel 31
#include "bpq32.h"
#include "tncinfo.h"
#define WSA_ACCEPT WM_USER + 1
#define WSA_DATA WM_USER + 2
#define WSA_CONNECT WM_USER + 3
static int Socket_Data(int sock, int error, int eventcode);
int KillTNC(struct TNCINFO * TNC);
int RestartTNC(struct TNCINFO * TNC);
int KillPopups(struct TNCINFO * TNC);
VOID MoveWindows(struct TNCINFO * TNC);
int SendReporttoWL2K(struct TNCINFO * TNC);
char * CheckAppl(struct TNCINFO * TNC, char * Appl);
int DoScanLine(struct TNCINFO * TNC, char * Buff, int Len);
BOOL KillOldTNC(char * Path);
int ARDOPSendData(struct TNCINFO * TNC, char * Buff, int Len);
VOID ARDOPSendCommand(struct TNCINFO * TNC, char * Buff, BOOL Queue);
VOID ARDOPSendPktCommand(struct TNCINFO * TNC, int Stream, char * Buff);
VOID SendToTNC(struct TNCINFO * TNC, int Stream, UCHAR * Encoded, int EncLen);
VOID ARDOPProcessDataPacket(struct TNCINFO * TNC, UCHAR * Type, UCHAR * Data, int Length);
void ARDOPSCSCheckRX(struct TNCINFO * TNC);
VOID ARDOPSCSPoll(struct TNCINFO * TNC);
VOID ARDOPDoTNCReinit(struct TNCINFO * TNC);
int SerialGetTCPMessage(struct TNCINFO * TNC, unsigned char * Buffer, int Len);
int SerialConnecttoTCP(struct TNCINFO * TNC);
int ARDOPWriteCommBlock(struct TNCINFO * TNC);
int ReadCOMBlockEx(HANDLE fd, char * Block, int MaxLength, BOOL * Error);
int Unstuff(UCHAR * MsgIn, UCHAR * MsgOut, int len);
BOOL WriteCommBlock(struct TNCINFO * TNC);
VOID AddVirtualKISSPort(struct TNCINFO * TNC, int Port, char * buf);
int ConfigVirtualKISSPort(struct TNCINFO * TNC, char * Cmd);
void ProcessKISSBytes(struct TNCINFO * TNC, UCHAR * Data, int Len);
void ProcessKISSPacket(struct TNCINFO * TNC, UCHAR * KISSBuffer, int Len);
int ARDOPProcessDEDFrame(struct TNCINFO * TNC, UCHAR * Msg, int framelen);
int ConnecttoARDOP(struct TNCINFO * TNC);
int standardParams(struct TNCINFO * TNC, char * buf);
#ifndef LINBPQ
BOOL CALLBACK EnumARDOPWindowsProc(HWND hwnd, LPARAM lParam);
#endif
static char ClassName[]="ARDOPSTATUS";
static char WindowTitle[] = "ARDOP";
static int RigControlRow = 165;
#define WINMOR
#define NARROWMODE 21
#define WIDEMODE 22
#ifndef LINBPQ
#include <commctrl.h>
#endif
extern char * PortConfig[33];
extern int SemHeldByAPI;
static RECT Rect;
extern struct TNCINFO * TNCInfo[41]; // Records are Malloc'd
static int ProcessLine(char * buf, int Port);
BOOL ARDOPStopPort(struct PORTCONTROL * PORT)
{
// Disable Port - close TCP Sockets or Serial Port
struct TNCINFO * TNC = PORT->TNC;
TNC->CONNECTED = FALSE;
TNC->Alerted = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE); // Make sure PTT is down
if (TNC->Streams[0].Attached)
TNC->Streams[0].ReportDISC = TRUE;
if (TNC->TCPSock)
{
shutdown(TNC->TCPSock, SD_BOTH);
Sleep(100);
closesocket(TNC->TCPSock);
}
if (TNC->TCPDataSock)
{
shutdown(TNC->TCPDataSock, SD_BOTH);
Sleep(100);
closesocket(TNC->TCPDataSock);
}
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
if (TNC->hDevice)
{
CloseCOMPort(TNC->hDevice);
TNC->hDevice = 0;
}
sprintf(PORT->TNC->WEB_COMMSSTATE, "%s", "Port Stopped");
MySetWindowText(PORT->TNC->xIDC_COMMSSTATE, PORT->TNC->WEB_COMMSSTATE);
return TRUE;
}
BOOL ARDOPStartPort(struct PORTCONTROL * PORT)
{
// Restart Port - Open Sockets or Serial Port
struct TNCINFO * TNC = PORT->TNC;
if (TNC->ARDOPCommsMode == 'T')
{
ConnecttoARDOP(TNC);
TNC->lasttime = time(NULL);;
}
sprintf(PORT->TNC->WEB_COMMSSTATE, "%s", "Port Restarted");
MySetWindowText(PORT->TNC->xIDC_COMMSSTATE, PORT->TNC->WEB_COMMSSTATE);
return TRUE;
}
int GenCRC16(unsigned char * Data, unsigned short length)
{
// For CRC-16-CCITT = x^16 + x^12 +x^5 + 1 intPoly = 1021 Init FFFF
// intSeed is the seed value for the shift register and must be in the range 0-&HFFFF
int intRegister = 0xffff; //intSeed
int i,j;
int Bit;
int intPoly = 0x8810; // This implements the CRC polynomial x^16 + x^12 +x^5 + 1
for (j = 0; j < (length); j++)
{
int Mask = 0x80; // Top bit first
for (i = 0; i < 8; i++) // for each bit processing MS bit first
{
Bit = Data[j] & Mask;
Mask >>= 1;
if (intRegister & 0x8000) // Then ' the MSB of the register is set
{
// Shift left, place data bit as LSB, then divide
// Register := shiftRegister left shift 1
// Register := shiftRegister xor polynomial
if (Bit)
intRegister = 0xFFFF & (1 + (intRegister << 1));
else
intRegister = 0xFFFF & (intRegister << 1);
intRegister = intRegister ^ intPoly;
}
else
{
// the MSB is not set
// Register is not divisible by polynomial yet.
// Just shift left and bring current data bit onto LSB of shiftRegister
if (Bit)
intRegister = 0xFFFF & (1 + (intRegister << 1));
else
intRegister = 0xFFFF & (intRegister << 1);
}
}
}
return intRegister;
}
BOOL checkcrc16(unsigned char * Data, unsigned short length)
{
int intRegister = 0xffff; //intSeed
int i,j;
int Bit;
int intPoly = 0x8810; // This implements the CRC polynomial x^16 + x^12 +x^5 + 1
for (j = 0; j < (length - 2); j++) // ' 2 bytes short of data length
{
int Mask = 0x80; // Top bit first
for (i = 0; i < 8; i++) // for each bit processing MS bit first
{
Bit = Data[j] & Mask;
Mask >>= 1;
if (intRegister & 0x8000) // Then ' the MSB of the register is set
{
// Shift left, place data bit as LSB, then divide
// Register := shiftRegister left shift 1
// Register := shiftRegister xor polynomial
if (Bit)
intRegister = 0xFFFF & (1 + (intRegister << 1));
else
intRegister = 0xFFFF & (intRegister << 1);
intRegister = intRegister ^ intPoly;
}
else
{
// the MSB is not set
// Register is not divisible by polynomial yet.
// Just shift left and bring current data bit onto LSB of shiftRegister
if (Bit)
intRegister = 0xFFFF & (1 + (intRegister << 1));
else
intRegister = 0xFFFF & (intRegister << 1);
}
}
}
if (Data[length - 2] == intRegister >> 8)
if (Data[length - 1] == (intRegister & 0xFF))
return TRUE;
return FALSE;
}
// Logging Interface. Used for Log over Serial Mode
BOOL ARDOPOpenLogFiles(struct TNCINFO * TNC)
{
UCHAR FN[MAX_PATH];
time_t T;
struct tm * tm;
T = time(NULL);
tm = gmtime(&T);
strlop(TNC->LogPath, 13);
strlop(TNC->LogPath, 32);
sprintf(FN,"%s/ARDOPDebugLog_%02d%02d_%d.txt", TNC->LogPath, tm->tm_mon + 1, tm->tm_mday, TNC->Port);
TNC->DebugHandle = fopen(FN, "ab");
sprintf(FN,"%s/ARDOPLog_%02d%02d_%d.txt", TNC->LogPath, tm->tm_mon + 1, tm->tm_mday, TNC->Port);
TNC->LogHandle = fopen(FN, "ab");
return (TNC->LogHandle != NULL);
}
VOID WritetoTrace(struct TNCINFO * TNC, char * Msg, int Len);
void SendARDOPorPacketData(struct TNCINFO * TNC, int Stream, UCHAR * Buff, int txlen)
{
struct STREAMINFO * STREAM = &TNC->Streams[Stream];
if (Stream == 0)
{
ARDOPSendData(TNC, Buff, txlen);
STREAM->BytesTXed += txlen;
WritetoTrace(TNC, Buff, txlen);
}
else
{
// Packet. Only works over Serial
PMSGWITHLEN buffptr;
UCHAR * buffp;
if (TNC->ARDOPCommsMode == 'T')
return;
buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return; // No buffers, so ignore
buffptr->Len = txlen + 1;
buffp = &buffptr->Data[0];
buffp[0] = 0; // CMD/Data Flag on front
memcpy(buffp +1, Buff, txlen);
C_Q_ADD(&STREAM->BPQtoPACTOR_Q, buffptr);
STREAM->FramesQueued++;
if (TNC->Timeout == 0) // if link idle can send now
ARDOPSCSPoll(TNC);
}
}
static int ProcessLine(char * buf, int Port)
{
UCHAR * ptr,* p_cmd;
char * p_ipad = 0;
char * p_port = 0;
char * PktPort = 0;
unsigned short WINMORport = 0;
int BPQport;
int len=510;
struct TNCINFO * TNC;
char errbuf[256];
strcpy(errbuf, buf);
ptr = strtok(buf, " \t\n\r");
if(ptr == NULL) return (TRUE);
if(*ptr =='#') return (TRUE); // comment
if(*ptr ==';') return (TRUE); // comment
// Must start ADDR or SERIAL
ptr = strtok(NULL, " \t\n\r");
BPQport = Port;
p_ipad = ptr;
if (_stricmp(buf, "ADDR") == 0 || _stricmp(buf, "TCPSERIAL") == 0)
{
TNC = TNCInfo[BPQport] = malloc(sizeof(struct TNCINFO));
memset(TNC, 0, sizeof(struct TNCINFO));
if (p_ipad == NULL)
p_ipad = strtok(NULL, " \t\n\r");
if (p_ipad == NULL) return (FALSE);
p_port = strtok(NULL, " \t\n\r");
if (p_port == NULL) return (FALSE);
PktPort = strlop(p_port, '/');
if (PktPort)
TNC->PacketPort = atoi(PktPort);
WINMORport = atoi(p_port);
TNC->destaddr.sin_family = AF_INET;
TNC->destaddr.sin_port = htons(WINMORport);
TNC->Datadestaddr.sin_family = AF_INET;
TNC->Datadestaddr.sin_port = htons(WINMORport+1);
TNC->HostName = malloc(strlen(p_ipad)+1);
if (TNC->HostName == NULL) return TRUE;
strcpy(TNC->HostName,p_ipad);
if (buf[0] == 'A')
TNC->ARDOPCommsMode = 'T'; // TCP
else
TNC->ARDOPCommsMode = 'E'; // TCPSERIAL (ESP01)
}
else if (_stricmp(buf, "SERIAL") == 0)
{
TNC = TNCInfo[BPQport] = malloc(sizeof(struct TNCINFO));
memset(TNC, 0, sizeof(struct TNCINFO));
if (p_ipad == NULL)
p_ipad = strtok(NULL, " \t\n\r");
if (p_ipad == NULL) return (FALSE);
p_port = strtok(NULL, " \t\n\r");
if (p_port == NULL) return (FALSE);
TNC->ARDOPSerialPort = _strdup(p_ipad);
TNC->ARDOPSerialSpeed = atoi(p_port);
TNC->ARDOPCommsMode = 'S';
}
else if (_stricmp(buf, "I2C") == 0)
{
TNC = TNCInfo[BPQport] = malloc(sizeof(struct TNCINFO));
memset(TNC, 0, sizeof(struct TNCINFO));
if (p_ipad == NULL)
p_ipad = strtok(NULL, " \t\n\r");
if (p_ipad == NULL) return (FALSE);
p_port = strtok(NULL, " \t\n\r");
if (p_port == NULL) return (FALSE);
TNC->ARDOPSerialPort = _strdup(p_ipad);
TNC->ARDOPSerialSpeed = atoi(p_port);
TNC->ARDOPCommsMode = 'I';
}
else
return FALSE; // Must start with ADDR
TNC->InitScript = malloc(1000);
TNC->InitScript[0] = 0;
ptr = strtok(NULL, " \t\n\r");
if (ptr)
{
if (_stricmp(ptr, "PTT") == 0)
{
ptr = strtok(NULL, " \t\n\r");
if (ptr)
{
DecodePTTString(TNC, ptr);
ptr = strtok(NULL, " \t\n\r");
}
}
}
if (ptr)
{
if (_memicmp(ptr, "PATH", 4) == 0)
{
p_cmd = strtok(NULL, "\n\r");
if (p_cmd) TNC->ProgramPath = _strdup(p_cmd);
}
}
TNC->MaxConReq = 10; // Default
TNC->OldMode = FALSE; // Default
// Read Initialisation lines
while(TRUE)
{
if (GetLine(buf) == 0)
return TRUE;
strcpy(errbuf, buf);
if (memcmp(buf, "****", 4) == 0)
return TRUE;
ptr = strchr(buf, ';');
if (ptr)
{
*ptr++ = 13;
*ptr = 0;
}
if ((_memicmp(buf, "CAPTURE", 7) == 0) || (_memicmp(buf, "PLAYBACK", 8) == 0))
{} // Ignore
else
/*
if (_memicmp(buf, "PATH", 4) == 0)
{
char * Context;
p_cmd = strtok_s(&buf[5], "\n\r", &Context);
if (p_cmd) TNC->ProgramPath = _strdup(p_cmd);
}
else
*/
if (_memicmp(buf, "PACKETCHANNELS", 14) == 0) // Packet Channels
TNC->PacketChannels = atoi(&buf[14]);
else
if (_memicmp(buf, "MAXCONREQ", 9) == 0) // Hold Time for Busy Detect
TNC->MaxConReq = atoi(&buf[9]);
else
if (_memicmp(buf, "STARTINROBUST", 13) == 0)
TNC->StartInRobust = TRUE;
else
if (_memicmp(buf, "ROBUST", 6) == 0)
{
if (_memicmp(&buf[7], "TRUE", 4) == 0)
TNC->Robust = TRUE;
strcat (TNC->InitScript, buf);
}
else
if (_memicmp(buf, "LOGDIR ", 7) == 0)
TNC->LogPath = _strdup(&buf[7]);
else
if (_memicmp(buf, "ENABLEPACKET", 12) == 0)
{
if (TNC->PacketChannels == 0)
TNC->PacketChannels = 5;
// AddVirtualKISSPort(TNC, Port, buf);
}
// else if (_memicmp(buf, "PAC ", 4) == 0 && _memicmp(buf, "PAC MODE", 8) != 0)
// {
// PAC MODE goes to TNC, others are parsed locally
//
// ConfigVirtualKISSPort(TNC, buf);
// }
else if (standardParams(TNC, buf) == FALSE)
strcat(TNC->InitScript, buf);
}
return (TRUE);
}
void ARDOPThread(struct TNCINFO * TNC);
VOID ARDOPProcessDataSocketData(int port);
int ConnecttoARDOP();
static VOID ARDOPProcessReceivedData(struct TNCINFO * TNC);
static VOID ARDOPProcessReceivedControl(struct TNCINFO * TNC);
int V4ProcessReceivedData(struct TNCINFO * TNC);
VOID ARDOPReleaseTNC(struct TNCINFO * TNC);
VOID SuspendOtherPorts(struct TNCINFO * ThisTNC);
VOID ReleaseOtherPorts(struct TNCINFO * ThisTNC);
VOID WritetoTrace(struct TNCINFO * TNC, char * Msg, int Len);
#define MAXBPQPORTS 32
static time_t ltime;
static SOCKADDR_IN sinx;
static SOCKADDR_IN rxaddr;
static int addrlen=sizeof(sinx);
unsigned short int compute_crc(unsigned char *buf,int len);
VOID ARDOPSendPktCommand(struct TNCINFO * TNC, int Stream, char * Buff)
{
// Encode and send to TNC. May be TCP or Serial
int EncLen;
UCHAR Encoded[256];
if (Stream == 0)
{
if (Buff[0] == 0) // Terminal Keepalive?
return;
}
else
{
if (Buff[1] == 0) // Terminal Keepalive?
return;
}
if (TNC->PacketSock) // Packet Data over separate TCP Connectoion?
{
// Chan, Cmd/Data, Len-1
int SentLen;
EncLen = sprintf(Encoded, "%c%c%c%s\r", Buff[0], 1, (int)strlen(Buff) -2, &Buff[1]);
SentLen = send(TNC->PacketSock, Encoded, EncLen, 0);
if (SentLen != EncLen)
{
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Pkt Write Failed for port %d - error code = %d\r\n", TNC->PacketPort, winerr);
WritetoConsole(ErrMsg);
closesocket(TNC->PacketSock);
TNC->PacketSock = 0;
}
return;
}
EncLen = sprintf(Encoded, "%s\r", Buff);
SendToTNC(TNC, Stream, Encoded, EncLen);
return;
}
VOID ARDOPSendCommand(struct TNCINFO * TNC, char * Buff, BOOL Queue)
{
// Encode and send to TNC. May be TCP or Serial
// Command Formst is C:TEXT<CR><CRC>
int i, EncLen;
UCHAR Encoded[260]; // could get 256 byte packet
if (Buff[0] == 0) // Terminal Keepalive?
return;
EncLen = sprintf(Encoded, "%s\r", Buff);
// it is possible for binary data to be dumped into the command
// handler if a session disconnects in middle of transfer
for (i = 0; i < EncLen; i++)
{
if (Encoded[i] > 0x7f)
return;
}
SendToTNC(TNC, 12, Encoded, EncLen); // Use streams 12 aad 13 for Host Mode Schannels 32 and 33
return;
}
VOID SendToTNC(struct TNCINFO * TNC, int Stream, UCHAR * Encoded, int EncLen)
{
int SentLen;
if (TNC->hDevice || (TNC->ARDOPCommsMode == 'E' && TNC->TCPSock))
{
// Serial mode. Queue to Hostmode driver
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return; // No buffers, so ignore
buffptr->Len = EncLen;
memcpy(&buffptr->Data[0], Encoded, EncLen);
C_Q_ADD(&TNC->Streams[Stream].BPQtoPACTOR_Q, buffptr);
TNC->Streams[Stream].FramesQueued++;
if (TNC->Timeout == 0) // if link idle can send now
ARDOPSCSPoll(TNC);
return;
}
if(TNC->ARDOPCommsMode == 'T' && TNC->TCPSock)
{
SentLen = send(TNC->TCPSock, Encoded, EncLen, 0);
if (SentLen != EncLen)
{
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Write Failed for port %d - error code = %d\r\n", TNC->Port, winerr);
WritetoConsole(ErrMsg);
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
TNC->CONNECTED = FALSE;
return;
}
}
}
VOID SendDataToTNC(struct TNCINFO * TNC, int Streem , UCHAR * Encoded, int EncLen)
{
int SentLen;
if (TNC->hDevice || (TNC->ARDOPCommsMode == 'E' && TNC->TCPSock))
{
// Serial mode. Queue to Hostmode driver
int Stream = 13; // use 12 and 13 for scs channels 32 and 33
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return; // No buffers, so ignore
buffptr->Len = EncLen;
memcpy(&buffptr->Data[0], Encoded, EncLen);
C_Q_ADD(&TNC->Streams[Stream].BPQtoPACTOR_Q, buffptr);
TNC->Streams[Stream].FramesQueued++;
if (TNC->Timeout == 0) // if link idle can send now
ARDOPSCSPoll(TNC);
return;
}
if(TNC->TCPDataSock)
{
SentLen = send(TNC->TCPDataSock, Encoded, EncLen, 0);
if (SentLen != EncLen)
{
// WINMOR doesn't seem to recover from a blocked write. For now just reset
// if (bytes == SOCKET_ERROR)
// {
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Write Failed for port %d - error code = %d\r\n", TNC->Port, winerr);
WritetoConsole(ErrMsg);
// if (winerr != WSAEWOULDBLOCK)
// {
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
TNC->CONNECTED = FALSE;
return;
}
}
}
int ARDOPSenPktdData(struct TNCINFO * TNC, int Stream, char * Buff, int Len)
{
// Encode and send to TNC. May be TCP or Serial
int EncLen;
UCHAR Msg[400];
UCHAR * Encoded = Msg;
*(Encoded++) = Len >> 8;
*(Encoded++) = Len & 0xff;
memcpy(Encoded, Buff, Len);
EncLen = Len + 2;
SendDataToTNC(TNC, Stream, Msg, EncLen);
return Len;
}
int ARDOPSendData(struct TNCINFO * TNC, char * Buff, int Len)
{
// Encode and send to TNC. May be TCP or Serial
int EncLen;
UCHAR Msg[400];
UCHAR * Encoded = Msg;
*(Encoded++) = Len >> 8;
*(Encoded++) = Len & 0xff;
memcpy(Encoded, Buff, Len);
EncLen = Len + 2;
SendDataToTNC(TNC, 13, Msg, EncLen);
return Len;
}
VOID ARDOPChangeMYC(struct TNCINFO * TNC, char * Call)
{
UCHAR TXMsg[100];
int datalen;
if (strcmp(Call, TNC->CurrentMYC) == 0)
return; // No Change
strcpy(TNC->CurrentMYC, Call);
// ARDOPSendCommand(TNC, "CODEC FALSE");
datalen = sprintf(TXMsg, "MYCALL %s", Call);
ARDOPSendCommand(TNC, TXMsg, TRUE);
// ARDOPSendCommand(TNC, "CODEC TRUE");
// TNC->StartSent = TRUE;
// ARDOPSendCommand(TNC, "MYCALL", TRUE);
}
static size_t ExtProc(int fn, int port, PDATAMESSAGE buff)
{
int datalen;
PMSGWITHLEN buffptr;
// char txbuff[500];
unsigned int bytes,txlen=0;
size_t Param;
int Stream = 0;
HKEY hKey=0;
struct TNCINFO * TNC = TNCInfo[port];
struct STREAMINFO * STREAM = &TNC->Streams[0];
struct ScanEntry * Scan;
if (TNC == NULL)
return 0; // Port not defined
if (TNC->CONNECTED == 0)
{
// clear Q if not connected
while(TNC->BPQtoWINMOR_Q)
{
buffptr = (PMSGWITHLEN)Q_REM(&TNC->BPQtoWINMOR_Q);
if (buffptr)
ReleaseBuffer(buffptr);
}
}
switch (fn)
{
case 7:
// approx 100 mS Timer. May now be needed, as Poll can be called more frequently in some circumstances
// Check session limit timer
if ((STREAM->Connecting || STREAM->Connected) && !STREAM->Disconnecting)
{
if (TNC->SessionTimeLimit && STREAM->ConnectTime && time(NULL) > (TNC->SessionTimeLimit + STREAM->ConnectTime))
{
ARDOPSendCommand(TNC, "DISCONNECT", TRUE);
STREAM->Disconnecting = TRUE;
}
}
if (TNC->ARDOPCommsMode != 'T') // S I or E
{
ARDOPSCSCheckRX(TNC);
ARDOPSCSPoll(TNC);
}
return 0;
case 1: // poll
// If not using serial interface, Rig Contol Frames are sent as
// ARDOP COmmand Frames. These are hex encoded
if (TNC->ARDOPCommsMode == 'T' && TNC->BPQtoRadio_Q)
{
PMSGWITHLEN buffptr;
buffptr = (PMSGWITHLEN)Q_REM(&TNC->BPQtoRadio_Q);
if (TNC->CONNECTED)
{
int len = (int)buffptr->Len;
UCHAR * ptr = &buffptr->Data[0];
char RigCommand[256] = "RADIOHEX ";
char * ptr2 = &RigCommand[9] ;
int i, j;
if (len < 120)
{
while (len--)
{
i = *(ptr++);
j = i >>4;
j += '0'; // ascii
if (j > '9')
j += 7;
*(ptr2++) = j;
j = i & 0xf;
j += '0'; // ascii
if (j > '9')
j += 7;
*(ptr2++) = j;
}
ARDOPSendCommand(TNC, RigCommand, FALSE);
}
}
ReleaseBuffer(buffptr);
}
while (TNC->PortRecord->UI_Q)
{
int datalen;
char * Buffer;
char FECMsg[512];
char Call[12] = " ";
struct _MESSAGE * buffptr;
int CallLen;
char * ptr = FECMsg;
buffptr = Q_REM(&TNC->PortRecord->UI_Q);
if (TNC->CONNECTED == 0 ||
TNC->Streams[0].Connecting ||
TNC->Streams[0].Connected)
{
// discard if TNC not connected or sesison active
ReleaseBuffer(buffptr);
continue;
}
datalen = buffptr->LENGTH - MSGHDDRLEN;
Buffer = &buffptr->DEST[0]; // Raw Frame
Buffer[datalen] = 0;
*ptr++ = '^'; // delimit fram ewith ^
// Frame has ax.25 format header. Convert to Text
CallLen = ConvFromAX25(Buffer + 7, Call); // Origin
memcpy(ptr, Call, CallLen);
ptr += CallLen;
*ptr++ = '>';
CallLen = ConvFromAX25(Buffer, Call); // Dest
memcpy(ptr, Call, CallLen);
ptr += CallLen;
Buffer += 14; // TO Digis
datalen -= 14;
while ((Buffer[-1] & 1) == 0)
{
*ptr++ = ',';
CallLen = ConvFromAX25(Buffer, Call);
memcpy(ptr, Call, CallLen);
ptr += CallLen;
Buffer += 7; // End of addr
datalen -= 7;
}
*ptr++ = '|'; // delimit calls
if (Buffer[0] == 3) // UI
{
Buffer += 2;
datalen -= 2;
}
memcpy(ptr, Buffer, datalen);
ptr += datalen;
*ptr++ = '^'; // delimit fram ewith ^
ARDOPSendData(TNC, FECMsg, (int)(ptr - FECMsg));
TNC->FECPending = 1;
ReleaseBuffer((UINT *)buffptr);
}
if (TNC->Busy) // Count down to clear
{
if ((TNC->BusyFlags & CDBusy) == 0) // TNC Has reported not busy
{
TNC->Busy--;
if (TNC->Busy == 0)
MySetWindowText(TNC->xIDC_CHANSTATE, "Clear");
strcpy(TNC->WEB_CHANSTATE, "Clear");
}
}
if (TNC->BusyDelay)
{
// Still Busy?
if (InterlockedCheckBusy(TNC) == FALSE)
{
// No, so send
// ARDOPSendCommand(TNC, "LISTEN TRUE", TRUE); // !!!! Temp bug workaround !!!!
ARDOPSendCommand(TNC, TNC->ConnectCmd, TRUE);
TNC->Streams[0].Connecting = TRUE;
memset(TNC->Streams[0].RemoteCall, 0, 10);
memcpy(TNC->Streams[0].RemoteCall, &TNC->ConnectCmd[8], (int)strlen(TNC->ConnectCmd)-10);
sprintf(TNC->WEB_TNCSTATE, "%s Connecting to %s", TNC->Streams[0].MyCall, TNC->Streams[0].RemoteCall);
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
free(TNC->ConnectCmd);
TNC->BusyDelay = 0;
}
else
{
// Wait Longer
TNC->BusyDelay--;
if (TNC->BusyDelay == 0)
{
// Timed out - Send Error Response
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return (0); // No buffers, so ignore
buffptr->Len = 39;
memcpy(&buffptr->Data[0], "Sorry, Can't Connect - Channel is busy\r", 39);
C_Q_ADD(&TNC->Streams[0].PACTORtoBPQ_Q, buffptr);
free(TNC->ConnectCmd);
sprintf(TNC->WEB_TNCSTATE, "In Use by %s", TNC->Streams[0].MyCall);
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
}
}
}
if (TNC->HeartBeat++ > 600 || (TNC->Streams[0].Connected && TNC->HeartBeat > 50)) // Every Minute unless connected
{
TNC->HeartBeat = 0;
if (TNC->CONNECTED)
{
// Probe link
if (TNC->Streams[0].Connecting || TNC->Streams[0].Connected)
fn =fn; //ARDOPSendCommand(TNC, "MODE", TRUE);
else
{
// if (time(NULL) - TNC->WinmorRestartCodecTimer > 300) // 5 mins
// {
// ARDOPSendCommand(TNC, "CODEC FALSE", TRUE);
// ARDOPSendCommand(TNC, "CODEC TRUE", TRUE);
// }
// else
ARDOPSendCommand(TNC, "STATE", TRUE);
}
}
}
if (TNC->FECMode)
{
if (TNC->FECIDTimer++ > 6000) // ID every 10 Mins
{
if (!TNC->Busy)
{
TNC->FECIDTimer = 0;
ARDOPSendCommand(TNC, "SENDID", TRUE);
}
}
}
// FECPending can be set if not in FEC Mode (eg beacon)
if (TNC->FECPending) // Check if FEC Send needed
{
if (TNC->Streams[0].BytesOutstanding) //Wait for data to be queued (async data session)
{
if (TNC->Busy == 0 && TNC->GavePermission == 0)
{
TNC->FECPending = 0;
ARDOPSendCommand(TNC,"FECSEND TRUE", TRUE);
}
}
}
if (TNC->DiscPending)
{
TNC->DiscPending--;
if (TNC->DiscPending == 0)
{
// Too long in Disc Pending - Kill and Restart TNC
if (TNC->PID)
KillTNC(TNC);
RestartTNC(TNC);
}
}
if (TNC->TimeSinceLast++ > 800) // Allow 10 secs for Keepalive
{
// Restart TNC
if (TNC->ProgramPath && TNC->CONNECTED && 0)
{
struct tm * tm;
char Time[80];
TNC->Restarts++;
TNC->LastRestart = time(NULL);
tm = gmtime(&TNC->LastRestart);
sprintf_s(Time, sizeof(Time),"%04d/%02d/%02d %02d:%02dZ",
tm->tm_year +1900, tm->tm_mon+1, tm->tm_mday, tm->tm_hour, tm->tm_min);
MySetWindowText(TNC->xIDC_RESTARTTIME, Time);
strcpy(TNC->WEB_RESTARTTIME, Time);
sprintf_s(Time, sizeof(Time),"%d", TNC->Restarts);
MySetWindowText(TNC->xIDC_RESTARTS, Time);
strcpy(TNC->WEB_RESTARTS, Time);
KillTNC(TNC);
RestartTNC(TNC);
TNC->TimeSinceLast = 0;
}
}
for (Stream = 0; Stream <= APMaxStreams; Stream++)
{
STREAM = &TNC->Streams[Stream];
if (STREAM->NeedDisc)
{
STREAM->NeedDisc--;
if (STREAM->NeedDisc == 0)
{
// Send the DISCONNECT
if (Stream == 0)
ARDOPSendCommand(TNC, "DISCONNECT", TRUE);
else
{
char Cmd[32];
sprintf(Cmd, "%cDISCONNECT", Stream);
ARDOPSendPktCommand(TNC, Stream, Cmd);
}
}
}
if (TNC->PortRecord->ATTACHEDSESSIONS[Stream] && STREAM->Attached == 0)
{
// New Attach
int calllen;
char Msg[80];
Debugprintf("ARDOP New Attach Stream %d DEDStream %d", Stream, STREAM->DEDStream);
STREAM->Attached = TRUE;
calllen = ConvFromAX25(TNC->PortRecord->ATTACHEDSESSIONS[Stream]->L4USER, TNC->Streams[Stream].MyCall);
TNC->Streams[Stream].MyCall[calllen] = 0;
if (Stream == 0)
{
// If Pactor, stop scanning and take out of listen mode.
// if (Stream == 0)
// STREAM->DEDStream = 31; // Pactor
// Stop Listening, and set MYCALL to user's call
ARDOPSendCommand(TNC, "LISTEN FALSE", TRUE);
ARDOPChangeMYC(TNC, TNC->Streams[0].MyCall);
TNC->SessionTimeLimit = TNC->DefaultSessionTimeLimit; // Reset Limit
// Stop other ports in same group
SuspendOtherPorts(TNC);
sprintf(TNC->WEB_TNCSTATE, "In Use by %s", TNC->Streams[0].MyCall);
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
// Stop Scanning
sprintf(Msg, "%d SCANSTOP", TNC->Port);
Rig_Command(-1, Msg);
}
else
{
// Packet Connect
}
}
if (STREAM->Attached)
CheckForDetach(TNC, Stream, STREAM, TidyClose, ForcedClose, CloseComplete);
}
if (TNC->CONNECTED == FALSE && TNC->CONNECTING == FALSE)
{
// See if time to reconnect
time(&ltime);
if (ltime - TNC->lasttime > 9 )
{
if (TNC->ARDOPCommsMode == 'T' && TNC->PortRecord->PORTCONTROL.PortStopped == 0)
ConnecttoARDOP(TNC);
TNC->lasttime = ltime;
}
}
// See if any frames for this port
for (Stream = 0; Stream <= APMaxStreams; Stream++)
{
STREAM = &TNC->Streams[Stream];
if (TNC->ARDOPCommsMode == 'T')
{
// For serial mode packets are taken from the queue by ARDOPSCSPoll
if (STREAM->BPQtoPACTOR_Q)
{
PMSGWITHLEN buffptr = (PMSGWITHLEN)Q_REM(&STREAM->BPQtoPACTOR_Q);
UCHAR * data = &buffptr->Data[0];
STREAM->FramesQueued--;
txlen = (int)buffptr->Len;
STREAM->BytesTXed += txlen;
if (Stream == 0)
{
bytes=ARDOPSendData(TNC, data, txlen);
WritetoTrace(TNC, data, txlen);
}
else
{
if (TNC->PacketSock)
{
// Using Packet over TCP)
// Chan, Cmd/Data, Len-1
UCHAR Encoded[280];
int EncLen;
int SentLen;
EncLen = sprintf(Encoded, "%c%c%c%s\r", Stream, 0, txlen - 1, data);
SentLen = send(TNC->PacketSock, Encoded, EncLen, 0);
if (SentLen != EncLen)
{
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Pkt Write Failed for port %d - error code = %d\r\n", TNC->PacketPort, winerr);
WritetoConsole(ErrMsg);
closesocket(TNC->PacketSock);
TNC->PacketSock = 0;
}
}
}
ReleaseBuffer(buffptr);
}
}
if (STREAM->PACTORtoBPQ_Q != 0)
{
buffptr = Q_REM(&STREAM->PACTORtoBPQ_Q);
datalen = (int)buffptr->Len;
buff->PORT = Stream; // Compatibility with Kam Driver
buff->PID = 0xf0;
memcpy(&buff->L2DATA, &buffptr->Data[0], datalen); // Data goes to + 7, but we have an extra byte
datalen += sizeof(void *) + 4;
PutLengthinBuffer(buff, datalen);
ReleaseBuffer(buffptr);
return (1);
}
if (STREAM->ReportDISC) // May need a delay so treat as a counter
{
STREAM->ReportDISC--;
if (STREAM->ReportDISC == 0)
{
buff->PORT = Stream;
// STREAM->Connected = 0;
// STREAM->Attached = 0;
return -1;
}
}
}
return (0);
case 2: // send
Stream = buff->PORT;
if (!TNC->CONNECTED)
{
// Send Error Response
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return (0); // No buffers, so ignore
buffptr->Len = sprintf(&buffptr->Data[0], "No Connection to ARDOP TNC\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0; // Don't try if not connected
}
STREAM = &TNC->Streams[Stream];
if (TNC->SwallowSignon)
{
TNC->SwallowSignon = FALSE; // Discard *** connected
return 0;
}
txlen = GetLengthfromBuffer(buff) - (sizeof(void *) + 4);
if (STREAM->Connected)
{
STREAM->PacketsSent++;
if (Stream == 0)
{
bytes=ARDOPSendData(TNC, &buff->L2DATA[0], txlen);
TNC->Streams[Stream].BytesOutstanding += bytes; // So flow control works - will be updated by BUFFER response
STREAM->BytesTXed += bytes;
WritetoTrace(TNC, &buff->L2DATA[0], txlen);
}
else
{
// Packet. Only works over Serial
PMSGWITHLEN buffptr;
UCHAR * buffp;
if (TNC->PacketSock)
{
// Using Packet over TCP)
// Chan, Cmd/Data, Len-1
UCHAR Encoded[280];
int EncLen;
int SentLen;
Encoded[0] = Stream;
Encoded[1] = 0; // Data
Encoded[2] = txlen - 1;
memcpy(&Encoded[3], &buff->L2DATA[0], txlen);
EncLen = txlen + 3;
SentLen = send(TNC->PacketSock, Encoded, EncLen, 0);
// We should increment outstanding here as TCP interface can fill buffer
// very quickly
TNC->Streams[Stream].BytesOutstanding += txlen;
if (SentLen != EncLen)
{
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Pkt Write Failed for port %d - error code = %d\r\n", TNC->PacketPort, winerr);
WritetoConsole(ErrMsg);
closesocket(TNC->PacketSock);
TNC->PacketSock = 0;
}
return 0;
}
if (TNC->ARDOPCommsMode == 'T')
return 0;
buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return 0; // No buffers, so ignore
buffptr->Len = txlen + 1;
buffp = &buffptr->Data[0];
buffp[0] = 0; // CMD/Data Flag on front
memcpy(buffp + 1, &buff->L2DATA[0], txlen);
C_Q_ADD(&TNC->Streams[Stream].BPQtoPACTOR_Q, buffptr);
STREAM->FramesQueued++;
if (TNC->Timeout == 0) // if link idle can send now
ARDOPSCSPoll(TNC);
return 0;
}
}
else
{
if (_memicmp(&buff->L2DATA[0], "D\r", 2) == 0 || _memicmp(&buff->L2DATA[0], "BYE\r", 4) == 0)
{
STREAM->ReportDISC = TRUE; // Tell Node
return 0;
}
if (TNC->FECMode)
{
char Buffer[300];
int len;
// Send FEC Data
buff->L2DATA[txlen] = 0;
len = sprintf(Buffer, "%-9s: %s", TNC->Streams[0].MyCall, &buff->L2DATA[0]);
ARDOPSendData(TNC, Buffer, len);
TNC->FECPending = 1;
return 0;
}
// See if Local command (eg RADIO)
if (_memicmp(&buff->L2DATA[0], "RADIO ", 6) == 0)
{
sprintf(&buff->L2DATA[0], "%d %s", TNC->Port, &buff->L2DATA[6]);
if (Rig_Command(TNC->PortRecord->ATTACHEDSESSIONS[0]->L4CROSSLINK->CIRCUITINDEX, &buff->L2DATA[0]))
{
}
else
{
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0) return 1; // No buffers, so ignore
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "%s", &buff->L2DATA[0]);
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
}
return 1;
}
// if (_memicmp(&buff[8], "PAC ", 4) == 0 && _memicmp(&buff[8], "PAC MODE", 8) != 0)
// {
// PAC MODE goes to TNC, others are parsed locally
// buff[7 + txlen] = 0;
// ConfigVirtualKISSPort(TNC, &buff[8]);
// return 1;
// }
if (_memicmp(&buff->L2DATA[0], "OVERRIDEBUSY", 12) == 0)
{
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
TNC->OverrideBusy = TRUE;
if (buffptr)
{
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} OK\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
}
return 0;
}
if (_memicmp(&buff->L2DATA[0], "MAXCONREQ", 9) == 0)
{
if (buff->L2DATA[9] != 13)
{
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
// Limit connects
int tries = atoi(&buff->L2DATA[10]);
if (tries > 10) tries = 10;
TNC->MaxConReq = tries;
if (buffptr)
{
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} OK\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
}
return 0;
}
}
if (_memicmp(&buff->L2DATA[0], "SessionTimeLimit", 16) == 0)
{
if (buff->L2DATA[16] != 13)
{
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
TNC->SessionTimeLimit = atoi(&buff->L2DATA[16]) * 60;
if (buffptr)
{
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} OK\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
}
return 0;
}
}
if (_memicmp(&buff->L2DATA[0], "ARQBW ", 6) == 0)
TNC->WinmorCurrentMode = 0; // So scanner will set next value
if (_memicmp(&buff->L2DATA[0], "CODEC TRUE", 9) == 0)
TNC->StartSent = TRUE;
if (_memicmp(&buff->L2DATA[0], "D\r", 2) == 0)
{
STREAM->ReportDISC = TRUE; // Tell Node
return 0;
}
if (_memicmp(&buff->L2DATA[0], "FEC\r", 4) == 0 || _memicmp(&buff->L2DATA[0], "FEC ", 4) == 0)
{
TNC->FECMode = TRUE;
TNC->FECIDTimer = 0;
// ARDOPSendCommand(TNC,"FECRCV TRUE");
return 0;
}
if (_memicmp(&buff->L2DATA[0], "PING ", 5) == 0)
{
if (InterlockedCheckBusy(TNC))
{
// Channel Busy. Unless override set, reject
if (TNC->OverrideBusy == 0)
{
// Reject
PMSGWITHLEN buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr)
{
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} Ping blocked by Busy\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
}
return 0;
}
}
TNC->OverrideBusy = FALSE;
}
// See if a Connect Command. If so, start codec and set Connecting
if (toupper(buff->L2DATA[0]) == 'C' && buff->L2DATA[1] == ' ' && txlen > 2) // Connect
{
char Connect[80];
char * ptr = strchr(&buff->L2DATA[2], 13);
if (ptr)
*ptr = 0;
_strupr(&buff->L2DATA[2]);
if (strlen(&buff->L2DATA[2]) > 9)
buff->L2DATA[11] = 0;
if (Stream == 0)
{
sprintf(Connect, "ARQCALL %s %d", &buff->L2DATA[2], TNC->MaxConReq);
ARDOPChangeMYC(TNC, TNC->Streams[0].MyCall);
// See if Busy
if (InterlockedCheckBusy(TNC))
{
// Channel Busy. Unless override set, wait
if (TNC->OverrideBusy == 0)
{
// Save Command, and wait up to 10 secs
sprintf(TNC->WEB_TNCSTATE, "Waiting for clear channel");
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
TNC->ConnectCmd = _strdup(Connect);
TNC->BusyDelay = TNC->BusyWait * 10; // BusyWait secs
return 0;
}
}
TNC->OverrideBusy = FALSE;
// ARDOPSendCommand(TNC, "LISTEN TRUE", TRUE); // !!!! Temp bug workaround !!!!
memset(TNC->Streams[0].RemoteCall, 0, 10);
strcpy(TNC->Streams[0].RemoteCall, &buff->L2DATA[2]);
sprintf(TNC->WEB_TNCSTATE, "%s Connecting to %s", STREAM->MyCall, STREAM->RemoteCall);
ARDOPSendCommand(TNC, Connect, TRUE);
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
}
else
{
// Packet Connect
sprintf(Connect, "%cPKTCALL %s %s", Stream, &buff->L2DATA[2], STREAM->MyCall);
ARDOPSendPktCommand(TNC, Stream, Connect);
}
STREAM->Connecting = TRUE;
STREAM->ConnectTime = time(NULL);
return 0;
}
buff->L2DATA[txlen - 1] = 0; // Remove CR
ARDOPSendCommand(TNC, &buff->L2DATA[0], TRUE);
}
return 0;
case 3:
// CHECK IF OK TO SEND (And check TNC Status)
Stream = (int)(size_t)buff;
// I think we should check buffer space for all comms modes
//if (!(TNC->ARDOPCommsMode == 'T'))
//{
// // if serial mode must check buffer space
{
int Queued;
int Outstanding = TNC->Streams[Stream].BytesOutstanding;
if (Stream == 0)
Queued = TNC->Streams[13].FramesQueued; // ARDOP Native Mode Send Queue
else
Queued = TNC->Streams[Stream].FramesQueued;
if (TNC->Mode == 'O') // OFDM version has more buffer space
{
if (Queued > 4 || Outstanding > 8500)
return (1 | (TNC->HostMode | TNC->CONNECTED) << 8 | STREAM->Disconnecting << 15);
}
else if (TNC->Mode == '3') // ARDOP3 has a bit more buffer space
{
if (Queued > 4 || Outstanding > 5000)
return (1 | (TNC->HostMode | TNC->CONNECTED) << 8 | STREAM->Disconnecting << 15);
}
else
{
if (Queued > 4 || Outstanding > 2000)
return (1 | (TNC->HostMode | TNC->CONNECTED) << 8 | STREAM->Disconnecting << 15);
}
}
if (TNC->Streams[Stream].Attached == 0)
return TNC->CONNECTED << 8 | 1;
return (TNC->CONNECTED << 8 | TNC->Streams[Stream].Disconnecting << 15); // OK
case 4: // reinit7
return 0;
case 5: // Close
if (TNC->CONNECTED)
{
if (TNC->WeStartedTNC)
{
GetSemaphore(&Semaphore, 52);
ARDOPSendCommand(TNC, "CLOSE", FALSE);
FreeSemaphore(&Semaphore);
Sleep(100);
}
}
if (TNC->TCPSock)
{
shutdown(TNC->TCPSock, SD_BOTH);
Sleep(100);
closesocket(TNC->TCPSock);
}
if (TNC->TCPDataSock)
{
shutdown(TNC->TCPDataSock, SD_BOTH);
Sleep(100);
closesocket(TNC->TCPDataSock);
}
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
return 0;
case 6: // Scan Stop Interface
Param = (size_t)buff;
if (Param == 2) // Check Permission (Shouldn't happen)
{
Debugprintf("Scan Check Permission called on ARDOP");
return 1; // OK to change
}
if (Param == 1) // Request Permission
{
if (TNC->ARDOPCommsMode == 'T') // TCP Mode
{
if (!TNC->CONNECTED)
return 0; // No connection so no interlock
}
else
{
// Serial Modes
if (!TNC->HostMode)
return 0; // No connection so no interlock
}
if (TNC->ConnectPending == 0 && TNC->PTTState == 0)
{
ARDOPSendCommand(TNC, "LISTEN FALSE", TRUE);
TNC->GavePermission = TRUE;
return 0; // OK to Change
}
if (TNC->ConnectPending)
TNC->ConnectPending--; // Time out if set too long
return TRUE;
}
if (Param == 3) // Release Permission
{
if (TNC->GavePermission)
{
TNC->GavePermission = FALSE;
if (TNC->ARDOPCurrentMode[0] != 'S') // Skip
ARDOPSendCommand(TNC, "LISTEN TRUE", TRUE);
}
return 0;
}
// Param is Address of a struct ScanEntry
Scan = (struct ScanEntry *)buff;
if (Scan->ARDOPMode[0] == 0)
{
// Not specified, so no change from previous
return 0;
}
if (strcmp(Scan->ARDOPMode, TNC->ARDOPCurrentMode) != 0)
{
// Mode changed
char CMD[32];
if (TNC->ARDOPCurrentMode[0] == 'S') // Skip
ARDOPSendCommand(TNC, "LISTEN TRUE", TRUE);
// Debugprintf("ARDOPMODE %s", Scan->ARDOPMode);
memcpy(TNC->ARDOPCurrentMode, Scan->ARDOPMode, 6);
if (Scan->ARDOPMode[0] == 'S') // SKIP - Dont Allow Connects
{
if (TNC->ARDOPCurrentMode[0] != 'S')
{
ARDOPSendCommand(TNC, "LISTEN FALSE", TRUE);
TNC->ARDOPCurrentMode[0] = 'S';
}
TNC->WL2KMode = 0;
return 0;
}
if (strchr(Scan->ARDOPMode, 'F'))
sprintf(CMD, "ARQBW %sORCED", Scan->ARDOPMode);
else if (strchr(Scan->ARDOPMode, 'M'))
sprintf(CMD, "ARQBW %sAX", Scan->ARDOPMode);
else
sprintf(CMD, "ARQBW %s", Scan->ARDOPMode); // ARDOPOFDM doesn't use MAX/FORCED
ARDOPSendCommand(TNC, CMD, TRUE);
return 0;
}
return 0;
}
return 0;
}
VOID ARDOPReleaseTNC(struct TNCINFO * TNC)
{
// Set mycall back to Node or Port Call, and Start Scanner
UCHAR TXMsg[1000];
ARDOPChangeMYC(TNC, TNC->NodeCall);
ARDOPSendCommand(TNC, "LISTEN TRUE", TRUE);
strcpy(TNC->WEB_TNCSTATE, "Free");
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
// Start Scanner
// Start Scanner
if (TNC->DefaultRadioCmd)
{
sprintf(TXMsg, "%d %s", TNC->Port, TNC->DefaultRadioCmd);
Rig_Command(-1, TXMsg);
}
sprintf(TXMsg, "%d SCANSTART 15", TNC->Port);
Rig_Command(-1, TXMsg);
ReleaseOtherPorts(TNC);
}
VOID ARDOPSuspendPort(struct TNCINFO * TNC)
{
ARDOPSendCommand(TNC, "LISTEN FALSE", TRUE);
}
VOID ARDOPReleasePort(struct TNCINFO * TNC)
{
ARDOPSendCommand(TNC, "LISTEN TRUE", TRUE);
}
extern char WebProcTemplate[];
extern char sliderBit[];
static int WebProc(struct TNCINFO * TNC, char * Buff, BOOL LOCAL)
{
int Len = sprintf(Buff, WebProcTemplate, TNC->Port, TNC->Port, "ARDOP Status", "ARDOP Status");
if (TNC->TXFreq)
Len += sprintf(&Buff[Len], sliderBit, TNC->TXOffset, TNC->TXOffset);
Len += sprintf(&Buff[Len], "<table style=\"text-align: left; width: 500px; font-family: monospace; align=center \" border=1 cellpadding=2 cellspacing=2>");
Len += sprintf(&Buff[Len], "<tr><td width=110px>Comms State</td><td>%s</td></tr>", TNC->WEB_COMMSSTATE);
Len += sprintf(&Buff[Len], "<tr><td>TNC State</td><td>%s</td></tr>", TNC->WEB_TNCSTATE);
Len += sprintf(&Buff[Len], "<tr><td>Mode</td><td>%s</td></tr>", TNC->WEB_MODE);
Len += sprintf(&Buff[Len], "<tr><td>Channel State</td><td>%s &nbsp; %s</td></tr>", TNC->WEB_CHANSTATE, TNC->WEB_LEVELS);
Len += sprintf(&Buff[Len], "<tr><td>Proto State</td><td>%s</td></tr>", TNC->WEB_PROTOSTATE);
Len += sprintf(&Buff[Len], "<tr><td>Traffic</td><td>%s</td></tr>", TNC->WEB_TRAFFIC);
// Len += sprintf(&Buff[Len], "<tr><td>TNC Restarts</td><td></td></tr>", TNC->WEB_RESTARTS);
Len += sprintf(&Buff[Len], "</table>");
Len += sprintf(&Buff[Len], "<textarea rows=10 style=\"width:500px; height:250px;\" id=textarea >%s</textarea>", TNC->WebBuffer);
Len = DoScanLine(TNC, Buff, Len);
return Len;
}
VOID * ARDOPExtInit(EXTPORTDATA * PortEntry)
{
int i, port;
char Msg[255];
char * ptr;
APPLCALLS * APPL;
struct TNCINFO * TNC;
char Aux[100] = "MYAUX ";
char Appl[11];
char * TempScript;
int Stream;
//
// Will be called once for each WINMOR port
//
// The Socket to connect to is in IOBASE
//
port = PortEntry->PORTCONTROL.PORTNUMBER;
ReadConfigFile(port, ProcessLine);
TNC = TNCInfo[port];
if (TNC == NULL)
{
// Not defined in Config file
sprintf(Msg," ** Error - no info in BPQ32.cfg for this port\n");
WritetoConsole(Msg);
return ExtProc;
}
TNC->Port = port;
if (TNC->LogPath)
ARDOPOpenLogFiles(TNC);
TNC->ARDOPBuffer = malloc(8192);
TNC->ARDOPDataBuffer = malloc(16384);
TNC->ARDOPAPRS = zalloc(512);
TNC->ARDOPAPRSLen = 0;
if (TNC->ProgramPath)
TNC->WeStartedTNC = RestartTNC(TNC);
TNC->Hardware = H_ARDOP;
if (TNC->BusyWait == 0)
TNC->BusyWait = 10;
if (TNC->BusyHold == 0)
TNC->BusyHold = 1;
TNC->PortRecord = PortEntry;
if (PortEntry->PORTCONTROL.PORTCALL[0] == 0)
memcpy(TNC->NodeCall, MYNODECALL, 10);
else
ConvFromAX25(&PortEntry->PORTCONTROL.PORTCALL[0], TNC->NodeCall);
if (PortEntry->PORTCONTROL.PORTINTERLOCK && TNC->RXRadio == 0 && TNC->TXRadio == 0)
TNC->RXRadio = TNC->TXRadio = PortEntry->PORTCONTROL.PORTINTERLOCK;
PortEntry->PORTCONTROL.PROTOCOL = 10;
PortEntry->PORTCONTROL.PORTQUALITY = 0;
for (Stream = 1; Stream <= APMaxStreams; Stream++)
{
TNC->Streams[Stream].DEDStream = Stream;
}
if (TNC->PacketChannels > APMaxStreams)
TNC->PacketChannels = APMaxStreams;
PortEntry->MAXHOSTMODESESSIONS = TNC->PacketChannels + 1;
PortEntry->SCANCAPABILITIES = SIMPLE; // Scan Control - pending connect only
PortEntry->PERMITGATEWAY = TRUE; // Can change ax.25 call on each stream
PortEntry->PORTCONTROL.UICAPABLE = TRUE;
if (PortEntry->PORTCONTROL.PORTPACLEN == 0)
PortEntry->PORTCONTROL.PORTPACLEN = 236;
TNC->SuspendPortProc = ARDOPSuspendPort;
TNC->ReleasePortProc = ARDOPReleasePort;
PortEntry->PORTCONTROL.PORTSTARTCODE = ARDOPStartPort;
PortEntry->PORTCONTROL.PORTSTOPCODE = ARDOPStopPort;
TNC->ModemCentre = 1500; // ARDOP is always 1500 Offset
ptr=strchr(TNC->NodeCall, ' ');
if (ptr) *(ptr) = 0; // Null Terminate
// Set Essential Params and MYCALL
// Put overridable ones on front, essential ones on end
TempScript = malloc(1000);
strcpy(TempScript, "INITIALIZE\r");
strcat(TempScript, "VERSION\r");
strcat(TempScript, "CWID False\r");
strcat(TempScript, "PROTOCOLMODE ARQ\r");
strcat(TempScript, "ARQTIMEOUT 90\r");
// strcat(TempScript, "ROBUST False\r");
strcat(TempScript, TNC->InitScript);
free(TNC->InitScript);
TNC->InitScript = TempScript;
// Set MYCALL
// strcat(TNC->InitScript,"FECRCV True\r");
// strcat(TNC->InitScript,"AUTOBREAK True\r");
sprintf(Msg, "MYCALL %s\r", TNC->NodeCall);
strcat(TNC->InitScript, Msg);
// strcat(TNC->InitScript,"PROCESSID\r");
// strcat(TNC->InitScript,"CODEC TRUE\r");
// strcat(TNC->InitScript,"LISTEN TRUE\r");
for (i = 0; i < 32; i++)
{
APPL=&APPLCALLTABLE[i];
if (APPL->APPLCALL_TEXT[0] > ' ')
{
char * ptr;
memcpy(Appl, APPL->APPLCALL_TEXT, 10);
ptr=strchr(Appl, ' ');
if (ptr)
{
*ptr++ = ',';
*ptr = 0;
}
strcat(Aux, Appl);
}
}
if (strlen(Aux) > 8)
{
Aux[strlen(Aux) - 1] = '\r';
strcat(TNC->InitScript, Aux);
}
strcpy(TNC->CurrentMYC, TNC->NodeCall);
if (TNC->WL2K == NULL)
if (PortEntry->PORTCONTROL.WL2KInfo.RMSCall[0]) // Alrerady decoded
TNC->WL2K = &PortEntry->PORTCONTROL.WL2KInfo;
if (TNC->destaddr.sin_family == 0)
{
// not defined in config file, so use localhost and port from IOBASE
TNC->destaddr.sin_family = AF_INET;
TNC->destaddr.sin_port = htons(PortEntry->PORTCONTROL.IOBASE);
TNC->Datadestaddr.sin_family = AF_INET;
TNC->Datadestaddr.sin_port = htons(PortEntry->PORTCONTROL.IOBASE+1);
TNC->HostName=malloc(10);
if (TNC->HostName != NULL)
strcpy(TNC->HostName,"127.0.0.1");
}
PortEntry->PORTCONTROL.TNC = TNC;
TNC->WebWindowProc = WebProc;
TNC->WebWinX = 520;
TNC->WebWinY = 500;
TNC->WebBuffer = zalloc(5000);
TNC->WEB_COMMSSTATE = zalloc(100);
TNC->WEB_TNCSTATE = zalloc(100);
TNC->WEB_CHANSTATE = zalloc(100);
TNC->WEB_BUFFERS = zalloc(100);
TNC->WEB_PROTOSTATE = zalloc(100);
TNC->WEB_RESTARTTIME = zalloc(100);
TNC->WEB_RESTARTS = zalloc(100);
TNC->WEB_MODE = zalloc(20);
TNC->WEB_TRAFFIC = zalloc(100);
TNC->WEB_LEVELS =zalloc(32);
#ifndef LINBPQ
CreatePactorWindow(TNC, ClassName, WindowTitle, RigControlRow, PacWndProc, 500, 450, ForcedClose);
CreateWindowEx(0, "STATIC", "Comms State", WS_CHILD | WS_VISIBLE, 10,6,120,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_COMMSSTATE = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE, 120,6,386,20, TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "TNC State", WS_CHILD | WS_VISIBLE, 10,28,106,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_TNCSTATE = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE, 120,28,520,20, TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "Mode", WS_CHILD | WS_VISIBLE, 10,50,80,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_MODE = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE, 120,50,200,20, TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "Channel State", WS_CHILD | WS_VISIBLE, 10,72,110,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_CHANSTATE = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE, 120,72,82,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_LEVELS = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE, 200,72,200,20, TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "Proto State", WS_CHILD | WS_VISIBLE,10,94,80,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_PROTOSTATE = CreateWindowEx(0, "STATIC", "", WS_CHILD | WS_VISIBLE,120,94,374,20 , TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "Traffic", WS_CHILD | WS_VISIBLE,10,116,80,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_TRAFFIC = CreateWindowEx(0, "STATIC", "0 0 0 0", WS_CHILD | WS_VISIBLE,120,116,374,20 , TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "TNC Restarts", WS_CHILD | WS_VISIBLE,10,138,100,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_RESTARTS = CreateWindowEx(0, "STATIC", "0", WS_CHILD | WS_VISIBLE,120,138,40,20 , TNC->hDlg, NULL, hInstance, NULL);
CreateWindowEx(0, "STATIC", "Last Restart", WS_CHILD | WS_VISIBLE,140,138,100,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->xIDC_RESTARTTIME = CreateWindowEx(0, "STATIC", "Never", WS_CHILD | WS_VISIBLE,250,138,200,20, TNC->hDlg, NULL, hInstance, NULL);
TNC->hMonitor= CreateWindowEx(0, "LISTBOX", "", WS_CHILD | WS_VISIBLE | LBS_NOINTEGRALHEIGHT |
LBS_DISABLENOSCROLL | WS_HSCROLL | WS_VSCROLL,
0,170,250,300, TNC->hDlg, NULL, hInstance, NULL);
TNC->ClientHeight = 450;
TNC->ClientWidth = 500;
TNC->hMenu = CreatePopupMenu();
AppendMenu(TNC->hMenu, MF_STRING, WINMOR_KILL, "Kill ARDOP TNC");
AppendMenu(TNC->hMenu, MF_STRING, WINMOR_RESTART, "Kill and Restart ARDOP TNC");
AppendMenu(TNC->hMenu, MF_STRING, WINMOR_RESTARTAFTERFAILURE, "Restart TNC after failed Connection");
AppendMenu(TNC->hMenu, MF_STRING, ARDOP_ABORT, "Abort Current Session");
CheckMenuItem(TNC->hMenu, WINMOR_RESTARTAFTERFAILURE, (TNC->RestartAfterFailure) ? MF_CHECKED : MF_UNCHECKED);
MoveWindows(TNC);
#endif
if (TNC->ARDOPCommsMode == 'T')
{
Consoleprintf("ARDOP Host %s %d", TNC->HostName, htons(TNC->destaddr.sin_port));
ConnecttoARDOP(TNC);
}
else if (TNC->ARDOPCommsMode == 'E')
{
Consoleprintf("ARDOP TCPSerial %s:%d", TNC->HostName, htons(TNC->destaddr.sin_port));
SerialConnecttoTCP(TNC);
}
else if (TNC->ARDOPCommsMode == 'S')
{
TNC->PortRecord->PORTCONTROL.SerialPortName = _strdup(TNC->ARDOPSerialPort); // for common routines
Consoleprintf("ARDOP Serial %s", TNC->ARDOPSerialPort);
OpenCOMMPort(TNC, TNC->ARDOPSerialPort, TNC->ARDOPSerialSpeed, FALSE);
}
else if (TNC->ARDOPCommsMode == 'I')
{
#ifdef WIN32
sprintf(Msg,"ARDOP I2C is not supported on WIN32 systems\n");
WritetoConsoleLocal(Msg);
#else
#ifdef NOI2C
sprintf(Msg,"I2C is not supported on this systems\n");
WritetoConsoleLocal(Msg);
#else
char i2cname[30];
int fd;
int retval;
if (strlen(TNC->ARDOPSerialPort) < 3)
{
sprintf(i2cname, "/dev/i2c-%s", TNC->ARDOPSerialPort);
TNC->ARDOPSerialPort = _strdup(i2cname);
}
else
strcpy(i2cname, TNC->ARDOPSerialPort);
TNC->PortRecord->PORTCONTROL.SerialPortName = _strdup(i2cname); // for common routines
Consoleprintf("ARDOP I2C Bus %s Addr %d ", i2cname, TNC->ARDOPSerialSpeed);
// Open and configure the i2c interface
fd = TNC->hDevice = open(i2cname, O_RDWR);
if (fd < 0)
printf("Cannot find i2c bus %s \n", i2cname);
else
{
retval = ioctl(TNC->hDevice, I2C_SLAVE, TNC->ARDOPSerialSpeed);
if(retval == -1)
printf("Cannot open i2c device %x\n", TNC->ARDOPSerialSpeed);
ioctl(TNC->hDevice, I2C_TIMEOUT, 10); //100 mS
}
#endif
#endif
}
time(&TNC->lasttime); // Get initial time value
return ExtProc;
}
VOID TNCLost(struct TNCINFO * TNC)
{
int Stream = 0;
struct STREAMINFO * STREAM;
if (TNC->TCPSock)
closesocket(TNC->TCPSock);
if (TNC->TCPDataSock)
closesocket(TNC->TCPDataSock);
if (TNC->PacketSock)
closesocket(TNC->PacketSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
TNC->PacketSock = 0;
TNC->CONNECTED = FALSE;
for (Stream = 0; Stream <= APMaxStreams; Stream++)
{
STREAM = &TNC->Streams[Stream];
STREAM->BytesOutstanding = 0;
if (Stream == 0)
{
sprintf(TNC->WEB_TRAFFIC, "Sent %d RXed %d Queued %d",
STREAM->BytesTXed - STREAM->BytesOutstanding, STREAM->BytesRXed, STREAM->BytesOutstanding);
MySetWindowText(TNC->xIDC_TRAFFIC, TNC->WEB_TRAFFIC);
}
if (STREAM->Attached)
{
STREAM->Connected = FALSE;
STREAM->Connecting = FALSE;
STREAM->ReportDISC = TRUE;
}
}
}
int ConnecttoARDOP(struct TNCINFO * TNC)
{
_beginthread(ARDOPThread, 0, (void *)TNC);
return 0;
}
VOID ARDOPThread(struct TNCINFO * TNC)
{
// Opens sockets and looks for data on control and data sockets.
// Socket may be TCP/IP or Serial
char Msg[255];
int err, i, ret;
u_long param=1;
BOOL bcopt=TRUE;
struct hostent * HostEnt;
fd_set readfs;
fd_set errorfs;
struct timeval timeout;
char * ptr1, * ptr2;
PMSGWITHLEN buffptr;
char Cmd[64];
if (TNC->HostName == NULL)
return;
TNC->BusyFlags = 0;
TNC->CONNECTING = TRUE;
Sleep(3000); // Allow init to complete
#ifdef WIN32
if (strcmp(TNC->HostName, "127.0.0.1") == 0)
{
// can only check if running on local host
TNC->PID = GetListeningPortsPID(TNC->destaddr.sin_port);
if (TNC->PID == 0)
{
TNC->CONNECTING = FALSE;
return; // Not listening so no point trying to connect
}
// Get the File Name in case we want to restart it.
if (TNC->ProgramPath == NULL)
{
if (GetModuleFileNameExPtr)
{
HANDLE hProc;
char ExeName[256] = "";
hProc = OpenProcess(PROCESS_QUERY_INFORMATION |PROCESS_VM_READ, FALSE, TNC->PID);
if (hProc)
{
GetModuleFileNameExPtr(hProc, 0, ExeName, 255);
CloseHandle(hProc);
TNC->ProgramPath = _strdup(ExeName);
}
}
}
}
#endif
// // If we started the TNC make sure it is still running.
// if (!IsProcess(TNC->PID))
// {
// RestartTNC(TNC);
// Sleep(3000);
// }
TNC->destaddr.sin_addr.s_addr = inet_addr(TNC->HostName);
TNC->Datadestaddr.sin_addr.s_addr = inet_addr(TNC->HostName);
if (TNC->destaddr.sin_addr.s_addr == INADDR_NONE)
{
// Resolve name to address
HostEnt = gethostbyname (TNC->HostName);
if (!HostEnt)
{
TNC->CONNECTING = FALSE;
return; // Resolve failed
}
memcpy(&TNC->destaddr.sin_addr.s_addr,HostEnt->h_addr,4);
memcpy(&TNC->Datadestaddr.sin_addr.s_addr,HostEnt->h_addr,4);
}
if (TNC->TCPSock)
{
Debugprintf("ARDOP Closing Sock %d", TNC->TCPSock);
closesocket(TNC->TCPSock);
}
if (TNC->TCPDataSock)
{
Debugprintf("ARDOP Closing Sock %d", TNC->TCPDataSock);
closesocket(TNC->TCPDataSock);
}
if (TNC->PacketSock)
{
Debugprintf("ARDOP Closing Sock %d", TNC->PacketSock);
closesocket(TNC->PacketSock);
}
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
TNC->PacketSock = 0;
TNC->TCPSock=socket(AF_INET,SOCK_STREAM,0);
if (TNC->TCPSock == INVALID_SOCKET)
{
i=sprintf(Msg, "Socket Failed for ARDOP socket - error code = %d\r\n", WSAGetLastError());
WritetoConsole(Msg);
TNC->CONNECTING = FALSE;
return;
}
TNC->TCPDataSock=socket(AF_INET,SOCK_STREAM,0);
if (TNC->TCPDataSock == INVALID_SOCKET)
{
i=sprintf(Msg, "Socket Failed for ARDOP Data socket - error code = %d\r\n", WSAGetLastError());
WritetoConsole(Msg);
TNC->CONNECTING = FALSE;
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
return;
}
setsockopt(TNC->TCPSock, SOL_SOCKET, SO_REUSEADDR, (const char FAR *)&bcopt, 4);
setsockopt(TNC->TCPDataSock, SOL_SOCKET, SO_REUSEADDR, (const char FAR *)&bcopt, 4);
// setsockopt(TNC->TCPDataSock, IPPROTO_TCP, TCP_NODELAY, (const char FAR *)&bcopt, 4);
sinx.sin_family = AF_INET;
sinx.sin_addr.s_addr = INADDR_ANY;
sinx.sin_port = 0;
if (connect(TNC->TCPSock,(LPSOCKADDR) &TNC->destaddr,sizeof(TNC->destaddr)) == 0)
{
//
// Connected successful
//
}
else
{
if (TNC->Alerted == FALSE)
{
sprintf(Msg, "Connect Failed for ARDOP socket - error code = %d Port %d\n",
WSAGetLastError(), htons(TNC->destaddr.sin_port));
WritetoConsole(Msg);
sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC failed");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->Alerted = TRUE;
}
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
TNC->CONNECTING = FALSE;
return;
}
// Connect Data Port
if (connect(TNC->TCPDataSock,(LPSOCKADDR) &TNC->Datadestaddr,sizeof(TNC->Datadestaddr)) == 0)
{
//
// Connected successful
//
}
else
{
if (TNC->Alerted == FALSE)
{
err=WSAGetLastError();
i=sprintf(Msg, "Connect Failed for ARDOP Data socket - error code = %d\r\n", err);
WritetoConsole(Msg);
sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC failed");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->Alerted = TRUE;
}
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
TNC->CONNECTING = FALSE;
return;
}
if (TNC->PacketPort)
{
struct sockaddr_in destaddr;
TNC->PacketSock = socket(AF_INET,SOCK_STREAM,0);
if (TNC->PacketSock == INVALID_SOCKET)
{
i=sprintf(Msg, "Socket Failed for ARDOP Packet socket - error code = %d\r\n", WSAGetLastError());
WritetoConsole(Msg);
TNC->PacketSock = 0;
}
else
{
setsockopt(TNC->PacketSock, SOL_SOCKET, SO_REUSEADDR, (const char FAR *)&bcopt, 4);
// setsockopt(TNC->PacketSock, IPPROTO_TCP, TCP_NODELAY, (const char FAR *)&bcopt, 4);
destaddr.sin_addr.s_addr = inet_addr(TNC->HostName);
destaddr.sin_family = AF_INET;
destaddr.sin_port = htons(TNC->PacketPort);
if (connect(TNC->PacketSock,(LPSOCKADDR) &destaddr, sizeof(destaddr)) == 0)
{
// Connected successful
}
else
{
if (TNC->Alerted == FALSE)
{
err=WSAGetLastError();
i=sprintf(Msg, "Connect Failed for ARDOP Packet socket - error code = %d\r\n", err);
WritetoConsole(Msg);
TNC->Alerted = TRUE;
}
closesocket(TNC->PacketSock);
TNC->PacketSock = 0;
}
}
}
#ifndef LINBPQ
// FreeSemaphore(&Semaphore);
EnumWindows(EnumARDOPWindowsProc, (LPARAM)TNC);
// GetSemaphore(&Semaphore, 52);
#endif
Sleep(1000);
TNC->LastFreq = 0; // so V4 display will be updated
TNC->CONNECTING = FALSE;
TNC->CONNECTED = TRUE;
TNC->BusyFlags = 0;
TNC->InputLen = 0;
// Send INIT script
// ARDOP needs each command in a separate send
ptr1 = &TNC->InitScript[0];
// We should wait for first RDY. Cheat by queueing a null command
GetSemaphore(&Semaphore, 52);
while(TNC->BPQtoWINMOR_Q)
{
buffptr = (PMSGWITHLEN)Q_REM(&TNC->BPQtoWINMOR_Q);
if (buffptr)
ReleaseBuffer(buffptr);
}
buffptr = (PMSGWITHLEN)GetBuff();
buffptr->Len = 0;
C_Q_ADD(&TNC->BPQtoWINMOR_Q, buffptr);
while (ptr1 && ptr1[0])
{
ptr2 = strchr(ptr1, 13);
if (ptr2)
*(ptr2) = 0;
// if Date or Time command add current time
if (_memicmp(ptr1, "DATETIME", 4) == 0)
{
time_t T;
struct tm * tm;
T = time(NULL);
tm = gmtime(&T);
sprintf(Cmd, "DATETIME %02d %02d %02d %02d %02d %02d",
tm->tm_mday, tm->tm_mon + 1, tm->tm_year - 100,
tm->tm_hour, tm->tm_min, tm->tm_sec);
ptr1 = Cmd;
}
ARDOPSendCommand(TNC, ptr1, TRUE);
if (ptr2)
*(ptr2++) = 13; // Put CR back for next time
ptr1 = ptr2;
}
TNC->Alerted = TRUE;
sprintf(TNC->WEB_COMMSSTATE, "Connected to ARDOP TNC");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
FreeSemaphore(&Semaphore);
sprintf(Msg, "Connected to ARDOP TNC Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
while (TNC->CONNECTED)
{
FD_ZERO(&readfs);
FD_ZERO(&errorfs);
FD_SET(TNC->TCPSock,&readfs);
FD_SET(TNC->TCPSock,&errorfs);
if (TNC->CONNECTED) FD_SET(TNC->TCPDataSock,&readfs);
// FD_ZERO(&writefs);
// if (TNC->BPQtoWINMOR_Q) FD_SET(TNC->TCPDataSock,&writefs); // Need notification of busy clearing
if (TNC->PacketSock)
{
FD_SET(TNC->PacketSock,&errorfs);
FD_SET(TNC->PacketSock,&readfs);
}
// FD_ZERO(&writefs);
// if (TNC->BPQtoWINMOR_Q) FD_SET(TNC->TCPDataSock,&writefs); // Need notification of busy clearing
if (TNC->CONNECTING || TNC->CONNECTED) FD_SET(TNC->TCPDataSock,&errorfs);
timeout.tv_sec = 600;
timeout.tv_usec = 0; // We should get messages more frequently that this
if (TNC->PacketSock)
ret = select((int)TNC->PacketSock + 1, &readfs, NULL, &errorfs, &timeout);
else
ret = select((int)TNC->TCPDataSock + 1, &readfs, NULL, &errorfs, &timeout);
if (ret == SOCKET_ERROR)
{
Debugprintf("ARDOP Select failed %d ", WSAGetLastError());
goto Lost;
}
if (ret > 0)
{
// See what happened
if (FD_ISSET(TNC->TCPSock, &readfs))
{
GetSemaphore(&Semaphore, 52);
ARDOPProcessReceivedControl(TNC);
FreeSemaphore(&Semaphore);
}
if (FD_ISSET(TNC->TCPDataSock, &readfs))
{
GetSemaphore(&Semaphore, 52);
ARDOPProcessReceivedData(TNC);
FreeSemaphore(&Semaphore);
}
if (FD_ISSET(TNC->PacketSock, &readfs))
{
int InputLen, Used;
UCHAR Buffer[4096];
InputLen = recv(TNC->PacketSock, Buffer, 4096, 0);
if (InputLen == 0 || InputLen == SOCKET_ERROR)
{
sprintf(Msg, "ARDOP Connection lost for Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC lost");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->CONNECTED = FALSE;
TNC->Alerted = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE); // Make sure PTT is down
TNCLost(TNC);
return;
}
// Could be more than one frame in buffer
while (InputLen > 0)
{
GetSemaphore(&Semaphore, 52);
Used = ARDOPProcessDEDFrame(TNC, Buffer, InputLen);
FreeSemaphore(&Semaphore);
if (Used == 0)
break; // need to check
InputLen -= Used;
if (InputLen > 0)
memmove(Buffer, &Buffer[Used], InputLen);
}
}
if (FD_ISSET(TNC->TCPSock, &errorfs))
{
Lost:
sprintf(Msg, "ARDOP Connection lost for Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC lost");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->CONNECTED = FALSE;
TNC->Alerted = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE); // Make sure PTT is down
if (TNC->Streams[0].Attached)
TNC->Streams[0].ReportDISC = TRUE;
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
return;
}
if (FD_ISSET(TNC->TCPDataSock, &errorfs))
{
sprintf(Msg, "ARDOP Connection lost for Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC lost");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->CONNECTED = FALSE;
TNC->Alerted = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE); // Make sure PTT is down
if (TNC->Streams[0].Attached)
TNC->Streams[0].ReportDISC = TRUE;
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
return;
}
if (FD_ISSET(TNC->PacketSock, &errorfs))
{
sprintf(Msg, "ARDOP Packet Connection lost for Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
TNC->Alerted = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE); // Make sure PTT is down
if (TNC->Streams[0].Attached)
TNC->Streams[0].ReportDISC = TRUE;
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
return;
}
continue;
}
else
{
// 60 secs without data. Shouldn't happen
sprintf(Msg, "ARDOP No Data Timeout Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
// sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC lost");
// GetSemaphore(&Semaphore, 52);
// MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
// FreeSemaphore(&Semaphore);
TNC->CONNECTED = FALSE;
TNC->Alerted = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE); // Make sure PTT is down
if (TNC->Streams[0].Attached)
TNC->Streams[0].ReportDISC = TRUE;
GetSemaphore(&Semaphore, 52);
ARDOPSendCommand(TNC, "CODEC FALSE", FALSE);
FreeSemaphore(&Semaphore);
shutdown(TNC->TCPSock, SD_BOTH);
shutdown(TNC->TCPDataSock, SD_BOTH);
Sleep(100);
closesocket(TNC->TCPSock);
closesocket(TNC->TCPDataSock);
TNC->TCPSock = 0;
TNC->TCPDataSock = 0;
if (TNC->PID && TNC->WeStartedTNC)
{
// KillTNC(TNC);
}
return;
}
}
sprintf(Msg, "ARDOP Thread Terminated Port %d\r\n", TNC->Port);
WritetoConsole(Msg);
}
#ifndef LINBPQ
BOOL CALLBACK EnumARDOPWindowsProc(HWND hwnd, LPARAM lParam)
{
char wtext[100];
struct TNCINFO * TNC = (struct TNCINFO *)lParam;
UINT ProcessId;
GetWindowText(hwnd,wtext,99);
if (memcmp(wtext,"ARDOP_Win ", 10) == 0)
{
GetWindowThreadProcessId(hwnd, &ProcessId);
if (TNC->PID == ProcessId)
{
// Our Process
sprintf (wtext, "ARDOP Virtual TNC - BPQ %s", TNC->PortRecord->PORTCONTROL.PORTDESCRIPTION);
SetWindowText(hwnd, wtext);
return FALSE;
}
}
return (TRUE);
}
#endif
VOID ARDOPProcessResponse(struct TNCINFO * TNC, UCHAR * Buffer, int MsgLen)
{
PMSGWITHLEN buffptr;
struct STREAMINFO * STREAM = &TNC->Streams[0];
Buffer[MsgLen - 1] = 0; // Remove CR
if (_memicmp(Buffer, "RDY", 3) == 0)
return; // RDY not used now
if (_memicmp(Buffer, "RADIOHEX ", 9) == 0)
{
// Parameter is block to send to radio, in hex
char c;
int val;
char * ptr1 = &Buffer[9];
UCHAR * ptr2 = Buffer;
PMSGWITHLEN buffptr;
int Len;
// if not configured to use PTC Rig Control, Ignore
if (TNC->RIG->PORT == NULL || TNC->RIG->PORT->PTC == NULL)
return;
buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == NULL)
return;
while (c = *(ptr1++))
{
val = c - 0x30;
if (val > 15) val -= 7;
val <<= 4;
c = *(ptr1++) - 0x30;
if (c > 15) c -= 7;
val |= c;
*(ptr2++) = val;
}
*(ptr2) = 0;
Len = (int)(ptr2 - Buffer);
buffptr->Len = Len;
memcpy(&buffptr->Data[0], Buffer, Len);
C_Q_ADD(&TNC->RadiotoBPQ_Q, buffptr);
// WriteCOMBlock(hRIGDevice, ptrParams, ptr2 - ptrParams);
return;
}
if (_memicmp(Buffer, "INPUTPEAKS", 10) == 0)
{
sscanf(&Buffer[10], "%i %i", &TNC->InputLevelMin, &TNC->InputLevelMax);
sprintf(TNC->WEB_LEVELS, "Input peaks %s", &Buffer[10]);
MySetWindowText(TNC->xIDC_LEVELS, TNC->WEB_LEVELS);
return; // Response shouldn't go to user
}
if (_memicmp(Buffer, "LISTEN NOW", 10) == 0)
return; // Response shouldn't go to user
if (_memicmp(Buffer, "ARQCALL ", 7) == 0)
return; // Response shouldn't go to user
if (_memicmp(Buffer, "FAULT failure to Restart Sound card", 20) == 0)
{
Debugprintf(Buffer);
// Force a restart
ARDOPSendCommand(TNC, "CODEC FALSE", TRUE);
ARDOPSendCommand(TNC, "CODEC TRUE", TRUE);
}
else
{
TNC->TimeSinceLast = 0;
}
if (_memicmp(Buffer, "STATE ", 6) == 0)
{
if (_memicmp(&Buffer[6], "OFFLINE", 7) == 0)
{
// Force a restart
ARDOPSendCommand(TNC, "CODEC FALSE", TRUE);
ARDOPSendCommand(TNC, "CODEC TRUE", TRUE);
}
return;
}
if (_memicmp(Buffer, "PTT T", 5) == 0)
{
TNC->Busy = TNC->BusyHold * 10; // BusyHold delay
TNC->PTTState = TRUE;
if (TNC->PTTMode)
Rig_PTT(TNC, TRUE);
return;
}
if (_memicmp(Buffer, "PTT F", 5) == 0)
{
TNC->PTTState = FALSE;
if (TNC->PTTMode)
Rig_PTT(TNC, FALSE);
return;
}
if (_memicmp(Buffer, "BUSY TRUE", 9) == 0)
{
TNC->BusyFlags |= CDBusy;
TNC->Busy = TNC->BusyHold * 10; // BusyHold delay
MySetWindowText(TNC->xIDC_CHANSTATE, "Busy");
strcpy(TNC->WEB_CHANSTATE, "Busy");
TNC->WinmorRestartCodecTimer = time(NULL);
return;
}
if (_memicmp(Buffer, "BUSY FALSE", 10) == 0)
{
TNC->BusyFlags &= ~CDBusy;
if (TNC->Busy)
strcpy(TNC->WEB_CHANSTATE, "BusyHold");
else
strcpy(TNC->WEB_CHANSTATE, "Clear");
MySetWindowText(TNC->xIDC_CHANSTATE, TNC->WEB_CHANSTATE);
TNC->WinmorRestartCodecTimer = time(NULL);
return;
}
if (_memicmp(Buffer, "TARGET", 6) == 0)
{
TNC->ConnectPending = 6; // This comes before Pending
Debugprintf(Buffer);
WritetoTrace(TNC, Buffer, MsgLen - 1);
memcpy(TNC->TargetCall, &Buffer[7], 10);
return;
}
if (_memicmp(Buffer, "OFFSET", 6) == 0)
{
// WritetoTrace(TNC, Buffer, MsgLen - 5);
return;
}
if (_memicmp(Buffer, "BUFFER", 6) == 0)
{
sscanf(&Buffer[7], "%d", &STREAM->BytesOutstanding);
if (STREAM->BytesOutstanding == 0)
{
// all sent
if (STREAM->Disconnecting) // Disconnect when all sent
{
if (STREAM->NeedDisc == 0)
STREAM->NeedDisc = 1;
}
}
else
{
// Make sure Node Keepalive doesn't kill session.
TRANSPORTENTRY * SESS = TNC->PortRecord->ATTACHEDSESSIONS[0];
if (SESS)
{
SESS->L4KILLTIMER = 0;
SESS = SESS->L4CROSSLINK;
if (SESS)
SESS->L4KILLTIMER = 0;
}
}
sprintf(TNC->WEB_TRAFFIC, "Sent %d RXed %d Queued %d",
STREAM->BytesTXed - STREAM->BytesOutstanding, STREAM->BytesRXed, STREAM->BytesOutstanding);
MySetWindowText(TNC->xIDC_TRAFFIC, TNC->WEB_TRAFFIC);
return;
}
if (_memicmp(Buffer, "CONNECTED ", 10) == 0)
{
char Call[11];
char * ptr;
APPLCALLS * APPL;
char * ApplPtr = APPLS;
int App;
char Appl[10];
struct WL2KInfo * WL2K = TNC->WL2K;
int Speed = 0;
Debugprintf(Buffer);
WritetoTrace(TNC, Buffer, MsgLen - 1);
STREAM->ConnectTime = time(NULL);
STREAM->BytesRXed = STREAM->BytesTXed = STREAM->PacketsSent = 0;
memcpy(Call, &Buffer[10], 10);
ptr = strchr(Call, ' ');
if (ptr) *ptr = 0;
// Get Speed
ptr = strchr(&Buffer[10], ' ');
if (ptr)
{
Speed = atoi(ptr);
if (Speed == 200)
TNC->WL2KMode = 40;
else if (Speed == 500)
TNC->WL2KMode = 41;
else if (Speed == 1000)
TNC->WL2KMode = 42;
else if (Speed == 2000)
TNC->WL2KMode = 43;
}
TNC->HadConnect = TRUE;
if (TNC->PortRecord->ATTACHEDSESSIONS[0] == 0)
{
TRANSPORTENTRY * SESS;
// Incoming Connect
TNC->SessionTimeLimit = TNC->DefaultSessionTimeLimit; // Reset Limit
// Stop other ports in same group
SuspendOtherPorts(TNC);
ProcessIncommingConnectEx(TNC, Call, 0, TRUE, TRUE);
SESS = TNC->PortRecord->ATTACHEDSESSIONS[0];
SESS->Mode = TNC->WL2KMode;
TNC->ConnectPending = FALSE;
if (TNC->RIG && TNC->RIG != &TNC->DummyRig && strcmp(TNC->RIG->RigName, "PTT"))
{
sprintf(TNC->WEB_TNCSTATE, "%s Connected to %s Inbound Freq %s", TNC->Streams[0].RemoteCall, TNC->TargetCall, TNC->RIG->Valchar);
SESS->Frequency = (int)(atof(TNC->RIG->Valchar) * 1000000.0) + 1500; // Convert to Centre Freq
}
else
{
sprintf(TNC->WEB_TNCSTATE, "%s Connected to %s Inbound", TNC->Streams[0].RemoteCall, TNC->TargetCall);
if (WL2K)
{
SESS->Frequency = WL2K->Freq;
}
}
if (WL2K)
strcpy(SESS->RMSCall, WL2K->RMSCall);
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
// Check for ExcludeList
if (ExcludeList[0])
{
if (CheckExcludeList(SESS->L4USER) == FALSE)
{
char Status[64];
TidyClose(TNC, 0);
sprintf(Status, "%d SCANSTART 15", TNC->Port);
Rig_Command(-1, Status);
Debugprintf("ARDOP Call from %s rejected", Call);
return;
}
}
// IF WE HAVE A PERMITTED CALLS LIST, SEE IF HE IS IN IT
if (TNC->PortRecord->PORTCONTROL.PERMITTEDCALLS)
{
UCHAR * ptr = TNC->PortRecord->PORTCONTROL.PERMITTEDCALLS;
while (TRUE)
{
if (memcmp(SESS->L4USER, ptr, 6) == 0) // Ignore SSID
break;
ptr += 7;
if ((*ptr) == 0) // Not in list
{
char Status[64];
TidyClose(TNC, 0);
sprintf(Status, "%d SCANSTART 15", TNC->Port);
Rig_Command(-1, Status);
Debugprintf("ARDOP Call from %s not in ValidCalls - rejected", Call);
return;
}
}
}
// See which application the connect is for
for (App = 0; App < 32; App++)
{
APPL=&APPLCALLTABLE[App];
memcpy(Appl, APPL->APPLCALL_TEXT, 10);
ptr=strchr(Appl, ' ');
if (ptr)
*ptr = 0;
if (_stricmp(TNC->TargetCall, Appl) == 0)
break;
}
if (App < 32)
{
char AppName[13];
memcpy(AppName, &ApplPtr[App * sizeof(CMDX)], 12);
AppName[12] = 0;
if (TNC->SendTandRtoRelay && memcmp(AppName, "RMS ", 4) == 0
&& (strstr(Call, "-T" ) || strstr(Call, "-R")))
strcpy(AppName, "RELAY ");
// Make sure app is available
if (CheckAppl(TNC, AppName))
{
MsgLen = sprintf(Buffer, "%s\r", AppName);
buffptr = GetBuff();
if (buffptr == 0)
{
return; // No buffers, so ignore
}
buffptr->Len = MsgLen;
memcpy(&buffptr->Data[0], Buffer, MsgLen);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
TNC->SwallowSignon = TRUE;
// Save Appl Call in case needed for
}
else
{
char Msg[] = "Application not available\r\n";
// Send a Message, then a disconenct
// Send CTEXT First
if (TNC->Streams[0].BPQtoPACTOR_Q) //Used for CTEXT
{
PMSGWITHLEN buffptr = (PMSGWITHLEN)Q_REM(&TNC->Streams[0].BPQtoPACTOR_Q);
UCHAR * data = &buffptr->Data[0];
int txlen = (int)(buffptr->Len);
SendARDOPorPacketData(TNC, 0, data, txlen);
}
SendARDOPorPacketData(TNC, 0, Msg, (int)strlen(Msg));
STREAM->NeedDisc = 100; // 10 secs
}
}
return;
}
else
{
// Connect Complete
char Reply[80];
int ReplyLen;
buffptr = GetBuff();
if (buffptr == 0)
{
return; // No buffers, so ignore
}
ReplyLen = sprintf(Reply, "*** Connected to %s\r", Call);
buffptr->Len = ReplyLen;
memcpy(&buffptr->Data[0], Reply, ReplyLen);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
TNC->Streams[0].Connecting = FALSE;
TNC->Streams[0].Connected = TRUE; // Subsequent data to data channel
if (TNC->RIG && TNC->RIG->Valchar[0])
sprintf(TNC->WEB_TNCSTATE, "%s Connected to %s Outbound Freq %s", TNC->Streams[0].MyCall, TNC->Streams[0].RemoteCall, TNC->RIG->Valchar);
else
sprintf(TNC->WEB_TNCSTATE, "%s Connected to %s Outbound", TNC->Streams[0].MyCall, TNC->Streams[0].RemoteCall);
MySetWindowText(TNC->xIDC_TNCSTATE, TNC->WEB_TNCSTATE);
UpdateMH(TNC, Call, '+', 'O');
return;
}
}
if (_memicmp(Buffer, "DISCONNECTED", 12) == 0
|| _memicmp(Buffer, "STATUS CONNECT TO", 17) == 0
|| _memicmp(Buffer, "STATUS ARQ TIMEOUT FROM PROTOCOL STATE", 24) == 0
// || _memicmp(Buffer, "NEWSTATE DISC", 13) == 0
|| _memicmp(Buffer, "ABORT", 5) == 0)
{
TNC->ConnectPending = FALSE; // Cancel Scan Lock
if (TNC->FECMode)
return;
if (TNC->StartSent)
{
TNC->StartSent = FALSE; // Disconnect reported following start codec
return;
}
if (TNC->Streams[0].Connecting)
{
// Report Connect Failed, and drop back to command mode
TNC->Streams[0].Connecting = FALSE;
buffptr = GetBuff();
if (buffptr == 0)
{
return; // No buffers, so ignore
}
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "*** Failure with %s\r", TNC->Streams[0].RemoteCall);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
if (TNC->RestartAfterFailure)
{
if (TNC->PID)
KillTNC(TNC);
RestartTNC(TNC);
}
return;
}
WritetoTrace(TNC, Buffer, MsgLen - 1);
// Release Session3
if (TNC->Streams[0].Connected)
{
// Create a traffic record
char logmsg[120];
time_t Duration;
Duration = time(NULL) - STREAM->ConnectTime;
if (Duration == 0)
Duration = 1;
sprintf(logmsg,"Port %2d %9s Bytes Sent %d BPS %d Bytes Received %d BPS %d Time %d Seconds",
TNC->Port, STREAM->RemoteCall,
STREAM->BytesTXed, (int)(STREAM->BytesTXed/Duration),
STREAM->BytesRXed, (int)(STREAM->BytesRXed/Duration), (int)Duration);
Debugprintf(logmsg);
}
TNC->Streams[0].Connecting = FALSE;
TNC->Streams[0].Connected = FALSE; // Back to Command Mode
TNC->Streams[0].ReportDISC = TRUE; // Tell Node
if (TNC->Streams[0].Disconnecting) //
ARDOPReleaseTNC(TNC);
TNC->Streams[0].Disconnecting = FALSE;
return;
}
if (_memicmp(Buffer, "MODE", 4) == 0)
{
strcpy(TNC->WEB_MODE, &Buffer[5]);
MySetWindowText(TNC->xIDC_MODE, &Buffer[5]);
return;
}
if (_memicmp(Buffer, "STATUS ", 7) == 0)
{
return;
}
if (_memicmp(Buffer, "RADIOMODELS", 11) == 0)
return;
if (_memicmp(&Buffer[0], "PENDING", 7) == 0) // Save Pending state for scan control
{
TNC->ConnectPending = 6; // Time out after 6 Scanintervals
return;
}
// REJECTEDBW and REJECTEDBUSY are sent to both calling and called
if (_memicmp(&Buffer[0], "REJECTEDBUSY", 12) == 0)
{
TNC->ConnectPending = FALSE;
if (TNC->Streams[0].Connecting)
{
// Report Connect Failed, and drop back to command mode
TNC->Streams[0].Connecting = FALSE;
buffptr = GetBuff();
if (buffptr == 0)
{
return; // No buffers, so ignore
}
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} Connection to %s Rejected - Channel Busy\r", TNC->Streams[0].RemoteCall);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
return;
}
}
if (_memicmp(&Buffer[0], "REJECTEDBW", 10) == 0)
{
TNC->ConnectPending = FALSE;
if (TNC->Streams[0].Connecting)
{
// Report Connect Failed, and drop back to command mode
TNC->Streams[0].Connecting = FALSE;
buffptr = GetBuff();
if (buffptr == 0)
{
return; // No buffers, so ignore
}
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} Connection to %s Rejected - Incompatible Bandwidth\r", TNC->Streams[0].RemoteCall);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
return;
}
}
if (_memicmp(&Buffer[0], "CANCELPENDING", 13) == 0
|| _memicmp(&Buffer[0], "REJECTEDB", 9) == 0) //REJECTEDBUSY or REJECTEDBW
{
TNC->ConnectPending = FALSE;
return;
}
if (_memicmp(Buffer, "FAULT", 5) == 0)
{
Debugprintf(Buffer);
WritetoTrace(TNC, Buffer, MsgLen - 1);
// return;
}
if (_memicmp(Buffer, "NEWSTATE", 8) == 0)
{
TNC->WinmorRestartCodecTimer = time(NULL);
MySetWindowText(TNC->xIDC_PROTOSTATE, &Buffer[9]);
strcpy(TNC->WEB_PROTOSTATE, &Buffer[9]);
if (_memicmp(&Buffer[9], "DISC", 4) == 0)
{
TNC->DiscPending = FALSE;
TNC->ConnectPending = FALSE;
return;
}
if (strcmp(&Buffer[9], "ISS") == 0) // Save Pending state for scan control
TNC->TXRXState = 'S';
else if (strcmp(&Buffer[9], "IRS") == 0)
TNC->TXRXState = 'R';
return;
}
if ((_memicmp(Buffer, "FAULT Not from state FEC", 24) == 0) || (_memicmp(Buffer, "FAULT Blocked by Busy Lock", 24) == 0))
{
if (TNC->FECMode)
{
Sleep(1000);
// if (TNC->FEC1600)
// ARDOPSendCommand(TNC,"FECSEND 1600");
// else
// ARDOPSendCommand(TNC,"FECSEND 500");
return;
}
}
if (_memicmp(Buffer, "PLAYBACKDEVICES", 15) == 0)
{
TNC->PlaybackDevices = _strdup(&Buffer[16]);
}
// Others should be responses to commands
if (_memicmp(Buffer, "BLOCKED", 6) == 0)
{
WritetoTrace(TNC, Buffer, MsgLen - 1);
return;
}
if (_memicmp(Buffer, "OVER", 4) == 0)
{
WritetoTrace(TNC, Buffer, MsgLen - 1);
return;
}
if (_memicmp(Buffer, "PING ", 5) == 0)
{
char Call[32];
// Make sure not Echoed PING
// c:ping gm8bpq-1 5
// c:PING GM8BPQ>GM8BPQ-1 15 98
if (strchr(Buffer, '>') == 0) // Echoed
return;
WritetoTrace(TNC, Buffer, MsgLen - 1);
// Release scanlock after another interval (to allow time for response to be sent)
// ?? use cancelpending TNC->ConnectPending = 1;
memcpy(Call, &Buffer[5], 20);
strlop(Call, '>');
UpdateMH(TNC, Call, '!', 'I');
return;
}
if (_memicmp(Buffer, "VERSION ", 8) == 0)
{
// If contains "OFDM" or "ARDOP3" increase data session busy level
if (strstr(&Buffer[8], "OFDM"))
TNC->Mode = 'O';
else if (strstr(&Buffer[8], "TNC_3"))
TNC->Mode = '3';
else
TNC->Mode = 0;
}
if (_memicmp(Buffer, "PINGACK ", 8) == 0)
{
WritetoTrace(TNC, Buffer, MsgLen - 1);
// Drop through to return to user
}
if (_memicmp(Buffer, "CQ ", 3) == 0 && MsgLen > 10)
{
char Call[32];
char * Loc;
WritetoTrace(TNC, Buffer, MsgLen - 1);
// Update MH
{
memcpy(Call, &Buffer[3], 32);
Loc = strlop(Call, ' ');
strlop(Loc, ']');
UpdateMHEx(TNC, Call, '!', 'I', &Loc[1], TRUE);
}
// Drop through to go to user if attached but not connected
}
// Return others to user (if attached but not connected)
if (TNC->Streams[0].Attached == 0)
return;
if (TNC->Streams[0].Connected || TNC->Streams[0].Connecting)
return;
if (MsgLen > 200)
MsgLen = 200;
buffptr = GetBuff();
if (buffptr == 0)
{
return; // No buffers, so ignore
}
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} %s\r", Buffer);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
}
static VOID ARDOPProcessReceivedData(struct TNCINFO * TNC)
{
int InputLen, MsgLen;
// May get several messages per packet
// May get message split over packets
// Data has a length field
// ARQ|FEC|ERR|, 2 byte count (Hex 0001 <20> FFFF), binary data<74>
// New standard doesnt have d:
if (TNC->DataInputLen > 16000) // Shouldnt have packets longer than this
TNC->DataInputLen=0;
// OFDM can return large packets (up to 10160)
// in serial mode, data has been put in input buffer by comms code
if (TNC->ARDOPCommsMode == 'T')
{
InputLen=recv(TNC->TCPDataSock, &TNC->ARDOPDataBuffer[TNC->DataInputLen], 16384 - TNC->DataInputLen, 0);
if (InputLen == 0 || InputLen == SOCKET_ERROR)
{
TNCLost(TNC);
return;
}
TNC->DataInputLen += InputLen;
}
loop:
if (TNC->OldMode)
goto OldRX;
else
{ // No D:
// Data = check we have it all
int DataLen = (TNC->ARDOPDataBuffer[0] << 8) + TNC->ARDOPDataBuffer[1]; // HI First
UCHAR DataType[4];
UCHAR * Data;
if (TNC->DataInputLen < DataLen + 2)
return; // Wait for more
MsgLen = DataLen + 2; // Len
memcpy(DataType, &TNC->ARDOPDataBuffer[2] , 3);
DataType[3] = 0;
Data = &TNC->ARDOPDataBuffer[5];
DataLen -= 3;
ARDOPProcessDataPacket(TNC, DataType, Data, DataLen);
// See if anything else in buffer
TNC->DataInputLen -= MsgLen;
if (TNC->DataInputLen == 0)
return;
memmove(TNC->ARDOPDataBuffer, &TNC->ARDOPDataBuffer[MsgLen], TNC->DataInputLen);
goto loop;
}
OldRX:
if (TNC->DataInputLen < 8)
return; // Wait for more to arrive (?? timeout??)
if (TNC->ARDOPDataBuffer[1] = ':') // At least message looks reasonable
{
if (TNC->ARDOPDataBuffer[0] == 'd')
{
// Data = check we have it all
int DataLen = (TNC->ARDOPDataBuffer[2] << 8) + TNC->ARDOPDataBuffer[3]; // HI First
// unsigned short CRC;
UCHAR DataType[4];
UCHAR * Data;
if (TNC->DataInputLen < DataLen + 4)
return; // Wait for more
MsgLen = DataLen + 4; // d: Len CRC
memcpy(DataType, &TNC->ARDOPDataBuffer[4] , 3);
DataType[3] = 0;
Data = &TNC->ARDOPDataBuffer[7];
DataLen -= 3;
ARDOPProcessDataPacket(TNC, DataType, Data, DataLen);
// See if anything else in buffer
TNC->DataInputLen -= MsgLen;
if (TNC->DataInputLen == 0)
return;
memmove(TNC->ARDOPDataBuffer, &TNC->ARDOPDataBuffer[MsgLen], TNC->DataInputLen);
goto loop;
}
else
// Duff - clear input buffer
TNC->DataInputLen = 0;
}
return;
}
static VOID ARDOPProcessReceivedControl(struct TNCINFO * TNC)
{
int InputLen, MsgLen;
char * ptr, * ptr2;
char Buffer[4096];
// May get several messages per packet
// May get message split over packets
// Commands end with CR.
if (TNC->InputLen > 8000) // Shouldnt have packets longer than this
TNC->InputLen=0;
// in serial mode, data has been put in input buffer by comms code
if (TNC->ARDOPCommsMode == 'T')
{
// I don't think it likely we will get packets this long, but be aware...
InputLen=recv(TNC->TCPSock, &TNC->ARDOPBuffer[TNC->InputLen], 8192 - TNC->InputLen, 0);
if (InputLen == 0 || InputLen == SOCKET_ERROR)
{
TNCLost(TNC);
return;
}
TNC->InputLen += InputLen;
}
loop:
ptr = memchr(TNC->ARDOPBuffer, '\r', TNC->InputLen);
if (ptr == 0) // CR in buffer
return; // Wait for it
ptr2 = &TNC->ARDOPBuffer[TNC->InputLen];
if ((ptr2 - ptr) == 1) // CR (no CRC in new version)
{
// Usual Case - single meg in buffer
ARDOPProcessResponse(TNC, TNC->ARDOPBuffer, TNC->InputLen);
TNC->InputLen=0;
return;
}
else
{
// buffer contains more that 1 message
MsgLen = TNC->InputLen - (int)(ptr2-ptr) + 1; // Include CR
memcpy(Buffer, TNC->ARDOPBuffer, MsgLen);
ARDOPProcessResponse(TNC, Buffer, MsgLen);
if (TNC->InputLen < MsgLen)
{
TNC->InputLen = 0;
return;
}
memmove(TNC->ARDOPBuffer, ptr + 1, TNC->InputLen-MsgLen);
TNC->InputLen -= MsgLen;
goto loop;
}
return;
}
VOID ARDOPProcessDataPacket(struct TNCINFO * TNC, UCHAR * Type, UCHAR * Data, int Length)
{
// Info on Data Socket - just packetize and send on
struct STREAMINFO * STREAM = &TNC->Streams[0];
int PacLen = 236;
PMSGWITHLEN buffptr;
TNC->TimeSinceLast = 0;
if (strcmp(Type, "IDF") == 0)
{
// Place ID frames in Monitor Window and MH
char Call[20];
char * Loc;
// GM8BPQ-2:[IO68VL]
//ID:GM8BPQ-2 IO68VL :
// GM8BPQ-2:[IO68VL]
//ID:GM8BPQ-2 [IO68vl]:
//ID:HB9AVK JN47HG :
// TX BPQ IDF GM8BPQ-2:[IO68VL]
// RX Rick IDF ID:GM8BPQ-2 [IO68vl]:
// TX Rick IDF GM8BPQ-2:[IO68VL]
// RX BPQ IDF ID:GM8BPQ-2 IO68VL :
//ID:GM8BPQ-2 [IO68vl] :
Data[Length] = 0;
WritetoTrace(TNC, Data, Length);
Debugprintf("ARDOP IDF %s", Data);
// Loos like transmitted ID doesnt have ID:
if (memcmp(Data, "ID:", 3) == 0) // These seem to be received ID's
{
memcpy(Call, &Data[3], 20);
Loc = strlop(Call, ' ');
strlop(Loc, ']');
UpdateMHEx(TNC, Call, '!', 'I', &Loc[1], TRUE);
}
return;
}
STREAM->BytesRXed += Length;
Data[Length] = 0;
Debugprintf("ARDOP: RXD %d bytes", Length);
sprintf(TNC->WEB_TRAFFIC, "Sent %d RXed %d Queued %d",
STREAM->BytesTXed - STREAM->BytesOutstanding, STREAM->BytesRXed, STREAM->BytesOutstanding);
MySetWindowText(TNC->xIDC_TRAFFIC, TNC->WEB_TRAFFIC);
if (TNC->FECMode)
{
Length = (int)strlen(Data);
if (Data[Length - 1] == 10)
Data[Length - 1] = 13;
}
if (strcmp(Type, "FEC") == 0)
{
// May be an APRS Message
// These are delimired with ^ characters
// As they are likely to be split across
// FEC blocks they need to be recombined
char * ptr = Data;
char * ptr1;
char * ptr2;
char c;
int Len = Length;
Debugprintf(Data);
if (*ptr == '^' || TNC->ARDOPAPRSLen)
{
// New Packet or continuation
while (Len--)
{
c = *(ptr++);
if (c == '^')
{
// may be start or end
Debugprintf("Start/end of beacon Len = %d", TNC->ARDOPAPRSLen);
if (TNC->ARDOPAPRSLen == 0)
continue; // Start
// Validate and Process Block
Debugprintf("beacon %s", TNC->ARDOPAPRS);
ptr1 = TNC->ARDOPAPRS;
ptr2 = strchr(ptr1, '>');
if (ptr2 && (ptr2 - ptr1) < 10)
{
// Could be APRS
// if ((memcmp(ptr2 + 1, "AP", 2) == 0) || (memcmp(ptr2 + 1, "BE", 2) == 0))
if (1) // People using other dests
{
int APLen;
// assume it is
char * ptr3 = strchr(ptr2, '|');
struct _MESSAGE * buffptr;
if (ptr3 == 0)
{
TNC->ARDOPAPRSLen = 0;
Debugprintf("no |");
continue;
}
buffptr = GetBuff();
*(ptr3++) = 0; // Terminate TO call
APLen = TNC->ARDOPAPRSLen - (int)(ptr3 - ptr1);
TNC->ARDOPAPRSLen = 0;
Debugprintf("Good APRS %d Left", Len);
// Convert to ax.25 format
if (buffptr == 0)
continue; // No buffers, so ignore
buffptr->PORT = TNC->Port;
ConvToAX25(ptr1, buffptr->ORIGIN);
ConvToAX25(ptr2 + 1, buffptr->DEST);
buffptr->ORIGIN[6] |= 1; // Set end of address
buffptr->CTL = 3;
buffptr->PID = 0xF0;
memcpy(buffptr->L2DATA, ptr3, APLen);
buffptr->LENGTH = 16 + MSGHDDRLEN + APLen;
time(&buffptr->Timestamp);
BPQTRACE((MESSAGE *)buffptr, TRUE);
}
else
{
Debugprintf("Not APRS");
TNC->ARDOPAPRSLen = 0; // in case not aprs
}
}
else
{
Debugprintf("cant be APRS");
TNC->ARDOPAPRSLen = 0; // in case not aprs
}
continue;
}
// Normal Char
TNC->ARDOPAPRS[TNC->ARDOPAPRSLen++] = c;
if (TNC->ARDOPAPRSLen == 512)
TNC->ARDOPAPRSLen = 0;
}
// End of packet.
Debugprintf("End of Packet Len %d", TNC->ARDOPAPRSLen);
}
// FEC but not APRS. Discard if connected
if (TNC->Streams[0].Connected)
return;
}
WritetoTrace(TNC, Data, Length);
// We can get messages of form ARQ [ConReq2000M: GM8BPQ-2 > OE3FQU]
// Noe (V2) [ConReq2500 > G8XXX]
// when not connected.
if (TNC->Streams[0].Connected == FALSE)
{
if (strcmp(Type, "ARQ") == 0)
{
if (Data[1] == '[')
{
// Log to MH
char Call[20];
char * ptr;
// Add a Newline for monitoring
Data[Length++] = 13;
Data[Length] = 0;
ptr = strchr(Data, ':');
if (ptr)
{
memcpy(Call, &ptr[2], 20);
strlop(Call, ' ');
UpdateMH(TNC, Call, '!', 'I');
}
}
}
}
if (TNC->Streams[0].Attached == 0)
return;
// May need to fragment
while (Length)
{
int Fraglen = Length;
if (Length > PACLEN)
Fraglen = PACLEN;
Length -= Fraglen;
buffptr = GetBuff();
if (buffptr == 0)
return; // No buffers, so ignore
memcpy(&buffptr->Data[0], Data, Fraglen);
Data += Fraglen;
buffptr->Len = Fraglen;
C_Q_ADD(&TNC->Streams[0].PACTORtoBPQ_Q, buffptr);
}
return;
}
/*
INT_PTR CALLBACK ConfigDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
int Cmd = LOWORD(wParam);
switch (message)
{
case WM_INITDIALOG:
{
struct TNCINFO * TNC = (struct TNCINFO * )lParam;
char * ptr1, *ptr2;
int ptr3 = 0;
char Line[1000];
int len;
ptr1 = TNC->CaptureDevices;
if (!ptr1)
return 0; // No Devices
while (ptr2 = strchr(ptr1, ','))
{
len = ptr2 - ptr1;
memcpy(&Line[ptr3], ptr1, len);
ptr3 += len;
Line[ptr3++] = '\r';
Line[ptr3++] = '\n';
ptr1 = ++ptr2;
}
Line[ptr3] = 0;
strcat(Line, ptr1);
SetDlgItemText(hDlg, IDC_CAPTURE, Line);
ptr3 = 0;
ptr1 = TNC->PlaybackDevices;
if (!ptr1)
return 0; // No Devices
while (ptr2 = strchr(ptr1, ','))
{
len = ptr2 - ptr1;
memcpy(&Line[ptr3], ptr1, len);
ptr3 += len;
Line[ptr3++] = '\r';
Line[ptr3++] = '\n';
ptr1 = ++ptr2;
}
Line[ptr3] = 0;
strcat(Line, ptr1);
SetDlgItemText(hDlg, IDC_PLAYBACK, Line);
SendDlgItemMessage(hDlg, IDC_PLAYBACK, EM_SETSEL, -1, 0);
// KillTNC(TNC);
return TRUE;
}
case WM_SIZING:
{
return TRUE;
}
case WM_ACTIVATE:
// SendDlgItemMessage(hDlg, IDC_MESSAGE, EM_SETSEL, -1, 0);
break;
case WM_COMMAND:
if (Cmd == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
return (INT_PTR)TRUE;
break;
}
return (INT_PTR)FALSE;
}
*/
VOID TidyClose(struct TNCINFO * TNC, int Stream)
{
// If all acked, send disc
if (TNC->Streams[Stream].BytesOutstanding == 0)
{
if (Stream == 0)
ARDOPSendCommand(TNC, "DISCONNECT", TRUE);
else
{
char Cmd[32];
sprintf(Cmd, "%cDISCONNECT", 1);
ARDOPSendPktCommand(TNC, Stream, Cmd);
}
}
}
VOID ForcedClose(struct TNCINFO * TNC, int Stream)
{
if (Stream == 0)
ARDOPSendCommand(TNC, "ABORT", TRUE);
else
{
char Cmd[32];
sprintf(Cmd, "%cDISCONNECT", Stream);
ARDOPSendPktCommand(TNC, Stream, Cmd);
}
}
VOID CloseComplete(struct TNCINFO * TNC, int Stream)
{
if (Stream == 0)
{
ARDOPReleaseTNC(TNC);
if (TNC->FECMode)
{
TNC->FECMode = FALSE;
ARDOPSendCommand(TNC, "SENDID", TRUE);
}
}
}
VOID ARDOPAbort(struct TNCINFO * TNC)
{
ARDOPSendCommand(TNC, "ABORT", TRUE);
}
// Host Mode Stuff (we reuse some routines in SCSPactor)
VOID ARDOPCRCStuffAndSend(struct TNCINFO * TNC, UCHAR * Msg, int Len)
{
unsigned short int crc;
UCHAR StuffedMsg[500];
int i, j;
Msg[3] |= TNC->Toggle;
// Debugprintf("ARDOP TX Toggle %x", TNC->Toggle);
crc = compute_crc(&Msg[2], Len-2);
crc ^= 0xffff;
Msg[Len++] = (crc&0xff);
Msg[Len++] = (crc>>8);
for (i = j = 2; i < Len; i++)
{
StuffedMsg[j++] = Msg[i];
if (Msg[i] == 170)
{
StuffedMsg[j++] = 0;
}
}
if (j != i)
{
Len = j;
memcpy(Msg, StuffedMsg, j);
}
TNC->TXLen = Len;
Msg[0] = 170;
Msg[1] = 170;
ARDOPWriteCommBlock(TNC);
TNC->Retries = 5;
}
VOID ARDOPExitHost(struct TNCINFO * TNC)
{
UCHAR * Poll = TNC->TXBuffer;
// Try to exit Host Mode
TNC->TXBuffer[2] = 31;
TNC->TXBuffer[3] = 0x41;
TNC->TXBuffer[4] = 0x5;
memcpy(&TNC->TXBuffer[5], "JHOST0", 6);
ARDOPCRCStuffAndSend(TNC, Poll, 11);
return;
}
VOID ARDOPDoTermModeTimeout(struct TNCINFO * TNC)
{
UCHAR * Poll = TNC->TXBuffer;
if (TNC->ReinitState == 0)
{
//Checking if in Terminal Mode - Try to set back to Term Mode
TNC->ReinitState = 1;
ARDOPExitHost(TNC);
TNC->Retries = 1;
return;
}
if (TNC->ReinitState == 1)
{
// Forcing back to Term Mode
TNC->ReinitState = 0;
ARDOPDoTNCReinit(TNC); // See if worked
return;
}
if (TNC->ReinitState == 3)
{
// Entering Host Mode
// Assume ok
TNC->HostMode = TRUE;
TNC->IntCmdDelay = 10;
return;
}
}
VOID ARDOPDoTNCReinit(struct TNCINFO * TNC)
{
UCHAR * Poll = TNC->TXBuffer;
if (TNC->ReinitState == 0)
{
// Just Starting - Send a TNC Mode Command to see if in Terminal or Host Mode
Poll[0] = 13;
Poll[1] = 0x1B;
TNC->TXLen = 2;
// Debugprintf("Sending CR ESC, Mode %c", TNC->ARDOPCommsMode);
if (TNC->ARDOPCommsMode == 'E')
{
if (TNC->TCPCONNECTED)
{
int SentLen = send(TNC->TCPSock, TNC->TXBuffer, TNC->TXLen, 0);
if (SentLen != TNC->TXLen)
{
// ARDOP doesn't seem to recover from a blocked write. For now just reset
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Write Failed for port %d - error code = %d\r\n", TNC->Port, winerr);
WritetoConsole(ErrMsg);
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
TNC->TCPCONNECTED = FALSE;
}
}
TNC->TNCOK = FALSE;
sprintf(TNC->WEB_COMMSSTATE,"%s Initialising TNC", TNC->ARDOPSerialPort);
SetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->Timeout = 20; // 2 secs
TNC->Retries = 1;
return;
}
if (TNC->hDevice == 0) // Dont try to init if device not open
{
if (TNC->PortRecord->PORTCONTROL.PortStopped == 0)
OpenCOMMPort(TNC, TNC->ARDOPSerialPort, TNC->ARDOPSerialSpeed, TRUE);
TNC->Timeout = 100; // 10 secs
TNC->Retries = 1;
return;
}
TNC->TNCOK = FALSE;
sprintf(TNC->WEB_COMMSSTATE,"%s Initialising TNC", TNC->ARDOPSerialPort);
SetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
if (ARDOPWriteCommBlock(TNC) == FALSE)
{
if (TNC->hDevice)
{
Debugprintf("ARDOPWriteCommBlock Failed Mode %c", TNC->ARDOPCommsMode);
CloseCOMPort(TNC->hDevice);
}
if (TNC->ARDOPCommsMode == 'S')
{
OpenCOMMPort(TNC, TNC->ARDOPSerialPort, TNC->ARDOPSerialSpeed, TRUE);
}
else
{
#ifdef WIN32
#else
#ifdef NOI2C
#else
char i2cname[30];
int fd;
int retval;
// Open and configure the i2c interface
if (strlen(TNC->ARDOPSerialPort) < 3)
sprintf(i2cname, "/dev/i2c-%s", TNC->ARDOPSerialPort);
else
strcpy(i2cname, TNC->ARDOPSerialPort);
fd = TNC->hDevice = open(i2cname, O_RDWR);
if (fd < 0)
printf("Cannot find i2c bus %s\n", i2cname);
else
{
retval = ioctl(fd, I2C_SLAVE, TNC->ARDOPSerialSpeed);
if(retval == -1)
printf("Cannot open i2c device %x\n", TNC->ARDOPSerialSpeed);
ioctl(fd, I2C_TIMEOUT, 10);
}
#endif
#endif
}
}
TNC->Retries = 1;
}
if (TNC->ReinitState == 1) // Forcing back to Term
TNC->ReinitState = 0;
if (TNC->ReinitState == 2) // In Term State, Sending Initialisation Commands
{
Debugprintf("DOTNCReinit %d Complete - Entering Hostmode", TNC->Port);
TNC->TXBuffer[2] = 0;
TNC->Toggle = 0;
memcpy(Poll, "JHOST4\r", 7);
TNC->TXLen = 7;
ARDOPWriteCommBlock(TNC);
// Timeout will enter host mode
TNC->Timeout = 1;
TNC->Retries = 1;
TNC->Toggle = 0;
TNC->ReinitState = 3; // Set toggle force bit
TNC->OKToChangeFreq = 1; // In case failed whilst waiting for permission
TNC->CONNECTED = TRUE;
return;
}
}
VOID ARDOPProcessTermModeResponse(struct TNCINFO * TNC)
{
UCHAR * Poll = TNC->TXBuffer;
char * ptr1, * ptr2;
int len;
if (TNC->ReinitState == 0)
{
// Testing if in Term Mode. It is, so can now send Init Commands
TNC->InitPtr = TNC->InitScript;
TNC->ReinitState = 2;
// Send ARDOP to make sure TNC is in a known state
strcpy(Poll, "ARDOP\r");
// OpenLogFile(TNC->Port);
// WriteLogLine(TNC->Port, Poll, 7);
// CloseLogFile(TNC->Port);
TNC->TXLen = 6;
ARDOPWriteCommBlock(TNC);
TNC->Timeout = 60; // 6 secs
return;
}
if (TNC->ReinitState == 2)
{
// Sending Init Commands
// Send INIT script
ptr1 = &TNC->InitScript[0];
/* GetSemaphore(&Semaphore, 52);
while(TNC->BPQtoWINMOR_Q)
{
void ** buffptr = Q_REM(&TNC->BPQtoWINMOR_Q);
if (buffptr)
ReleaseBuffer(buffptr);
}
FreeSemaphore(&Semaphore, 52);
*/
while (ptr1 && ptr1[0])
{
ptr2 = strchr(ptr1, 13);
if (ptr2 == 0)
break;
len = (int)(ptr2 - ptr1) + 1;
memcpy(Poll, ptr1, len);
// if Date or Time command add current time
if (_memicmp(ptr1, "DATETIME", 4) == 0)
{
time_t T;
struct tm * tm;
T = time(NULL);
tm = gmtime(&T);
len = sprintf(Poll, "DATETIME %02d %02d %02d %02d %02d %02d\r",
tm->tm_mday, tm->tm_mon + 1, tm->tm_year - 100,
tm->tm_hour, tm->tm_min, tm->tm_sec);
}
// if RADIOPTTON ? or RADIOPTTOFF ? replace ?
// with correct string
if (_memicmp(ptr1, "RADIOPTTOFF ?", 13) == 0)
{
int Len = TNC->RIG->PTTOffLen;
UCHAR * Cmd = TNC->RIG->PTTOff;
char Hex[256];
char * hexptr = Hex;
int i, j;
while(Len--)
{
i = *(Cmd++);
j = i >>4;
j += '0'; // ascii
if (j > '9')
j += 7;
*(hexptr++) = j;
j = i & 0xf;
j += '0'; // ascii
if (j > '9')
j += 7;
*(hexptr++) = j;
}
*(hexptr++) = 0;
len = sprintf(Poll, "RADIOPTTOFF %s\r", Hex);
}
if (_memicmp(ptr1, "RADIOPTTON ?", 12) == 0)
{
int Len = TNC->RIG->PTTOnLen;
UCHAR * Cmd = TNC->RIG->PTTOn;
char Hex[256];
char * hexptr = Hex;
int i, j;
while(Len--)
{
i = *(Cmd++);
j = i >>4;
j += '0'; // ascii
if (j > '9')
j += 7;
*(hexptr++) = j;
j = i & 0xf;
j += '0'; // ascii
if (j > '9')
j += 7;
*(hexptr++) = j;
}
*(hexptr++) = 0;
len = sprintf(Poll, "RADIOPTTON %s\r", Hex);
}
TNC->TXLen = len;
ARDOPWriteCommBlock(TNC);
Sleep(50);
TNC->Timeout = 60; // 6 secs
if (ptr2)
*(ptr2++) = 13; // Put CR back for next time
ptr1 = ptr2;
}
// All Sent - enter Host Mode
ARDOPDoTNCReinit(TNC); // Send Next Command
return;
}
}
int ARDOPProcessDEDFrame(struct TNCINFO * TNC, UCHAR * Msg, int framelen)
{
PMSGWITHLEN buffptr;
UCHAR * Buffer; // Data portion of frame
unsigned int Stream = 0, RealStream;
if (Msg[0] == 255 && Msg[1] == 255)
{
goto tcpHostFrame;
}
if (TNC->HostMode == 0)
return framelen;
// Check toggle
// Debugprintf("ARDOP RX Toggle = %x MSG[3] = %x", TNC->Toggle, Msg[3]);
if (TNC->Toggle != (Msg[3] & 0x80))
{
Debugprintf("ARDOP PTC Seq Error");
return framelen; // should check if retrying
}
// Any valid frame is an ACK
TNC->Toggle ^= 0x80; // update toggle
TNC->Timeout = 0;
Msg[3] &= 0x7f; // remove toggle
if (TNC->TNCOK == FALSE)
{
// Just come up
TNC->TNCOK = TRUE;
sprintf(TNC->WEB_COMMSSTATE,"%s TNC link OK", TNC->ARDOPSerialPort);
SetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
}
tcpHostFrame:
Stream = RealStream = Msg[2];
// See if Poll Reply or Data
if (Msg[3] == 1 && Stream > 0 && Stream <= APMaxStreams)
{
// Ardop Packet Data. Probably Buffer Status
int Len = (int)strlen(&Msg[4]);
if (memcmp(&Msg[4], "Queued ", 7) == 0)
{
int Count = atoi(&Msg[11]);
TNC->Streams[Stream].BytesOutstanding = Count;
}
return Len + 5;
}
if (Stream == 32) // Native Mode Command Response
{
if (Msg[3] == 1) // Null terminated response
{
int Len = (int)strlen(&Msg[4]) + 1;
ARDOPProcessResponse(TNC, &Msg[4], Len);
return Len + 5;
}
if (Msg[3] == 0) // Success, no response
return 5;
if (Msg[3] == 7) // Status Reports
{
int Len = Msg[4] + 1;
// may have more than one message in buffer,
// so pass to unpack
memcpy(&TNC->ARDOPBuffer[TNC->InputLen],&Msg[5], Len);
TNC->InputLen += Len;
ARDOPProcessReceivedControl(TNC);
return Len + 5;
}
return 0;
}
if (Stream == 33) // Native Mode Data
{
// May be connected, FEC or ID
if (Msg[3] == 1) // Null terminated response
return 0;
if (Msg[3] == 0) // Success, no response
return 0;
if (Msg[3] == 7) // Data
{
int Len = Msg[4] + 1;
// may have more than one message in buffer,
// so pass to unpack
memcpy(&TNC->ARDOPDataBuffer[TNC->DataInputLen],&Msg[5], Len);
TNC->DataInputLen += Len;
ARDOPProcessReceivedData(TNC);
return 0;
}
return 0;
}
if (Stream == 34) // Native Mode Log
{
int Len = Msg[4] + 1;
char timebuf[32];
char Line[256];
char * ptr, * ptr2;
#ifdef WIN32
SYSTEMTIME st;
#else
struct timespec tp;
int hh;
int mm;
int ss;
#endif
if (TNC->LogHandle == 0 && TNC->DebugHandle == 0)
return 0;
#ifdef WIN32
GetSystemTime(&st);
sprintf(timebuf, "%02d:%02d:%02d.%03d ",
st.wHour, st.wMinute, st.wSecond, st.wMilliseconds);
#else
clock_gettime(CLOCK_REALTIME, &tp);
ss = tp.tv_sec % 86400; // Secs int day
hh = ss / 3600;
mm = (ss - (hh * 3600)) / 60;
ss = ss % 60;
sprintf(timebuf, "%02d:%02d:%02d.%03d ",
hh, mm, ss, (int)tp.tv_nsec/1000000);
#endif
// Messages may be blocked with top bit of first byte set
ptr = &Msg[5];
ptr2 = Line;
while(Len--)
{
int c = (*ptr++);
if (c & 0x80)
{
// New Message
c &= 0x7f;
*ptr2 = 0;
fputs(Line, TNC->DebugHandle); // rest of last line
if (TNC->LastLogType < '7')
fputs(Line, TNC->LogHandle);
TNC->LastLogType = c;
// Timestamp new message and add type
fputs(timebuf, TNC->DebugHandle);
fputc(c, TNC->DebugHandle);
fputc(' ', TNC->DebugHandle);
if (TNC->LastLogType < '7')
{
fputs(timebuf, TNC->LogHandle);
fputc(c, TNC->LogHandle);
fputc(' ', TNC->LogHandle);
}
ptr2 = Line;
}
else
*(ptr2++) = c;
}
*ptr2 = 0;
fputs(Line, TNC->DebugHandle); // rest of last line
fflush(TNC->DebugHandle);
if (TNC->LastLogType < '7')
{
fputs(Line, TNC->LogHandle);
fflush(TNC->LogHandle);
}
return 0;
}
if (Msg[3] == 4 || Msg[3] == 5)
{
MESSAGE Monframe;
// Packet Monitor Data.
// DED Host uses 4 and 5 as Null Terminated ascii encoded header
// and 6 byte count format info.
// In ARDOP Native mode I pass both header and data
// in byte count raw format, as there is no point
// in ascii coding then converting back to pass to
// monitor code
// The First byte is TX/RX Flag
int Len = Msg[4]; // Would be +1 but first is Flag
memset(&Monframe, 0, sizeof(Monframe));
memcpy(Monframe.DEST, &Msg[6], Len);
Monframe.LENGTH = Len + MSGHDDRLEN;
Monframe.PORT = TNC->Port | Msg[5]; // or in TX Flag
time(&Monframe.Timestamp);
if (Msg[3] == 5) // More to come
{
// Save the header till the data arrives
if (TNC->Monframe)
free(TNC->Monframe);
TNC->Monframe = malloc(sizeof(MESSAGE));
if (TNC->Monframe)
memcpy(TNC->Monframe, &Monframe, sizeof(MESSAGE));
return Len + 6;
}
BPQTRACE((MESSAGE *)&Monframe, TRUE);
return Len + 6;
}
if (Msg[3] == 6)
{
// Second part of I or UI
int Len = Msg[4] + 1;
MESSAGE Monframe;
UCHAR * ptr = (UCHAR *)&Monframe;
memset(&Monframe, 0, sizeof(Monframe));
if (TNC->Monframe)
{
memcpy(&Monframe, TNC->Monframe, TNC->Monframe->LENGTH);
memcpy(&ptr[TNC->Monframe->LENGTH], &Msg[5], Len);
Monframe.LENGTH += Len;
time(&Monframe.Timestamp);
BPQTRACE((MESSAGE *)&Monframe, TRUE);
free(TNC->Monframe);
TNC->Monframe = NULL;
}
return Len + 6;
}
if (Msg[3] == 0)
{
// Success - Nothing Follows
if (Stream < 32)
if (TNC->Streams[Stream].CmdSet)
return 4; // Response to Command Set
if ((TNC->TXBuffer[3] & 1) == 0) // Data
return 4;
if (Stream > 0 && Stream <= APMaxStreams)
{
// Packet Response
return 4;
}
// If the response to a Command, then we should convert to a text OK" for forward scripts, etc
if (TNC->TXBuffer[5] == 'G') // Poll
return 4;
if (TNC->TXBuffer[5] == 'C') // Connect - reply we need is async
return 4;
if (TNC->TXBuffer[5] == 'L') // Shouldnt happen!
return 4;
if (TNC->TXBuffer[5] == '#') // Shouldnt happen!
return 4;
if (TNC->TXBuffer[5] == '%' && TNC->TXBuffer[6] == 'W') // Scan Control - Response to W1
if (TNC->InternalCmd)
return 4; // Just Ignore
if (TNC->TXBuffer[5] == 'J') // JHOST
{
if (TNC->TXBuffer[10] == '0') // JHOST0
{
TNC->Timeout = 1; //
return 4;
}
}
if (TNC->Streams[Stream].Connected)
return 4;
buffptr = GetBuff();
if (buffptr == NULL) return 4; // No buffers, so ignore
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0],"ARDOP} Ok\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
// C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 4;
}
if (Msg[3] > 0 && Msg[3] < 6)
{
// Success with message - null terminated
UCHAR * ptr;
int len;
if (Msg[2] == 0xff) // General Poll Response
{
UCHAR * Poll = TNC->TXBuffer;
UCHAR Chan = Msg[4] - 1;
if (Chan == 255) // Nothing doing
return 0;
if (Msg[5] != 0)
{
// More than one to poll - save the list of channels to poll
strcpy(TNC->NexttoPoll, &Msg[5]);
}
// Poll the channel that had data
Poll[2] = Chan; // Channel
Poll[3] = 0x1; // Command
if (Chan == 254) // Status - Send Extended Status (G3)
{
Poll[4] = 1; // Len-1
Poll[5] = 'G'; // Extended Status Poll
Poll[6] = '3';
}
else
{
Poll[4] = 0; // Len-1
Poll[5] = 'G'; // Poll
}
ARDOPCRCStuffAndSend(TNC, Poll, Poll[4] + 6);
TNC->InternalCmd = FALSE;
return 0;
}
Buffer = &Msg[4];
ptr = strchr(Buffer, 0);
if (ptr == 0)
return 0;
*(ptr++) = 13;
*(ptr) = 0;
len = (int)(ptr - Buffer);
if (len > 256)
return 0;
if (Stream > 0 && Stream <= APMaxStreams)
{
// Packet Mode Response. Could be command response or status.
struct STREAMINFO * STREAM = &TNC->Streams[Stream];
PMSGWITHLEN buffptr;
if (strstr(Buffer, "Incoming"))
{
// incoming call. Check which application it is for
char Call[11];
char TargetCall[11] = "";
char * ptr;
APPLCALLS * APPL;
char * ApplPtr = APPLS;
int App;
char Appl[10];
TRANSPORTENTRY * SESS;
Buffer[len-1] = 0;
WritetoTrace(TNC, Buffer, len);
STREAM->ConnectTime = time(NULL);
STREAM->BytesRXed = STREAM->BytesTXed = STREAM->PacketsSent = 0;
memcpy(Call, &Buffer[19], 10);
ptr = strchr(Call, ' ');
if (ptr) *ptr = 0;
ptr = strstr(&Buffer[19], " to ");
if (ptr)
{
memcpy(TargetCall, ptr + 4, 10);
ptr = strchr(TargetCall, 13);
if (ptr)
*ptr = 0;
}
ProcessIncommingConnectEx(TNC, Call, Stream, TRUE, FALSE);
SESS = TNC->PortRecord->ATTACHEDSESSIONS[Stream];
// Check for ExcludeList
if (ExcludeList[0])
{
if (CheckExcludeList(SESS->L4USER) == FALSE)
{
TidyClose(TNC, 0);
Debugprintf("ARDOP Packet Call from %s rejected", Call);
return 0;
}
}
// IF WE HAVE A PERMITTED CALLS LIST, SEE IF HE IS IN IT
if (TNC->PortRecord->PORTCONTROL.PERMITTEDCALLS)
{
UCHAR * ptr = TNC->PortRecord->PORTCONTROL.PERMITTEDCALLS;
while (TRUE)
{
if (memcmp(SESS->L4USER, ptr, 6) == 0) // Ignore SSID
break;
ptr += 7;
if ((*ptr) == 0) // Not in list
{
TidyClose(TNC, 0);
Debugprintf("ARDOP Packet Call from %s not in ValidCalls - rejected", Call);
return 0;
}
}
}
// See which application the connect is for
for (App = 0; App < 32; App++)
{
APPL=&APPLCALLTABLE[App];
memcpy(Appl, APPL->APPLCALL_TEXT, 10);
ptr=strchr(Appl, ' ');
if (ptr)
*ptr = 0;
if (_stricmp(TargetCall, Appl) == 0)
break;
}
if (App < 32)
{
char AppName[13];
memcpy(AppName, &ApplPtr[App * sizeof(CMDX)], 12);
AppName[12] = 0;
// Make sure app is available
if (CheckAppl(TNC, AppName))
{
char ApplCmd[80];
int Len = sprintf(ApplCmd, "%s\r", AppName);
buffptr = GetBuff();
if (buffptr == 0)
{
return len + 5; // No buffers, so ignore
}
buffptr->Len = Len;
memcpy(&buffptr->Data[0], ApplCmd, Len);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
TNC->SwallowSignon = TRUE;
// Save Appl Call in case needed for
}
else
{
char Msg[] = "Application not available\r\n";
// Send a Message, then a disconenct
// Send CTEXT First
if (STREAM->BPQtoPACTOR_Q) //Used for CTEXT
{
PMSGWITHLEN buffptr = Q_REM(&STREAM->BPQtoPACTOR_Q);
UCHAR * data = &buffptr->Data[0];
int txlen = (int)buffptr->Len;
SendARDOPorPacketData(TNC, Stream, data, txlen);
ReleaseBuffer(buffptr);
}
SendARDOPorPacketData(TNC, Stream, Msg, (int)strlen(Msg));
STREAM->NeedDisc = 100; // 10 secs
}
}
STREAM->Connected = TRUE;
return len + 5;
}
// Send to host
buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr == 0)
return len + 5; // No buffers, so ignore
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "%s", Buffer);
C_Q_ADD(&STREAM->PACTORtoBPQ_Q, buffptr);
// Unless Connected response close session
STREAM->Connecting = FALSE;
if (strstr(Buffer, "Connected"))
STREAM->Connected = TRUE;
else
if (strstr(Buffer, "Failure with"))
STREAM->ReportDISC = 10; // Gives time for failure message to display
else
if (strstr(Buffer, "Busy from"))
STREAM->ReportDISC = 10; // Gives time for failure message to display
else
if (strstr(Buffer, "Disconnected from"))
{
if (STREAM->Disconnecting) // We requested disconnect
{
STREAM->Connecting = FALSE;
STREAM->Connected = FALSE; // Back to Command Mode
STREAM->Disconnecting = FALSE;
}
else
{
STREAM->Connected = 0;
STREAM->ReportDISC = 10;
}
}
else
STREAM->NeedDisc = 10;
return len + 5;
}
// See if we need to process locally (Response to our command, Incoming Call, Disconencted, etc
if (Msg[3] < 3) // 1 or 2 - Success or Fail
{
return 0;
}
if (Msg[3] == 3) // Status
{
struct STREAMINFO * STREAM = &TNC->Streams[Stream];
return 0;
}
// 1, 2, 4, 5 - pass to Appl
return 0;
}
if (Msg[3] == 6)
{
return 0;
}
if (Msg[3] == 7) // Data
{
if (Stream > 0 && Stream <= APMaxStreams)
{
// Packet Response
int len = Msg[4] + 1;
if (TNC->Streams[Stream].Connected == 0)
return len + 5;
buffptr = GetBuff();
if (buffptr == NULL) return 0; // No buffers, so ignore
buffptr->Len = len;
memcpy((UCHAR *)&buffptr->Data[0], &Msg[5], buffptr->Len);
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return len + 5;
}
if (Stream == 32) // Command string
{
int Len = Msg[4] + 1;
ARDOPProcessResponse(TNC, &Msg[5], Len);
return 0;
}
if (Msg[2] == 0xfe) // Status Poll Response
{
return 0;
}
if (Msg[2] == 248) // Log Message
{
// Monitor Data - Length format
// first 4 bytes contain a 32 bits long timestamp.
// That timestamp holds the number of seconds that elapsed since date 01.01.2000 at 00:00:00.
// The MS byte is sent first. The timestamp can be corrected to the usual C timestamp (seconds
//since 01.01.1970, 00:00:00) simply by adding 946684800 (seconds) to it.
// Teminated with LF
int datalen = Msg[4] + 1;
time_t timestamp = (Msg[5] << 24) + (Msg[6] << 16)
+ (Msg[7] << 8) + Msg[8] + 946684800;
char c;
char timebuf[32] = "HH:MM:SS.MMM";
struct tm * tm;
if (TNC->LogHandle == 0 || TNC->DebugHandle == 0)
return 0;
tm = gmtime(&timestamp);
sprintf(timebuf, "%02d:%02d:%02d. ",
tm->tm_hour, tm->tm_min, tm->tm_sec);
// ARDOP Messages have a millisec time in first 3 bytes
// and a log type in 4th
c = Msg[12]; // Type
memcpy(&timebuf[9], &Msg[9], 3); // copy millisecs
fputs(timebuf, TNC->DebugHandle);
fputc(c, TNC->DebugHandle);
fputc(' ', TNC->DebugHandle);
fwrite(&Msg[13], 1, datalen - 8, TNC->DebugHandle);
fflush(TNC->DebugHandle);
if (c < '7')
{
// All types below debug go to log file
fputs(timebuf, TNC->LogHandle);
fputc(c, TNC->LogHandle);
fputc(' ', TNC->LogHandle);
fwrite(&Msg[13], 1, datalen - 8, TNC->LogHandle);
fflush(TNC->LogHandle);
}
return 0;
}
if (Msg[2] == 253) // Rig Port Response
{
// Queue for Rig Control Driver
int datalen = Msg[4] + 1;
PMSGWITHLEN buffptr;
// if not configured to use PTC Rig Control, Ignore
if (TNC->RIG->PORT == NULL || TNC->RIG->PORT->PTC == NULL)
return 0;
buffptr = (PMSGWITHLEN)GetBuff();
if (buffptr)
{
buffptr->Len = datalen;
memcpy(&buffptr->Data[0], &Msg[5], datalen);
C_Q_ADD(&TNC->RadiotoBPQ_Q, buffptr);
}
return 0;
}
if (Msg[2] == 250) // KISS
{
// Pass to KISS Code
int datalen = Msg[4] + 1;
void ** buffptr = NULL;
ProcessKISSBytes(TNC, &Msg[5], datalen);
return datalen + 5;
}
return 0;
}
return 0;
}
void ARDOPSCSCheckRX(struct TNCINFO * TNC)
{
int Length, Len = 0;
unsigned short crc;
char UnstuffBuffer[500];
if (TNC->RXLen == 500)
TNC->RXLen = 0;
if (TNC->ARDOPCommsMode == 'I')
{
unsigned char Buffer[33];
BOOL Error;
int gotThisTime = 0, i2clen;
// i2c mode always returns as much as requested or error
// First two bytes of block are length
// if (TNC->hDevice < 0)
// return;
while ((TNC->RXLen + Len) < 460)
{
i2clen = ReadCOMBlockEx(TNC->hDevice, Buffer, 33, &Error);
if (i2clen < 33 || i2clen == 5)
return;
if (Error)
{
Debugprintf("ARDOP i2c returned %d bytes Error %d", i2clen, Error);
return;
}
gotThisTime = Buffer[0];
if (gotThisTime == 0)
{
if (Len)
break; // Something to process
return; // No More
}
// if (gotThisTime != 7)
// Debugprintf("ARDOP i2c Len %d RXL %d %x %x %x %x %x %x %x %x %x %x %x %x",
// gotThisTime, TNC->RXLen + Len,
// Buffer[0], Buffer[1], Buffer[2], Buffer[3],
// Buffer[4], Buffer[5], Buffer[6], Buffer[7],
// Buffer[8], Buffer[9], Buffer[10], Buffer[11]);
memcpy(&TNC->RXBuffer[TNC->RXLen + Len], &Buffer[1], gotThisTime);
Len += gotThisTime;
if (Buffer[0] < 32)
break; // no more
}
}
else if (TNC->ARDOPCommsMode =='E') //Serial over TCP
Len = SerialGetTCPMessage(TNC, &TNC->RXBuffer[TNC->RXLen], 500 - TNC->RXLen);
else
if (TNC->hDevice)
Len = ReadCOMBlock(TNC->hDevice, &TNC->RXBuffer[TNC->RXLen], 500 - TNC->RXLen);
if (Len == 0)
return;
TNC->RXLen += Len;
Length = TNC->RXLen;
// DED mode doesn't have an end of frame delimiter. We need to know if we have a full frame
// Fortunately this is a polled protocol, so we only get one frame at a time
// If first char != 170, then probably a Terminal Mode Frame. Wait for CR on end
// If first char is 170, we could check rhe length field, but that could be corrupt, as
// we haen't checked CRC. All I can think of is to check the CRC and if it is ok, assume frame is
// complete. If CRC is duff, we will eventually time out and get a retry. The retry code
// can clear the RC buffer
if (TNC->RXBuffer[0] != 170)
{
// Char Mode Frame I think we need to see cmd: on end
// If we think we are in host mode, then to could be noise - just discard.
if (TNC->HostMode)
{
TNC->RXLen = 0; // Ready for next frame
return;
}
TNC->RXBuffer[TNC->RXLen] = 0;
// if (TNC->Streams[Stream].RXBuffer[TNC->Streams[Stream].RXLen-2] != ':')
if (strlen(TNC->RXBuffer) < TNC->RXLen)
TNC->RXLen = 0;
if ((strstr(TNC->RXBuffer, "cmd: ") == 0) && (strstr(TNC->RXBuffer, "pac: ") == 0))
return; // Wait for rest of frame
// Complete Char Mode Frame
// OpenLogFile(TNC->Port);
// WriteLogLine(TNC->Port, TNC->RXBuffer, (int)strlen(TNC->RXBuffer));
// CloseLogFile(TNC->Port);
TNC->RXLen = 0; // Ready for next frame
if (TNC->HostMode == 0)
{
// We think TNC is in Terminal Mode
ARDOPProcessTermModeResponse(TNC);
return;
}
// We thought it was in Host Mode, but are wrong.
TNC->HostMode = FALSE;
return;
}
if (TNC->HostMode == FALSE)
{
TNC->RXLen = 0; // clear input and wait for char mode response
return;
}
// Receiving a Host Mode frame
if (Length < 6) // Minimum Frame Sise
return;
if (TNC->RXBuffer[2] == 170)
{
// Retransmit Request
TNC->RXLen = 0;
return; // Ignore for now
}
// Can't unstuff into same buffer - fails if partial msg received, and we unstuff twice
Length = Unstuff(&TNC->RXBuffer[2], &UnstuffBuffer[2], Length - 2);
if (Length == -1)
{
// Unstuff returned an errors (170 not followed by 0)
TNC->RXLen = 0;
return; // Ignore for now
}
crc = compute_crc(&UnstuffBuffer[2], Length);
if (crc == 0xf0b8) // Good CRC
{
TNC->RXLen = 0; // Ready for next frame
UnstuffBuffer[0] = 0; // Make sure not seen as TCP Frame
ARDOPProcessDEDFrame(TNC, UnstuffBuffer, Length);
// If there are more channels to poll (more than 1 entry in general poll response,
// and link is not active, poll the next one
if (TNC->Timeout == 0)
{
UCHAR * Poll = TNC->TXBuffer;
if (TNC->NexttoPoll[0])
{
UCHAR Chan = TNC->NexttoPoll[0] - 1;
memmove(&TNC->NexttoPoll[0], &TNC->NexttoPoll[1], 19);
Poll[2] = Chan; // Channel
Poll[3] = 0x1; // Command
if (Chan == 254) // Status - Send Extended Status (G3)
{
Poll[4] = 1; // Len-1
Poll[5] = 'G'; // Extended Status Poll
Poll[6] = '3';
}
else
{
Poll[4] = 0; // Len-1
Poll[5] = 'G'; // Poll
}
ARDOPCRCStuffAndSend(TNC, Poll, Poll[4] + 6);
TNC->InternalCmd = FALSE;
return;
}
else
{
// if last message wasn't a general poll, send one now
if (TNC->PollSent)
return;
TNC->PollSent = TRUE;
// Use General Poll (255)
Poll[2] = 255 ; // Channel
Poll[3] = 0x1; // Command
Poll[4] = 0; // Len-1
Poll[5] = 'G'; // Poll
ARDOPCRCStuffAndSend(TNC, Poll, 6);
TNC->InternalCmd = FALSE;
}
}
return;
}
// Bad CRC - assume incomplete frame, and wait for rest. If it was a full bad frame, timeout and retry will recover link.
return;
}
VOID ARDOPSCSPoll(struct TNCINFO * TNC)
{
UCHAR * Poll = TNC->TXBuffer;
int Stream = 0;
if (TNC->Timeout)
{
TNC->Timeout--;
if (TNC->Timeout) // Still waiting
return;
TNC->Retries--;
if(TNC->Retries >= 0)
{
if (TNC->HostMode)
Debugprintf("ARDOP Timeout - Retransmit PTC Block");
ARDOPWriteCommBlock(TNC); // Retransmit Block
return;
}
// Retried out.
if (TNC->HostMode == 0)
{
ARDOPDoTermModeTimeout(TNC);
return;
}
// Retried out in host mode - Clear any connection and reinit the TNC
Debugprintf("ARDOP - Link to TNC Lost");
TNC->TNCOK = FALSE;
sprintf(TNC->WEB_COMMSSTATE,"%s Open but TNC not responding", TNC->PortRecord->PORTCONTROL.SerialPortName);
SetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
// Clear anything from UI_Q
while (TNC->PortRecord->UI_Q)
{
void ** buffptr = Q_REM(&TNC->PortRecord->UI_Q);
ReleaseBuffer(buffptr);
}
TNC->HostMode = 0;
TNC->ReinitState = 0;
TNC->CONNECTED = FALSE;
// Disconenct any attached sessions
for (Stream = 0; Stream <= APMaxStreams; Stream++)
{
if (TNC->PortRecord->ATTACHEDSESSIONS[Stream]) // Connected
{
TNC->Streams[Stream].Connected = FALSE; // Back to Command Mode
TNC->Streams[Stream].ReportDISC = TRUE; // Tell Node
}
}
}
if (TNC->Timeout)
return; // We've sent something
// if we have just restarted or TNC appears to be in terminal mode, run Initialisation Sequence
if (!TNC->HostMode)
{
ARDOPDoTNCReinit(TNC);
return;
}
TNC->PollSent = FALSE;
if (TNC->TNCOK && TNC->BPQtoRadio_Q)
{
int datalen;
PMSGWITHLEN buffptr;
buffptr = (PMSGWITHLEN)Q_REM(&TNC->BPQtoRadio_Q);
datalen = (int)buffptr->Len;
Poll[2] = 253; // Radio Channel
Poll[3] = 0; // Data?
Poll[4] = datalen - 1;
memcpy(&Poll[5], &buffptr->Data[0], datalen);
ReleaseBuffer(buffptr);
ARDOPCRCStuffAndSend(TNC, Poll, datalen + 5);
return;
}
for (Stream = 0; Stream < 14; Stream++) // Priority to commands
{
if (TNC->TNCOK && TNC->Streams[Stream].BPQtoPACTOR_Q)
{
int datalen;
PMSGWITHLEN buffptr;
char * Buffer;
buffptr=Q_REM(&TNC->Streams[Stream].BPQtoPACTOR_Q);
TNC->Streams[Stream].FramesQueued--;
datalen = (int)buffptr->Len;
Buffer = &buffptr->Data[0]; // Data portion of frame
Poll[3]= 0;
if (Stream > 11)
Poll[2] = Stream + 20; // 12 and 13 to Channels 32 and 33
else
if (Stream == 0)
Poll[2] = 33;
else
{
// Packet Frame
Poll[2] = Stream;
Poll[3] = Buffer[0]; // First Byte is CMD/Data FLag
datalen--;
Buffer++;
}
Poll[4] = datalen - 1;
memcpy(&Poll[5], Buffer, datalen);
ReleaseBuffer(buffptr);
ARDOPCRCStuffAndSend(TNC, Poll, datalen + 5);
return;
}
}
//0x04421CB0 aa aa 21 00 07 00 06 48 65 6c 6c 6f 0d c8 3e 38 42 50 51 2d 32 20 35 0d 4a 8a 4d 38 42 50 51 <20><>!....Hello.<2E>>8BPQ-2 5.J<>M8BPQ
//0x04421CCF 2d 31 30 2c 47 4d 38 42 50 51 2d 35 2c 47 4d 38 42 50 51 2c 47 4d 38 42 50 51 2d 31 35 0d 00 -10,GM8BPQ-5,GM8BPQ,GM8BPQ-15..
//0x04421CEE 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ...............................
if (TNC->TNCOK && TNC->KISSTX_Q)
{
int datalen;
PMSGWITHLEN buffptr;
buffptr = (PMSGWITHLEN)Q_REM(&TNC->KISSTX_Q);
datalen = (int)buffptr->Len;
Poll[2] = 250; // KISS Channel
Poll[3] = 0; // Data
Poll[4] = datalen - 1;
memcpy(&Poll[5], &buffptr->Data[0], datalen);
ReleaseBuffer(buffptr);
ARDOPCRCStuffAndSend(TNC, Poll, datalen + 5);
return;
}
TNC->PollSent = TRUE;
// Use General Poll (255)
Poll[2] = 255 ; // Channel
Poll[3] = 0x1; // Command
if (TNC->ReinitState == 3)
{
TNC->ReinitState = 0;
Poll[3] = 0x41;
}
Poll[4] = 0; // Len-1
Poll[5] = 'G'; // Poll
ARDOPCRCStuffAndSend(TNC, Poll, 6);
TNC->InternalCmd = FALSE;
return;
}
// ARDOP Serial over TCP Routines
// Probably only for Teensy with ESP01. Runs SCS Emulator over a TCP Link
VOID SerialConnecttoTCPThread(struct TNCINFO * TNC);
int SerialConnecttoTCP(struct TNCINFO * TNC)
{
_beginthread(SerialConnecttoTCPThread, 0, (void *)TNC);
return 0;
}
VOID SerialConnecttoTCPThread(struct TNCINFO * TNC)
{
char Msg[255];
int i;
u_long param = 1;
BOOL bcopt=TRUE;
struct hostent * HostEnt;
SOCKADDR_IN sinx;
int addrlen=sizeof(sinx);
if (TNC->HostName == NULL)
return;
Sleep(5000); // Allow init to complete
while(1)
{
if (TNC->TCPCONNECTED)
{
Sleep(57000);
continue;
}
TNC->BusyFlags = 0;
TNC->CONNECTING = TRUE;
TNC->destaddr.sin_addr.s_addr = inet_addr(TNC->HostName);
TNC->Datadestaddr.sin_addr.s_addr = inet_addr(TNC->HostName);
if (TNC->destaddr.sin_addr.s_addr == INADDR_NONE)
{
// Resolve name to address
HostEnt = gethostbyname (TNC->HostName);
if (!HostEnt)
{
TNC->CONNECTING = FALSE;
Sleep (57000);
return; // Resolve failed
}
memcpy(&TNC->destaddr.sin_addr.s_addr,HostEnt->h_addr,4);
memcpy(&TNC->Datadestaddr.sin_addr.s_addr,HostEnt->h_addr,4);
}
if (TNC->TCPSock)
{
Debugprintf("ARDOP Closing Sock %d", TNC->TCPSock);
closesocket(TNC->TCPSock);
}
TNC->TCPSock = 0;
TNC->TCPSock = socket(AF_INET,SOCK_STREAM,0);
if (TNC->TCPSock == INVALID_SOCKET)
{
i=sprintf(Msg, "Socket Failed for ARDOP socket - error code = %d\r\n", WSAGetLastError());
WritetoConsole(Msg);
TNC->CONNECTING = FALSE;
return;
}
setsockopt(TNC->TCPSock, SOL_SOCKET, SO_REUSEADDR, (const char FAR *)&bcopt, 4);
// setsockopt(TNC->TCPDataSock, IPPROTO_TCP, TCP_NODELAY, (const char FAR *)&bcopt, 4);
sinx.sin_family = AF_INET;
sinx.sin_addr.s_addr = INADDR_ANY;
sinx.sin_port = 0;
if (connect(TNC->TCPSock,(LPSOCKADDR) &TNC->destaddr,sizeof(TNC->destaddr)) == 0)
{
//
// Connected successful
TNC->TCPCONNECTED = TRUE;
ioctl(TNC->TCPSock, FIONBIO, &param);
Debugprintf("ARDOP TCPSerial connected, Socket %d", TNC->TCPSock);
}
else
{
if (TNC->Alerted == FALSE)
{
sprintf(Msg, "Connect Failed for ARDOP socket - error code = %d Port %d\n",
WSAGetLastError(), htons(TNC->destaddr.sin_port));
WritetoConsole(Msg);
sprintf(TNC->WEB_COMMSSTATE, "Connection to TNC failed");
MySetWindowText(TNC->xIDC_COMMSSTATE, TNC->WEB_COMMSSTATE);
TNC->Alerted = TRUE;
}
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
TNC->CONNECTING = FALSE;
Sleep (57000); // 1 Mins
}
}
}
int SerialGetTCPMessage(struct TNCINFO * TNC, unsigned char * Buffer, int Len)
{
int index=0;
ULONG param = 1;
if (TNC->TCPCONNECTED)
{
int InputLen;
// Poll TCP COnnection for data
// May have several messages per packet, or message split over packets
InputLen = recv(TNC->TCPSock, Buffer, Len, 0);
if (InputLen < 0)
{
int err = WSAGetLastError();
if (err == 10035 || err == 11)
{
InputLen = 0;
return 0;
}
Debugprintf("ARDOP Serial TCP RX Error %d received for socket %d", err, TNC->TCPSock);
TNC->TCPCONNECTED = 0;
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
return 0;
}
if (InputLen > 0)
return InputLen;
else
{
Debugprintf("ARDOP Serial TCP Close received for socket %d", TNC->TCPSock);
TNC->TCPCONNECTED = 0;
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
return 0;
}
}
return 0;
}
int ARDOPWriteCommBlock(struct TNCINFO * TNC)
{
if (TNC->ARDOPCommsMode == 'E')
{
if (TNC->TCPCONNECTED)
{
int SentLen = send(TNC->TCPSock, TNC->TXBuffer, TNC->TXLen, 0);
if (SentLen != TNC->TXLen)
{
// ARDOP doesn't seem to recover from a blocked write. For now just reset
int winerr=WSAGetLastError();
char ErrMsg[80];
sprintf(ErrMsg, "ARDOP Write Failed for port %d - error code = %d\r\n", TNC->Port, winerr);
WritetoConsole(ErrMsg);
closesocket(TNC->TCPSock);
TNC->TCPSock = 0;
TNC->TCPCONNECTED = FALSE;
}
}
TNC->Timeout = 20; // 2 secs
return 1;
}
if (TNC->hDevice)
return (WriteCommBlock(TNC));
else
{
TNC->Timeout = 20; // 2 secs
return 0;
}
}
// Teensy Combined ARDOP/AX.25 Support
#define FEND 0xC0 // KISS CONTROL CODES
#define FESC 0xDB
#define TFEND 0xDC
#define TFESC 0xDD
/*
VOID ARAXINIT(struct PORTCONTROL * PORT)
{
char Msg[80] = "";
memcpy(Msg, PORT->PORTDESCRIPTION, 30);
sprintf(Msg, "%s\n", Msg);
WritetoConsoleLocal(Msg);
}
VOID ARAXTX(struct PORTCONTROL * PORT, PMESSAGE Buffer)
{
// KISS Encode and Queue to Host Mode KISS Queue
UINT * TXMsg = NULL; // KISS Message to queue to Hostmode KISS Queue
UCHAR * TXPtr;
int TXLen = 0;
struct _LINKTABLE * ACKWORD = Buffer->Linkptr;
struct _MESSAGE * Message = (struct _MESSAGE *)Buffer;
UCHAR c;
char * ptr1;
int Len;
struct TNCINFO * TNC = PORT->TNC;
if (TNC && TNC->CONNECTED) // Have a Host Session
TXMsg = GetBuff(); // KISS Message to queue to Hostmode KISS Queue
if (TXMsg == NULL) // No Session or No buffers
{
// Reset any ACKMODE Timer and release buffer C_Q_ADD(&TRACE_Q, Buffer);
struct _LINKTABLE * LINK = Buffer->Linkptr;
if (LINK)
{
if (LINK->L2TIMER)
LINK->L2TIMER = LINK->L2TIME;
Buffer->Linkptr = 0; // CLEAR FLAG FROM BUFFER
}
C_Q_ADD(&TRACE_Q, Buffer);
return;
}
TXPtr = (UCHAR *)&TXMsg[2];
ptr1 = &Message->DEST[0];
Len = Message->LENGTH - 7;
*(TXPtr++) = FEND;
if (ACKWORD) // Frame Needs ACK
{
*TXPtr++ = 0x0c; // ACK OPCODE
ACKWORD -= (UINT)LINKS; // Con only send 16 bits, so use offset into LINKS
*TXPtr++ = ACKWORD & 0xff;
*TXPtr++ = (ACKWORD >> 8) &0xff;
// have to reset flag so trace doesnt clear it
Buffer->Linkptr = 0;
TXLen = 4;struct _LINKTABLE *
}
else
{
*TXPtr++ = 0;
TXLen = 2;
}
while (Len--)
{
c = *(ptr1++);
switch (c)
{
case FEND:
(*TXPtr++) = FESC;
(*TXPtr++) = TFEND;
TXLen += 2;
break;
case FESC:
(*TXPtr++) = FESC;
(*TXPtr++) = TFESC;
TXLen += 2;
break;
default:
(*TXPtr++) = c;
TXLen++;
}
if (TXLen > 250)
{
// Queue frame to KISS Channel and get another buffer
// can take up to 256, but sometimes add 2 at a time
TXMsg[1] = (int)(TXPtr - (UCHAR *)&TXMsg[2]);
}
}
(*TXPtr++) = FEND;
TXLen++;
TXMsg[1] = TXLen;
C_Q_ADD(&TNC->KISSTX_Q, TXMsg);
// Pass buffer to trace routines
C_Q_ADD(&TRACE_Q, Buffer);
}
VOID ARAXRX()
{
}
VOID ARAXTIMER()
{
}
VOID ARAXCLOSE()
{
}
BOOL ARAXTXCHECK()
{
return 0;
}
#define DATABYTES 400000 // WAS 320000
extern UCHAR * NEXTFREEDATA; // ADDRESS OF NEXT FREE BYTE in shared memory
extern UCHAR DATAAREA[DATABYTES];
VOID AddVirtualKISSPort(struct TNCINFO * TNC, int ARDOPPort, char * buf)
{
// Adds a Virtual KISS port for simultaneous ARDOP and Packet on Teensy TNC
// or ARDOP_PTC ovet a single host mode port.
// Not needed if using TCP interface as that uses KISS over TCP
struct PORTCONTROL * PORTVEC=PORTTABLE;
struct PORTCONTROL * PORT;
int pl = sizeof(struct PORTCONTROL);
int mh = MHENTRIES * sizeof(MHSTRUC);
int space = (int)(&DATAAREA[DATABYTES] - NEXTFREEDATA);
char Msg[64];
unsigned char * ptr3;
unsigned int3;
int newPortNumber = 0;
if (TNC->ARDOPCommsMode == 'T') // TCP
return;
if (buf[12] == '=')
newPortNumber = atoi(&buf[13]);
if (space < (pl + mh))
{
WritetoConsoleLocal("Insufficient space to add ARDOP/Packet Port\n");
return;
}
while (PORTVEC->PORTPOINTER)
{
PORTVEC=PORTVEC->PORTPOINTER;
}
// PORTVEC is now last port in chain
ptr3 = NEXTFREEDATA;
PORT = (struct PORTCONTROL *)ptr3;
ptr3 += sizeof (struct PORTCONTROL);
// Round to word boundary (for ARM5 etc)
int3 = (int)ptr3;
int3 += 3;
int3 &= 0xfffffffc;
ptr3 = (UCHAR *)int3;
PORTVEC->PORTPOINTER = PORT; // Chain to previous last port
if (newPortNumber == 0)
newPortNumber = 32;;
if (GetPortTableEntryFromPortNum(newPortNumber))
{
// Number in use
// If user specified search up, if default search down
if (newPortNumber == 32)
while(newPortNumber && GetPortTableEntryFromPortNum(newPortNumber)) // Try next lower
newPortNumber--;
else
while(newPortNumber < 32 && GetPortTableEntryFromPortNum(newPortNumber)) // Try next highest
newPortNumber++;
}
if (newPortNumber == 0 || newPortNumber > 32)
{
WritetoConsoleLocal("No free Port Number to add ARDOP/Packet Port\n");
return;
}
NUMBEROFPORTS++;
PORT->PORTNUMBER = newPortNumber;
PORT->PortSlot = PORTVEC->PortSlot + 1;
sprintf(Msg, "Packet Port for ARDOP Port %d ", ARDOPPort);
memcpy(PORT->PORTDESCRIPTION, Msg, 30);
PORT->TNC = TNC;
TNC->VirtualPORT = PORT; // Link TNC and PORT both ways
PORT->PORTINITCODE = ARAXINIT;
PORT->PORTTIMERCODE = ARAXTIMER;
PORT->PORTRXROUTINE = ARAXRX;
PORT->PORTTXROUTINE = ARAXTX;
PORT->PORTCLOSECODE = ARAXCLOSE;
PORT->PORTTXCHECKCODE = ARAXTXCHECK;
// Default L2 Params
PORT->PORTN2 = 5;
PORT->PORTT1 = 5000/333; // FRACK 5 secs
// As we use IPoll we can set RESPTIME very long and FRACK short
PORT->PORTT2 = 20000/333; // RESPTIME
PORT->PORTPACLEN = 128;
PORT->PORTWINDOW = 1;
// ADD MH AREA IF NEEDED
NEEDMH = 1; // Include MH in Command List
PORT->PORTMHEARD = (PMHSTRUC)ptr3;
ptr3 += MHENTRIES * sizeof(MHSTRUC);
// Round to word boundary (for ARM5 etc)
int3 = (int)ptr3;
int3 += 3;
int3 &= 0xfffffffc;
ptr3 = (UCHAR *)int3;
NEXTFREEDATA = ptr3;
return;
}
int ConfigVirtualKISSPort(struct TNCINFO * TNC, char * Cmd)
{
struct PORTCONTROL * PORT = TNC->VirtualPORT;
void ** buffptr = GetBuff();
char * Context;
char * Param;
char * Command;
int Value;
int Stream = 0;
if (buffptr == NULL)
return 0;
if (PORT == NULL)
{
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} Packet Mode nor Enabled\r");
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
_strupr(Cmd);
Command = strtok_s(&Cmd[4], " \n\r", &Context);
Param = strtok_s(NULL, " \n\r", &Context);
if (Param)
Value = atoi(Param);
if (strcmp(Command, "PACLEN") == 0)
{
if (Param == NULL)
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s %d\r", Command, PORT->PORTPACLEN);
else
{
if (Value > 0 && Value <= 256)
PORT->PORTPACLEN = Value;
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s now %d\r", Command, PORT->PORTPACLEN);
}
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
else if (strcmp(Command, "RETRIES") == 0)
{
if (Param == NULL)
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s %d\r", Command, PORT->PORTN2);
else
{
if (Value > 0 && Value <= 16)
PORT->PORTN2 = Value;
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s now %d\r", Command, PORT->PORTN2);
}
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
else if (strcmp(Command, "WINDOW") == 0 || strcmp(Command, "MAXFRAME") == 0)
{
if (Param == NULL)
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s %d\r", Command, PORT->PORTWINDOW);
else
{
if (Value > 0 && Value <= 7)
PORT->PORTWINDOW = Value;
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s now %d\r", Command, PORT->PORTWINDOW);
}
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
else if (strcmp(Command, "FRACK") == 0)
{
if (Param == NULL)
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s %d mS\r", Command, PORT->PORTT1 * 333);
else
{
if (Value > 0 && Value <= 20000)
PORT->PORTT1 = Value / 333;
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s now %d mS\r", Command, PORT->PORTT1 * 333);
}
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
else if (strcmp(Command, "RESPTIME") == 0)
{
if (Param == NULL)
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s %d mS\r", Command, PORT->PORTT2 * 333);
else
{
if (Value > 0 && Value <= 20000)
PORT->PORTT2 = Value / 333;
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} PAC %s now %d mS\r", Command, PORT->PORTT2 * 333);
}
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
else
buffptr->Len = sprintf((UCHAR *)&buffptr->Data[0], "ARDOP} Invalid Command %s\r", Cmd);
C_Q_ADD(&TNC->Streams[Stream].PACTORtoBPQ_Q, buffptr);
return 0;
}
*/
void ProcessKISSBytes(struct TNCINFO * TNC, UCHAR * Data, int Len)
{
// Kiss data received from TNC but not necessarrily a full packet
// and could be multiple packets
// The TNC record is for the ARDOP Port, but we need to queue data
// to the linked Virtual Packet Port
struct PORTCONTROL * PORT = TNC->VirtualPORT;
UCHAR * KISSBuffer = TNC->KISSBuffer;
UCHAR c;
UCHAR * inptr = Data;
int outptr = TNC->KISSInputLen;
if (PORT == NULL)
return;
while(Len--)
{
c = *(inptr++);
if (TNC->ESCFLAG)
{
//
// FESC received - next should be TFESC or TFEND
TNC->ESCFLAG = FALSE;
if (c == TFESC)
c = FESC;
if (c == TFEND)
c = FEND;
}
else
{
switch (c)
{
case FEND:
//
// Either start of message or message complete
//
if (outptr == 0)
{
// Start of Message. If polling, extend timeout
continue;
}
ProcessKISSPacket(TNC, KISSBuffer, outptr);
outptr = 0;
return;
case FESC:
TNC->ESCFLAG = TRUE;
continue;
}
}
//
// Ok, a normal char
//
KISSBuffer[outptr++] = c;
}
if (outptr > 510)
outptr = 0; // Protect Buffer
TNC->KISSInputLen = outptr;
return;
}
void ProcessKISSPacket(struct TNCINFO * TNC, UCHAR * KISSBuffer, int Len)
{
if (KISSBuffer[0] == 0x0c) // ACK Frame
{
// ACK FRAME - reset link timer
struct _LINKTABLE * LINK;
UINT ACKWORD = KISSBuffer[1] | KISSBuffer[2] << 8;
LINK = LINKS + ACKWORD;
if (LINK->L2TIMER)
LINK->L2TIMER = LINK->L2TIME;
return;
}
if (KISSBuffer[0] == 0) // Data Frame
{
PDATAMESSAGE Buffer = (PDATAMESSAGE)GetBuff();
if (Buffer)
{
memcpy(&Buffer->PID, &KISSBuffer[1], --Len);
Len += sizeof(void *) + 3;
PutLengthinBuffer(Buffer, Len);
C_Q_ADD(&TNC->VirtualPORT->PORTRX_Q, (UINT *)Buffer);
}
}
}