@@ -874,11 +874,15 @@ static int filt_t (pfstate_t *pf)
874
874
875
875
case 'p' : /* Position */
876
876
if (* infop == '!' ) return (1 );
877
- if (* infop == '\'' ) return (1 );
878
877
if (* infop == '/' ) return (1 );
879
878
if (* infop == '=' ) return (1 );
880
879
if (* infop == '@' ) return (1 );
881
- if (* infop == '`' ) return (1 );
880
+ if (* infop == '\'' ) return (1 ); // MIC-E
881
+ if (* infop == '`' ) return (1 ); // MIC-E
882
+
883
+ // What if we have "_" symbol code for weather?
884
+ // Still consider as position.
885
+ // The same packet can match more than one type here.
882
886
break ;
883
887
884
888
case 'o' : /* Object */
@@ -897,6 +901,11 @@ static int filt_t (pfstate_t *pf)
897
901
if (* infop == '?' ) return (1 );
898
902
break ;
899
903
904
+ case 'c' : /* station Capabilities - my extension */
905
+ /* Most often used for IGate statistics. */
906
+ if (* infop == '<' ) return (1 );
907
+ break ;
908
+
900
909
case 's' : /* Status */
901
910
if (* infop == '>' ) return (1 );
902
911
break ;
@@ -910,16 +919,32 @@ static int filt_t (pfstate_t *pf)
910
919
if (* infop == '{' ) return (1 );
911
920
break ;
912
921
922
+ case 'h' : /* third party Header - my extension */
923
+ if (* infop == '}' ) return (1 );
924
+ break ;
925
+
913
926
case 'w' : /* Weather */
914
- if (* infop == '@' ) return (1 );
915
- if (* infop == '*' ) return (1 );
916
- if (* infop == '_' ) return (1 );
927
+
928
+ if (* infop == '*' ) return (1 ); // Peet Bros
929
+ if (* infop == '_' ) return (1 ); // Weather report, no position.
930
+ if (strncmp (infop , "!!" , 2 ) == 0 ) return (1 ); // Ultimeter 2000.
917
931
918
932
/* '$' is normally raw GPS. Check for special case. */
933
+
919
934
if (strncmp (infop , "$ULTW" , 5 ) == 0 ) return (1 );
920
935
921
- /* TODO: Positions !=/@ can be weather. */
922
- /* Need to check for _ symbol. */
936
+ /* Positions !=/@ with symbol code _ are weather. */
937
+
938
+ if (strchr ("!=/@" , * infop ) != NULL &&
939
+ pf -> decoded .g_symbol_code == '_' ) return (1 );
940
+
941
+ /* Object with _ symbol is also weather. APRS protocol spec page 66. */
942
+
943
+ if (* infop == ';' &&
944
+ pf -> decoded .g_symbol_code == '_' ) return (1 );
945
+
946
+ // TODO: need more test cases at end for new weather cases.
947
+
923
948
break ;
924
949
925
950
case 'n' : /* NWS format */
@@ -1059,76 +1084,194 @@ static int filt_r (pfstate_t *pf, char *sdist)
1059
1084
*
1060
1085
* Description:
1061
1086
*
1087
+ * s/pri
1088
+ * s/pri/alt
1089
+ * s/pri/alt/
1090
+ * s/pri/alt/over
1091
+ *
1062
1092
* "pri" is zero or more symbols from the primary symbol set.
1093
+ * Symbol codes are any printable ASCII character other than | or ~.
1094
+ * (Zero symbols here would be sensible only if later alt part is specified.)
1063
1095
* "alt" is one or more symbols from the alternate symbol set.
1064
- * "over" is overlay characters. Overlays apply only to the alternate symbol set.
1096
+ * "over" is overlay characters for the alternate symbol set.
1097
+ * Only upper case letters, digits, and \ are allowed here.
1098
+ * If the last part is not specified, any overlay or lack of overlay, is ignored.
1099
+ * If the last part is specified, only the listed overlays will match.
1100
+ * An explicit lack of overlay is represented by the \ character.
1065
1101
*
1066
1102
* Examples:
1067
- * s/-> Allow house and car from primary symbol table.
1068
- * s//# Allow alternate table digipeater, with or without overlay.
1069
- * s//#/\ Allow alternate table digipeater, only if no overlay.
1070
- * s//#/SL1 Allow alternate table digipeater, with overlay S, L, or 1
1103
+ * s/O Balloon.
1104
+ * s/-> House or car from primary symbol table.
1105
+ *
1106
+ * s//# Alternate table digipeater, with or without overlay.
1107
+ * s//#/\ Alternate table digipeater, only if no overlay.
1108
+ * s//#/SL1 Alternate table digipeater, with overlay S, L, or 1.
1109
+ * s//#/SL\ Alternate table digipeater, with S, L, or no overlay.
1110
+ *
1111
+ * s/s/s Any variation of watercraft. Either symbol table. With or without overlay.
1112
+ * s/s/s/ Ship or ship sideview, only if no overlay.
1113
+ * s//s/J Jet Ski.
1114
+ *
1115
+ * What if you want to use the / symbol when / is being used as a delimiter here? Recall that you
1116
+ * can use some other special character after the initial lower case letter and this becomes the
1117
+ * delimiter for the rest of the specification.
1118
+ *
1119
+ * Examples:
1120
+ *
1121
+ * s:/ Red Dot.
1122
+ * s::/ Waypoint Destination, with or without overlay.
1123
+ * s:/:/ Either Red Dot or Waypoint Destination.
1124
+ * s:/:/: Either Red Dot or Waypoint Destination, no overlay.
1125
+ *
1126
+ * Bad example:
1127
+ *
1128
+ * Someone tried using this to include ballons: s/'/O/-/#/_
1129
+ * probably following the buddy filter pattern of / between each alternative.
1130
+ * There should be an error message because it has more than 3 delimiter characters.
1131
+ *
1071
1132
*
1072
1133
*------------------------------------------------------------------------------*/
1073
1134
1074
1135
static int filt_s (pfstate_t * pf )
1075
1136
{
1076
1137
char str [MAX_TOKEN_LEN ];
1077
1138
char * cp ;
1078
- char sep [2 ];
1079
- char * pri , * alt , * over ;
1139
+ char sep [2 ]; // Delimiter character. Typically / but it could be different.
1140
+ char * pri = NULL , * alt = NULL , * over = NULL , * extra = NULL ;
1141
+ char * x ;
1080
1142
1081
1143
1082
1144
strlcpy (str , pf -> token_str , sizeof (str ));
1083
1145
sep [0 ] = str [1 ];
1084
1146
sep [1 ] = '\0' ;
1085
1147
cp = str + 2 ;
1086
1148
1087
- // TODO: check here.
1149
+
1150
+ // First, separate the parts and do a strict syntax check.
1088
1151
1089
1152
pri = strsep (& cp , sep );
1090
1153
1091
- if (pri == NULL ) {
1154
+ if (pri != NULL ) {
1155
+
1156
+ // Zero length is acceptable if alternate symbol(s) specified. Will check that later.
1157
+
1158
+ for (x = pri ; * x != '\0' ; x ++ ) {
1159
+ if ( ! isprint (* x ) || * x == '|' || * x == '~' ) {
1160
+ print_error (pf , "Symbol filter, primary must be printable ASCII character(s) other than | or ~." );
1161
+ return (-1 );
1162
+ }
1163
+ }
1164
+
1165
+ alt = strsep (& cp , sep );
1166
+
1167
+ if (alt != NULL ) {
1168
+
1169
+ // Zero length after second / would be pointless.
1170
+
1171
+ if (strlen (alt ) == 0 ) {
1172
+ print_error (pf , "Nothing specified for alternate symbol table." );
1173
+ return (-1 );
1174
+ }
1175
+
1176
+ for (x = alt ; * x != '\0' ; x ++ ) {
1177
+ if ( ! isprint (* x ) || * x == '|' || * x == '~' ) {
1178
+ print_error (pf , "Symbol filter, alternate must be printable ASCII character(s) other than | or ~." );
1179
+ return (-1 );
1180
+ }
1181
+ }
1182
+
1183
+ over = strsep (& cp , sep );
1184
+
1185
+ if (over != NULL ) {
1186
+
1187
+ // Zero length is acceptable and is not the same as missing.
1188
+
1189
+ for (x = over ; * x != '\0' ; x ++ ) {
1190
+ if ( (! isupper (* x )) && (! isdigit (* x )) && * x != '\\' ) {
1191
+ print_error (pf , "Symbol filter, overlay must be upper case letter, digit, or \\." );
1192
+ return (-1 );
1193
+ }
1194
+ }
1195
+
1196
+ extra = strsep (& cp , sep );
1197
+
1198
+ if (extra != NULL ) {
1199
+ print_error (pf , "More than 3 delimiter characters in Symbol filter." );
1200
+ return (-1 );
1201
+ }
1202
+ }
1203
+ }
1204
+ else {
1205
+ // No alt part is OK if at least one primary symbol was specified.
1206
+ if (strlen (pri ) == 0 ) {
1207
+ print_error (pf , "No symbols specified for Symbol filter." );
1208
+ return (-1 );
1209
+ }
1210
+ }
1211
+ }
1212
+ else {
1092
1213
print_error (pf , "Missing arguments for Symbol filter." );
1093
1214
return (-1 );
1094
1215
}
1095
1216
1096
- if (pf -> decoded .g_symbol_table == '/' && strchr (pri , pf -> decoded .g_symbol_code ) != NULL ) {
1097
- /* Found in primary symbols. All done. */
1098
- return (1 );
1217
+
1218
+ // This applies only for Position, Object, Item.
1219
+ // decode_aprs() should set symbol code to space to mean undefined.
1220
+
1221
+ if (pf -> decoded .g_symbol_code == ' ' ) {
1222
+ return (0 );
1223
+ }
1224
+
1225
+
1226
+ // Look for Primary symbols.
1227
+
1228
+ if (pf -> decoded .g_symbol_table == '/' ) {
1229
+ if (pri != NULL && strlen (pri ) > 0 ) {
1230
+ return (strchr (pri , pf -> decoded .g_symbol_code ) != NULL );
1231
+ }
1099
1232
}
1100
1233
1101
- alt = strsep (& cp , sep );
1102
1234
if (alt == NULL ) {
1103
1235
return (0 );
1104
1236
}
1105
- if (strlen (alt ) == 0 ) {
1106
- /* We have s/.../ */
1107
- print_error (pf , "Missing alternate symbols for Symbol filter." );
1108
- return (-1 );
1109
- }
1110
1237
1111
1238
//printf ("alt=\"%s\" sym='%c'\n", alt, pf->decoded.g_symbol_code);
1112
1239
1113
- if (strchr (alt , pf -> decoded .g_symbol_code ) == NULL ) {
1114
- /* Not found in alternate symbols. Reject. */
1115
- return (0 );
1116
- }
1240
+ // Look for Alternate symbols.
1117
1241
1118
- over = strsep (& cp , sep );
1119
- if (over == NULL ) {
1120
- /* alternate, with or without overlay. */
1121
- return (pf -> decoded .g_symbol_table != '/' );
1122
- }
1242
+ if (strchr (alt , pf -> decoded .g_symbol_code ) != NULL ) {
1123
1243
1124
- // printf ("over=\"%s\" table='%c'\n", over, pf->decoded.g_symbol_table);
1244
+ // We have a match but that might not be enough.
1245
+ // We must see if there was an overlay part specified.
1125
1246
1126
- if (strlen (over ) == 0 ) {
1127
- return (pf -> decoded .g_symbol_table == '\\' );
1247
+ if (over != NULL ) {
1248
+
1249
+ if (strlen (over ) > 0 ) {
1250
+
1251
+ // Non-zero length overlay part was specified.
1252
+ // Need to match one of them.
1253
+
1254
+ return (strchr (over , pf -> decoded .g_symbol_table ) != NULL );
1255
+ }
1256
+ else {
1257
+
1258
+ // Zero length overlay part was specified.
1259
+ // We must have no overlay, i.e. table is \.
1260
+
1261
+ return (pf -> decoded .g_symbol_table == '\\' );
1262
+ }
1263
+ }
1264
+ else {
1265
+
1266
+ // No check of overlay part. Just make sure it is not primary table.
1267
+
1268
+ return (pf -> decoded .g_symbol_table != '/' );
1269
+ }
1128
1270
}
1129
1271
1130
- return (strchr (over , pf -> decoded .g_symbol_table ) != NULL );
1131
- }
1272
+ return (0 );
1273
+
1274
+ } /* end filt_s */
1132
1275
1133
1276
1134
1277
/*------------------------------------------------------------------------------
@@ -1494,10 +1637,15 @@ int main ()
1494
1637
pftest (126 , "t/n" , "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#" , 1 );
1495
1638
pftest (127 , "t/" , "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#" , 0 );
1496
1639
1497
- pftest (130 , "r/42.6/-71.3/10" , "WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1498
- pftest (131 , "r/42.6/-71.3/10" , "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32" , 0 );
1640
+ pftest (128 , "t/c" , "S0RCE>DEST:<stationcapabilities" , 1 );
1641
+ pftest (129 , "t/h" , "S0RCE>DEST:<stationcapabilities" , 0 );
1642
+ pftest (130 , "t/h" , "S0RCE>DEST:}thirdpartyheaderwhatever" , 1 );
1643
+ pftest (131 , "t/c" , "S0RCE>DEST:}thirdpartyheaderwhatever" , 0 );
1644
+
1645
+ pftest (140 , "r/42.6/-71.3/10" , "WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1646
+ pftest (141 , "r/42.6/-71.3/10" , "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32" , 0 );
1499
1647
1500
- pftest (140 , "( t/t & b/WB2OSZ ) | ( t/o & ! r/42.6/-71.3/1 )" , "WB2OSZ>APDW12:;home *111111z4237.14N/07120.83W-Chelmsford MA" , 1 );
1648
+ pftest (145 , "( t/t & b/WB2OSZ ) | ( t/o & ! r/42.6/-71.3/1 )" , "WB2OSZ>APDW12:;home *111111z4237.14N/07120.83W-Chelmsford MA" , 1 );
1501
1649
1502
1650
pftest (150 , "s/->" , "WB2OSZ-5>APDW12:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1503
1651
pftest (151 , "s/->" , "WB2OSZ-5>APDW12:!4237.14N/07120.83W-PHG7140Chelmsford MA" , 1 );
@@ -1515,13 +1663,28 @@ int main ()
1515
1663
pftest (160 , "s//#/LS1" , "WB2OSZ-5>APDW12:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1516
1664
pftest (161 , "s//#/LS1" , "WB2OSZ-5>APDW12:!4237.14N\\07120.83W#PHG7140Chelmsford MA" , 0 );
1517
1665
pftest (162 , "s//#/LS1" , "WB2OSZ-5>APDW12:!4237.14N/07120.83W#PHG7140Chelmsford MA" , 0 );
1666
+ pftest (163 , "s//#/LS\\" , "WB2OSZ-5>APDW12:!4237.14N\\07120.83W#PHG7140Chelmsford MA" , 1 );
1518
1667
1519
- pftest (170 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1520
- pftest (171 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1*,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1521
- pftest (172 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1522
- pftest (173 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3*,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1523
- pftest (174 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4*:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1524
- pftest (175 , "v/DIGI9/DIGI2" , "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1668
+ pftest (170 , "s:/" , "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA" , 1 );
1669
+ pftest (171 , "s:/" , "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA" , 0 );
1670
+ pftest (172 , "s::/" , "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA" , 0 );
1671
+ pftest (173 , "s::/" , "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA" , 1 );
1672
+ pftest (174 , "s:/:/" , "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA" , 1 );
1673
+ pftest (175 , "s:/:/" , "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA" , 1 );
1674
+ pftest (176 , "s:/:/" , "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA" , 1 );
1675
+ pftest (177 , "s:/:/:X" , "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA" , 1 );
1676
+
1677
+ // FIXME: Different on Windows and 64 bit Linux.
1678
+ //pftest (178, "s:/:/:", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1);
1679
+
1680
+ pftest (179 , "s:/:/:\\" , "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA" , 0 );
1681
+
1682
+ pftest (180 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1683
+ pftest (181 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1*,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1684
+ pftest (182 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 1 );
1685
+ pftest (183 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3*,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1686
+ pftest (184 , "v/DIGI2/DIGI3" , "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4*:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1687
+ pftest (185 , "v/DIGI9/DIGI2" , "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA" , 0 );
1525
1688
1526
1689
/* Test error reporting. */
1527
1690
@@ -1540,12 +1703,23 @@ int main ()
1540
1703
pftest (226 , "i/30/8/" , "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001" , 1 );
1541
1704
pftest (227 , "i/30/8" , "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001" , 1 );
1542
1705
1543
- // FIXME: behaves differently on Windows and Linux
1706
+ // FIXME: behaves differently on Windows and Linux. Why?
1707
+ // On Windows we have our own version of strsep because it's not in the MS library.
1708
+ // It must behave differently than the Linux version when nothing follows the last separator.
1544
1709
//pftest (228, "i/30/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1);
1545
1710
1546
1711
pftest (229 , "i/30" , "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001" , 1 );
1547
1712
pftest (230 , "i/" , "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001" , -1 );
1548
1713
1714
+ pftest (240 , "s/" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1715
+ pftest (241 , "s/'/O/-/#/_" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1716
+ pftest (242 , "s/O/O/c" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1717
+ pftest (243 , "s/O/O/1/2" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1718
+ pftest (244 , "s/O/|/1" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1719
+ pftest (245 , "s//" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1720
+ pftest (246 , "s///" , "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA" , -1 );
1721
+
1722
+
1549
1723
// TODO: to be continued...
1550
1724
1551
1725
if (error_count > 0 ) {
0 commit comments