Skip to content

Commit 74a5c34

Browse files
committed
AIS refinements.
1 parent b99f9f3 commit 74a5c34

File tree

5 files changed

+213
-21
lines changed

5 files changed

+213
-21
lines changed

src/ais.c

+179-10
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,42 @@
5050
#include "textcolor.h"
5151
#include "ais.h"
5252

53+
// Lengths, in bits, for the AIS message types.
54+
55+
#define NUM_TYPES 27
56+
static const struct {
57+
short min;
58+
short max;
59+
} valid_len[NUM_TYPES+1] = {
60+
{ -1, -1 }, // 0 not used
61+
{ 168, 168 }, // 1
62+
{ 168, 168 }, // 2
63+
{ 168, 168 }, // 3
64+
{ 168, 168 }, // 4
65+
{ 424, 424 }, // 5
66+
{ 72, 1008 }, // 6 multipurpose
67+
{ 72, 168 }, // 7 increments of 32 bits
68+
{ 168, 1008 }, // 8 multipurpose
69+
{ 168, 168 }, // 9
70+
{ 72, 72 }, // 10
71+
{ 168, 168 }, // 11
72+
{ 72, 1008 }, // 12
73+
{ 72, 168 }, // 13 increments of 32 bits
74+
{ 40, 1008 }, // 14
75+
{ 88, 160 }, // 15
76+
{ 96, 114 }, // 16 96 or 114, not range
77+
{ 80, 816 }, // 17
78+
{ 168, 168 }, // 18
79+
{ 312, 312 }, // 19
80+
{ 72, 160 }, // 20
81+
{ 272, 360 }, // 21
82+
{ 168, 168 }, // 22
83+
{ 160, 160 }, // 23
84+
{ 160, 168 }, // 24
85+
{ 40, 168 }, // 25
86+
{ 60, 1064 }, // 26
87+
{ 96, 168 } // 27 96 or 168, not range
88+
};
5389

5490
/*-------------------------------------------------------------------
5591
*
@@ -113,23 +149,54 @@ static double get_field_latlon (unsigned char *base, unsigned int start, unsigne
113149
// Latitude of 0x3412140 (91 deg) means not available.
114150
// Longitude of 0x6791AC0 (181 deg) means not available.
115151
return ((double)get_field_signed(base, start, len) / 600000.0);
152+
153+
// Message type 27 uses lower resolution, 17 & 18 bits rather than 27 & 28.
154+
// It encodes minutes/10 rather than normal minutes/10000.
116155
}
117156

118157
static float get_field_speed (unsigned char *base, unsigned int start, unsigned int len)
119158
{
120159
// Raw 1023 means not available.
121160
// Multiply by 0.1 to get knots.
122-
return ((float)get_field_signed(base, start, len) * 0.1);
161+
// For aircraft it is knots, not deciknots.
162+
return ((float)get_field(base, start, len) * 0.1);
123163
}
124164

125165
static float get_field_course (unsigned char *base, unsigned int start, unsigned int len)
126166
{
127167
// Raw 3600 means not available.
128168
// Multiply by 0.1 to get degrees
129-
return ((float)get_field_signed(base, start, len) * 0.1);
169+
return ((float)get_field(base, start, len) * 0.1);
170+
}
171+
172+
static int get_field_ascii (unsigned char *base, unsigned int start, unsigned int len)
173+
{
174+
assert (len == 6);
175+
int ch = get_field(base, start, len);
176+
if (ch < 32) ch += 64;
177+
return (ch);
178+
}
179+
180+
static void get_field_string (unsigned char *base, unsigned int start, unsigned int len, char *result)
181+
{
182+
assert (len % 6 == 0);
183+
int nc = len / 6; // Number of characters.
184+
// Caller better provide space for at least this +1.
185+
// No bounds checking here.
186+
for (int i = 0; i < nc; i++) {
187+
result[i] = get_field_ascii (base, start + i * 6, 6);
188+
}
189+
result[nc] = '\0';
190+
// Officially it should be terminated/padded with @ but we also see trailing spaces.
191+
char *p = strchr(result, '@');
192+
if (p != NULL) *p = '\0';
193+
for (int k = strlen(result) - 1; k >= 0 && result[k] == ' '; k--) {
194+
result[k] = '\0';
195+
}
130196
}
131197

132198

199+
133200
/*-------------------------------------------------------------------
134201
*
135202
* Convert between 6 bit values and printable characters used in
@@ -238,18 +305,22 @@ void ais_to_nmea (unsigned char *ais, int ais_len, char *nmea, int nmea_size)
238305
* mssi 9 digit identifier.
239306
* odlat latitude.
240307
* odlon longitude.
241-
* ofknots speed.
308+
* ofknots speed, knots.
242309
* ofcourse direction of travel.
310+
* ofalt_m altitude, meters.
311+
* symtab APRS symbol table.
312+
* symbol APRS symbol code.
243313
*
244314
* Returns: 0 for success, -1 for error.
245315
*
246316
*--------------------------------------------------------------------*/
247317

248-
// Maximum NMEA sentence length is 82 according to some people.
318+
// Maximum NMEA sentence length is 82, including CR/LF.
249319
// Make buffer considerably larger to be safe.
250320
#define NMEA_MAX_LEN 240
251321

252-
int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mssi, int mssi_size, double *odlat, double *odlon, float *ofknots, float *ofcourse)
322+
int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mssi, int mssi_size, double *odlat, double *odlon,
323+
float *ofknots, float *ofcourse, float *ofalt_m, char *symtab, char *symbol, char *comment, int comment_size)
253324
{
254325
char stemp[NMEA_MAX_LEN]; /* Make copy because parsing is destructive. */
255326

@@ -258,6 +329,7 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
258329
*odlon = G_UNKNOWN;
259330
*ofknots = G_UNKNOWN;
260331
*ofcourse = G_UNKNOWN;
332+
*ofalt_m = G_UNKNOWN;
261333

262334
strlcpy (stemp, sentence, sizeof(stemp));
263335

@@ -349,18 +421,24 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
349421
}
350422
}
351423

424+
352425
// Extract the fields of interest from a few message types.
353426
// Don't get too carried away.
354427

355428
int type = get_field(ais, 0, 6);
429+
430+
if (type >= 1 && type <= 27) {
431+
snprintf (mssi, mssi_size, "%d", get_field(ais, 8, 30));
432+
}
356433
switch (type) {
357434

358435
case 1: // Position Report Class A
359436
case 2:
360437
case 3:
361438

362439
snprintf (descr, descr_size, "AIS %d: Position Report Class A", type);
363-
snprintf (mssi, mssi_size, "%d", get_field(ais, 8, 30));
440+
*symtab = '/';
441+
*symbol = 's'; // Power boat (ship) side view
364442
*odlon = get_field_latlon(ais, 61, 28);
365443
*odlat = get_field_latlon(ais, 89, 27);
366444
*ofknots = get_field_speed(ais, 50, 10);
@@ -370,7 +448,8 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
370448
case 4: // Base Station Report
371449

372450
snprintf (descr, descr_size, "AIS %d: Base Station Report", type);
373-
snprintf (mssi, mssi_size, "%d", get_field(ais, 8, 30));
451+
*symtab = '\\';
452+
*symbol = 'L'; // Lighthouse
374453
//year = get_field(ais, 38, 14);
375454
//month = get_field(ais, 52, 4);
376455
//day = get_field(ais, 56, 5);
@@ -381,22 +460,71 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
381460
*odlat = get_field_latlon(ais, 107, 27);
382461
break;
383462

463+
case 5: // Static and Voyage Related Data
464+
465+
snprintf (descr, descr_size, "AIS %d: Static and Voyage Related Data", type);
466+
*symtab = '/';
467+
*symbol = 's'; // Power boat (ship) side view
468+
{
469+
char callsign[12];
470+
char shipname[24];
471+
char destination[24];
472+
get_field_string(ais, 70, 42, callsign);
473+
get_field_string(ais, 112, 120, shipname);
474+
get_field_string(ais, 302, 120, destination);
475+
476+
if (strlen(destination) > 0) {
477+
snprintf (comment, comment_size, "%s, %s, dest. %s", shipname, callsign, destination);
478+
}
479+
else {
480+
snprintf (comment, comment_size, "%s, %s", shipname, callsign);
481+
}
482+
}
483+
break;
484+
485+
486+
case 9: // Standard SAR Aircraft Position Report
487+
488+
snprintf (descr, descr_size, "AIS %d: SAR Aircraft Position Report", type);
489+
*symtab = '/';
490+
*symbol = '\''; // Small AIRCRAFT
491+
*ofalt_m = get_field(ais, 38, 12); // meters, 4095 means not available
492+
*odlon = get_field_latlon(ais, 61, 28);
493+
*odlat = get_field_latlon(ais, 89, 27);
494+
*ofknots = get_field_speed(ais, 50, 10) * 10.0; // plane is knots, not knots/10
495+
*ofcourse = get_field_course(ais, 116, 12);
496+
break;
497+
384498
case 18: // Standard Class B CS Position Report
499+
// As an oversimplification, Class A is commercial, B is recreational.
385500

386501
snprintf (descr, descr_size, "AIS %d: Standard Class B CS Position Report", type);
387-
snprintf (mssi, mssi_size, "%d", get_field(ais, 8, 30));
502+
*symtab = '/';
503+
*symbol = 'Y'; // YACHT (sail)
388504
*odlon = get_field_latlon(ais, 57, 28);
389505
*odlat = get_field_latlon(ais, 85, 27);
390506
break;
391507

392508
case 19: // Extended Class B CS Position Report
393509

394510
snprintf (descr, descr_size, "AIS %d: Extended Class B CS Position Report", type);
395-
snprintf (mssi, mssi_size, "%d", get_field(ais, 8, 30));
511+
*symtab = '/';
512+
*symbol = 'Y'; // YACHT (sail)
396513
*odlon = get_field_latlon(ais, 57, 28);
397514
*odlat = get_field_latlon(ais, 85, 27);
398515
break;
399516

517+
case 27: // Long Range AIS Broadcast message
518+
519+
snprintf (descr, descr_size, "AIS %d: Long Range AIS Broadcast message", type);
520+
*symtab = '\\';
521+
*symbol = 's'; // OVERLAY SHIP/boat (top view)
522+
*odlon = get_field_latlon(ais, 44, 18) * 1000; // Note: minutes/10 rather than usual /10000.
523+
*odlat = get_field_latlon(ais, 62, 17) * 1000;
524+
*ofknots = get_field_speed(ais, 79, 6) * 10; // Note: knots, not deciknots.
525+
*ofcourse = get_field_course(ais, 85, 9) * 10; // Note: degrees, not decidegrees.
526+
break;
527+
400528
default:
401529
snprintf (descr, descr_size, "AIS message type %d", type);
402530
break;
@@ -406,4 +534,45 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
406534

407535
} /* end ais_parse */
408536

409-
// end ais.c
537+
538+
539+
/*-------------------------------------------------------------------
540+
*
541+
* Name: ais_check_length
542+
*
543+
* Purpose: Verify frame length against expected.
544+
*
545+
* Inputs: type Message type, 1 - 27.
546+
*
547+
* length Number of data octets in in frame.
548+
*
549+
* Returns: -1 Invalid message type.
550+
* 0 Good length.
551+
* 1 Unexpected lenth.
552+
*
553+
*--------------------------------------------------------------------*/
554+
555+
int ais_check_length (int type, int length)
556+
{
557+
if (type >= 1 && type <= NUM_TYPES) {
558+
int b = length * 8;
559+
if (b >= valid_len[type].min && b <= valid_len[type].max) {
560+
return (0); // Good.
561+
}
562+
else {
563+
//text_color_set (DW_COLOR_ERROR);
564+
//dw_printf("AIS ERROR: type %d, has %d bits when %d to %d expected.\n",
565+
// type, b, valid_len[type].min, valid_len[type].max);
566+
return (1); // Length out of range.
567+
}
568+
}
569+
else {
570+
//text_color_set (DW_COLOR_ERROR);
571+
//dw_printf("AIS ERROR: message type %d is invalid.\n", type);
572+
return (-1); // Invalid type.
573+
}
574+
575+
} // end ais_check_length
576+
577+
578+
// end ais.c

src/ais.h

+4-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,7 @@
22

33
void ais_to_nmea (unsigned char *ais, int ais_len, char *nema, int nema_size);
44

5-
int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mssi, int mssi_size, double *odlat, double *odlon, float *ofknots, float *ofcourse);
5+
int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mssi, int mssi_size, double *odlat, double *odlon,
6+
float *ofknots, float *ofcourse, float *ofalt_m, char *symtab, char *symbol, char *comment, int comment_size);
7+
8+
int ais_check_length (int type, int length);

src/decode_aprs.c

+5-1
Original file line numberDiff line numberDiff line change
@@ -2335,13 +2335,17 @@ static void aprs_user_defined (decode_aprs_t *A, char *info, int ilen)
23352335
else if (info[0] == '{' && info[1] == USER_DEF_USER_ID && info[2] == USER_DEF_TYPE_AIS) {
23362336
double lat, lon;
23372337
float knots, course;
2338+
float alt_meters;
23382339

2339-
ais_parse (info+3, 0, A->g_msg_type, sizeof(A->g_msg_type), A->g_name, sizeof(A->g_name), &lat, &lon, &knots, &course);
2340+
ais_parse (info+3, 0, A->g_msg_type, sizeof(A->g_msg_type), A->g_name, sizeof(A->g_name),
2341+
&lat, &lon, &knots, &course, &alt_meters, &(A->g_symbol_table), &(A->g_symbol_code),
2342+
A->g_comment, sizeof(A->g_comment));
23402343

23412344
A->g_lat = lat;
23422345
A->g_lon = lon;
23432346
A->g_speed_mph = DW_KNOTS_TO_MPH(knots);
23442347
A->g_course = course;
2348+
A->g_altitude_ft = DW_METERS_TO_FEET(alt_meters);
23452349
strcpy (A->g_mfr, "");
23462350
}
23472351
else if (strncmp(info, "{{", 2) == 0) {

src/demod.c

+16-3
Original file line numberDiff line numberDiff line change
@@ -622,6 +622,22 @@ int demod_init (struct audio_s *pa)
622622
default: /* Not AFSK */
623623
{
624624

625+
// For AIS we will accept only a good CRC without any fixup attempts.
626+
// Even with that, there are still a lot of CRC false matches with random noise.
627+
628+
if (save_audio_config_p->achan[chan].modem_type == MODEM_AIS) {
629+
if (save_audio_config_p->achan[chan].fix_bits != RETRY_NONE) {
630+
text_color_set(DW_COLOR_INFO);
631+
dw_printf ("Channel %d: FIX_BITS option has been turned off for AIS.\n", chan);
632+
save_audio_config_p->achan[chan].fix_bits = RETRY_NONE;
633+
}
634+
if (save_audio_config_p->achan[chan].passall != 0) {
635+
text_color_set(DW_COLOR_INFO);
636+
dw_printf ("Channel %d: PASSALL option has been turned off for AIS.\n", chan);
637+
save_audio_config_p->achan[chan].passall = 0;
638+
}
639+
}
640+
625641
if (strcmp(save_audio_config_p->achan[chan].profiles, "") == 0) {
626642

627643
/* Apply default if not set earlier. */
@@ -632,13 +648,10 @@ int demod_init (struct audio_s *pa)
632648
/* We want higher performance to be the default. */
633649
/* "MODEM 9600 -" can be used on very slow CPU if necessary. */
634650

635-
//#ifndef __arm__
636651
strlcpy (save_audio_config_p->achan[chan].profiles, "+", sizeof(save_audio_config_p->achan[chan].profiles));
637-
//#endif
638652
}
639653

640654

641-
642655
#ifdef TUNE_ZEROSTUFF
643656
zerostuff = TUNE_ZEROSTUFF;
644657
#endif

src/hdlc_rec2.c

+9-6
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@
104104
#include "demod_9600.h" /* for descramble() */
105105
#include "audio.h" /* for struct audio_s */
106106
//#include "ax25_pad.h" /* for AX25_MAX_ADDR_LEN */
107-
107+
#include "ais.h"
108108

109109
//#define DEBUG 1
110110
//#define DEBUGx 1
@@ -766,11 +766,14 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t
766766

767767
if (actual_fcs == expected_fcs && save_audio_config_p->achan[chan].modem_type == MODEM_AIS) {
768768

769-
// Sanity check for AIS does not seem feasible.
770-
// Could possibly check length if we knew all the valid possibilities.
771-
772-
multi_modem_process_rec_frame (chan, subchan, slice, H.frame_buf, H.frame_len - 2, alevel, retry_conf.retry, 0); /* len-2 to remove FCS. */
773-
return 1; /* success */
769+
// Sanity check for AIS.
770+
if (ais_check_length((H.frame_buf[0] >> 2) & 0x3f, H.frame_len - 2) == 0) {
771+
multi_modem_process_rec_frame (chan, subchan, slice, H.frame_buf, H.frame_len - 2, alevel, retry_conf.retry, 0); /* len-2 to remove FCS. */
772+
return 1; /* success */
773+
}
774+
else {
775+
return 0; /* did not pass sanity check */
776+
}
774777
}
775778
else if (actual_fcs == expected_fcs &&
776779
sanity_check (H.frame_buf, H.frame_len - 2, retry_conf.retry, save_audio_config_p->achan[chan].sanity_test)) {

0 commit comments

Comments
 (0)