@@ -476,7 +476,7 @@ packet_t ax25_from_text (char *monitor, int strict)
476
476
return (NULL );
477
477
}
478
478
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 )) {
480
480
text_color_set (DW_COLOR_ERROR );
481
481
dw_printf ("Failed to create packet from text. Bad source address\n" );
482
482
ax25_delete (this_p );
@@ -500,7 +500,7 @@ packet_t ax25_from_text (char *monitor, int strict)
500
500
return (NULL );
501
501
}
502
502
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 )) {
504
504
text_color_set (DW_COLOR_ERROR );
505
505
dw_printf ("Failed to create packet from text. Bad destination address\n" );
506
506
ax25_delete (this_p );
@@ -524,7 +524,7 @@ packet_t ax25_from_text (char *monitor, int strict)
524
524
525
525
// JWL 10:38 this_p->num_addr++;
526
526
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 )) {
528
528
text_color_set (DW_COLOR_ERROR );
529
529
dw_printf ("Failed to create packet from text. Bad digipeater address\n" );
530
530
ax25_delete (this_p );
@@ -582,12 +582,9 @@ packet_t ax25_from_frame_debug (unsigned char *fbuf, int flen, alevel_t alevel,
582
582
packet_t ax25_from_frame (unsigned char * fbuf , int flen , alevel_t alevel )
583
583
#endif
584
584
{
585
- //unsigned char *pf;
586
585
packet_t this_p ;
587
586
588
- //int a;
589
- //int addr_bytes;
590
-
587
+
591
588
/*
592
589
* First make sure we have an acceptable length:
593
590
*
@@ -687,7 +684,10 @@ packet_t ax25_dup (packet_t copy_from)
687
684
*
688
685
* Purpose: Parse address with optional ssid.
689
686
*
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*"
691
691
*
692
692
* strict - TRUE for strict checking (6 characters, no lower case,
693
693
* SSID must be in range of 0 to 15).
@@ -710,8 +710,12 @@ packet_t ax25_dup (packet_t copy_from)
710
710
*
711
711
*------------------------------------------------------------------------------*/
712
712
713
+ static const char * position_name [1 + AX25_MAX_ADDRS ] = {
714
+ "" , "Destination " , "Source " ,
715
+ "Digi1 " , "Digi2 " , "Digi3 " , "Digi4 " ,
716
+ "Digi5 " , "Digi6 " , "Digi7 " , "Digi8 " };
713
717
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 )
715
719
{
716
720
char * p ;
717
721
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
722
726
* out_ssid = 0 ;
723
727
* out_heard = 0 ;
724
728
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
+
725
736
//dw_printf ("ax25_parse_addr in: %s\n", in_addr);
726
737
738
+ if (position < -1 ) position = -1 ;
739
+ if (position > AX25_REPEATER_8 ) position = AX25_REPEATER_8 ;
740
+ position ++ ; /* Adjust for position_name above. */
741
+
727
742
maxlen = strict ? 6 : (AX25_MAX_ADDR_LEN - 1 );
728
743
p = in_addr ;
729
744
i = 0 ;
730
745
for (p = in_addr ; isalnum (* p ); p ++ ) {
731
746
if (i >= maxlen ) {
732
747
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 );
734
749
return 0 ;
735
750
}
736
751
out_addr [i ++ ] = * p ;
737
752
out_addr [i ] = '\0' ;
738
753
if (strict && islower (* p )) {
739
754
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 );
741
756
return 0 ;
742
757
}
743
758
}
@@ -748,21 +763,21 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
748
763
for (p ++ ; isalnum (* p ); p ++ ) {
749
764
if (j >= 2 ) {
750
765
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 );
752
767
return 0 ;
753
768
}
754
769
sstr [j ++ ] = * p ;
755
770
sstr [j ] = '\0' ;
756
771
if (strict && ! isdigit (* p )) {
757
772
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 );
759
774
return 0 ;
760
775
}
761
776
}
762
777
k = atoi (sstr );
763
778
if (k < 0 || k > 15 ) {
764
779
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 );
766
781
return 0 ;
767
782
}
768
783
* out_ssid = k ;
@@ -775,7 +790,7 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
775
790
776
791
if (* p != '\0' ) {
777
792
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 );
779
794
return 0 ;
780
795
}
781
796
@@ -786,6 +801,73 @@ int ax25_parse_addr (char *in_addr, int strict, char *out_addr, int *out_ssid, i
786
801
} /* end ax25_parse_addr */
787
802
788
803
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
+
789
871
/*------------------------------------------------------------------------------
790
872
*
791
873
* Name: ax25_unwrap_third_party
@@ -862,7 +944,7 @@ void ax25_set_addr (packet_t this_p, int n, char *ad)
862
944
/*
863
945
* Set existing address position.
864
946
*/
865
- ax25_parse_addr (ad , 0 , atemp , & ssid_temp , & heard_temp );
947
+ ax25_parse_addr (n , ad , 0 , atemp , & ssid_temp , & heard_temp );
866
948
867
949
memset (this_p -> frame_data + n * 7 , ' ' << 1 , 6 );
868
950
@@ -951,7 +1033,11 @@ void ax25_insert_addr (packet_t this_p, int n, char *ad)
951
1033
952
1034
SET_LAST_ADDR_FLAG ;
953
1035
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 );
955
1041
memset (this_p -> frame_data + n * 7 , ' ' << 1 , 6 );
956
1042
for (i = 0 ; i < 6 && atemp [i ] != '\0' ; i ++ ) {
957
1043
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)
1160
1246
if (ssid != 0 ) {
1161
1247
snprintf (sstr , sizeof (sstr ), "-%d" , ssid );
1162
1248
strlcat (station , sstr , 10 );
1163
- }
1164
- }
1249
+ }
1250
+
1251
+ } /* end ax25_get_addr_with_ssid */
1165
1252
1166
1253
1167
1254
/*------------------------------------------------------------------------------
@@ -2064,6 +2151,27 @@ int ax25_get_pid (packet_t this_p)
2064
2151
* There is a very very small probability that two unrelated
2065
2152
* packets will result in the same checksum, and the
2066
2153
* 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.
2067
2175
*
2068
2176
*------------------------------------------------------------------------------*/
2069
2177
@@ -2079,6 +2187,20 @@ unsigned short ax25_dedupe_crc (packet_t pp)
2079
2187
ax25_get_addr_with_ssid (pp , AX25_DESTINATION , dest );
2080
2188
info_len = ax25_get_info (pp , & pinfo );
2081
2189
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
+
2082
2204
crc = 0xffff ;
2083
2205
crc = crc16 ((unsigned char * )src , strlen (src ), crc );
2084
2206
crc = crc16 ((unsigned char * )dest , strlen (dest ), crc );
@@ -2159,6 +2281,11 @@ unsigned short ax25_m_m_crc (packet_t pp)
2159
2281
* as hexadecimal for troubleshooting? Maybe an option so the
2160
2282
* packet raw data is in hexadecimal but an extracted
2161
2283
* 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.
2162
2289
*
2163
2290
*------------------------------------------------------------------*/
2164
2291
@@ -2184,15 +2311,20 @@ void ax25_safe_print (char *pstr, int len, int ascii_only)
2184
2311
{
2185
2312
ch = * ((unsigned char * )pstr );
2186
2313
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 ||
2188
2320
(ascii_only && ch >= 0x80 ) ) {
2189
2321
2190
2322
/* Control codes and delete. */
2191
2323
/* UTF-8 does not use fe and ff except in a possible */
2192
2324
/* "Byte Order Mark" (BOM) at the beginning. */
2193
2325
2194
2326
snprintf (safe_str + safe_len , sizeof (safe_str )- safe_len , "<0x%02x>" , ch );
2195
- safe_len += 6 ;
2327
+ safe_len += 6 ;
2196
2328
}
2197
2329
else {
2198
2330
/* Let everything else thru so we can handle UTF-8 */
0 commit comments