Skip to content

Commit 1053bc4

Browse files
committed
Better duplicate checking for IGate and debug option to see what is going on.
1 parent 1376c0c commit 1053bc4

15 files changed

+663
-173
lines changed

Makefile.linux

+11-10
Original file line numberDiff line numberDiff line change
@@ -584,29 +584,30 @@ check : dtest ttest tttexttest pftest tlmtest lltest enctest kisstest check-mode
584584
# Can we encode and decode at popular data rates?
585585

586586
check-modem1200 : gen_packets atest
587-
gen_packets -n 100 -o /tmp/test1.wav
588-
atest -F0 -PE -L70 -G71 /tmp/test1.wav
589-
atest -F1 -PE -L73 -G75 /tmp/test1.wav
587+
./gen_packets -n 100 -o /tmp/test1.wav
588+
./atest -F0 -PE -L70 -G71 /tmp/test1.wav
589+
./atest -F1 -PE -L73 -G75 /tmp/test1.wav
590590
#rm /tmp/test1.wav
591591

592592
check-modem300 : gen_packets atest
593-
gen_packets -B300 -n 100 -o /tmp/test3.wav
594-
atest -B300 -F0 -L68 -G69 /tmp/test3.wav
595-
atest -B300 -F1 -L73 -G75 /tmp/test3.wav
593+
./gen_packets -B300 -n 100 -o /tmp/test3.wav
594+
./atest -B300 -F0 -L68 -G69 /tmp/test3.wav
595+
./atest -B300 -F1 -L73 -G75 /tmp/test3.wav
596596
rm /tmp/test3.wav
597597

598598
check-modem9600 : gen_packets atest
599-
gen_packets -B9600 -n 100 -o /tmp/test9.wav
600-
atest -B9600 -F0 -L57 -G59 /tmp/test9.wav
601-
atest -B9600 -F1 -L66 -G67 /tmp/test9.wav
599+
./gen_packets -B9600 -n 100 -o /tmp/test9.wav
600+
./atest -B9600 -F0 -L57 -G59 /tmp/test9.wav
601+
./atest -B9600 -F1 -L66 -G67 /tmp/test9.wav
602602
rm /tmp/test9.wav
603603

604604

605605

606606
# Unit test for inner digipeater algorithm
607607

608608
.PHONY : dtest
609-
dtest : digipeater.c pfilter.o ax25_pad.o dedupe.o fcs_calc.o tq.o textcolor.o \
609+
dtest : digipeater.c dedupe.c \
610+
pfilter.o ax25_pad.o fcs_calc.o tq.o textcolor.o \
610611
decode_aprs.o dwgpsnmea.o dwgps.o dwgpsd.o serial_port.o latlong.o telemetry.o symbols.o tt_text.o misc.a
611612
$(CC) $(CFLAGS) -DDIGITEST -o $@ $^ $(LDFLAGS)
612613
./dtest

Makefile.win

+2-1
Original file line numberDiff line numberDiff line change
@@ -283,7 +283,8 @@ atest9 : atest.c demod.c dsp.c demod_afsk.c demod_9600.c hdlc_rec.c hdlc_rec2.c
283283
# Unit test for inner digipeater algorithm
284284

285285
.PHONY: dtest
286-
dtest : digipeater.c pfilter.o ax25_pad.o dedupe.o fcs_calc.o tq.o textcolor.o \
286+
dtest : digipeater.c dedupe.c \
287+
pfilter.o ax25_pad.o fcs_calc.o tq.o textcolor.o \
287288
decode_aprs.o dwgpsnmea.o dwgps.o serial_port.o latlong.o telemetry.o symbols.o tt_text.o misc.a regex.a
288289
$(CC) $(CFLAGS) -DDIGITEST -o $@ $^
289290
./dtest

ax25_pad.c

+153-21
Original file line numberDiff line numberDiff line change
@@ -476,7 +476,7 @@ packet_t ax25_from_text (char *monitor, int strict)
476476
return (NULL);
477477
}
478478

479-
if ( ! ax25_parse_addr (pa, strict, atemp, &ssid_temp, &heard_temp)) {
479+
if ( ! ax25_parse_addr (AX25_SOURCE, pa, strict, atemp, &ssid_temp, &heard_temp)) {
480480
text_color_set(DW_COLOR_ERROR);
481481
dw_printf ("Failed to create packet from text. Bad source address\n");
482482
ax25_delete (this_p);
@@ -500,7 +500,7 @@ packet_t ax25_from_text (char *monitor, int strict)
500500
return (NULL);
501501
}
502502

503-
if ( ! ax25_parse_addr (pa, strict, atemp, &ssid_temp, &heard_temp)) {
503+
if ( ! ax25_parse_addr (AX25_DESTINATION, pa, strict, atemp, &ssid_temp, &heard_temp)) {
504504
text_color_set(DW_COLOR_ERROR);
505505
dw_printf ("Failed to create packet from text. Bad destination address\n");
506506
ax25_delete (this_p);
@@ -524,7 +524,7 @@ packet_t ax25_from_text (char *monitor, int strict)
524524

525525
// JWL 10:38 this_p->num_addr++;
526526

527-
if ( ! ax25_parse_addr (pa, strict, atemp, &ssid_temp, &heard_temp)) {
527+
if ( ! ax25_parse_addr (k, pa, strict, atemp, &ssid_temp, &heard_temp)) {
528528
text_color_set(DW_COLOR_ERROR);
529529
dw_printf ("Failed to create packet from text. Bad digipeater address\n");
530530
ax25_delete (this_p);
@@ -582,12 +582,9 @@ packet_t ax25_from_frame_debug (unsigned char *fbuf, int flen, alevel_t alevel,
582582
packet_t ax25_from_frame (unsigned char *fbuf, int flen, alevel_t alevel)
583583
#endif
584584
{
585-
//unsigned char *pf;
586585
packet_t this_p;
587586

588-
//int a;
589-
//int addr_bytes;
590-
587+
591588
/*
592589
* First make sure we have an acceptable length:
593590
*
@@ -687,7 +684,10 @@ packet_t ax25_dup (packet_t copy_from)
687684
*
688685
* Purpose: Parse address with optional ssid.
689686
*
690-
* Inputs: in_addr - Input such as "WB2OSZ-15*"
687+
* Inputs: position - AX25_DESTINATION, AX25_SOURCE, AX25_REPEATER_1...
688+
* Used for more specific error message. -1 if not used.
689+
*
690+
* in_addr - Input such as "WB2OSZ-15*"
691691
*
692692
* strict - TRUE for strict checking (6 characters, no lower case,
693693
* SSID must be in range of 0 to 15).
@@ -710,8 +710,12 @@ packet_t ax25_dup (packet_t copy_from)
710710
*
711711
*------------------------------------------------------------------------------*/
712712

713+
static const char *position_name[1 + AX25_MAX_ADDRS] = {
714+
"", "Destination ", "Source ",
715+
"Digi1 ", "Digi2 ", "Digi3 ", "Digi4 ",
716+
"Digi5 ", "Digi6 ", "Digi7 ", "Digi8 " };
713717

714-
int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, int *out_heard)
718+
int ax25_parse_addr (int position, char *in_addr, int strict, char *out_addr, int *out_ssid, int *out_heard)
715719
{
716720
char *p;
717721
char sstr[8]; /* Should be 1 or 2 digits for SSID. */
@@ -722,22 +726,33 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
722726
*out_ssid = 0;
723727
*out_heard = 0;
724728

729+
if (strict && strlen(in_addr) >= 2 && strncmp(in_addr, "qA", 2) == 0) {
730+
731+
text_color_set(DW_COLOR_ERROR);
732+
dw_printf ("%sAddress \"%s\" is a \"q-construct\" used for communicating\n", position_name[position], in_addr);
733+
dw_printf ("with APRS Internet Servers. It was not expected here.\n");
734+
}
735+
725736
//dw_printf ("ax25_parse_addr in: %s\n", in_addr);
726737

738+
if (position < -1) position = -1;
739+
if (position > AX25_REPEATER_8) position = AX25_REPEATER_8;
740+
position++; /* Adjust for position_name above. */
741+
727742
maxlen = strict ? 6 : (AX25_MAX_ADDR_LEN-1);
728743
p = in_addr;
729744
i = 0;
730745
for (p = in_addr; isalnum(*p); p++) {
731746
if (i >= maxlen) {
732747
text_color_set(DW_COLOR_ERROR);
733-
dw_printf ("Address is too long. \"%s\" has more than %d characters.\n", in_addr, maxlen);
748+
dw_printf ("%sAddress is too long. \"%s\" has more than %d characters.\n", position_name[position], in_addr, maxlen);
734749
return 0;
735750
}
736751
out_addr[i++] = *p;
737752
out_addr[i] = '\0';
738753
if (strict && islower(*p)) {
739754
text_color_set(DW_COLOR_ERROR);
740-
dw_printf ("Address has lower case letters. \"%s\" must be all upper case.\n", in_addr);
755+
dw_printf ("%sAddress has lower case letters. \"%s\" must be all upper case.\n", position_name[position], in_addr);
741756
return 0;
742757
}
743758
}
@@ -748,21 +763,21 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
748763
for (p++; isalnum(*p); p++) {
749764
if (j >= 2) {
750765
text_color_set(DW_COLOR_ERROR);
751-
dw_printf ("SSID is too long. SSID part of \"%s\" has more than 2 characters.\n", in_addr);
766+
dw_printf ("%sSSID is too long. SSID part of \"%s\" has more than 2 characters.\n", position_name[position], in_addr);
752767
return 0;
753768
}
754769
sstr[j++] = *p;
755770
sstr[j] = '\0';
756771
if (strict && ! isdigit(*p)) {
757772
text_color_set(DW_COLOR_ERROR);
758-
dw_printf ("SSID must be digits. \"%s\" has letters in SSID.\n", in_addr);
773+
dw_printf ("%sSSID must be digits. \"%s\" has letters in SSID.\n", position_name[position], in_addr);
759774
return 0;
760775
}
761776
}
762777
k = atoi(sstr);
763778
if (k < 0 || k > 15) {
764779
text_color_set(DW_COLOR_ERROR);
765-
dw_printf ("SSID out of range. SSID of \"%s\" not in range of 0 to 15.\n", in_addr);
780+
dw_printf ("%sSSID out of range. SSID of \"%s\" not in range of 0 to 15.\n", position_name[position], in_addr);
766781
return 0;
767782
}
768783
*out_ssid = k;
@@ -775,7 +790,7 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
775790

776791
if (*p != '\0') {
777792
text_color_set(DW_COLOR_ERROR);
778-
dw_printf ("Invalid character \"%c\" found in address \"%s\".\n", *p, in_addr);
793+
dw_printf ("Invalid character \"%c\" found in %saddress \"%s\".\n", *p, position_name[position], in_addr);
779794
return 0;
780795
}
781796

@@ -786,6 +801,73 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
786801
} /* end ax25_parse_addr */
787802

788803

804+
/*-------------------------------------------------------------------
805+
*
806+
* Name: ax25_check_addresses
807+
*
808+
* Purpose: Check addresses of given packet and print message if any issues.
809+
* We call this when receiving and transmitting.
810+
*
811+
* Inputs: pp - packet object pointer.
812+
*
813+
* Errors: Print error message.
814+
*
815+
* Returns: 1 for all valid. 0 if not.
816+
*
817+
* Examples: I was surprised to get this from an APRS-IS server with
818+
* a lower case source address.
819+
*
820+
* n1otx>APRS,TCPIP*,qAC,THIRD:@141335z4227.48N/07111.73W_348/005g014t044r000p000h60b10075.wview_5_20_2
821+
*
822+
* I haven't gotten to the bottom of this yet but it sounds
823+
* like "q constructs" are somehow getting on to the air when
824+
* they should only appear in conversations with IGate servers.
825+
*
826+
* https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/topics/678
827+
*
828+
* WB0VGI-7>APDW12,W0YC-5*,qAR,AE0RF-10:}N0DZQ-10>APWW10,TCPIP,WB0VGI-7*:;145.230MN*080306z4607.62N/09230.58WrKE0ACL/R 145.230- T146.2 (Pine County ARES)
829+
*
830+
* Typical result:
831+
*
832+
* Digipeater WIDE2 (probably N3LEE-4) audio level = 28(10/6) [NONE] __|||||||
833+
* [0.5] VE2DJE-9>P_0_P?,VE2PCQ-3,K1DF-7,N3LEE-4,WIDE2*:'{S+l <0x1c>>/
834+
* Invalid character "_" in MIC-E destination/latitude.
835+
* Invalid character "_" in MIC-E destination/latitude.
836+
* Invalid character "?" in MIC-E destination/latitude.
837+
* Invalid MIC-E N/S encoding in 4th character of destination.
838+
* Invalid MIC-E E/W encoding in 6th character of destination.
839+
* MIC-E, normal car (side view), Unknown manufacturer, Returning
840+
* N 00 00.0000, E 005 55.1500, 0 MPH
841+
* Invalid character "_" found in Destination address "P_0_P?".
842+
*
843+
* *** The origin and journey of this packet should receive some scrutiny. ***
844+
*
845+
*--------------------------------------------------------------------*/
846+
847+
int ax25_check_addresses (packet_t pp)
848+
{
849+
int n;
850+
char addr[AX25_MAX_ADDR_LEN];
851+
char ignore1[AX25_MAX_ADDR_LEN];
852+
int ignore2, ignore3;
853+
int all_ok = 1;
854+
855+
for (n = 0; n < ax25_get_num_addr(pp); n++) {
856+
ax25_get_addr_with_ssid (pp, n, addr);
857+
all_ok &= ax25_parse_addr (n, addr, 1, ignore1, &ignore2, &ignore3);
858+
}
859+
860+
if (! all_ok) {
861+
text_color_set(DW_COLOR_ERROR);
862+
dw_printf ("\n");
863+
dw_printf ("*** The origin and journey of this packet should receive some scrutiny. ***\n");
864+
dw_printf ("\n");
865+
}
866+
867+
return (all_ok);
868+
} /* end ax25_check_addresses */
869+
870+
789871
/*------------------------------------------------------------------------------
790872
*
791873
* Name: ax25_unwrap_third_party
@@ -862,7 +944,7 @@ void ax25_set_addr (packet_t this_p, int n, char *ad)
862944
/*
863945
* Set existing address position.
864946
*/
865-
ax25_parse_addr (ad, 0, atemp, &ssid_temp, &heard_temp);
947+
ax25_parse_addr (n, ad, 0, atemp, &ssid_temp, &heard_temp);
866948

867949
memset (this_p->frame_data + n*7, ' ' << 1, 6);
868950

@@ -951,7 +1033,11 @@ void ax25_insert_addr (packet_t this_p, int n, char *ad)
9511033

9521034
SET_LAST_ADDR_FLAG;
9531035

954-
ax25_parse_addr (ad, 0, atemp, &ssid_temp, &heard_temp);
1036+
// Why aren't we setting 'strict' here?
1037+
// Messages from IGate have q-constructs.
1038+
// We use this to parse it and later remove unwanted parts.
1039+
1040+
ax25_parse_addr (n, ad, 0, atemp, &ssid_temp, &heard_temp);
9551041
memset (this_p->frame_data + n*7, ' ' << 1, 6);
9561042
for (i=0; i<6 && atemp[i] != '\0'; i++) {
9571043
this_p->frame_data[n*7+i] = atemp[i] << 1;
@@ -1160,8 +1246,9 @@ void ax25_get_addr_with_ssid (packet_t this_p, int n, char *station)
11601246
if (ssid != 0) {
11611247
snprintf (sstr, sizeof(sstr), "-%d", ssid);
11621248
strlcat (station, sstr, 10);
1163-
}
1164-
}
1249+
}
1250+
1251+
} /* end ax25_get_addr_with_ssid */
11651252

11661253

11671254
/*------------------------------------------------------------------------------
@@ -2064,6 +2151,27 @@ int ax25_get_pid (packet_t this_p)
20642151
* There is a very very small probability that two unrelated
20652152
* packets will result in the same checksum, and the
20662153
* undesired dropping of the packet.
2154+
*
2155+
* There is a 1 / 65536 chance of getting a false positive match
2156+
* which is good enough for this application.
2157+
* We could reduce that with a 32 bit CRC instead of reusing
2158+
* code from the AX.25 frame CRC calculation.
2159+
*
2160+
* Version 1.3: We exclude any trailing CR/LF at the end of the info part
2161+
* so we can detect duplicates that are received only over the
2162+
* air and those which have gone thru an IGate where the process
2163+
* removes any trailing CR/LF. Example:
2164+
*
2165+
* Original via RF only:
2166+
* W1TG-1>APU25N,N3LEE-10*,WIDE2-1:<IGATE,MSG_CNT=30,LOC_CNT=61<0x0d>
2167+
*
2168+
* When we get the same thing via APRS-IS:
2169+
* W1TG-1>APU25N,K1FFK,WIDE2*,qAR,WB2ZII-15:<IGATE,MSG_CNT=30,LOC_CNT=61
2170+
*
2171+
* (Actually there is a trailing space. Maybe some systems
2172+
* change control characters to space???)
2173+
* Hmmmm. I guess we should ignore trailing space as well for
2174+
* duplicate detection and suppression.
20672175
*
20682176
*------------------------------------------------------------------------------*/
20692177

@@ -2079,6 +2187,20 @@ unsigned short ax25_dedupe_crc (packet_t pp)
20792187
ax25_get_addr_with_ssid(pp, AX25_DESTINATION, dest);
20802188
info_len = ax25_get_info (pp, &pinfo);
20812189

2190+
while (info_len >= 1 && (pinfo[info_len-1] == '\r' ||
2191+
pinfo[info_len-1] == '\n' ||
2192+
pinfo[info_len-1] == ' ')) {
2193+
2194+
// Temporary for debugging!
2195+
2196+
// if (pinfo[info_len-1] == ' ') {
2197+
// text_color_set(DW_COLOR_ERROR);
2198+
// dw_printf ("DEBUG: ax25_dedupe_crc ignoring trailing space.\n");
2199+
// }
2200+
2201+
info_len--;
2202+
}
2203+
20822204
crc = 0xffff;
20832205
crc = crc16((unsigned char *)src, strlen(src), crc);
20842206
crc = crc16((unsigned char *)dest, strlen(dest), crc);
@@ -2159,6 +2281,11 @@ unsigned short ax25_m_m_crc (packet_t pp)
21592281
* as hexadecimal for troubleshooting? Maybe an option so the
21602282
* packet raw data is in hexadecimal but an extracted
21612283
* comment displays UTF-8? Or a command line option for only ASCII?
2284+
*
2285+
* Trailing space:
2286+
* I recently noticed a case where a packet has space character
2287+
* at the end. If the last character of the line is a space,
2288+
* this will be displayed in hexadecimal to make it obvious.
21622289
*
21632290
*------------------------------------------------------------------*/
21642291

@@ -2184,15 +2311,20 @@ void ax25_safe_print (char *pstr, int len, int ascii_only)
21842311
{
21852312
ch = *((unsigned char *)pstr);
21862313

2187-
if (ch < ' ' || ch == 0x7f || ch == 0xfe || ch == 0xff ||
2314+
if (ch == ' ' && (len == 1 || pstr[1] == '\0')) {
2315+
2316+
snprintf (safe_str + safe_len, sizeof(safe_str)-safe_len, "<0x%02x>", ch);
2317+
safe_len += 6;
2318+
}
2319+
else if (ch < ' ' || ch == 0x7f || ch == 0xfe || ch == 0xff ||
21882320
(ascii_only && ch >= 0x80) ) {
21892321

21902322
/* Control codes and delete. */
21912323
/* UTF-8 does not use fe and ff except in a possible */
21922324
/* "Byte Order Mark" (BOM) at the beginning. */
21932325

21942326
snprintf (safe_str + safe_len, sizeof(safe_str)-safe_len, "<0x%02x>", ch);
2195-
safe_len += 6;
2327+
safe_len += 6;
21962328
}
21972329
else {
21982330
/* Let everything else thru so we can handle UTF-8 */

ax25_pad.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -307,7 +307,8 @@ extern void ax25_delete (packet_t pp);
307307
#endif
308308

309309

310-
extern int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, int *out_heard);
310+
extern int ax25_parse_addr (int position, char *in_addr, int strict, char *out_addr, int *out_ssid, int *out_heard);
311+
extern int ax25_check_addresses (packet_t pp);
311312

312313
extern packet_t ax25_unwrap_third_party (packet_t from_pp);
313314

0 commit comments

Comments
 (0)