@@ -87,6 +87,10 @@ static const struct {
87
87
{ 96 , 168 } // 27 96 or 168, not range
88
88
};
89
89
90
+ static void save_ship_data (char * mssi , char * shipname , char * callsign , char * destination );
91
+ static void get_ship_data (char * mssi , char * comment , int comment_size );
92
+
93
+
90
94
/*-------------------------------------------------------------------
91
95
*
92
96
* Functions to get and set element of a bit vector.
@@ -144,29 +148,68 @@ static int get_field_signed (unsigned char *base, unsigned int start, unsigned i
144
148
return (result );
145
149
}
146
150
147
- static double get_field_latlon (unsigned char * base , unsigned int start , unsigned int len )
151
+ static double get_field_lat (unsigned char * base , unsigned int start , unsigned int len )
148
152
{
149
153
// Latitude of 0x3412140 (91 deg) means not available.
150
- // Longitude of 0x6791AC0 (181 deg) means not available.
151
- return ((double )get_field_signed (base , start , len ) / 600000.0 );
154
+ // Message type 27 uses lower resolution, 17 bits rather than 27.
155
+ // It encodes minutes/10 rather than normal minutes/10000.
156
+
157
+ int n = get_field_signed (base , start , len );
158
+ if (len == 17 ) {
159
+ return ((n == 91 * 600 ) ? G_UNKNOWN : (double )n / 600.0 );
160
+ }
161
+ else {
162
+ return ((n == 91 * 600000 ) ? G_UNKNOWN : (double )n / 600000.0 );
163
+ }
164
+ }
152
165
153
- // Message type 27 uses lower resolution, 17 & 18 bits rather than 27 & 28.
166
+ static double get_field_lon (unsigned char * base , unsigned int start , unsigned int len )
167
+ {
168
+ // Longitude of 0x6791AC0 (181 deg) means not available.
169
+ // Message type 27 uses lower resolution, 18 bits rather than 28.
154
170
// It encodes minutes/10 rather than normal minutes/10000.
171
+
172
+ int n = get_field_signed (base , start , len );
173
+ if (len == 18 ) {
174
+ return ((n == 181 * 600 ) ? G_UNKNOWN : (double )n / 600.0 );
175
+ }
176
+ else {
177
+ return ((n == 181 * 600000 ) ? G_UNKNOWN : (double )n / 600000.0 );
178
+ }
155
179
}
156
180
157
181
static float get_field_speed (unsigned char * base , unsigned int start , unsigned int len )
158
182
{
159
183
// Raw 1023 means not available.
160
184
// Multiply by 0.1 to get knots.
161
185
// For aircraft it is knots, not deciknots.
162
- return ((float )get_field (base , start , len ) * 0.1 );
186
+
187
+ // Message type 27 uses lower resolution, 6 bits rather than 10.
188
+ // It encodes minutes/10 rather than normal minutes/10000.
189
+
190
+ int n = get_field (base , start , len );
191
+ if (len == 6 ) {
192
+ return ((n == 63 ) ? G_UNKNOWN : (float )n );
193
+ }
194
+ else {
195
+ return ((n == 1023 ) ? G_UNKNOWN : (float )n * 0.1 );
196
+ }
163
197
}
164
198
165
199
static float get_field_course (unsigned char * base , unsigned int start , unsigned int len )
166
200
{
167
201
// Raw 3600 means not available.
168
202
// Multiply by 0.1 to get degrees
169
- return ((float )get_field (base , start , len ) * 0.1 );
203
+ // Message type 27 uses lower resolution, 9 bits rather than 12.
204
+ // It encodes degrees rather than normal degrees/10.
205
+
206
+ int n = get_field (base , start , len );
207
+ if (len == 9 ) {
208
+ return ((n == 360 ) ? G_UNKNOWN : (float )n );
209
+ }
210
+ else {
211
+ return ((n == 3600 ) ? G_UNKNOWN : (float )n * 0.1 );
212
+ }
170
213
}
171
214
172
215
static int get_field_ascii (unsigned char * base , unsigned int start , unsigned int len )
@@ -428,7 +471,7 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
428
471
int type = get_field (ais , 0 , 6 );
429
472
430
473
if (type >= 1 && type <= 27 ) {
431
- snprintf (mssi , mssi_size , "%d " , get_field (ais , 8 , 30 ));
474
+ snprintf (mssi , mssi_size , "%09d " , get_field (ais , 8 , 30 ));
432
475
}
433
476
switch (type ) {
434
477
@@ -439,10 +482,11 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
439
482
snprintf (descr , descr_size , "AIS %d: Position Report Class A" , type );
440
483
* symtab = '/' ;
441
484
* symbol = 's' ; // Power boat (ship) side view
442
- * odlon = get_field_latlon (ais , 61 , 28 );
443
- * odlat = get_field_latlon (ais , 89 , 27 );
485
+ * odlon = get_field_lon (ais , 61 , 28 );
486
+ * odlat = get_field_lat (ais , 89 , 27 );
444
487
* ofknots = get_field_speed (ais , 50 , 10 );
445
488
* ofcourse = get_field_course (ais , 116 , 12 );
489
+ get_ship_data (mssi , comment , comment_size );
446
490
break ;
447
491
448
492
case 4 : // Base Station Report
@@ -456,8 +500,10 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
456
500
//hour = get_field(ais, 61, 5);
457
501
//minute = get_field(ais, 66, 6);
458
502
//second = get_field(ais, 72, 6);
459
- * odlon = get_field_latlon (ais , 79 , 28 );
460
- * odlat = get_field_latlon (ais , 107 , 27 );
503
+ * odlon = get_field_lon (ais , 79 , 28 );
504
+ * odlat = get_field_lat (ais , 107 , 27 );
505
+ // Is this suitable or not? Doesn't hurt, I suppose.
506
+ get_ship_data (mssi , comment , comment_size );
461
507
break ;
462
508
463
509
case 5 : // Static and Voyage Related Data
@@ -472,13 +518,8 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
472
518
get_field_string (ais , 70 , 42 , callsign );
473
519
get_field_string (ais , 112 , 120 , shipname );
474
520
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
- }
521
+ save_ship_data (mssi , shipname , callsign , destination );
522
+ get_ship_data (mssi , comment , comment_size );
482
523
}
483
524
break ;
484
525
@@ -489,10 +530,12 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
489
530
* symtab = '/' ;
490
531
* symbol = '\'' ; // Small AIRCRAFT
491
532
* 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
533
+ * odlon = get_field_lon (ais , 61 , 28 );
534
+ * odlat = get_field_lat (ais , 89 , 27 );
535
+ * ofknots = get_field_speed (ais , 50 , 10 ); // plane is knots, not knots/10
536
+ if (* ofknots != G_UNKNOWN ) * ofknots = * ofknots * 10.0 ;
495
537
* ofcourse = get_field_course (ais , 116 , 12 );
538
+ get_ship_data (mssi , comment , comment_size );
496
539
break ;
497
540
498
541
case 18 : // Standard Class B CS Position Report
@@ -501,28 +544,31 @@ int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mss
501
544
snprintf (descr , descr_size , "AIS %d: Standard Class B CS Position Report" , type );
502
545
* symtab = '/' ;
503
546
* symbol = 'Y' ; // YACHT (sail)
504
- * odlon = get_field_latlon (ais , 57 , 28 );
505
- * odlat = get_field_latlon (ais , 85 , 27 );
547
+ * odlon = get_field_lon (ais , 57 , 28 );
548
+ * odlat = get_field_lat (ais , 85 , 27 );
549
+ get_ship_data (mssi , comment , comment_size );
506
550
break ;
507
551
508
552
case 19 : // Extended Class B CS Position Report
509
553
510
554
snprintf (descr , descr_size , "AIS %d: Extended Class B CS Position Report" , type );
511
555
* symtab = '/' ;
512
556
* symbol = 'Y' ; // YACHT (sail)
513
- * odlon = get_field_latlon (ais , 57 , 28 );
514
- * odlat = get_field_latlon (ais , 85 , 27 );
557
+ * odlon = get_field_lon (ais , 57 , 28 );
558
+ * odlat = get_field_lat (ais , 85 , 27 );
559
+ get_ship_data (mssi , comment , comment_size );
515
560
break ;
516
561
517
562
case 27 : // Long Range AIS Broadcast message
518
563
519
564
snprintf (descr , descr_size , "AIS %d: Long Range AIS Broadcast message" , type );
520
565
* symtab = '\\' ;
521
566
* 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.
567
+ * odlon = get_field_lon (ais , 44 , 18 ); // Note: minutes/10 rather than usual /10000.
568
+ * odlat = get_field_lat (ais , 62 , 17 );
569
+ * ofknots = get_field_speed (ais , 79 , 6 ); // Note: knots, not deciknots.
570
+ * ofcourse = get_field_course (ais , 85 , 9 ); // Note: degrees, not decidegrees.
571
+ get_ship_data (mssi , comment , comment_size );
526
572
break ;
527
573
528
574
default :
@@ -575,4 +621,90 @@ int ais_check_length (int type, int length)
575
621
} // end ais_check_length
576
622
577
623
624
+
625
+ /*-------------------------------------------------------------------
626
+ *
627
+ * Name: save_ship_data
628
+ *
629
+ * Purpose: Save shipname, etc., from "Static and Voyage Related Data"
630
+ * so it can be combined later with the position reports.
631
+ *
632
+ * Inputs: mssi
633
+ * shipname
634
+ * callsign
635
+ * destination
636
+ *
637
+ *--------------------------------------------------------------------*/
638
+
639
+ struct ship_data_s {
640
+ struct ship_data_s * pnext ;
641
+ char mssi [9 + 1 ];
642
+ char shipname [20 + 1 ];
643
+ char callsign [7 + 1 ];
644
+ char destination [20 + 1 ];
645
+ };
646
+
647
+ // Just use a single linked list for now.
648
+ // If I get ambitious, I might use a hash table.
649
+ // I don't think we need a critical region because all channels
650
+ // should be serialized thru the receive queue.
651
+
652
+ static struct ship_data_s * ships = NULL ;
653
+
654
+
655
+ static void save_ship_data (char * mssi , char * shipname , char * callsign , char * destination )
656
+ {
657
+ // Get list node, either existing or new.
658
+ struct ship_data_s * p = ships ;
659
+ while (p != NULL ) {
660
+ if (strcmp (mssi , p -> mssi ) == 0 ) {
661
+ break ;
662
+ }
663
+ p = p -> pnext ;
664
+ }
665
+ if (p == NULL ) {
666
+ p = calloc (sizeof (struct ship_data_s ),1 );
667
+ p -> pnext = ships ;
668
+ ships = p ;
669
+ }
670
+
671
+ strlcpy (p -> mssi , mssi , sizeof (p -> mssi ));
672
+ strlcpy (p -> shipname , shipname , sizeof (p -> shipname ));
673
+ strlcpy (p -> callsign , callsign , sizeof (p -> callsign ));
674
+ strlcpy (p -> destination , destination , sizeof (p -> destination ));
675
+ }
676
+
677
+ /*-------------------------------------------------------------------
678
+ *
679
+ * Name: save_ship_data
680
+ *
681
+ * Purpose: Get ship data for specified mssi.
682
+ *
683
+ * Inputs: mssi
684
+ *
685
+ * Outputs: comment - If mssi is found, return in single string here,
686
+ * suitable for the comment field.
687
+ *
688
+ *--------------------------------------------------------------------*/
689
+
690
+ static void get_ship_data (char * mssi , char * comment , int comment_size )
691
+ {
692
+ struct ship_data_s * p = ships ;
693
+ while (p != NULL ) {
694
+ if (strcmp (mssi , p -> mssi ) == 0 ) {
695
+ break ;
696
+ }
697
+ p = p -> pnext ;
698
+ }
699
+ if (p != NULL ) {
700
+ if (strlen (p -> destination ) > 0 ) {
701
+ snprintf (comment , comment_size , "%s, %s, dest. %s" , p -> shipname , p -> callsign , p -> destination );
702
+ }
703
+ else {
704
+ snprintf (comment , comment_size , "%s, %s" , p -> shipname , p -> callsign );
705
+ }
706
+ }
707
+ }
708
+
709
+
578
710
// end ais.c
0 commit comments