618 lines
14 KiB
C
618 lines
14 KiB
C
/*
|
|
Copyright 2001-2018 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
|
|
*/
|
|
|
|
// Module for writing ADIF compatible log records
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#define _CRT_SECURE_NO_WARNINGS
|
|
#define _CRT_SECURE_NO_DEPRECATE
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include "time.h"
|
|
#include "CHeaders.h"
|
|
#include "tncinfo.h"
|
|
#include "adif.h"
|
|
#include "telnetserver.h"
|
|
|
|
/*
|
|
|
|
Typical File Header
|
|
|
|
Written by RMS Trimode ver: 1.3.21.0 <adif_ver:5>2.2.6<eoh>
|
|
|
|
Example TriMode Record
|
|
|
|
<call:6>VA2ROR
|
|
<qso_date:8>20190201
|
|
<time_on:6>140801
|
|
<time_off:6>140958
|
|
<mode:0>
|
|
<gridsquare:6>JG28DK
|
|
<band:3>20M
|
|
<freq:9>14.109000
|
|
<comment:145>0|2019-02-01 14:09:58|RMS Trimode|1.3.21.0|CMS|ZS1RS|VA2ROR|[AirMail-3.5.036-B2FHIM$]
|
|
|Pactor 3|14109000|1957|311|FQ|1|0|4098|93|117|JG28DK|JF96HD
|
|
<eor>
|
|
|
|
Info Used by RMS Analyser, Logged as an adif <comment> field
|
|
|
|
0|<date time>|<program name>|<program version>|<name of cms site used>|<rms site callsign>|<user
|
|
callsign>|<user sid>|<mode>|<frequency>|<range>|<bearing>|<termination>|<messages sent>|<messages
|
|
received>|<bytes sent>|<bytes received>|<holding time>
|
|
|
|
<date time> in the format YYYY-MM-DD HH:MM:SS and always in UTC.
|
|
<program name> is the name of the program posting this record.
|
|
<program version> is the version number of the program.
|
|
<name of CMS site used> is the city name assigned to the CMS.
|
|
<rms site callsign> is the radio callsign used by the RMS server including any SSID.
|
|
<user callsign> is the radio callsign used by the user including any SSID.
|
|
<user sid> is the SID used by the user's client program.
|
|
<mode> is the air protocol used such as "Pactor 3" or "Packet 1200".
|
|
At the present time modes used or expected include Pactor 1, Pactor 2, Pactor 3, Pactor 4, Packet 1200,
|
|
Packet 9600, Winmor 500, Winmor 1600, and Robust Packet.
|
|
<frequency> is the frequency used in Hertz.
|
|
|
|
<range> if the distance to the user's site (if known).
|
|
<bearing> is the bearing to the user's site (if known).
|
|
<termination> is the last command sent or received.
|
|
Normally there will be "FQ" but could be "FF", "FC" "PR", "PQ", "FS", "F>", or ">" or one of the keyboard
|
|
commands. It should limited to one or two characters.
|
|
<messages sent> through <holding time> are integers and may be 0 but not empty.
|
|
An example:
|
|
|
|
0|2011-05-20 12:34.56|RMS Packet|1.2.3.4|San Diego|KN6KB-10|W1ABC|
|
|
[Paclink-4.2.0.0-N00B2FIHM$]| Packet 1200|145090000|45|244|FQ|
|
|
1|0|2454|120|135
|
|
|
|
*/
|
|
|
|
VOID FromLOC(char * Locator, double * pLat, double * pLon);
|
|
double Distance(double laa, double loa, double lah, double loh, BOOL KM);
|
|
double Bearing(double lat2, double lon2, double lat1, double lon1);
|
|
|
|
extern UCHAR LogDirectory[260];
|
|
extern char LOC[7];
|
|
extern char TextVerstring[50];
|
|
struct WL2KInfo * WL2KReports;
|
|
|
|
extern char WL2KModes[55][18];
|
|
|
|
BOOL ADIFLogEnabled = FALSE;
|
|
|
|
char ADIFLogName[80] = "ADIF.adi";
|
|
|
|
char * stristr (char *ch1, char *ch2);
|
|
|
|
VOID CountMessages(ADIF * ADIF)
|
|
{
|
|
while (ADIF->FBBIndex)
|
|
{
|
|
// We have messages
|
|
|
|
if (ADIF->FBBLen[ADIF->FBBIndex - 1]) // don't count rejected messages
|
|
{
|
|
if (ADIF->Dirn == 'S')
|
|
{
|
|
ADIF->Sent++;
|
|
ADIF->BytesSent += ADIF->FBBLen[ADIF->FBBIndex - 1];
|
|
}
|
|
else
|
|
{
|
|
ADIF->Received++;
|
|
ADIF->BytesReceived += ADIF->FBBLen[ADIF->FBBIndex - 1];
|
|
}
|
|
}
|
|
ADIF->FBBIndex--;
|
|
}
|
|
}
|
|
|
|
|
|
BOOL UpdateADIFRecord(ADIF * ADIF, char * Msg, char Dirn)
|
|
{
|
|
// Entered each time a text message is received on CMS session
|
|
|
|
// To get transfer stats we have to monitor the B2 protocol messages, tracking proposals, accept/reject and completion
|
|
|
|
size_t Len;
|
|
|
|
// Always keep info so we can sent as Winlink Session Record
|
|
// even if ADIF logging is disabled.
|
|
|
|
// if (ADIFLogEnabled == FALSE)
|
|
// return TRUE;
|
|
|
|
if (ADIF == NULL)
|
|
return TRUE;
|
|
|
|
if (ADIF->StartTime == 0)
|
|
{
|
|
ADIF->StartTime = time(NULL);
|
|
// ADIF->Mode = 49; // Unused value (unforunately 0 = PKT1200)
|
|
}
|
|
|
|
// Try to build complete lines
|
|
|
|
if (Dirn == 'R')
|
|
{
|
|
Len = strlen(&ADIF->PartMessageRX[0]);
|
|
|
|
if (Len + strlen(Msg) < 512)
|
|
strcat(ADIF->PartMessageRX, Msg);
|
|
|
|
if (strstr(Msg, "<cr>"))
|
|
Msg = ADIF->PartMessageRX;
|
|
else
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
Len = strlen(&ADIF->PartMessageTX[0]);
|
|
|
|
if (Len + strlen(Msg) < 256)
|
|
strcat(ADIF->PartMessageTX, Msg);
|
|
|
|
if (strstr(Msg, "<cr>"))
|
|
Msg = ADIF->PartMessageTX;
|
|
else
|
|
return TRUE;
|
|
}
|
|
|
|
switch (Dirn)
|
|
{
|
|
case 'R':
|
|
|
|
if (Msg[0] == '[')
|
|
{
|
|
if (ADIF->ServerSID[0] == 0)
|
|
{
|
|
char * endsid = strchr(Msg, ']');
|
|
|
|
if (endsid && (endsid - Msg) < 78)
|
|
memcpy(ADIF->ServerSID, Msg, ++endsid - Msg);
|
|
}
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
if (Msg[0] == '[')
|
|
{
|
|
if (ADIF->UserSID[0] == 0)
|
|
{
|
|
char * endsid = strchr(Msg, ']');
|
|
|
|
if (endsid && (endsid - Msg) < 78)
|
|
memcpy(ADIF->UserSID, Msg, ++endsid - Msg);
|
|
}
|
|
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
if (ADIF->LOC[0] == 0 && Msg[0] == ';' && stristr(Msg, "DE "))
|
|
{
|
|
// Look for ; GM8BPQ-10 DE G8BPQ (IO92KX)
|
|
// AirMail EA8URF de KG5VSG (GK86qo) QTC: 1 209
|
|
|
|
|
|
// Paclink-Unix Sends
|
|
|
|
// ; VE7SPR-10 DE N7NIX QTC 1
|
|
|
|
char * StartLoc = strchr(Msg, '(');
|
|
char * EndLoc = strchr(Msg, ')');
|
|
|
|
if (StartLoc == NULL || EndLoc == NULL)
|
|
return TRUE;
|
|
|
|
if ((EndLoc - StartLoc) < 10)
|
|
memcpy(ADIF->LOC, StartLoc + 1, (EndLoc - StartLoc) - 1);
|
|
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
// Look for Proposals
|
|
|
|
// FC EM 3909_GM8BPQ 3520 1266 0<cr>
|
|
|
|
if (memcmp(Msg, "FC EM ", 6) == 0)
|
|
{
|
|
char * ptr, *Context;
|
|
|
|
// We need to detect first of a block of proposals so we can count any acked messages
|
|
|
|
if (ADIF->GotFC == 0) // Last was not FC
|
|
CountMessages(ADIF); // This acks last batch
|
|
|
|
ADIF->GotFC = 1;
|
|
|
|
ptr = strtok_s(&Msg[6], " \r", &Context); // BID
|
|
|
|
if (ptr)
|
|
{
|
|
ptr = strtok_s(NULL, " \r", &Context);
|
|
|
|
if (ptr)
|
|
{
|
|
ADIF->FBBLen[ADIF->FBBIndex] = atoi(ptr); // Not really sure we need lengths, but no harm
|
|
|
|
if (ADIF->FBBIndex++ == 5)
|
|
ADIF->FBBIndex = 4; // Proect ourselves
|
|
|
|
}
|
|
|
|
ADIF->Dirn = Dirn;
|
|
}
|
|
strcpy(ADIF->Termination, "FC");
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
if (memcmp(Msg, "FS ", 3) == 0)
|
|
{
|
|
// As we only count sent messages must check for rejections;
|
|
|
|
// FS YYY<cr>
|
|
|
|
int i = 0;
|
|
char c;
|
|
|
|
ADIF->GotFC = 0; // Ready for next batch
|
|
|
|
while (i < ADIF->FBBIndex)
|
|
{
|
|
c = Msg[i + 3];
|
|
|
|
if ((c == '-') || (c == 'N') || (c == 'R') || (c == 'E')) // Not wanted
|
|
{
|
|
ADIF->FBBLen[i] = 0; // Clear corresponding length
|
|
}
|
|
i++;
|
|
}
|
|
|
|
strcpy(ADIF->Termination, "FS");
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
if (strcmp(Msg, "FF<cr>") == 0 || strcmp(Msg, "FQ<cr>") == 0)
|
|
{
|
|
// Need to count any complete messages
|
|
|
|
CountMessages(ADIF); // This acks last batch
|
|
|
|
memcpy(ADIF->Termination, Msg, 2);
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
if (Msg[0] == ';' && Msg[3] == ':' )
|
|
memcpy(ADIF->Termination, &Msg[1],2);
|
|
|
|
Msg[0] = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
typedef struct BandLimits
|
|
{
|
|
char Name[10];
|
|
double lower;
|
|
double upper;
|
|
} BandLimits;
|
|
|
|
BandLimits Bands[] =
|
|
{
|
|
{"2190m", .1357, .1378},
|
|
{"630m", .472, .479},
|
|
{"560m", .501, .504},
|
|
{"160m", 1.8, 2.0},
|
|
{"80m", 3.5, 4.0},
|
|
{"60m", 5.06, 5.45},
|
|
{"40m", 7.0, 7.3},
|
|
{"30m", 10.1, 10.15},
|
|
{"20m", 14.0, 14.35},
|
|
{"17m", 18.068, 18.168},
|
|
{"15m", 21.0, 21.45},
|
|
{"12m", 24.890, 24.99},
|
|
{"10m", 28.0, 29.7},
|
|
{"6m", 50, 54},
|
|
{"4m", 70, 72},
|
|
{"2m", 144, 148},
|
|
{"1.25m", 222, 225},
|
|
{"70cm", 420, 450},
|
|
{"33cm", 902, 928},
|
|
{"23cm", 1240, 1300},
|
|
{"13cm", 2300, 2450},
|
|
{"9cm", 3300, 3500},
|
|
{"6cm", 5650, 5925},
|
|
{"3cm", 10000, 10500},
|
|
{"6mm", 47000, 47200},
|
|
{"4mm", 75500, 81000},
|
|
{"2.5mm", 119980, 120020},
|
|
{"2mm", 142000, 149000},
|
|
{"1mm", 241000, 250000},
|
|
{"unknown", 0, 9999999999}
|
|
};
|
|
|
|
int FreqCount = sizeof(Bands)/sizeof(struct BandLimits);
|
|
|
|
char ADIFModes [55][18] = {
|
|
"PKT", "PKT", "PKT", "PKT", "PKT", "PKT", "PKT", "", "", "", // 0 - 9
|
|
"", "PAC", "", "", "PAC/PAC2", "", "PAC/PAC3", "", "", "", "PAC/PAK4", // 10 - 20
|
|
"WINMOR", "WINMOR", "", "", "", "", "", "", "", // 21 - 29
|
|
"Robust Packet", "", "", "", "", "", "", "", "", "", // 30 - 39
|
|
"ARDOP", "ARDOP", "ARDOP", "ARDOP", "ARDOP", "", "", "", "", "", // 40 - 49
|
|
"VARA", "VARAFM", "VARAFM96", "VARA500", "VARA2750"};
|
|
|
|
|
|
BOOL WriteADIFRecord(ADIF * ADIF)
|
|
{
|
|
UCHAR Value[MAX_PATH];
|
|
time_t T;
|
|
struct tm * tm;
|
|
struct tm * starttm;
|
|
struct tm endtm;
|
|
|
|
FILE * Handle;
|
|
char Comment[256] = "";
|
|
int CommentLen;
|
|
char Date[32];
|
|
int Dist = 0;
|
|
int intBearing = 0;
|
|
struct stat STAT;
|
|
|
|
double Lat, Lon;
|
|
double myLat, myLon;
|
|
|
|
if (ADIFLogEnabled == FALSE)
|
|
return TRUE;
|
|
|
|
if (ADIF == NULL)
|
|
return TRUE;
|
|
|
|
T = time(NULL);
|
|
tm = gmtime(&T);
|
|
|
|
memcpy(&endtm, tm, sizeof(endtm));
|
|
|
|
if (LogDirectory[0] == 0)
|
|
{
|
|
strcpy(Value, "logs/BPQ_CMS_ADIF");
|
|
}
|
|
else
|
|
{
|
|
strcpy(Value, LogDirectory);
|
|
strcat(Value, "/");
|
|
strcat(Value, "logs/BPQ_CMS_ADIF");
|
|
}
|
|
|
|
sprintf(Value, "%s_%04d%02d.adi", Value,
|
|
tm->tm_year +1900, tm->tm_mon+1);
|
|
|
|
STAT.st_size = 0;
|
|
stat(Value, &STAT);
|
|
|
|
Handle = fopen(Value, "ab");
|
|
|
|
if (Handle == NULL)
|
|
return FALSE;
|
|
|
|
if (STAT.st_size == 0)
|
|
{
|
|
// New File - Write Header
|
|
|
|
char Header[256];
|
|
int Len;
|
|
|
|
Len = sprintf(Header, "Written by BPQ32 ver: %s <adif_ver:5>2.2.6<eoh>\r\n", TextVerstring);
|
|
fwrite(Header, 1, Len, Handle);
|
|
}
|
|
|
|
// Extract Info we need
|
|
|
|
// Distance and Bearing
|
|
|
|
if (LOC[0] && ADIF->LOC[0])
|
|
{
|
|
FromLOC(LOC, &myLat, &myLon);
|
|
FromLOC(ADIF->LOC, &Lat, &Lon);
|
|
|
|
Dist = (int)Distance(myLat, myLon, Lat, Lon, 0);
|
|
intBearing = (int)Bearing(Lat, Lon, myLat, myLon);
|
|
}
|
|
|
|
starttm = gmtime(&ADIF->StartTime);
|
|
|
|
//<call:6>VA2ROR
|
|
|
|
fprintf(Handle, "<call:%d>%s", (int)strlen(ADIF->Call), ADIF->Call);
|
|
|
|
//<qso_date:8>20190201
|
|
//<time_on:6>140801
|
|
//<time_off:6>140958
|
|
|
|
fprintf(Handle, "<qso_date:8>%04d%02d%02d<time_on:6>%02d%02d%02d<time_off:6>%02d%02d%02d",
|
|
starttm->tm_year + 1900, starttm->tm_mon + 1, starttm->tm_mday,
|
|
starttm->tm_hour, starttm->tm_min, starttm->tm_sec,
|
|
endtm.tm_hour, endtm.tm_min, endtm.tm_sec);
|
|
|
|
//<mode:0>
|
|
|
|
if (ADIFModes[ADIF->Mode][0])
|
|
{
|
|
// Send Mode, and Submode if present
|
|
|
|
char Mode[32];
|
|
char * SubMode;
|
|
|
|
strcpy(Mode, ADIFModes[ADIF->Mode]);
|
|
SubMode = strlop(Mode, '/');
|
|
|
|
fprintf(Handle, "<mode:%d>%s", (int)strlen(Mode), Mode);
|
|
if (SubMode)
|
|
fprintf(Handle, "<submode:%d>%s", (int)strlen(SubMode), SubMode);
|
|
|
|
}
|
|
//<gridsquare:6>JG28DK. Doc wants it even if empty
|
|
|
|
fprintf(Handle, "<gridsquare:%d>%s", (int)strlen(ADIF->LOC), ADIF->LOC);
|
|
|
|
|
|
//<band:3>20M
|
|
//<freq:9>14.109000
|
|
|
|
if (ADIF->Freq > 1500)
|
|
{
|
|
char Freqstr[32];
|
|
int i = 0;
|
|
BandLimits * Band = &Bands[0];
|
|
double Freq = ADIF->Freq / 1000000.0;
|
|
|
|
while (i++ < FreqCount)
|
|
{
|
|
if (Band->lower <= Freq && Band->upper >= Freq)
|
|
break;
|
|
|
|
Band++;
|
|
}
|
|
|
|
sprintf(Freqstr, "%.6f", ADIF->Freq / 1000000.0);
|
|
fprintf(Handle, "<band:%d>%s", (int)strlen(Band->Name), Band->Name);
|
|
fprintf(Handle, "<freq:%d>%s", (int)strlen(Freqstr), Freqstr);
|
|
}
|
|
else
|
|
fprintf(Handle, "<band:0><freq:0>");
|
|
|
|
|
|
|
|
|
|
|
|
// Do Comment
|
|
|
|
//0|2019-02-01 14:09:58|RMS Trimode|1.3.21.0|CMS|ZS1RS|VA2ROR|[AirMail-3.5.036-B2FHIM$]
|
|
//|Pactor 3|14109000|1957|311|FQ|1|0|4098|93|117|JG28DK|JF96HD
|
|
|
|
sprintf (Date, "%04d-%02d-%02d %02d:%02d:%02d",
|
|
endtm.tm_year + 1900, endtm.tm_mon + 1, endtm.tm_mday, endtm.tm_hour, endtm.tm_min, endtm.tm_sec);
|
|
|
|
CommentLen = sprintf(Comment, "0|%s|%s|%s|%s|%s|%s|%s|%s|%lld|%d|%d|%s|%d|%d|%d|%d|%d|%s|%s",
|
|
Date,
|
|
"BPQ32",
|
|
TextVerstring,
|
|
"CMS",
|
|
ADIF->CMSCall,
|
|
ADIF->Call,
|
|
ADIF->UserSID,
|
|
WL2KModes[ADIF->Mode],
|
|
ADIF->Freq,
|
|
Dist,
|
|
intBearing,
|
|
ADIF->Termination,
|
|
ADIF->Sent,
|
|
ADIF->Received,
|
|
ADIF->BytesSent,
|
|
ADIF->BytesReceived,
|
|
(int)(T - ADIF->StartTime),
|
|
ADIF->LOC,
|
|
LOC);
|
|
|
|
fprintf(Handle, "<comment:%d>%s", CommentLen, Comment);
|
|
fprintf(Handle, "<eor>\r\n");
|
|
fclose(Handle);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
VOID ADIFWriteFreqList()
|
|
{
|
|
// Write info needed for RMS Analyser to a file in similar format to Trimode
|
|
|
|
UCHAR Value[MAX_PATH];
|
|
FILE * Handle;
|
|
struct WL2KInfo * WL2KReport = WL2KReports;
|
|
char Locator[16];
|
|
char Call[16];
|
|
int i, freqCount = 0;
|
|
long long Freqs[100] = {0};
|
|
|
|
if (WL2KReport == NULL)
|
|
return;
|
|
|
|
if (LogDirectory[0] == 0)
|
|
{
|
|
strcpy(Value, "logs/BPQAnalyser.ini");
|
|
}
|
|
else
|
|
{
|
|
strcpy(Value, LogDirectory);
|
|
strcat(Value, "/");
|
|
strcat(Value, "logs/BPQAnalyser.ini");
|
|
}
|
|
|
|
Handle = fopen(Value, "wb");
|
|
|
|
if (Handle == NULL)
|
|
return;
|
|
|
|
while (WL2KReport)
|
|
{
|
|
strcpy(Call, WL2KReport->BaseCall);
|
|
strcpy(Locator, WL2KReport->GridSquare);
|
|
|
|
// if freq not in list add it
|
|
|
|
i = 0;
|
|
|
|
while (i < freqCount)
|
|
{
|
|
if (WL2KReport->Freq == Freqs[i])
|
|
break;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
if (i == freqCount)
|
|
Freqs[freqCount++] = WL2KReport->Freq;
|
|
|
|
|
|
WL2KReport = WL2KReport->Next;
|
|
}
|
|
|
|
fprintf(Handle, "[Site Properties]\r\n");
|
|
fprintf(Handle, "Grid Square=%s\r\n", Locator);
|
|
fprintf(Handle, "[Registration]\r\n");
|
|
fprintf(Handle, "Base Callsign=%s\r\n", Call);
|
|
|
|
fprintf(Handle, "[Channels]\r\n");
|
|
|
|
|
|
for (i = 0; i < freqCount; i++)
|
|
fprintf(Handle, "Frequency %d=%lld\r\n" , i + 1, Freqs[i]);
|
|
|
|
fclose(Handle);
|
|
}
|