50
50
#include "textcolor.h"
51
51
#include "ais.h"
52
52
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
+ };
53
89
54
90
/*-------------------------------------------------------------------
55
91
*
@@ -113,23 +149,54 @@ static double get_field_latlon (unsigned char *base, unsigned int start, unsigne
113
149
// Latitude of 0x3412140 (91 deg) means not available.
114
150
// Longitude of 0x6791AC0 (181 deg) means not available.
115
151
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.
116
155
}
117
156
118
157
static float get_field_speed (unsigned char * base , unsigned int start , unsigned int len )
119
158
{
120
159
// Raw 1023 means not available.
121
160
// 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 );
123
163
}
124
164
125
165
static float get_field_course (unsigned char * base , unsigned int start , unsigned int len )
126
166
{
127
167
// Raw 3600 means not available.
128
168
// 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
+ }
130
196
}
131
197
132
198
199
+
133
200
/*-------------------------------------------------------------------
134
201
*
135
202
* 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)
238
305
* mssi 9 digit identifier.
239
306
* odlat latitude.
240
307
* odlon longitude.
241
- * ofknots speed.
308
+ * ofknots speed, knots .
242
309
* ofcourse direction of travel.
310
+ * ofalt_m altitude, meters.
311
+ * symtab APRS symbol table.
312
+ * symbol APRS symbol code.
243
313
*
244
314
* Returns: 0 for success, -1 for error.
245
315
*
246
316
*--------------------------------------------------------------------*/
247
317
248
- // Maximum NMEA sentence length is 82 according to some people.
318
+ // Maximum NMEA sentence length is 82, including CR/LF.
249
319
// Make buffer considerably larger to be safe.
250
320
#define NMEA_MAX_LEN 240
251
321
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 )
253
324
{
254
325
char stemp [NMEA_MAX_LEN ]; /* Make copy because parsing is destructive. */
255
326
@@ -258,6 +329,7 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
258
329
* odlon = G_UNKNOWN ;
259
330
* ofknots = G_UNKNOWN ;
260
331
* ofcourse = G_UNKNOWN ;
332
+ * ofalt_m = G_UNKNOWN ;
261
333
262
334
strlcpy (stemp , sentence , sizeof (stemp ));
263
335
@@ -349,18 +421,24 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
349
421
}
350
422
}
351
423
424
+
352
425
// Extract the fields of interest from a few message types.
353
426
// Don't get too carried away.
354
427
355
428
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
+ }
356
433
switch (type ) {
357
434
358
435
case 1 : // Position Report Class A
359
436
case 2 :
360
437
case 3 :
361
438
362
439
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
364
442
* odlon = get_field_latlon (ais , 61 , 28 );
365
443
* odlat = get_field_latlon (ais , 89 , 27 );
366
444
* 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
370
448
case 4 : // Base Station Report
371
449
372
450
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
374
453
//year = get_field(ais, 38, 14);
375
454
//month = get_field(ais, 52, 4);
376
455
//day = get_field(ais, 56, 5);
@@ -381,22 +460,71 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
381
460
* odlat = get_field_latlon (ais , 107 , 27 );
382
461
break ;
383
462
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
+
384
498
case 18 : // Standard Class B CS Position Report
499
+ // As an oversimplification, Class A is commercial, B is recreational.
385
500
386
501
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)
388
504
* odlon = get_field_latlon (ais , 57 , 28 );
389
505
* odlat = get_field_latlon (ais , 85 , 27 );
390
506
break ;
391
507
392
508
case 19 : // Extended Class B CS Position Report
393
509
394
510
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)
396
513
* odlon = get_field_latlon (ais , 57 , 28 );
397
514
* odlat = get_field_latlon (ais , 85 , 27 );
398
515
break ;
399
516
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
+
400
528
default :
401
529
snprintf (descr , descr_size , "AIS message type %d" , type );
402
530
break ;
@@ -406,4 +534,45 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
406
534
407
535
} /* end ais_parse */
408
536
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
0 commit comments