OpenCPN Partial API docs
Loading...
Searching...
No Matches
ais_decoder.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 *
5 ***************************************************************************
6 * Copyright (C) 2010 by David S. Register *
7 * *
8 * This program is free software; you can redistribute it and/or modify *
9 * it under the terms of the GNU General Public License as published by *
10 * the Free Software Foundation; either version 2 of the License, or *
11 * (at your option) any later version. *
12 * *
13 * This program is distributed in the hope that it will be useful, *
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
16 * GNU General Public License for more details. *
17 * *
18 * You should have received a copy of the GNU General Public License *
19 * along with this program; if not, write to the *
20 * Free Software Foundation, Inc., *
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
22 ***************************************************************************
23 */
24
25// For compilers that support precompilation, includes "wx.h".
26#include <wx/wxprec.h>
27
28#ifndef WX_PRECOMP
29#include <wx/wx.h>
30#endif // precompiled headers
31
32#include <algorithm>
33#include <cstdio>
34#include <fstream>
35
36#ifdef __MINGW32__
37#undef IPV6STRICT // mingw FTBS fix: missing struct ip_mreq
38#include <windows.h>
39#endif
40
41#include <wx/datetime.h>
42#include <wx/event.h>
43#include <wx/log.h>
44#include <wx/string.h>
45#include <wx/textfile.h>
46#include <wx/timer.h>
47#include <wx/tokenzr.h>
48#include <wx/filename.h>
49
50// Be sure to include these before ais_decoder.h
51// to avoid a conflict with rapidjson/fwd.h
52#include "rapidjson/document.h"
53#include "rapidjson/writer.h"
54#include "rapidjson/stringbuffer.h"
55
56#include "model/ais_decoder.h"
58#include "model/meteo_points.h"
59#include "model/ais_target_data.h"
61#include "model/config_vars.h"
62#include "model/geodesic.h"
63#include "model/georef.h"
64#include "model/idents.h"
65#include "model/multiplexer.h"
66#include "model/navutil_base.h"
67#include "model/own_ship.h"
68#include "model/route_point.h"
69#include "model/select.h"
70#include "SoundFactory.h"
71#include "model/track.h"
72#include "N2KParser.h"
73
74#if !defined(NAN)
75static const long long lNaN = 0xfff8000000000000;
76#define NAN (*(double *)&lNaN)
77#endif
78
79wxEvtHandler *g_pais_alert_dialog_active;
80
81AisDecoder *g_pAIS;
82Select *pSelectAIS;
83bool g_bUseOnlyConfirmedAISName;
84wxString GetShipNameFromFile(int);
85wxString AISTargetNameFileName;
86bool isBuoyMmsi(const int);
87extern Multiplexer *g_pMUX;
88int g_OwnShipmmsi;
89
90wxDEFINE_EVENT(EVT_N0183_VDO, ObservedEvt);
91wxDEFINE_EVENT(EVT_N0183_VDM, ObservedEvt);
92wxDEFINE_EVENT(EVT_N0183_FRPOS, ObservedEvt);
93wxDEFINE_EVENT(EVT_N0183_CDDSC, ObservedEvt);
94wxDEFINE_EVENT(EVT_N0183_CDDSE, ObservedEvt);
95wxDEFINE_EVENT(EVT_N0183_TLL, ObservedEvt);
96wxDEFINE_EVENT(EVT_N0183_TTM, ObservedEvt);
97wxDEFINE_EVENT(EVT_N0183_OSD, ObservedEvt);
98wxDEFINE_EVENT(EVT_N0183_WPL, ObservedEvt);
99wxDEFINE_EVENT(EVT_SIGNALK, ObservedEvt);
100wxDEFINE_EVENT(EVT_N2K_129038, ObservedEvt);
101wxDEFINE_EVENT(EVT_N2K_129039, ObservedEvt);
102wxDEFINE_EVENT(EVT_N2K_129041, ObservedEvt);
103wxDEFINE_EVENT(EVT_N2K_129794, ObservedEvt);
104wxDEFINE_EVENT(EVT_N2K_129809, ObservedEvt);
105wxDEFINE_EVENT(EVT_N2K_129810, ObservedEvt);
106wxDEFINE_EVENT(EVT_N2K_129793, ObservedEvt);
107
108BEGIN_EVENT_TABLE(AisDecoder, wxEvtHandler)
109EVT_TIMER(TIMER_AIS1, AisDecoder::OnTimerAIS)
110EVT_TIMER(TIMER_DSC, AisDecoder::OnTimerDSC)
111END_EVENT_TABLE()
112
113static const double ms_to_knot_factor = 1.9438444924406;
114
115static int n_msgs;
116static int n_msg1;
117static int n_msg5;
118static int n_msg24;
119static bool b_firstrx;
120static int first_rx_ticks;
121static int rx_ticks;
122static double arpa_ref_hdg = NAN;
123
124static inline double GeodesicRadToDeg(double rads) {
125 return rads * 180.0 / M_PI;
126}
127
128static inline double MS2KNOTS(double ms) { return ms * 1.9438444924406; }
129
130int AisMeteoNewMmsi(int, int, int, int, int);
131int origin_mmsi = 0;
132
133void AISshipNameCache(AisTargetData *pTargetData,
134 AIS_Target_Name_Hash *AISTargetNamesC,
135 AIS_Target_Name_Hash *AISTargetNamesNC, long mmsi);
136
137AisDecoder::AisDecoder(AisDecoderCallbacks callbacks)
138 : m_signalk_selfid(""), m_callbacks(callbacks) {
139 // Load cached AIS target names from a file
140 AISTargetNamesC = new AIS_Target_Name_Hash;
141 AISTargetNamesNC = new AIS_Target_Name_Hash;
142
143 if (g_benableAISNameCache) {
144 if (wxFileName::FileExists(AISTargetNameFileName)) {
145 wxTextFile infile;
146 if (infile.Open(AISTargetNameFileName)) {
147 AIS_Target_Name_Hash *HashFile = AISTargetNamesNC;
148 wxString line = infile.GetFirstLine();
149 while (!infile.Eof()) {
150 if (line.IsSameAs(wxT("+++==Confirmed Entry's==+++")))
151 HashFile = AISTargetNamesC;
152 else {
153 if (line.IsSameAs(wxT("+++==Non Confirmed Entry's==+++")))
154 HashFile = AISTargetNamesNC;
155 else {
156 wxStringTokenizer tokenizer(line, _T(","));
157 int mmsi = wxAtoi(tokenizer.GetNextToken());
158 wxString name = tokenizer.GetNextToken().Trim();
159 (*HashFile)[mmsi] = name;
160 }
161 }
162 line = infile.GetNextLine();
163 }
164 }
165 infile.Close();
166 }
167 }
168
169 BuildERIShipTypeHash();
170
171 g_pais_alert_dialog_active = nullptr;
172 m_bAIS_Audio_Alert_On = false;
173
174 m_n_targets = 0;
175
176 m_bAIS_AlertPlaying = false;
177
178 TimerAIS.SetOwner(this, TIMER_AIS1);
179 TimerAIS.Start(TIMER_AIS_MSEC, wxTIMER_CONTINUOUS);
180
181 m_ptentative_dsctarget = NULL;
182 m_dsc_timer.SetOwner(this, TIMER_DSC);
183
184 // Create/connect a dynamic event handler slot for wxEVT_OCPN_DATASTREAM(s)
185 // FIXME delete Connect(wxEVT_OCPN_DATASTREAM,
186 // (wxObjectEventFunction)(wxEventFunction)&AisDecoder::OnEvtAIS);
187 // Connect(EVT_OCPN_SIGNALKSTREAM,
188 // (wxObjectEventFunction)(wxEventFunction)&AisDecoder::OnEvtSignalK);
189 InitCommListeners();
190}
191
192AisDecoder::~AisDecoder(void) {
193 // for (const auto &it : GetTargetList()) {
194 // AisTargetData *td = it.second;
195 //
196 // delete td;
197 // }
198
199 // Write mmsi-shipsname to file in a safe way
200 wxTempFile outfile;
201 if (outfile.Open(AISTargetNameFileName)) {
202 wxString content;
203 content = wxT("+++==Confirmed Entry's==+++");
204 AIS_Target_Name_Hash::iterator it;
205 for (it = AISTargetNamesC->begin(); it != AISTargetNamesC->end(); ++it) {
206 content.append(_T("\r\n"));
207 content.append(wxString::Format(wxT("%i"), it->first));
208 content.append(_T(",")).append(it->second);
209 }
210 content.append(_T("\r\n"));
211 content.append(_T("+++==Non Confirmed Entry's==+++"));
212 for (it = AISTargetNamesNC->begin(); it != AISTargetNamesNC->end(); ++it) {
213 content.append(_T("\r\n"));
214 content.append(wxString::Format(wxT("%i"), it->first));
215 content.append(_T(",")).append(it->second);
216 }
217 outfile.Write(content);
218 outfile.Commit();
219 }
220
221 AISTargetNamesC->clear();
222 delete AISTargetNamesC;
223 AISTargetNamesNC->clear();
224 delete AISTargetNamesNC;
225
226 clear_hash_ERI();
227
228 m_dsc_timer.Stop();
229 m_AIS_Audio_Alert_Timer.Stop();
230 TimerAIS.Stop();
231
232#ifdef AIS_DEBUG
233 printf(
234 "First message[1, 2] ticks: %d Last Message [1,2]ticks %d Difference: "
235 "%d\n",
236 first_rx_ticks, rx_ticks, rx_ticks - first_rx_ticks);
237#endif
238}
239
240bool IsTargetOnTheIgnoreList(const int &mmsi) {
241 // Check the MMSI-Prop list if the target shall be ignored
242 for (unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
243 if (mmsi == g_MMSI_Props_Array[i]->MMSI) {
244 MmsiProperties *props = g_MMSI_Props_Array[i];
245 if (props->m_bignore) {
246 return true;
247 }
248 }
249 }
250 return false;
251}
252
253void AisDecoder::InitCommListeners(void) {
254 // Initialize the comm listeners
255
256 auto &msgbus = NavMsgBus::GetInstance();
257
258 // NMEA0183
259 // VDM
260 Nmea0183Msg n0183_msg_VDM("VDM");
261 listener_N0183_VDM.Listen(n0183_msg_VDM, this, EVT_N0183_VDM);
262 Bind(EVT_N0183_VDM, [&](ObservedEvt ev) {
263 auto ptr = ev.GetSharedPtr();
264 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
265 HandleN0183_AIS(n0183_msg);
266 });
267
268 // FRPOS
269 Nmea0183Msg n0183_msg_FRPOS("FRPOS");
270 listener_N0183_FRPOS.Listen(n0183_msg_FRPOS, this, EVT_N0183_FRPOS);
271
272 Bind(EVT_N0183_FRPOS, [&](ObservedEvt ev) {
273 auto ptr = ev.GetSharedPtr();
274 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
275 HandleN0183_AIS(n0183_msg);
276 });
277
278 // CDDSC
279 Nmea0183Msg n0183_msg_CDDSC("CDDSC");
280 listener_N0183_CDDSC.Listen(n0183_msg_CDDSC, this, EVT_N0183_CDDSC);
281 Bind(EVT_N0183_CDDSC, [&](ObservedEvt ev) {
282 auto ptr = ev.GetSharedPtr();
283 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
284 HandleN0183_AIS(n0183_msg);
285 });
286
287 // CDDSE
288 Nmea0183Msg n0183_msg_CDDSE("CDDSE");
289 listener_N0183_CDDSE.Listen(n0183_msg_CDDSE, this, EVT_N0183_CDDSE);
290 Bind(EVT_N0183_CDDSE, [&](ObservedEvt ev) {
291 auto ptr = ev.GetSharedPtr();
292 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
293 HandleN0183_AIS(n0183_msg);
294 });
295
296 // TLL
297 Nmea0183Msg n0183_msg_TLL("TLL");
298 listener_N0183_TLL.Listen(n0183_msg_TLL, this, EVT_N0183_TLL);
299
300 Bind(EVT_N0183_TLL, [&](ObservedEvt ev) {
301 auto ptr = ev.GetSharedPtr();
302 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
303 HandleN0183_AIS(n0183_msg);
304 });
305
306 // TTM
307 Nmea0183Msg n0183_msg_ttm("TTM");
308 listener_N0183_TTM.Listen(n0183_msg_ttm, this, EVT_N0183_TTM);
309 Bind(EVT_N0183_TTM, [&](ObservedEvt ev) {
310 auto ptr = ev.GetSharedPtr();
311 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
312 HandleN0183_AIS(n0183_msg);
313 });
314
315 // OSD
316 Nmea0183Msg n0183_msg_OSD("OSD");
317 listener_N0183_OSD.Listen(n0183_msg_OSD, this, EVT_N0183_OSD);
318 Bind(EVT_N0183_OSD, [&](ObservedEvt ev) {
319 auto ptr = ev.GetSharedPtr();
320 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
321 HandleN0183_AIS(n0183_msg);
322 });
323
324 // WPL
325 Nmea0183Msg n0183_msg_WPL("WPL");
326 listener_N0183_WPL.Listen(n0183_msg_WPL, this, EVT_N0183_WPL);
327 Bind(EVT_N0183_WPL, [&](ObservedEvt ev) {
328 auto ptr = ev.GetSharedPtr();
329 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
330 HandleN0183_AIS(n0183_msg);
331 });
332
333 // SignalK
334 SignalkMsg sk_msg;
335 listener_SignalK.Listen(sk_msg, this, EVT_SIGNALK);
336 Bind(EVT_SIGNALK, [&](ObservedEvt ev) {
337 HandleSignalK(UnpackEvtPointer<SignalkMsg>(ev));
338 });
339
340 // AIS Class A PGN 129038
341 //-----------------------------
342 Nmea2000Msg n2k_msg_129038(static_cast<uint64_t>(129038));
343 listener_N2K_129038.Listen(n2k_msg_129038, this, EVT_N2K_129038);
344 Bind(EVT_N2K_129038, [&](ObservedEvt ev) {
345 HandleN2K_129038(UnpackEvtPointer<Nmea2000Msg>(ev));
346 });
347
348 // AIS Class B PGN 129039
349 //-----------------------------
350 Nmea2000Msg n2k_msg_129039(static_cast<uint64_t>(129039));
351 listener_N2K_129039.Listen(n2k_msg_129039, this, EVT_N2K_129039);
352 Bind(EVT_N2K_129039, [&](ObservedEvt ev) {
353 HandleN2K_129039(UnpackEvtPointer<Nmea2000Msg>(ev));
354 });
355
356 // AIS ATON PGN 129041
357 //-----------------------------
358 Nmea2000Msg n2k_msg_129041(static_cast<uint64_t>(129041));
359 listener_N2K_129041.Listen(n2k_msg_129041, this, EVT_N2K_129041);
360 Bind(EVT_N2K_129041, [&](ObservedEvt ev) {
361 HandleN2K_129041(UnpackEvtPointer<Nmea2000Msg>(ev));
362 });
363
364 // AIS static data class A PGN 129794
365 //-----------------------------
366 Nmea2000Msg n2k_msg_129794(static_cast<uint64_t>(129794));
367 listener_N2K_129794.Listen(n2k_msg_129794, this, EVT_N2K_129794);
368 Bind(EVT_N2K_129794, [&](ObservedEvt ev) {
369 HandleN2K_129794(UnpackEvtPointer<Nmea2000Msg>(ev));
370 });
371
372 // AIS static data class B part A PGN 129809
373 //-----------------------------
374 Nmea2000Msg n2k_msg_129809(static_cast<uint64_t>(129809));
375 listener_N2K_129809.Listen(n2k_msg_129809, this, EVT_N2K_129809);
376 Bind(EVT_N2K_129809, [&](ObservedEvt ev) {
377 HandleN2K_129809(UnpackEvtPointer<Nmea2000Msg>(ev));
378 });
379
380 // AIS static data class B part B PGN 129810
381 //-----------------------------
382 Nmea2000Msg n2k_msg_129810(static_cast<uint64_t>(129810));
383 listener_N2K_129810.Listen(n2k_msg_129810, this, EVT_N2K_129810);
384 Bind(EVT_N2K_129810, [&](ObservedEvt ev) {
385 HandleN2K_129810(UnpackEvtPointer<Nmea2000Msg>(ev));
386 });
387
388 // AIS Base Station report PGN 129793
389 //-----------------------------
390 Nmea2000Msg n2k_msg_129793(static_cast<uint64_t>(129793));
391 listener_N2K_129793.Listen(n2k_msg_129793, this, EVT_N2K_129793);
392 Bind(EVT_N2K_129793, [&](ObservedEvt ev) {
393 HandleN2K_129793(UnpackEvtPointer<Nmea2000Msg>(ev));
394 });
395}
396
397bool AisDecoder::HandleN0183_AIS(std::shared_ptr<const Nmea0183Msg> n0183_msg) {
398 std::string str = n0183_msg->payload;
399 wxString sentence(str.c_str());
400 DecodeN0183(sentence);
402 return true;
403}
404
405bool AisDecoder::HandleN2K_129038(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
406 std::vector<unsigned char> v = n2k_msg->payload;
407
408 uint8_t MessageID;
409 tN2kAISRepeat Repeat;
410 uint32_t UserID;
411 double Latitude;
412 double Longitude;
413 bool Accuracy;
414 bool RAIM;
415 uint8_t Seconds;
416 double COG;
417 double SOG;
418 double Heading;
419 double ROT;
420 tN2kAISNavStatus NavStat = N2kaisns_Under_Way_Motoring;
421 tN2kAISTransceiverInformation AISTransceiverInformation;
422
423 if (ParseN2kPGN129038(v, MessageID, Repeat, UserID, Latitude, Longitude,
424 Accuracy, RAIM, Seconds, COG, SOG, Heading, ROT,
425 NavStat, AISTransceiverInformation)) {
426 int mmsi = UserID;
427 // Stop here if the target shall be ignored
428 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi)) return false;
429
430 // Is this target already in the global target list?
431 // Search the current AISTargetList for an MMSI match
432 long mmsi_long = mmsi;
433 std::shared_ptr<AisTargetData> pTargetData = 0;
434 bool bnewtarget = false;
435
436 auto it = AISTargetList.find(mmsi);
437 if (it == AISTargetList.end()) // not found
438 {
439 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
440 bnewtarget = true;
441 m_n_targets++;
442 } else {
443 pTargetData = it->second; // find current entry
444 }
445
446 wxDateTime now = wxDateTime::Now();
447 now.MakeUTC();
448
449 // Populate the target_data
450 pTargetData->MMSI = mmsi;
451 pTargetData->MID = MessageID;
452 pTargetData->MMSI = mmsi;
453 pTargetData->Class = AIS_CLASS_A;
454 // Check for SART and friends by looking at first two digits of MMSI
455 if (97 == pTargetData->MMSI / 10000000) {
456 pTargetData->Class = AIS_SART;
457 // won't get a static report, so fake it here
458 pTargetData->StaticReportTicks = now.GetTicks();
459 }
460 pTargetData->NavStatus = (ais_nav_status)NavStat;
461 if (!N2kIsNA(SOG)) pTargetData->SOG = MS2KNOTS(SOG);
462 if (!N2kIsNA(COG)) pTargetData->COG = GeodesicRadToDeg(COG);
463 if (!N2kIsNA(Heading)) pTargetData->HDG = GeodesicRadToDeg(Heading);
464 if (!N2kIsNA(Longitude)) pTargetData->Lon = Longitude;
465 if (!N2kIsNA(Latitude)) pTargetData->Lat = Latitude;
466
467 pTargetData->ROTAIS = ROT;
468
469 double rot_dir = 1.0;
470
471 // FIXME (dave)
472 // if (ROT == 128)
473 // pTargetData->ROTAIS = -128; // not available codes as -128
474 // else if ((ROT & 0x80) == 0x80) {
475 // pTargetData->ROTAIS = ROT - 256; // convert to twos complement
476 // rot_dir = -1.0;
477 // }
478 //
479 // pTargetData->ROTIND = round(rot_dir * pow((ROT / 4.733), 2));
480
481 pTargetData->b_OwnShip =
482 AISTransceiverInformation ==
483 tN2kAISTransceiverInformation::N2kaisown_information_not_broadcast;
484 pTargetData->b_active = true;
485 pTargetData->b_lost = false;
486 pTargetData->b_positionOnceValid = true;
487 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
488 pTargetData->PositionReportTicks = now.GetTicks();
489
490 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
491 CommitAISTarget(pTargetData, "", true, bnewtarget);
492
494 return true;
495 } else
496 return false;
497}
498
499// AIS position reports for Class B
500bool AisDecoder::HandleN2K_129039(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
501 std::vector<unsigned char> v = n2k_msg->payload;
502
503 // Input:
504 // - N2kMsg NMEA2000 message to decode
505 // bool ParseN2kPGN129039(std::vector<unsigned char> &v, uint8_t &MessageID,
506 // tN2kAISRepeat &Repeat, uint32_t &UserID,
507 // double &Latitude, double &Longitude, bool
508 // &Accuracy, bool &RAIM, uint8_t &Seconds, double
509 // &COG, double &SOG, tN2kAISTransceiverInformation
510 // &AISTransceiverInformation, double &Heading,
511 // tN2kAISUnit &Unit, bool &Display, bool &DSC, bool
512 // &Band, bool &Msg22, tN2kAISMode &Mode, bool
513 // &State);
514
515 uint8_t MessageID;
516 tN2kAISRepeat Repeat;
517 uint32_t UserID;
518 double Latitude;
519 double Longitude;
520 bool Accuracy;
521 bool RAIM;
522 uint8_t Seconds;
523 double COG;
524 double SOG;
525 double Heading;
526 tN2kAISNavStatus NavStat = N2kaisns_Under_Way_Motoring;
527 tN2kAISTransceiverInformation AISTransceiverInformation;
528 tN2kAISUnit Unit;
529 bool DSC, Band, Msg22, State, Display;
530 tN2kAISMode Mode;
531
532 if (ParseN2kPGN129039(v, MessageID, Repeat, UserID, Latitude, Longitude,
533 Accuracy, RAIM, Seconds, COG, SOG,
534 AISTransceiverInformation, Heading, Unit, Display, DSC,
535 Band, Msg22, Mode, State)) {
536 int mmsi = UserID;
537 // Stop here if the target shall be ignored
538 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi)) return false;
539
540 // Is this target already in the global target list?
541 // Search the current AISTargetList for an MMSI match
542 long mmsi_long = mmsi;
543 std::shared_ptr<AisTargetData> pTargetData = 0;
544 bool bnewtarget = false;
545
546 auto it = AISTargetList.find(mmsi);
547 if (it == AISTargetList.end()) // not found
548 {
549 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
550 bnewtarget = true;
551 m_n_targets++;
552 } else {
553 pTargetData = it->second; // find current entry
554 }
555
556 wxDateTime now = wxDateTime::Now();
557 now.MakeUTC();
558
559 // Populate the target_data
560 pTargetData->MMSI = mmsi;
561 pTargetData->MID = MessageID;
562 if (!isBuoyMmsi(mmsi))
563 pTargetData->Class = AIS_CLASS_B;
564 else
565 pTargetData->Class = AIS_BUOY;
566
567 pTargetData->NavStatus = UNDEFINED; // Class B targets have no status.
568 if (!N2kIsNA(SOG)) pTargetData->SOG = MS2KNOTS(SOG);
569 if (!N2kIsNA(COG)) pTargetData->COG = GeodesicRadToDeg(COG);
570 if (!N2kIsNA(Heading)) pTargetData->HDG = GeodesicRadToDeg(Heading);
571 if (!N2kIsNA(Longitude)) pTargetData->Lon = Longitude;
572 if (!N2kIsNA(Latitude)) pTargetData->Lat = Latitude;
573
574 pTargetData->b_positionOnceValid = true;
575 pTargetData->b_active = true;
576 pTargetData->b_lost = false;
577 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
578 pTargetData->PositionReportTicks = now.GetTicks();
579 pTargetData->b_OwnShip =
580 AISTransceiverInformation ==
581 tN2kAISTransceiverInformation::N2kaisown_information_not_broadcast;
582
583 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
584 CommitAISTarget(pTargetData, "", true, bnewtarget);
585
587 return true;
588 } else
589 return false;
590
591 return true;
592}
593
594bool AisDecoder::HandleN2K_129041(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
595 std::vector<unsigned char> v = n2k_msg->payload;
596
597 tN2kAISAtoNReportData data;
598
599#if 0
600 struct tN2kAISAtoNReportData {
601 uint8_t MessageID;
602 tN2kAISRepeat Repeat;
603 uint32_t UserID;
604 double Longitude;
605 double Latitude;
606 bool Accuracy;
607 bool RAIM;
608 uint8_t Seconds;
609 double Length;
610 double Beam;
611 double PositionReferenceStarboard ;
612 double PositionReferenceTrueNorth;
613 tN2kAISAtoNType AtoNType;
614 bool OffPositionIndicator;
615 bool VirtualAtoNFlag;
616 bool AssignedModeFlag;
617 tN2kGNSStype GNSSType;
618 uint8_t AtoNStatus;
619 tN2kAISTransceiverInformation AISTransceiverInformation;
620 char AtoNName[34 + 1];
621#endif
622
623 if (ParseN2kPGN129041(v, data)) {
624 int mmsi = data.UserID;
625 // Stop here if the target shall be ignored
626 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi)) return false;
627
628 // Is this target already in the global target list?
629 // Search the current AISTargetList for an MMSI match
630 long mmsi_long = mmsi;
631 std::shared_ptr<AisTargetData> pTargetData = 0;
632 bool bnewtarget = false;
633
634 auto it = AISTargetList.find(mmsi);
635 if (it == AISTargetList.end()) // not found
636 {
637 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
638 bnewtarget = true;
639 m_n_targets++;
640 } else {
641 pTargetData = it->second; // find current entry
642 }
643
644 // Populate the target_data
645 pTargetData->MMSI = mmsi;
646
647 wxDateTime now = wxDateTime::Now();
648 now.MakeUTC();
649
650 int offpos = data.OffPositionIndicator; // off position flag
651 int virt = data.VirtualAtoNFlag; // virtual flag
652
653 if (virt)
654 pTargetData->NavStatus = ATON_VIRTUAL;
655 else
656 pTargetData->NavStatus = ATON_REAL;
657
658 pTargetData->m_utc_sec = data.Seconds;
659
660 if (pTargetData->m_utc_sec <= 59) {
661 pTargetData->NavStatus += 1;
662 if (offpos) pTargetData->NavStatus += 1;
663 }
664
665 data.AtoNName[34] = 0;
666 strncpy(pTargetData->ShipName, data.AtoNName, SHIP_NAME_LEN - 1);
667 pTargetData->ShipName[sizeof(pTargetData->ShipName) - 1] = '\0';
668 pTargetData->b_nameValid = true;
669 pTargetData->MID = 124; // Indicates a name from n2k
670
671 pTargetData->ShipType = data.AtoNType;
672 pTargetData->Class = AIS_ATON;
673
674 if (!N2kIsNA(data.Longitude)) pTargetData->Lon = data.Longitude;
675 if (!N2kIsNA(data.Latitude)) pTargetData->Lat = data.Latitude;
676 pTargetData->b_positionDoubtful = false;
677 pTargetData->b_positionOnceValid = true; // Got the position at least once
678 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
679 pTargetData->PositionReportTicks = now.GetTicks();
680
681 // FIXME (dave) Populate more fiddly static data
682
683 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
684 CommitAISTarget(pTargetData, "", true, bnewtarget);
685
686 touch_state.Notify();
687 return true;
688 } else
689 return false;
690}
691
692// AIS static data class A
693bool AisDecoder::HandleN2K_129794(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
694 std::vector<unsigned char> v = n2k_msg->payload;
695
696 uint8_t MessageID;
697 tN2kAISRepeat Repeat;
698 uint32_t UserID;
699 uint32_t IMOnumber;
700 char Callsign[21];
701 char Name[SHIP_NAME_LEN];
702 uint8_t VesselType;
703 double Length;
704 double Beam;
705 double PosRefStbd;
706 double PosRefBow;
707 uint16_t ETAdate;
708 double ETAtime;
709 double Draught;
710 char Destination[DESTINATION_LEN];
711 tN2kAISVersion AISversion;
712 tN2kGNSStype GNSStype;
713 tN2kAISDTE DTE;
714 tN2kAISTranceiverInfo AISinfo;
715
716 if (ParseN2kPGN129794(v, MessageID, Repeat, UserID, IMOnumber, Callsign, Name,
717 VesselType, Length, Beam, PosRefStbd, PosRefBow,
718 ETAdate, ETAtime, Draught, Destination, AISversion,
719 GNSStype, DTE, AISinfo)) {
720 int mmsi = UserID;
721 // Stop here if the target shall be ignored
722 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi)) return false;
723
724 // Is this target already in the global target list?
725 // Search the current AISTargetList for an MMSI match
726 long mmsi_long = mmsi;
727 std::shared_ptr<AisTargetData> pTargetData = 0;
728 bool bnewtarget = false;
729
730 auto it = AISTargetList.find(mmsi);
731 if (it == AISTargetList.end()) // not found
732 {
733 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
734 bnewtarget = true;
735 m_n_targets++;
736 } else {
737 pTargetData = it->second; // find current entry
738 }
739
740 // Populate the target_data
741 pTargetData->MMSI = mmsi;
742 strncpy(pTargetData->ShipName, Name, SHIP_NAME_LEN - 1);
743 pTargetData->ShipName[sizeof(pTargetData->ShipName) - 1] = '\0';
744 Name[sizeof(Name) - 1] = 0;
745 pTargetData->b_nameValid = true;
746 pTargetData->MID = 124; // Indicates a name from n2k
747
748 pTargetData->b_OwnShip =
749 AISinfo ==
750 tN2kAISTranceiverInfo::N2kaisti_Own_information_not_broadcast;
751
752 pTargetData->DimA = PosRefBow;
753 pTargetData->DimB = Length - PosRefBow;
754 pTargetData->DimC = Beam - PosRefStbd;
755 pTargetData->DimD = PosRefStbd;
756 pTargetData->Draft = Draught;
757 pTargetData->IMO = IMOnumber;
758 strncpy(pTargetData->CallSign, Callsign, CALL_SIGN_LEN - 1);
759 pTargetData->CallSign[sizeof(pTargetData->CallSign) - 1] = '\0';
760 pTargetData->ShipType = (unsigned char)VesselType;
761 strncpy(pTargetData->Destination, Destination, DESTINATION_LEN - 1);
762 pTargetData->Destination[sizeof(pTargetData->Destination) - 1] = '\0';
763 Destination[sizeof(Destination) - 1] = 0;
764
765 if (!N2kIsNA(ETAdate) && !N2kIsNA(ETAtime)) {
766 long secs = (ETAdate * 24 * 3600) + wxRound(ETAtime);
767 wxDateTime t((time_t)secs);
768 if (t.IsValid()) {
769 wxDateTime tz = t.ToUTC();
770 pTargetData->ETA_Mo = tz.GetMonth() + 1;
771 pTargetData->ETA_Day = tz.GetDay();
772 pTargetData->ETA_Hr = tz.GetHour();
773 pTargetData->ETA_Min = tz.GetMinute();
774 }
775 }
776
777 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
778 CommitAISTarget(pTargetData, "", true, bnewtarget);
779
781 return true;
782 } else
783 return false;
784}
785// AIS static data class B part A
786bool AisDecoder::HandleN2K_129809(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
787 std::vector<unsigned char> v = n2k_msg->payload;
788
789 uint8_t MessageID;
790 tN2kAISRepeat Repeat;
791 uint32_t UserID;
792 char Name[21];
793
794 if (ParseN2kPGN129809(v, MessageID, Repeat, UserID, Name)) {
795 int mmsi = UserID;
796 // Stop here if the target shall be ignored
797 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi)) return false;
798
799 // Is this target already in the global target list?
800 // Search the current AISTargetList for an MMSI match
801 long mmsi_long = mmsi;
802 std::shared_ptr<AisTargetData> pTargetData = 0;
803 bool bnewtarget = false;
804
805 auto it = AISTargetList.find(mmsi);
806 if (it == AISTargetList.end()) // not found
807 {
808 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
809 bnewtarget = true;
810 m_n_targets++;
811 } else {
812 pTargetData = it->second; // find current entry
813 }
814
815 // Populate the target_data
816 pTargetData->MMSI = mmsi;
817 Name[sizeof(Name) - 1] = 0;
818 strncpy(pTargetData->ShipName, Name, SHIP_NAME_LEN - 1);
819 pTargetData->b_nameValid = true;
820 pTargetData->MID = 124; // Indicates a name from n2k
821
822 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
823 CommitAISTarget(pTargetData, "", true, bnewtarget);
824
826 return true;
827
828 } else
829 return false;
830}
831
832// AIS static data class B part B
833bool AisDecoder::HandleN2K_129810(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
834 std::vector<unsigned char> v = n2k_msg->payload;
835
836 uint8_t MessageID;
837 tN2kAISRepeat Repeat;
838 uint32_t UserID;
839 uint8_t VesselType;
840 char Vendor[20];
841 char Callsign[20];
842 double Length;
843 double Beam;
844 double PosRefStbd;
845 double PosRefBow;
846 uint32_t MothershipID;
847
848 if (ParseN2kPGN129810(v, MessageID, Repeat, UserID, VesselType, Vendor,
849 Callsign, Length, Beam, PosRefStbd, PosRefBow,
850 MothershipID)) {
851 int mmsi = UserID;
852 // Stop here if the target shall be ignored
853 if (mmsi == g_OwnShipmmsi || IsTargetOnTheIgnoreList(mmsi)) return false;
854
855 // Is this target already in the global target list?
856 // Search the current AISTargetList for an MMSI match
857 long mmsi_long = mmsi;
858 std::shared_ptr<AisTargetData> pTargetData = 0;
859 bool bnewtarget = false;
860
861 auto it = AISTargetList.find(mmsi);
862 if (it == AISTargetList.end()) // not found
863 {
864 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
865 bnewtarget = true;
866 m_n_targets++;
867 } else {
868 pTargetData = it->second; // find current entry
869 }
870
871 // Populate the target_data
872 pTargetData->MMSI = mmsi;
873 pTargetData->DimA = PosRefBow;
874 pTargetData->DimB = Length - PosRefBow;
875 pTargetData->DimC = Beam - PosRefStbd;
876 pTargetData->DimD = PosRefStbd;
877 strncpy(pTargetData->CallSign, Callsign, CALL_SIGN_LEN - 1);
878 pTargetData->CallSign[sizeof(pTargetData->CallSign) - 1] = '\0';
879 pTargetData->ShipType = (unsigned char)VesselType;
880
881 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
882 CommitAISTarget(pTargetData, "", true, bnewtarget);
883
885 return true;
886 } else
887 return false;
888}
889
890// AIS Base Station Report
891bool AisDecoder::HandleN2K_129793(std::shared_ptr<const Nmea2000Msg> n2k_msg) {
892 std::vector<unsigned char> v = n2k_msg->payload;
893
894 uint8_t MessageID;
895 tN2kAISRepeat Repeat;
896 uint32_t UserID;
897 double Longitude;
898 double Latitude;
899 unsigned int SecondsSinceMidnight;
900 unsigned int DaysSinceEpoch;
901
902 if (ParseN2kPGN129793(v, MessageID, Repeat, UserID, Longitude, Latitude,
903 SecondsSinceMidnight, DaysSinceEpoch)) {
904 wxDateTime now = wxDateTime::Now();
905 now.MakeUTC();
906
907 // Is this target already in the global target list?
908 // Search the current AISTargetList for an MMSI match
909 int mmsi = UserID;
910 long mmsi_long = mmsi;
911 std::shared_ptr<AisTargetData> pTargetData = 0;
912 bool bnewtarget = false;
913
914 auto it = AISTargetList.find(mmsi);
915 if (it == AISTargetList.end()) // not found
916 {
917 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
918 bnewtarget = true;
919 m_n_targets++;
920 } else {
921 pTargetData = it->second; // find current entry
922 }
923
924 // Populate the target_data
925 pTargetData->MMSI = mmsi;
926 pTargetData->Class = AIS_BASE;
927
928 if (!N2kIsNA(Longitude)) pTargetData->Lon = Longitude;
929 if (!N2kIsNA(Latitude)) pTargetData->Lat = Latitude;
930 pTargetData->b_positionDoubtful = false;
931 pTargetData->b_positionOnceValid = true; // Got the position at least once
932 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
933 pTargetData->PositionReportTicks = now.GetTicks();
934
935 // FIXME (dave) Populate more fiddly static data
936
937 pSelectAIS->DeleteSelectablePoint((void *)(long)mmsi, SELTYPE_AISTARGET);
938 CommitAISTarget(pTargetData, "", true, bnewtarget);
939
941 return true;
942 } else
943 return false;
944}
945
946// void AisDecoder::HandleSignalK(std::shared_ptr<const SignalkMsg> sK_msg){
947// std::string msgTerminated = sK_msg->raw_message;;
948// wxString sentence(msgTerminated.c_str());
949// Decode(sentence);
950// touch_state.notify();
951// return true;
952// }
953
954void AisDecoder::BuildERIShipTypeHash(void) {
955 make_hash_ERI(8000, _("Vessel, type unknown"));
956 make_hash_ERI(8150, _("Freightbarge"));
957 make_hash_ERI(8160, _("Tankbarge"));
958 make_hash_ERI(8163, _("Tankbarge, dry cargo as if liquid (e.g. cement)"));
959 make_hash_ERI(8450, _("Service vessel, police patrol, port service"));
960 make_hash_ERI(8430, _("Pushboat, single"));
961 make_hash_ERI(8510, _("Object, not otherwise specified"));
962 make_hash_ERI(8470, _("Object, towed, not otherwise specified"));
963 make_hash_ERI(8490, _("Bunkership"));
964 make_hash_ERI(8010, _("Motor freighter"));
965 make_hash_ERI(8020, _("Motor tanker"));
966 make_hash_ERI(8021, _("Motor tanker, liquid cargo, type N"));
967 make_hash_ERI(8022, _("Motor tanker, liquid cargo, type C"));
968 make_hash_ERI(8023, _("Motor tanker, dry cargo as if liquid (e.g. cement)"));
969 make_hash_ERI(8030, _("Container vessel"));
970 make_hash_ERI(8040, _("Gas tanker"));
971 make_hash_ERI(8050, _("Motor freighter, tug"));
972 make_hash_ERI(8060, _("Motor tanker, tug"));
973 make_hash_ERI(8070, _("Motor freighter with one or more ships alongside"));
974 make_hash_ERI(8080, _("Motor freighter with tanker"));
975 make_hash_ERI(8090, _("Motor freighter pushing one or more freighters"));
976 make_hash_ERI(8100, _("Motor freighter pushing at least one tank-ship"));
977 make_hash_ERI(8110, _("Tug, freighter"));
978 make_hash_ERI(8120, _("Tug, tanker"));
979 make_hash_ERI(8130, _("Tug freighter, coupled"));
980 make_hash_ERI(8140, _("Tug, freighter/tanker, coupled"));
981 make_hash_ERI(8161, _("Tankbarge, liquid cargo, type N"));
982 make_hash_ERI(8162, _("Tankbarge, liquid cargo, type C"));
983 make_hash_ERI(8170, _("Freightbarge with containers"));
984 make_hash_ERI(8180, _("Tankbarge, gas"));
985 make_hash_ERI(8210, _("Pushtow, one cargo barge"));
986 make_hash_ERI(8220, _("Pushtow, two cargo barges"));
987 make_hash_ERI(8230, _("Pushtow, three cargo barges"));
988 make_hash_ERI(8240, _("Pushtow, four cargo barges"));
989 make_hash_ERI(8250, _("Pushtow, five cargo barges"));
990 make_hash_ERI(8260, _("Pushtow, six cargo barges"));
991 make_hash_ERI(8270, _("Pushtow, seven cargo barges"));
992 make_hash_ERI(8280, _("Pushtow, eight cargo barges"));
993 make_hash_ERI(8290, _("Pushtow, nine or more barges"));
994 make_hash_ERI(8310, _("Pushtow, one tank/gas barge"));
995 make_hash_ERI(8320,
996 _("Pushtow, two barges at least one tanker or gas barge"));
997 make_hash_ERI(8330,
998 _("Pushtow, three barges at least one tanker or gas barge"));
999 make_hash_ERI(8340,
1000 _("Pushtow, four barges at least one tanker or gas barge"));
1001 make_hash_ERI(8350,
1002 _("Pushtow, five barges at least one tanker or gas barge"));
1003 make_hash_ERI(8360,
1004 _("Pushtow, six barges at least one tanker or gas barge"));
1005 make_hash_ERI(8370,
1006 _("Pushtow, seven barges at least one tanker or gas barge"));
1007 make_hash_ERI(8380,
1008 _("Pushtow, eight barges at least one tanker or gas barge"));
1009 make_hash_ERI(
1010 8390, _("Pushtow, nine or more barges at least one tanker or gas barge"));
1011 make_hash_ERI(8400, _("Tug, single"));
1012 make_hash_ERI(8410, _("Tug, one or more tows"));
1013 make_hash_ERI(8420, _("Tug, assisting a vessel or linked combination"));
1014 make_hash_ERI(8430, _("Pushboat, single"));
1015 make_hash_ERI(8440, _("Passenger ship, ferry, cruise ship, red cross ship"));
1016 make_hash_ERI(8441, _("Ferry"));
1017 make_hash_ERI(8442, _("Red cross ship"));
1018 make_hash_ERI(8443, _("Cruise ship"));
1019 make_hash_ERI(8444, _("Passenger ship without accommodation"));
1020 make_hash_ERI(8460, _("Vessel, work maintenance craft, floating derrick, "
1021 "cable-ship, buoy-ship, dredge"));
1022 make_hash_ERI(8480, _("Fishing boat"));
1023 make_hash_ERI(8500, _("Barge, tanker, chemical"));
1024 make_hash_ERI(1500, _("General cargo Vessel maritime"));
1025 make_hash_ERI(1510, _("Unit carrier maritime"));
1026 make_hash_ERI(1520, _("Bulk carrier maritime"));
1027 make_hash_ERI(1530, _("Tanker"));
1028 make_hash_ERI(1540, _("Liquified gas tanker"));
1029 make_hash_ERI(1850, _("Pleasure craft, longer than 20 metres"));
1030 make_hash_ERI(1900, _("Fast ship"));
1031 make_hash_ERI(1910, _("Hydrofoil"));
1032}
1033
1034//----------------------------------------------------------------------------------
1035// Handle events from SignalK
1036//----------------------------------------------------------------------------------
1037void AisDecoder::HandleSignalK(std::shared_ptr<const SignalkMsg> sK_msg) {
1038 rapidjson::Document root;
1039
1040 root.Parse(sK_msg->raw_message);
1041
1042 if (root.HasParseError()) return;
1043
1044 if (root.HasMember("self")) {
1045 // m_signalk_selfid = _T("vessels.") + (root["self"].AsString());
1046 m_signalk_selfid =
1047 (root["self"]
1048 .GetString()); // Verified for OpenPlotter node.js server 1.20
1049 }
1050 if (m_signalk_selfid.IsEmpty()) {
1051 return; // Don't handle any messages (with out self) until we know how we
1052 // are
1053 }
1054 long mmsi = 0;
1055 int meteo_SiteID = 0;
1056 if (root.HasMember("context") && root["context"].IsString()) {
1057 wxString context = root["context"].GetString();
1058 if (context == m_signalk_selfid) {
1059#if 0
1060 wxLogMessage(_T("** Ignore context own ship.."));
1061#endif
1062 return;
1063 }
1064 wxString mmsi_string;
1065 if (context.StartsWith(_T("vessels.urn:mrn:imo:mmsi:"), &mmsi_string) ||
1066 context.StartsWith(_T("atons.urn:mrn:imo:mmsi:"), &mmsi_string) ||
1067 context.StartsWith(_T("aircraft.urn:mrn:imo:mmsi:"), &mmsi_string)) {
1068 // wxLogMessage(wxString::Format(_T("Context: %s, %s"),
1069 // context.c_str(), mmsi_string));
1070 if (mmsi_string.ToLong(&mmsi)) {
1071 // wxLogMessage(_T("Got MMSI from context."));
1072 } else {
1073 mmsi = 0;
1074 }
1075 } else if (context.StartsWith(_T("meteo.urn:mrn:imo:mmsi:"),
1076 &mmsi_string)) {
1077 // mmsi_string for a Meteo is like: 002655619:672707
1078 origin_mmsi = wxAtoi(wxString(mmsi_string).BeforeFirst(':'));
1079 meteo_SiteID = wxAtoi('1' + wxString(mmsi_string).AfterFirst(':'));
1080 // Preface "1" to distinguish e.g. "012345" from "12345"
1081 // Get a meteo mmsi_ID
1082 int meteo_mmsi = AisMeteoNewMmsi(origin_mmsi, 0, 0, 999, meteo_SiteID);
1083 if (meteo_mmsi)
1084 mmsi = meteo_mmsi;
1085 else
1086 mmsi = 0;
1087 }
1088 }
1089 if (mmsi == 0) {
1090 return; // Only handle ships with MMSI for now
1091 }
1092
1093 if (g_pMUX && g_pMUX->IsLogActive()) {
1094 wxString logmsg;
1095 logmsg.Printf("AIS :MMSI: %ld", mmsi);
1096 std::string source = sK_msg->source->to_string();
1097 g_pMUX->LogInputMessage(logmsg, source, false, false);
1098 }
1099
1100 // Stop here if the target shall be ignored
1101 if (IsTargetOnTheIgnoreList(mmsi)) return;
1102 // If self data is not set on the SK server, own ship mmsi
1103 // could fall through here but being set in options Own ship.
1104 if (mmsi == g_OwnShipmmsi) return;
1105
1106#if 0
1107 wxString dbg;
1108 wxJSONWriter writer;
1109 writer.Write(root, dbg);
1110
1111 wxString msg( _T("AisDecoder::OnEvtSignalK: ") );
1112 msg.append(dbg);
1113 wxLogMessage(msg);
1114#endif
1115 std::shared_ptr<AisTargetData> pTargetData = 0;
1116 std::shared_ptr<AisTargetData> pStaleTarget = NULL;
1117 bool bnewtarget = false;
1118 int last_report_ticks;
1119 wxDateTime now;
1120 getAISTarget(mmsi, pTargetData, pStaleTarget, bnewtarget, last_report_ticks,
1121 now);
1122 if (pTargetData) {
1123 pTargetData->MMSI = mmsi;
1124 getMmsiProperties(pTargetData);
1125 if (root.HasMember("updates") && root["updates"].IsArray()) {
1126 for (rapidjson::Value::ConstValueIterator itr = root["updates"].Begin();
1127 itr != root["updates"].End(); ++itr) {
1128 handleUpdate(pTargetData, bnewtarget, *itr);
1129 }
1130 }
1131
1132 // A SART can send wo any values first transmits. Detect class already here.
1133 if (97 == mmsi / 10000000) {
1134 pTargetData->Class = AIS_SART;
1135 } else if (1994 == mmsi / 100000) {
1136 // SignalK meteo data
1137 pTargetData->Class = AIS_METEO;
1138 pTargetData->met_data.original_mmsi = origin_mmsi;
1139 pTargetData->met_data.stationID = meteo_SiteID;
1140 /* Make a unique "shipname" for each station
1141 based on position inherited from meteo_SiteID */
1142 wxString met_name = pTargetData->ShipName;
1143 if (met_name.Find("METEO") == wxNOT_FOUND) {
1144 wxString s_id;
1145 int id1, id2;
1146 s_id << meteo_SiteID;
1147 id1 = wxAtoi(s_id.Mid(1, 3));
1148 id2 = wxAtoi(s_id.Mid(4, 3));
1149 met_name = "METEO ";
1150 met_name << wxString::Format("%03d", (id1 + id2)).Right(3);
1151 strncpy(pTargetData->ShipName, met_name, SHIP_NAME_LEN - 1);
1152 }
1153 pTargetData->b_nameValid = true;
1154 pTargetData->MID = 123; // Indicates a name from SignalK
1155 pTargetData->COG = -1.;
1156 pTargetData->HDG = 511;
1157 pTargetData->SOG = -1.;
1158 pTargetData->b_NoTrack = true;
1159 pTargetData->b_show_track = false;
1160 }
1161 pTargetData->b_OwnShip = false;
1162 AISTargetList[pTargetData->MMSI] = pTargetData;
1163 }
1164}
1165
1166void AisDecoder::handleUpdate(std::shared_ptr<AisTargetData> pTargetData,
1167 bool bnewtarget, const rapidjson::Value &update) {
1168 wxString sfixtime = "";
1169
1170 if (update.HasMember("timestamp")) {
1171 sfixtime = update["timestamp"].GetString();
1172 }
1173 if (update.HasMember("values") && update["values"].IsArray()) {
1174 for (rapidjson::Value::ConstValueIterator itr = update["values"].Begin();
1175 itr != update["values"].End(); ++itr) {
1176 updateItem(pTargetData, bnewtarget, *itr, sfixtime);
1177 }
1178 }
1179 wxDateTime now = wxDateTime::Now();
1180 pTargetData->m_utc_hour = now.ToUTC().GetHour();
1181 pTargetData->m_utc_min = now.ToUTC().GetMinute();
1182 pTargetData->m_utc_sec = now.ToUTC().GetSecond();
1183 // pTargetData->NavStatus = 15; // undefined
1184 pTargetData->b_active = true;
1185 pTargetData->b_lost = false;
1186
1187 if (pTargetData->b_positionOnceValid) {
1188 long mmsi_long = pTargetData->MMSI;
1189 SelectItem *pSel =
1190 pSelectAIS->AddSelectablePoint(pTargetData->Lat, pTargetData->Lon,
1191 (void *)mmsi_long, SELTYPE_AISTARGET);
1192 pSel->SetUserData(pTargetData->MMSI);
1193 }
1194 UpdateOneCPA(pTargetData.get());
1195 if (pTargetData->b_show_track) UpdateOneTrack(pTargetData.get());
1196}
1197
1198void AisDecoder::updateItem(std::shared_ptr<AisTargetData> pTargetData,
1199 bool bnewtarget, const rapidjson::Value &item,
1200 wxString &sfixtime) const {
1201 if (item.HasMember("path") && item.HasMember("value")) {
1202 const wxString &update_path = item["path"].GetString();
1203 if (update_path == _T("navigation.position")) {
1204 if (item["value"].HasMember("latitude") &&
1205 item["value"].HasMember("longitude")) {
1206 wxDateTime now = wxDateTime::Now();
1207 now.MakeUTC();
1208 double lat = item["value"]["latitude"].GetDouble();
1209 double lon = item["value"]["longitude"].GetDouble();
1210 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
1211 pTargetData->PositionReportTicks = now.GetTicks();
1212 pTargetData->StaticReportTicks = now.GetTicks();
1213 pTargetData->Lat = lat;
1214 pTargetData->Lon = lon;
1215 pTargetData->b_positionOnceValid = true;
1216 pTargetData->b_positionDoubtful = false;
1217 }
1218
1219 /* Not implemented in SK server (2024-01)
1220 if (item["value"].HasMember("altitude")) {
1221 pTargetData->altitude = item["value"]["altitude "].GetInt();
1222 }*/
1223 } else if (update_path == _T("navigation.speedOverGround") &&
1224 item["value"].IsNumber()) {
1225 pTargetData->SOG = item["value"].GetDouble() * ms_to_knot_factor;
1226 } else if (update_path == _T("navigation.courseOverGroundTrue") &&
1227 item["value"].IsNumber()) {
1228 pTargetData->COG = GEODESIC_RAD2DEG(item["value"].GetDouble());
1229 } else if (update_path == _T("navigation.headingTrue") &&
1230 item["value"].IsNumber()) {
1231 pTargetData->HDG = GEODESIC_RAD2DEG(item["value"].GetDouble());
1232 } else if (update_path == _T("navigation.rateOfTurn") &&
1233 item["value"].IsNumber()) {
1234 pTargetData->ROTAIS = 4.733 * sqrt(item["value"].GetDouble());
1235 } else if (update_path == _T("design.aisShipType")) {
1236 if (item["value"].HasMember("id")) {
1237 if (!pTargetData->b_isDSCtarget) {
1238 pTargetData->ShipType = item["value"]["id"].GetUint();
1239 }
1240 }
1241 } else if (update_path == _T("atonType")) {
1242 if (item["value"].HasMember("id")) {
1243 pTargetData->ShipType = item["value"]["id"].GetUint();
1244 }
1245 } else if (update_path == _T("virtual")) {
1246 if (item["value"].GetBool()) {
1247 pTargetData->NavStatus = ATON_VIRTUAL;
1248 } else {
1249 pTargetData->NavStatus = ATON_REAL;
1250 }
1251 } else if (update_path == _T("offPosition")) {
1252 if (item["value"].GetBool()) {
1253 if (ATON_REAL == pTargetData->NavStatus) {
1254 pTargetData->NavStatus = ATON_REAL_OFFPOSITION;
1255 } else if (ATON_VIRTUAL == pTargetData->NavStatus) {
1256 pTargetData->NavStatus = ATON_VIRTUAL_OFFPOSITION;
1257 }
1258 }
1259 } else if (update_path == _T("design.draft")) {
1260 if (item["value"].HasMember("maximum") &&
1261 item["value"]["maximum"].IsNumber()) {
1262 pTargetData->Draft = item["value"]["maximum"].GetDouble();
1263 pTargetData->Euro_Draft = item["value"]["maximum"].GetDouble();
1264 }
1265 if (item["value"].HasMember("current") &&
1266 item["value"]["current"].IsNumber()) {
1267 double draft = item["value"]["current"].GetDouble();
1268 if (draft > 0) {
1269 pTargetData->Draft = draft;
1270 pTargetData->Euro_Draft = draft;
1271 }
1272 }
1273 } else if (update_path == _T("design.length")) {
1274 if (pTargetData->DimB == 0) {
1275 if (item["value"].HasMember("overall")) {
1276 if (item["value"]["overall"].IsNumber()) {
1277 pTargetData->Euro_Length = item["value"]["overall"].GetDouble();
1278 pTargetData->DimA = item["value"]["overall"].GetDouble();
1279 }
1280 pTargetData->DimB = 0;
1281 }
1282 }
1283 } else if (update_path == _T("sensors.ais.class")) {
1284 wxString aisclass = item["value"].GetString();
1285 if (aisclass == _T("A")) {
1286 if (!pTargetData->b_isDSCtarget) pTargetData->Class = AIS_CLASS_A;
1287 } else if (aisclass == _T("B")) {
1288 if (!pTargetData->b_isDSCtarget) {
1289 if (!isBuoyMmsi(pTargetData->MMSI))
1290 pTargetData->Class = AIS_CLASS_B;
1291 else
1292 pTargetData->Class = AIS_BUOY;
1293
1294 // Class B targets have no status. Enforce this...
1295 pTargetData->NavStatus = UNDEFINED;
1296 }
1297 } else if (aisclass == _T("BASE")) {
1298 pTargetData->Class = AIS_BASE;
1299 } else if (aisclass == _T("ATON")) {
1300 pTargetData->Class = AIS_ATON;
1301 }
1302 } else if (update_path == _T("sensors.ais.fromBow")) {
1303 if (pTargetData->DimB == 0 && pTargetData->DimA != 0) {
1304 int length = pTargetData->DimA;
1305 if (item["value"].IsNumber()) {
1306 pTargetData->DimA = item["value"].GetDouble();
1307 pTargetData->DimB = length - item["value"].GetDouble();
1308 }
1309 }
1310 } else if (update_path == _T("design.beam")) {
1311 if (pTargetData->DimD == 0) {
1312 if (item["value"].IsNumber()) {
1313 pTargetData->Euro_Beam = item["value"].GetDouble();
1314 pTargetData->DimC = item["value"].GetDouble();
1315 }
1316 pTargetData->DimD = 0;
1317 }
1318 } else if (update_path == _T("sensors.ais.fromCenter")) {
1319 if (pTargetData->DimD == 0 && pTargetData->DimC != 0) {
1320 int beam = pTargetData->DimC;
1321 int center = beam / 2;
1322 if (item["value"].IsNumber()) {
1323 // FIXME (nohal): Dim* are int, but we have seen data streams with
1324 // doubles in them...
1325 pTargetData->DimC = center + item["value"].GetDouble();
1326 pTargetData->DimD = beam - pTargetData->DimC;
1327 }
1328 }
1329 } else if (update_path == _T("navigation.state")) {
1330 wxString state = item["value"].GetString();
1331 if (state == _T("motoring")) {
1332 pTargetData->NavStatus = UNDERWAY_USING_ENGINE;
1333 } else if (state == _T("anchored")) {
1334 pTargetData->NavStatus = AT_ANCHOR;
1335 } else if (state == _T("not under command")) {
1336 pTargetData->NavStatus = NOT_UNDER_COMMAND;
1337 } else if (state == _T("restricted manouverability")) {
1338 pTargetData->NavStatus = RESTRICTED_MANOEUVRABILITY;
1339 } else if (state == _T("constrained by draft")) {
1340 pTargetData->NavStatus = CONSTRAINED_BY_DRAFT;
1341 } else if (state == _T("moored")) {
1342 pTargetData->NavStatus = MOORED;
1343 } else if (state == _T("aground")) {
1344 pTargetData->NavStatus = AGROUND;
1345 } else if (state == _T("fishing")) {
1346 pTargetData->NavStatus = FISHING;
1347 } else if (state == _T("sailing")) {
1348 pTargetData->NavStatus = UNDERWAY_SAILING;
1349 } else if (state == _T("hazardous material high speed")) {
1350 pTargetData->NavStatus = HSC;
1351 } else if (state == _T("hazardous material wing in ground")) {
1352 pTargetData->NavStatus = WIG;
1353 } else if (state == _T("ais-sart")) {
1354 pTargetData->NavStatus = RESERVED_14;
1355 } else {
1356 pTargetData->NavStatus = UNDEFINED;
1357 }
1358 } else if (update_path == _T("navigation.destination.commonName")) {
1359 const wxString &destination = item["value"].GetString();
1360 pTargetData->Destination[0] = '\0';
1361 strncpy(pTargetData->Destination, destination.c_str(),
1362 DESTINATION_LEN - 1);
1363 } else if (update_path == _T("navigation.specialManeuver")) {
1364 if (strcmp("not available", item["value"].GetString()) != 0 &&
1365 pTargetData->IMO < 1) {
1366 const wxString &bluesign = item["value"].GetString();
1367 if (_T("not engaged") == bluesign) {
1368 pTargetData->blue_paddle = 1;
1369 }
1370 if (_T("engaged") == bluesign) {
1371 pTargetData->blue_paddle = 2;
1372 }
1373 pTargetData->b_blue_paddle =
1374 pTargetData->blue_paddle == 2 ? true : false;
1375 }
1376 } else if (update_path == _T("sensors.ais.designatedAreaCode")) {
1377 if (item["value"].GetInt() == 200) { // European inland
1378 pTargetData->b_hasInlandDac = true;
1379 }
1380 } else if (update_path == _T("sensors.ais.functionalId")) {
1381 if (item["value"].GetInt() == 10 && pTargetData->b_hasInlandDac) {
1382 // "Inland ship static and voyage related data"
1383 pTargetData->b_isEuroInland = true;
1384 }
1385
1386 // METEO Data
1387 } else if (update_path == "environment.date") {
1388 wxString issued = item["value"].GetString();
1389 if (issued.Len()) {
1390 // Parse ISO 8601 date/time
1391 wxDateTime tz;
1392 ParseGPXDateTime(tz, issued);
1393 pTargetData->met_data.day = tz.GetDay();
1394 pTargetData->met_data.hour = tz.GetHour();
1395 pTargetData->met_data.minute = tz.GetMinute();
1396 }
1397 } else if (update_path == "environment.wind.averageSpeed" &&
1398 item["value"].IsNumber()) {
1399 pTargetData->met_data.wind_kn = MS2KNOTS(item["value"].GetDouble());
1400 } else if (update_path == "environment.wind.gust" &&
1401 item["value"].IsNumber()) {
1402 pTargetData->met_data.wind_gust_kn = MS2KNOTS(item["value"].GetDouble());
1403 } else if (update_path == "environment.wind.directionTrue" &&
1404 item["value"].IsNumber()) {
1405 pTargetData->met_data.wind_dir =
1406 GEODESIC_RAD2DEG(item["value"].GetDouble());
1407 } else if (update_path == "environment.wind.gustDirectionTrue" &&
1408 item["value"].IsNumber()) {
1409 pTargetData->met_data.wind_gust_dir =
1410 GEODESIC_RAD2DEG(item["value"].GetDouble());
1411 } else if (update_path == "environment.outside.temperature" &&
1412 item["value"].IsNumber()) {
1413 pTargetData->met_data.air_temp = KelvinToC(item["value"].GetDouble());
1414 } else if (update_path == "environment.outside.relativeHumidity" &&
1415 item["value"].IsNumber()) {
1416 pTargetData->met_data.rel_humid = item["value"].GetDouble();
1417 } else if (update_path == "environment.outside.dewPointTemperature" &&
1418 item["value"].IsNumber()) {
1419 pTargetData->met_data.dew_point = KelvinToC(item["value"].GetDouble());
1420 } else if (update_path == "environment.outside.pressure" &&
1421 item["value"].IsNumber()) {
1422 pTargetData->met_data.airpress =
1423 static_cast<int>(item["value"].GetDouble() / 100);
1424 } else if (update_path == "environment.water.level" &&
1425 item["value"].IsNumber()) {
1426 pTargetData->met_data.water_lev_dev = item["value"].GetDouble();
1427 } else if (update_path == "environment.water.current.drift" &&
1428 item["value"].IsNumber()) { // surfcurrspd
1429 pTargetData->met_data.current = MS2KNOTS(item["value"].GetDouble());
1430 } else if (update_path == "environment.water.current.set" &&
1431 item["value"].IsNumber()) { // surfcurrdir
1432 pTargetData->met_data.curr_dir =
1433 GEODESIC_RAD2DEG(item["value"].GetDouble());
1434 } else if (update_path == "environment.water.levelTendencyValue" &&
1435 item["value"].IsNumber()) {
1436 pTargetData->met_data.water_lev_trend =
1437 static_cast<int>(item["value"].GetDouble());
1438 } else if (update_path == "environment.water.levelTendency") {
1439 // Don't use this text we parse it ourself.
1440 } else if (update_path == "environment.water.waves.significantHeight" &&
1441 item["value"].IsNumber()) {
1442 pTargetData->met_data.wave_height = item["value"].GetDouble();
1443 } else if (update_path == "environment.water.waves.period" &&
1444 item["value"].IsNumber()) {
1445 pTargetData->met_data.wave_period =
1446 static_cast<int>(item["value"].GetDouble());
1447 } else if (update_path == "environment.water.waves.directionTrue" &&
1448 item["value"].IsNumber()) {
1449 pTargetData->met_data.wave_dir =
1450 GEODESIC_RAD2DEG(item["value"].GetDouble());
1451 } else if (update_path == "environment.water.swell.height" &&
1452 item["value"].IsNumber()) {
1453 pTargetData->met_data.swell_height = item["value"].GetDouble();
1454 } else if (update_path == "environment.water.swell.period" &&
1455 item["value"].IsNumber()) {
1456 pTargetData->met_data.swell_per =
1457 static_cast<int>(item["value"].GetDouble());
1458 } else if (update_path == "environment.water.swell.directionTrue" &&
1459 item["value"].IsNumber()) {
1460 pTargetData->met_data.swell_dir =
1461 GEODESIC_RAD2DEG(item["value"].GetDouble());
1462 } else if (update_path == "environment.water.temperature" &&
1463 item["value"].IsNumber()) {
1464 pTargetData->met_data.water_temp = KelvinToC(item["value"].GetDouble());
1465 } else if (update_path == "environment.water.salinity" &&
1466 item["value"].IsNumber()) {
1467 pTargetData->met_data.salinity = item["value"].GetDouble();
1468 } else if (update_path == "environment.water.ice") {
1469 // Don't use. We parse it ourself
1470 } else if (update_path == "environment.water.iceValue" &&
1471 item["value"].IsNumber()) {
1472 pTargetData->met_data.ice = static_cast<int>(item["value"].GetDouble());
1473 } else if (update_path == "environment.water.seaStateValue" &&
1474 item["value"].IsNumber()) {
1475 pTargetData->met_data.seastate =
1476 static_cast<int>(item["value"].GetDouble());
1477 } else if (update_path == "environment.water.seaState") {
1478 // This is the parsed (air!) Bf-scale. Don't use
1479 } else if (update_path == "environment.outside.precipitation") {
1480 // Don't use. We parse it ourself
1481 } else if (update_path == "environment.outside.precipitationValue" &&
1482 item["value"].IsNumber()) {
1483 pTargetData->met_data.precipitation =
1484 static_cast<int>(item["value"].GetDouble());
1485 } else if (update_path == "environment.outside.pressureTendencyValue" &&
1486 item["value"].IsNumber()) {
1487 pTargetData->met_data.airpress_tend =
1488 static_cast<int>(item["value"].GetDouble());
1489 } else if (update_path == "environment.outside.pressureTendency") {
1490 // Parsed value, don't use, we do it ourself
1491 } else if (update_path == "environment.outside.horizontalVisibility" &&
1492 item["value"].IsNumber()) {
1493 pTargetData->met_data.hor_vis =
1494 GEODESIC_METERS2NM(item["value"].GetDouble());
1495 } else if (update_path ==
1496 "environment.outside.horizontalVisibility.overRange") {
1497 pTargetData->met_data.hor_vis_GT = item["value"].GetBool();
1498 } else if (update_path == _T("")) {
1499 if (item["value"].HasMember("name")) {
1500 const wxString &name = item["value"]["name"].GetString();
1501 strncpy(pTargetData->ShipName, name.c_str(), SHIP_NAME_LEN - 1);
1502 pTargetData->b_nameValid = true;
1503 pTargetData->MID = 123; // Indicates a name from SignalK
1504 } else if (item["value"].HasMember("registrations")) {
1505 const wxString &imo = item["value"]["registrations"]["imo"].GetString();
1506 pTargetData->IMO = wxAtoi(imo.Right(7));
1507 } else if (item["value"].HasMember("communication")) {
1508 const wxString &callsign =
1509 item["value"]["communication"]["callsignVhf"].GetString();
1510 strncpy(pTargetData->CallSign, callsign.c_str(), 7);
1511 }
1512 if (item["value"].HasMember("mmsi") &&
1513 1994 != (pTargetData->MMSI) / 100000) { // Meteo
1514 long mmsi;
1515 wxString tmp = item["value"]["mmsi"].GetString();
1516 if (tmp.ToLong(&mmsi)) {
1517 pTargetData->MMSI = mmsi;
1518
1519 if (97 == mmsi / 10000000) {
1520 pTargetData->Class = AIS_SART;
1521 }
1522 if (111 == mmsi / 1000000) {
1523 pTargetData->b_SarAircraftPosnReport = true;
1524 }
1525
1526 AISshipNameCache(pTargetData.get(), AISTargetNamesC, AISTargetNamesNC,
1527 mmsi);
1528 }
1529 }
1530 } else {
1531 wxLogMessage(wxString::Format(
1532 _T("** AisDecoder::updateItem: unhandled path %s"), update_path));
1533#if 1
1534 rapidjson::StringBuffer buffer;
1535 rapidjson::Writer<rapidjson::StringBuffer> writer(buffer);
1536 item.Accept(writer);
1537 wxString msg(_T("update: "));
1538 msg.append(buffer.GetString());
1539 wxLogMessage(msg);
1540#endif
1541 }
1542 }
1543}
1544
1545//----------------------------------------------------------------------------------
1546// Decode a single AIVDO sentence to a Generic Position Report
1547//----------------------------------------------------------------------------------
1548AisError AisDecoder::DecodeSingleVDO(const wxString &str, GenericPosDatEx *pos,
1549 wxString *accumulator) {
1550 // Make some simple tests for validity
1551 if (str.Len() > 128) return AIS_NMEAVDX_TOO_LONG;
1552
1553 if (!NMEACheckSumOK(str)) return AIS_NMEAVDX_CHECKSUM_BAD;
1554
1555 if (!pos) return AIS_GENERIC_ERROR;
1556
1557 if (!accumulator) return AIS_GENERIC_ERROR;
1558
1559 // We only process AIVDO messages
1560 if (!str.Mid(1, 5).IsSameAs(_T("AIVDO"))) return AIS_GENERIC_ERROR;
1561
1562 // Use a tokenizer to pull out the first 4 fields
1563 wxStringTokenizer tkz(str, _T(","));
1564
1565 wxString token;
1566 token = tkz.GetNextToken(); // !xxVDx
1567
1568 token = tkz.GetNextToken();
1569 int nsentences = atoi(token.mb_str());
1570
1571 token = tkz.GetNextToken();
1572 int isentence = atoi(token.mb_str());
1573
1574 token = tkz.GetNextToken(); // skip 2 fields
1575 token = tkz.GetNextToken();
1576
1577 wxString string_to_parse;
1578 string_to_parse.Clear();
1579
1580 // Fill the output structure with all NANs
1581 pos->kLat = NAN;
1582 pos->kLon = NAN;
1583 pos->kCog = NAN;
1584 pos->kSog = NAN;
1585 pos->kHdt = NAN;
1586 pos->kVar = NAN;
1587 pos->kHdm = NAN;
1588
1589 // Simple case first
1590 // First and only part of a one-part sentence
1591 if ((1 == nsentences) && (1 == isentence)) {
1592 string_to_parse = tkz.GetNextToken(); // the encapsulated data
1593 }
1594
1595 else if (nsentences > 1) {
1596 if (1 == isentence) {
1597 *accumulator = tkz.GetNextToken(); // the encapsulated data
1598 }
1599
1600 else {
1601 accumulator->Append(tkz.GetNextToken());
1602 }
1603
1604 if (isentence == nsentences) { // ready to parse
1605 string_to_parse = *accumulator;
1606 }
1607 }
1608
1609 if (string_to_parse.IsEmpty() &&
1610 (nsentences > 1)) { // not ready, so return with NAN
1611 return AIS_INCOMPLETE_MULTIPART; // and non-zero return
1612 }
1613
1614 // Create the bit accessible string
1615 AisBitstring strbit(string_to_parse.mb_str());
1616
1617 // auto TargetData = std::make_unique<AisTargetData>(
1618 // *AisTargetDataMaker::GetInstance().GetTargetData());
1619
1620 auto TargetData = AisTargetDataMaker::GetInstance().GetTargetData();
1621
1622 bool bdecode_result = Parse_VDXBitstring(&strbit, TargetData);
1623
1624 if (bdecode_result) {
1625 switch (TargetData->MID) {
1626 case 1:
1627 case 2:
1628 case 3:
1629 case 18: {
1630 if (!TargetData->b_positionDoubtful) {
1631 pos->kLat = TargetData->Lat;
1632 pos->kLon = TargetData->Lon;
1633 } else {
1634 pos->kLat = NAN;
1635 pos->kLon = NAN;
1636 }
1637
1638 if (TargetData->COG == 360.0)
1639 pos->kCog = NAN;
1640 else
1641 pos->kCog = TargetData->COG;
1642
1643 if (TargetData->SOG > 102.2)
1644 pos->kSog = NAN;
1645 else
1646 pos->kSog = TargetData->SOG;
1647
1648 if ((int)TargetData->HDG == 511)
1649 pos->kHdt = NAN;
1650 else
1651 pos->kHdt = TargetData->HDG;
1652
1653 // VDO messages do not contain variation or magnetic heading
1654 pos->kVar = NAN;
1655 pos->kHdm = NAN;
1656 break;
1657 }
1658 default:
1659 return AIS_GENERIC_ERROR; // unrecognised sentence
1660 }
1661
1662 return AIS_NoError;
1663 } else
1664 return AIS_GENERIC_ERROR;
1665}
1666
1667//----------------------------------------------------------------------------------------
1668// Decode NMEA VDM/VDO/FRPOS/DSCDSE/TTM/TLL/OSD/RSD/TLB/WPL sentence to AIS
1669// Target(s)
1670//----------------------------------------------------------------------------------------
1671
1672AisError AisDecoder::DecodeN0183(const wxString &str) {
1673 AisError ret = AIS_GENERIC_ERROR;
1674 wxString string_to_parse;
1675
1676 double gpsg_lat, gpsg_lon, gpsg_mins, gpsg_degs;
1677 double gpsg_cog, gpsg_sog, gpsg_utc_time;
1678 int gpsg_utc_hour = 0;
1679 int gpsg_utc_min = 0;
1680 int gpsg_utc_sec = 0;
1681 char gpsg_name_str[21];
1682 wxString gpsg_date;
1683
1684 bool bdecode_result = false;
1685
1686 int gpsg_mmsi = 0;
1687 int arpa_mmsi = 0;
1688 int aprs_mmsi = 0;
1689 int mmsi = 0;
1690
1691 long arpa_tgt_num = 0;
1692 double arpa_sog = 0.;
1693 double arpa_cog = 0.;
1694 double arpa_lat = 0.;
1695 double arpa_lon = 0.;
1696 double arpa_dist = 0.;
1697 double arpa_brg = 0.;
1698 wxString arpa_brgunit;
1699 wxString arpa_status;
1700 wxString arpa_distunit;
1701 wxString arpa_cogunit;
1702 wxString arpa_reftarget;
1703 double arpa_mins, arpa_degs;
1704 double arpa_utc_time;
1705 int arpa_utc_hour = 0;
1706 int arpa_utc_min = 0;
1707 int arpa_utc_sec = 0;
1708 char arpa_name_str[21];
1709 bool arpa_lost = true;
1710 bool arpa_nottracked = false;
1711
1712 double aprs_lat = 0.;
1713 double aprs_lon = 0.;
1714 char aprs_name_str[21];
1715 double aprs_mins, aprs_degs;
1716
1717 std::shared_ptr<AisTargetData> pTargetData = 0;
1718 std::shared_ptr<AisTargetData> pStaleTarget = NULL;
1719 bool bnewtarget = false;
1720 int last_report_ticks;
1721
1722 // Make some simple tests for validity
1723
1724 if (str.Len() > 128) return AIS_NMEAVDX_TOO_LONG;
1725
1726 if (!NMEACheckSumOK(str)) {
1727 return AIS_NMEAVDX_CHECKSUM_BAD;
1728 }
1729 if (str.Mid(1, 2).IsSameAs(_T("CD"))) {
1730 ProcessDSx(str);
1731 return AIS_NoError;
1732 } else if (str.Mid(3, 3).IsSameAs(_T("TTM"))) {
1733 //$--TTM,xx,x.x,x.x,a,x.x,x.x,a,x.x,x.x,a,c--c,a,a*hh <CR><LF>
1734 // or
1735 //$--TTM,xx,x.x,x.x,a,x.x,x.x,a,x.x,x.x,a,c--c,a,a,hhmmss.ss,a*hh<CR><LF>
1736 wxString string(str);
1737 wxStringTokenizer tkz(string, _T(",*"));
1738
1739 wxString token;
1740 token = tkz.GetNextToken(); // Sentence (xxTTM)
1741 token = tkz.GetNextToken(); // 1) Target Number
1742 token.ToLong(&arpa_tgt_num);
1743 token = tkz.GetNextToken(); // 2)Target Distance
1744 token.ToDouble(&arpa_dist);
1745 token = tkz.GetNextToken(); // 3) Bearing from own ship
1746 token.ToDouble(&arpa_brg);
1747 arpa_brgunit = tkz.GetNextToken(); // 4) Bearing Units
1748 if (arpa_brgunit == _T("R")) {
1749 if (std::isnan(arpa_ref_hdg)) {
1750 if (!std::isnan(gHdt))
1751 arpa_brg += gHdt;
1752 else
1753 arpa_brg += gCog;
1754 } else
1755 arpa_brg += arpa_ref_hdg;
1756 if (arpa_brg >= 360.) arpa_brg -= 360.;
1757 }
1758 token = tkz.GetNextToken(); // 5) Target speed
1759 token.ToDouble(&arpa_sog);
1760 token = tkz.GetNextToken(); // 6) Target Course
1761 token.ToDouble(&arpa_cog);
1762 arpa_cogunit = tkz.GetNextToken(); // 7) Course Units
1763 if (arpa_cogunit == _T("R")) {
1764 if (std::isnan(arpa_ref_hdg)) {
1765 if (!std::isnan(gHdt))
1766 arpa_cog += gHdt;
1767 else
1768 arpa_cog += gCog;
1769 } else
1770 arpa_cog += arpa_ref_hdg;
1771 if (arpa_cog >= 360.) arpa_cog -= 360.;
1772 }
1773 token = tkz.GetNextToken(); // 8) Distance of closest-point-of-approach
1774 token = tkz.GetNextToken(); // 9) Time until closest-point-of-approach "-"
1775 // means increasing
1776 arpa_distunit = tkz.GetNextToken(); // 10)Speed/ dist unit
1777 token = tkz.GetNextToken(); // 11) Target name
1778 if (token == wxEmptyString)
1779 token = wxString::Format(_T("ARPA %ld"), arpa_tgt_num);
1780 int len = wxMin(token.Length(), 20);
1781 strncpy(arpa_name_str, token.mb_str(), len);
1782 arpa_name_str[len] = 0;
1783 arpa_status = tkz.GetNextToken(); // 12) Target Status
1784 if (arpa_status != _T( "L" )) {
1785 arpa_lost = false;
1786 } else if (arpa_status != wxEmptyString)
1787 arpa_nottracked = true;
1788 arpa_reftarget = tkz.GetNextToken(); // 13) Reference Target
1789 if (tkz.HasMoreTokens()) {
1790 token = tkz.GetNextToken();
1791 token.ToDouble(&arpa_utc_time);
1792 arpa_utc_hour = (int)(arpa_utc_time / 10000.0);
1793 arpa_utc_min = (int)(arpa_utc_time / 100.0) - arpa_utc_hour * 100;
1794 arpa_utc_sec =
1795 (int)arpa_utc_time - arpa_utc_hour * 10000 - arpa_utc_min * 100;
1796 } else {
1797 arpa_utc_hour = wxDateTime::Now().ToUTC().GetHour();
1798 arpa_utc_min = wxDateTime::Now().ToUTC().GetMinute();
1799 arpa_utc_sec = wxDateTime::Now().ToUTC().GetSecond();
1800 }
1801
1802 if (arpa_distunit == _T("K")) {
1803 arpa_dist = fromUsrDistance(arpa_dist, DISTANCE_KM, g_iDistanceFormat);
1804 arpa_sog = fromUsrSpeed(arpa_sog, SPEED_KMH, g_iSpeedFormat);
1805 } else if (arpa_distunit == _T("S")) {
1806 arpa_dist = fromUsrDistance(arpa_dist, DISTANCE_MI, g_iDistanceFormat);
1807 arpa_sog = fromUsrSpeed(arpa_sog, SPEED_MPH, g_iSpeedFormat);
1808 }
1809
1810 mmsi = arpa_mmsi = 199200000 + arpa_tgt_num;
1811 // 199 is INMARSAT-A MID, should not occur ever in AIS
1812 // stream + we make sure we are out of the hashes for
1813 // GPSGate buddies by being above 1992*
1814 } else if (str.Mid(3, 3).IsSameAs(_T("TLL"))) {
1815 //$--TLL,xx,llll.lll,a,yyyyy.yyy,a,c--c,hhmmss.ss,a,a*hh<CR><LF>
1816 //"$RATLL,01,5603.370,N,01859.976,E,ALPHA,015200.36,T,*75\r\n"
1817 wxString aprs_tll_str;
1818 wxString string(str);
1819 wxStringTokenizer tkz(string, _T(",*"));
1820
1821 wxString token;
1822 aprs_tll_str = tkz.GetNextToken(); // Sentence (xxTLL)
1823 token = tkz.GetNextToken(); // 1) Target number 00 - 99
1824 token.ToLong(&arpa_tgt_num);
1825 token = tkz.GetNextToken(); // 2) Latitude, N/S
1826 token.ToDouble(&arpa_lat);
1827 arpa_degs = (int)(arpa_lat / 100.0);
1828 arpa_mins = arpa_lat - arpa_degs * 100.0;
1829 arpa_lat = arpa_degs + arpa_mins / 60.0;
1830 token = tkz.GetNextToken(); // hemisphere N or S
1831 if (token.Mid(0, 1).Contains(_T("S")) == true ||
1832 token.Mid(0, 1).Contains(_T("s")) == true)
1833 arpa_lat = 0. - arpa_lat;
1834 token = tkz.GetNextToken(); // 3) Longitude, E/W
1835 token.ToDouble(&arpa_lon);
1836 arpa_degs = (int)(arpa_lon / 100.0);
1837 arpa_mins = arpa_lon - arpa_degs * 100.0;
1838 arpa_lon = arpa_degs + arpa_mins / 60.0;
1839 token = tkz.GetNextToken(); // hemisphere E or W
1840 if (token.Mid(0, 1).Contains(_T("W")) == true ||
1841 token.Mid(0, 1).Contains(_T("w")) == true)
1842 arpa_lon = 0. - arpa_lon;
1843 token = tkz.GetNextToken(); // 4) Target name
1844 if (token == wxEmptyString)
1845 token = wxString::Format(_T("ARPA %d"), arpa_tgt_num);
1846 int len = wxMin(token.Length(), 20);
1847 strncpy(arpa_name_str, token.mb_str(), len);
1848 arpa_name_str[len] = 0;
1849 token = tkz.GetNextToken(); // 5) UTC of data
1850 token.ToDouble(&arpa_utc_time);
1851 arpa_utc_hour = (int)(arpa_utc_time / 10000.0);
1852 arpa_utc_min = (int)(arpa_utc_time / 100.0) - arpa_utc_hour * 100;
1853 arpa_utc_sec =
1854 (int)arpa_utc_time - arpa_utc_hour * 10000 - arpa_utc_min * 100;
1855 arpa_status =
1856 tkz.GetNextToken(); // 6) Target status: L = lost,tracked
1857 // target has beenlost Q = query,target in
1858 // the process of acquisition T = tracking
1859 if (arpa_status != _T("L"))
1860 arpa_lost = false;
1861 else if (arpa_status != wxEmptyString)
1862 arpa_nottracked = true;
1863 arpa_reftarget = tkz.GetNextToken(); // 7) Reference target=R,null
1864 // otherwise
1865 mmsi = arpa_mmsi =
1866 199200000 +
1867 arpa_tgt_num; // 199 is INMARSAT-A MID, should not occur ever in AIS
1868 // stream + we make sure we are out of the hashes for
1869 // GPSGate buddies by being above 1992*
1870 } else if (str.Mid(3, 3).IsSameAs(_T("OSD"))) {
1871 //$--OSD,x.x,A,x.x,a,x.x,a,x.x,x.x,a*hh <CR><LF>
1872 wxString string(str);
1873 wxStringTokenizer tkz(string, _T(",*"));
1874
1875 wxString token;
1876 token = tkz.GetNextToken(); // Sentence (xxOSD)
1877 token = tkz.GetNextToken(); // 1) Heading (true)
1878 token.ToDouble(&arpa_ref_hdg);
1879 // 2) speed
1880 // 3) Vessel Course, degrees True
1881 // 4) Course Reference, B/M/W/R/P (see note)
1882 // 5) Vessel Speed
1883 // 6) Speed Reference, B/M/W/R/P (see note)
1884 // 7) Vessel Set, degrees True - Manually entered
1885 // 8) Vessel drift (speed) - Manually entered
1886 // 9) Speed Units K = km/h; N = Knots; S = statute miles/h
1887
1888 } else if (g_bWplUsePosition && str.Mid(3, 3).IsSameAs(_T("WPL"))) {
1889 //** $--WPL,llll.ll,a,yyyyy.yy,a,c--c*hh<CR><LF>
1890 wxString string(str);
1891 wxStringTokenizer tkz(string, _T(",*"));
1892
1893 wxString token;
1894 token = tkz.GetNextToken(); // Sentence (xxWPL)
1895 token = tkz.GetNextToken(); // 1) Latitude, N/S
1896 token.ToDouble(&aprs_lat);
1897 aprs_degs = (int)(aprs_lat / 100.0);
1898 aprs_mins = aprs_lat - aprs_degs * 100.0;
1899 aprs_lat = aprs_degs + aprs_mins / 60.0;
1900 token = tkz.GetNextToken(); // 2) hemisphere N or S
1901 if (token.Mid(0, 1).Contains(_T("S")) == true ||
1902 token.Mid(0, 1).Contains(_T("s")) == true)
1903 aprs_lat = 0. - aprs_lat;
1904 token = tkz.GetNextToken(); // 3) Longitude, E/W
1905 token.ToDouble(&aprs_lon);
1906 aprs_degs = (int)(aprs_lon / 100.0);
1907 aprs_mins = aprs_lon - aprs_degs * 100.0;
1908 aprs_lon = aprs_degs + aprs_mins / 60.0;
1909 token = tkz.GetNextToken(); // 4) hemisphere E or W
1910 if (token.Mid(0, 1).Contains(_T("W")) == true ||
1911 token.Mid(0, 1).Contains(_T("w")) == true)
1912 aprs_lon = 0. - aprs_lon;
1913 token = tkz.GetNextToken(); // 5) Target name
1914 int len = wxMin(token.Length(), 20);
1915 strncpy(aprs_name_str, token.mb_str(), len + 1);
1916 if (0 == g_WplAction) { // APRS position reports
1917 int i, hash = 0;
1918 aprs_name_str[len] = 0;
1919 for (i = 0; i < len; i++) {
1920 hash = hash * 10;
1921 hash += (int)(aprs_name_str[i]);
1922 while (hash >= 100000) hash = hash / 100000;
1923 }
1924 mmsi = aprs_mmsi = 199300000 + hash;
1925 // 199 is INMARSAT-A MID, should not occur ever in AIS stream +
1926 // we make sure we are out of the hashes for GPSGate buddies
1927 // and ARPA by being above 1993*
1928 } else if (1 == g_WplAction) { // Create mark
1929 RoutePoint *pWP = new RoutePoint(aprs_lat, aprs_lon, wxEmptyString,
1930 aprs_name_str, wxEmptyString, false);
1931 pWP->m_bIsolatedMark = true;
1932 InsertWpt(pWP, true);
1933 }
1934 } else if (str.Mid(1, 5).IsSameAs(_T("FRPOS"))) {
1935 // parse a GpsGate Position message $FRPOS,.....
1936
1937 // Use a tokenizer to pull out the first 9 fields
1938 wxString string(str);
1939 wxStringTokenizer tkz(string, _T(",*"));
1940
1941 wxString token;
1942 token = tkz.GetNextToken(); // !$FRPOS
1943
1944 token = tkz.GetNextToken(); // latitude DDMM.MMMM
1945 token.ToDouble(&gpsg_lat);
1946 gpsg_degs = (int)(gpsg_lat / 100.0);
1947 gpsg_mins = gpsg_lat - gpsg_degs * 100.0;
1948 gpsg_lat = gpsg_degs + gpsg_mins / 60.0;
1949
1950 token = tkz.GetNextToken(); // hemisphere N or S
1951 if (token.Mid(0, 1).Contains(_T("S")) == true ||
1952 token.Mid(0, 1).Contains(_T("s")) == true)
1953 gpsg_lat = 0. - gpsg_lat;
1954
1955 token = tkz.GetNextToken(); // longitude DDDMM.MMMM
1956 token.ToDouble(&gpsg_lon);
1957 gpsg_degs = (int)(gpsg_lon / 100.0);
1958 gpsg_mins = gpsg_lon - gpsg_degs * 100.0;
1959 gpsg_lon = gpsg_degs + gpsg_mins / 60.0;
1960
1961 token = tkz.GetNextToken(); // hemisphere E or W
1962 if (token.Mid(0, 1).Contains(_T("W")) == true ||
1963 token.Mid(0, 1).Contains(_T("w")) == true)
1964 gpsg_lon = 0. - gpsg_lon;
1965
1966 token = tkz.GetNextToken(); // altitude AA.a
1967 // token.toDouble(&gpsg_alt);
1968
1969 token = tkz.GetNextToken(); // speed over ground SSS.SS knots
1970 token.ToDouble(&gpsg_sog);
1971
1972 token = tkz.GetNextToken(); // heading over ground HHH.hh degrees
1973 token.ToDouble(&gpsg_cog);
1974
1975 token = tkz.GetNextToken(); // date DDMMYY
1976 gpsg_date = token;
1977
1978 token = tkz.GetNextToken(); // time UTC hhmmss.dd
1979 token.ToDouble(&gpsg_utc_time);
1980 gpsg_utc_hour = (int)(gpsg_utc_time / 10000.0);
1981 gpsg_utc_min = (int)(gpsg_utc_time / 100.0) - gpsg_utc_hour * 100;
1982 gpsg_utc_sec =
1983 (int)gpsg_utc_time - gpsg_utc_hour * 10000 - gpsg_utc_min * 100;
1984
1985 // now comes the name, followed by in * and NMEA checksum
1986
1987 token = tkz.GetNextToken();
1988 int i, len, hash = 0;
1989 len = wxMin(wxStrlen(token), 20);
1990 strncpy(gpsg_name_str, token.mb_str(), len);
1991 gpsg_name_str[len] = 0;
1992 for (i = 0; i < len; i++) {
1993 hash = hash * 10;
1994 hash += (int)(token[i]);
1995 while (hash >= 100000) hash = hash / 100000;
1996 }
1997 // 199 is INMARSAT-A MID, should not occur ever in AIS stream
1998 gpsg_mmsi = 199000000 + hash;
1999 mmsi = gpsg_mmsi;
2000 } else if (!str.Mid(3, 2).IsSameAs(_T("VD"))) {
2001 return AIS_NMEAVDX_BAD;
2002 }
2003
2004 // OK, looks like the sentence is OK
2005
2006 // Use a tokenizer to pull out the first 4 fields
2007 wxString string(str);
2008 wxStringTokenizer tkz(string, _T(","));
2009
2010 wxString token;
2011 token = tkz.GetNextToken(); // !xxVDx
2012
2013 token = tkz.GetNextToken();
2014 nsentences = atoi(token.mb_str());
2015
2016 token = tkz.GetNextToken();
2017 isentence = atoi(token.mb_str());
2018
2019 token = tkz.GetNextToken();
2020 long lsequence_id = 0;
2021 token.ToLong(&lsequence_id);
2022
2023 token = tkz.GetNextToken();
2024 long lchannel;
2025 token.ToLong(&lchannel);
2026 // Now, some decisions
2027
2028 string_to_parse.Clear();
2029
2030 // Simple case first
2031 // First and only part of a one-part sentence
2032 if ((1 == nsentences) && (1 == isentence)) {
2033 string_to_parse = tkz.GetNextToken(); // the encapsulated data
2034 }
2035
2036 else if (nsentences > 1) {
2037 if (1 == isentence) {
2038 sentence_accumulator = tkz.GetNextToken(); // the encapsulated data
2039 }
2040
2041 else {
2042 sentence_accumulator += tkz.GetNextToken();
2043 }
2044
2045 if (isentence == nsentences) {
2046 string_to_parse = sentence_accumulator;
2047 }
2048 }
2049
2050 if (mmsi || (!string_to_parse.IsEmpty() &&
2051 (string_to_parse.Len() < AIS_MAX_MESSAGE_LEN))) {
2052 // Create the bit accessible string
2053 wxCharBuffer abuf = string_to_parse.ToUTF8();
2054 if (!abuf.data()) // badly formed sentence?
2055 return AIS_GENERIC_ERROR;
2056
2057 AisBitstring strbit(abuf.data());
2058
2059 // Extract the MMSI
2060 if (!mmsi) mmsi = strbit.GetInt(9, 30);
2061 long mmsi_long = mmsi;
2062
2063 // Ais8_001_31 || ais8_367_33 (class AIS_METEO) test for a new mmsi ID
2064 int origin_mmsi = 0;
2065 int messID = strbit.GetInt(1, 6);
2066 int dac = strbit.GetInt(41, 10);
2067 int fi = strbit.GetInt(51, 6);
2068 if (messID == 8) {
2069 int met_lon, met_lat;
2070 if (dac == 001 && fi == 31) {
2071 origin_mmsi = mmsi;
2072 met_lon = strbit.GetInt(57, 25);
2073 met_lat = strbit.GetInt(82, 24);
2074 mmsi = AisMeteoNewMmsi(mmsi, met_lat, met_lon, 25, 0);
2075 mmsi_long = mmsi;
2076
2077 } else if (dac == 367 && fi == 33) { // ais8_367_33
2078 // Check for a valid message size before further handling
2079 const int size = strbit.GetBitCount();
2080 if (size < 168) return AIS_GENERIC_ERROR;
2081 const int startb = 56;
2082 const int slot_size = 112;
2083 const int extra_bits = (size - startb) % slot_size;
2084 if (extra_bits > 0) return AIS_GENERIC_ERROR;
2085
2086 int mes_type = strbit.GetInt(57, 4);
2087 int site_ID = strbit.GetInt(77, 7);
2088 if (mes_type == 0) { // Location
2089 origin_mmsi = mmsi;
2090 met_lon = strbit.GetInt(90, 28);
2091 met_lat = strbit.GetInt(118, 27);
2092 mmsi = AisMeteoNewMmsi(mmsi, met_lat, met_lon, 28, site_ID);
2093 mmsi_long = mmsi;
2094 } else { // Other messsage types without position.
2095 // We need a previously received type 0, position message
2096 // to get use of any sensor report.
2097 int x_mmsi = AisMeteoNewMmsi(mmsi, 91, 181, 0, site_ID);
2098 if (x_mmsi) {
2099 origin_mmsi = mmsi;
2100 mmsi = x_mmsi;
2101 mmsi_long = mmsi;
2102 } else // So far no use for this report.
2103 return AIS_GENERIC_ERROR;
2104 }
2105 }
2106 }
2107
2108 // Check for own ship mmsi. It's not a valid AIS target.
2109 if (mmsi == g_OwnShipmmsi) return AIS_GENERIC_ERROR;
2110
2111 // Search the current AISTargetList for an MMSI match
2112 auto it = AISTargetList.find(mmsi);
2113 if (it == AISTargetList.end()) // not found
2114 {
2115 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
2116 bnewtarget = true;
2117 m_n_targets++;
2118
2119 if (origin_mmsi) { // New mmsi allocated for a Meteo station
2120 pTargetData->MMSI = mmsi;
2121 pTargetData->met_data.original_mmsi = origin_mmsi;
2122 }
2123 } else {
2124 pTargetData = it->second; // find current entry
2125
2126 if (!bnewtarget)
2127 pStaleTarget = pTargetData; // save a pointer to stale data
2128 if (origin_mmsi) { // Meteo point
2129 pTargetData->MMSI = mmsi;
2130 pTargetData->met_data.original_mmsi = origin_mmsi;
2131 }
2132 }
2133 for (unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
2134 MmsiProperties *props = g_MMSI_Props_Array[i];
2135 if (mmsi == props->MMSI) {
2136 // Check if this target has a dedicated tracktype
2137 if (TRACKTYPE_NEVER == props->TrackType) {
2138 pTargetData->b_show_track = false;
2139 } else if (TRACKTYPE_ALWAYS == props->TrackType) {
2140 pTargetData->b_show_track = true;
2141 }
2142
2143 // Check to see if this MMSI has been configured to be ignored
2144 // completely...
2145 if (props->m_bignore) return AIS_NoError;
2146 // Check to see if this MMSI wants VDM translated to VDO or whether we
2147 // want to persist it's track...
2148 else if (props->m_bVDM) {
2149 // Only single line VDM messages to be translated
2150 if (str.Mid(3, 9).IsSameAs(wxT("VDM,1,1,,"))) {
2151 int message_ID = strbit.GetInt(1, 6); // Parse on message ID
2152 // Only translate the dynamic positionreport messages (1, 2, 3 or
2153 // 18)
2154 if ((message_ID <= 3) || (message_ID == 18)) {
2155 // set OwnShip to prevent target from being drawn
2156 pTargetData->b_OwnShip = true;
2157 // Rename nmea sentence to AIVDO and calc a new checksum
2158 wxString aivdostr = str;
2159 aivdostr.replace(1, 5, "AIVDO");
2160 unsigned char calculated_checksum = 0;
2161 wxString::iterator i;
2162 for (i = aivdostr.begin() + 1; i != aivdostr.end() && *i != '*';
2163 ++i)
2164 calculated_checksum ^= static_cast<unsigned char>(*i);
2165 // if i is not at least 3 positons befoere end, there is no
2166 // checksum added so also no need to add one now.
2167 if (i <= aivdostr.end() - 3)
2168 aivdostr.replace(
2169 i + 1, i + 3,
2170 wxString::Format(_("%02X"), calculated_checksum));
2171
2172 gps_watchdog_timeout_ticks =
2173 60; // increase watchdog time up to 1 minute
2174 // add the changed sentence into nmea message system
2175 std::string full_sentence = aivdostr.ToStdString();
2176 std::string identifier("AIVDO");
2177 // We notify based on full message, including the Talker ID
2178 // notify message listener and also "ALL" N0183 messages, to
2179 // support plugin API using original talker id
2180 auto address = std::make_shared<NavAddr0183>("virtual");
2181 auto msg = std::make_shared<const Nmea0183Msg>(
2182 identifier, full_sentence, address);
2183 auto msg_all = std::make_shared<const Nmea0183Msg>(*msg, "ALL");
2184
2185 auto &msgbus = NavMsgBus::GetInstance();
2186
2187 msgbus.Notify(std::move(msg));
2188 msgbus.Notify(std::move(msg_all));
2189 }
2190 }
2191 return AIS_NoError;
2192 } else
2193 break;
2194 }
2195 }
2196
2197 // Grab the stale targets's last report time
2198 wxDateTime now = wxDateTime::Now();
2199 now.MakeGMT();
2200
2201 if (pStaleTarget)
2202 last_report_ticks = pStaleTarget->PositionReportTicks;
2203 else
2204 last_report_ticks = now.GetTicks();
2205
2206 // Delete the stale AIS Target selectable point
2207 if (pStaleTarget)
2208 pSelectAIS->DeleteSelectablePoint((void *)mmsi_long, SELTYPE_AISTARGET);
2209
2210 if (pTargetData) {
2211 if (gpsg_mmsi) {
2212 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
2213 pTargetData->PositionReportTicks = now.GetTicks();
2214 pTargetData->StaticReportTicks = now.GetTicks();
2215 pTargetData->m_utc_hour = gpsg_utc_hour;
2216 pTargetData->m_utc_min = gpsg_utc_min;
2217 pTargetData->m_utc_sec = gpsg_utc_sec;
2218 pTargetData->m_date_string = gpsg_date;
2219 pTargetData->MMSI = gpsg_mmsi;
2220 pTargetData->NavStatus = 0; // underway
2221 pTargetData->Lat = gpsg_lat;
2222 pTargetData->Lon = gpsg_lon;
2223 pTargetData->b_positionOnceValid = true;
2224 pTargetData->COG = gpsg_cog;
2225 pTargetData->SOG = gpsg_sog;
2226 pTargetData->ShipType = 52; // buddy
2227 pTargetData->Class = AIS_GPSG_BUDDY;
2228 memcpy(pTargetData->ShipName, gpsg_name_str, sizeof(gpsg_name_str));
2229 pTargetData->b_nameValid = true;
2230 pTargetData->b_active = true;
2231 pTargetData->b_lost = false;
2232
2233 bdecode_result = true;
2234 } else if (arpa_mmsi) {
2235 pTargetData->m_utc_hour = arpa_utc_hour;
2236 pTargetData->m_utc_min = arpa_utc_min;
2237 pTargetData->m_utc_sec = arpa_utc_sec;
2238 pTargetData->MMSI = arpa_mmsi;
2239 pTargetData->NavStatus = 15; // undefined
2240 if (str.Mid(3, 3).IsSameAs(_T("TLL"))) {
2241 if (!bnewtarget) {
2242 int age_of_last =
2243 (now.GetTicks() - pTargetData->PositionReportTicks);
2244 if (age_of_last > 0) {
2245 ll_gc_ll_reverse(pTargetData->Lat, pTargetData->Lon, arpa_lat,
2246 arpa_lon, &pTargetData->COG, &pTargetData->SOG);
2247 pTargetData->SOG = pTargetData->SOG * 3600 / age_of_last;
2248 }
2249 }
2250 pTargetData->Lat = arpa_lat;
2251 pTargetData->Lon = arpa_lon;
2252 } else if (str.Mid(3, 3).IsSameAs(_T("TTM"))) {
2253 if (arpa_dist != 0.) // Not a new or turned off target
2254 ll_gc_ll(gLat, gLon, arpa_brg, arpa_dist, &pTargetData->Lat,
2255 &pTargetData->Lon);
2256 else
2257 arpa_lost = true;
2258 pTargetData->COG = arpa_cog;
2259 pTargetData->SOG = arpa_sog;
2260 }
2261 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
2262 pTargetData->PositionReportTicks = now.GetTicks();
2263 pTargetData->StaticReportTicks = now.GetTicks();
2264 pTargetData->b_positionOnceValid = true;
2265 pTargetData->ShipType = 55; // arpa
2266 pTargetData->Class = AIS_ARPA;
2267
2268 memcpy(pTargetData->ShipName, arpa_name_str, sizeof(arpa_name_str));
2269 if (arpa_status != _T("Q"))
2270 pTargetData->b_nameValid = true;
2271 else
2272 pTargetData->b_nameValid = false;
2273 pTargetData->b_active = !arpa_lost;
2274 pTargetData->b_lost = arpa_nottracked;
2275
2276 bdecode_result = true;
2277 } else if (aprs_mmsi) {
2278 pTargetData->m_utc_hour = now.GetHour();
2279 pTargetData->m_utc_min = now.GetMinute();
2280 pTargetData->m_utc_sec = now.GetSecond();
2281 pTargetData->MMSI = aprs_mmsi;
2282 pTargetData->NavStatus = 15; // undefined
2283 if (!bnewtarget) {
2284 int age_of_last = (now.GetTicks() - pTargetData->PositionReportTicks);
2285 if (age_of_last > 0) {
2286 ll_gc_ll_reverse(pTargetData->Lat, pTargetData->Lon, aprs_lat,
2287 aprs_lon, &pTargetData->COG, &pTargetData->SOG);
2288 pTargetData->SOG = pTargetData->SOG * 3600 / age_of_last;
2289 }
2290 }
2291 pTargetData->LastPositionReportTicks = pTargetData->PositionReportTicks;
2292 pTargetData->PositionReportTicks = now.GetTicks();
2293 pTargetData->StaticReportTicks = now.GetTicks();
2294 pTargetData->Lat = aprs_lat;
2295 pTargetData->Lon = aprs_lon;
2296 pTargetData->b_positionOnceValid = true;
2297 pTargetData->ShipType = 56; // aprs
2298 pTargetData->Class = AIS_APRS;
2299 memcpy(pTargetData->ShipName, aprs_name_str, sizeof(aprs_name_str));
2300 pTargetData->b_nameValid = true;
2301 pTargetData->b_active = true;
2302 pTargetData->b_lost = false;
2303
2304 bdecode_result = true;
2305 } else {
2306 // The normal Plain-Old AIS target code path....
2307 bdecode_result =
2308 Parse_VDXBitstring(&strbit, pTargetData); // Parse the new data
2309 }
2310
2311 // Catch mmsi properties like track, persistent track, follower.
2312 getMmsiProperties(pTargetData);
2313
2314 // Update the most recent report period
2315 pTargetData->RecentPeriod =
2316 pTargetData->PositionReportTicks - last_report_ticks;
2317 }
2318 ret = AIS_NoError;
2319 } else {
2320 ret = AIS_Partial; // accumulating parts of a multi-sentence message
2321 pTargetData = 0;
2322 }
2323
2324 if (pTargetData) {
2325 // pTargetData is valid, either new or existing. Commit to GUI
2326 CommitAISTarget(pTargetData, str, bdecode_result, bnewtarget);
2327 }
2328
2329 n_msgs++;
2330#ifdef AIS_DEBUG
2331 if ((n_msgs % 10000) == 0)
2332 printf("n_msgs %10d m_n_targets: %6d n_msg1: %10d n_msg5+24: %10d \n",
2333 n_msgs, m_n_targets, n_msg1, n_msg5 + n_msg24);
2334#endif
2335
2336 return ret;
2337}
2338
2339void AisDecoder::CommitAISTarget(std::shared_ptr<AisTargetData> pTargetData,
2340 const wxString &str, bool message_valid,
2341 bool new_target) {
2342 m_pLatestTargetData = pTargetData;
2343
2344 if (!str.IsEmpty()) { // NMEA0183 message
2345 if (str.Mid(3, 3).IsSameAs(_T("VDO")))
2346 pTargetData->b_OwnShip = true;
2347 else
2348 pTargetData->b_OwnShip = false;
2349 }
2350
2351 if (!pTargetData->b_OwnShip) {
2352 // set mmsi-props to default values
2353 if (0 == m_persistent_tracks.count(pTargetData->MMSI)) {
2354 // Normal target
2355 pTargetData->b_PersistTrack = false;
2356 // Or first decode for this target
2357 for (unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
2358 if (pTargetData->MMSI == g_MMSI_Props_Array[i]->MMSI) {
2359 MmsiProperties *props = g_MMSI_Props_Array[i];
2360 pTargetData->b_mPropPersistTrack = props->m_bPersistentTrack;
2361 break;
2362 }
2363 }
2364 } else {
2365 // The track persistency enabled in the query window or mmsi-props
2366 }
2367 pTargetData->b_NoTrack = false;
2368 }
2369
2370 // If the message was decoded correctly
2371 // Update the AIS Target information
2372 if (message_valid) {
2373 // Print to name cache only if not mmsi = 0
2374 if (pTargetData->MMSI) {
2375 AISshipNameCache(pTargetData.get(), AISTargetNamesC, AISTargetNamesNC,
2376 pTargetData->MMSI);
2377 }
2378 AISTargetList[pTargetData->MMSI] =
2379 pTargetData; // update the hash table entry
2380
2381 if (!pTargetData->area_notices.empty()) {
2382 auto it = AIS_AreaNotice_Sources.find(pTargetData->MMSI);
2383 if (it == AIS_AreaNotice_Sources.end())
2384 AIS_AreaNotice_Sources[pTargetData->MMSI] = pTargetData;
2385 }
2386
2387 // If this is not an ownship message, update the AIS Target in the
2388 // Selectable list, and update the CPA info
2389 if (!pTargetData->b_OwnShip) {
2390 if (pTargetData->b_positionOnceValid) {
2391 long mmsi_long = pTargetData->MMSI;
2392 SelectItem *pSel = pSelectAIS->AddSelectablePoint(
2393 pTargetData->Lat, pTargetData->Lon, (void *)mmsi_long,
2394 SELTYPE_AISTARGET);
2395 pSel->SetUserData(pTargetData->MMSI);
2396 }
2397
2398 // Calculate CPA info for this target immediately
2399 UpdateOneCPA(pTargetData.get());
2400
2401 // Update this target's track
2402 if (pTargetData->b_show_track) UpdateOneTrack(pTargetData.get());
2403 }
2404 // TODO add ais message call
2405 plugin_msg.Notify(std::make_shared<AisTargetData>(*pTargetData), "");
2406 } else {
2407 // printf("Unrecognised AIS message ID: %d\n",
2408 // pTargetData->MID);
2409 if (new_target) {
2410 // delete pTargetData; // this target is not going to be used
2411 m_n_targets--;
2412 } else {
2413 // If this is not an ownship message, update the AIS Target in the
2414 // Selectable list even if the message type was not recognized
2415 if (!pTargetData->b_OwnShip) {
2416 if (pTargetData->b_positionOnceValid) {
2417 long mmsi_long = pTargetData->MMSI;
2418 SelectItem *pSel = pSelectAIS->AddSelectablePoint(
2419 pTargetData->Lat, pTargetData->Lon, (void *)mmsi_long,
2420 SELTYPE_AISTARGET);
2421 pSel->SetUserData(pTargetData->MMSI);
2422 }
2423 }
2424 }
2425 }
2426}
2427
2428void AisDecoder::getAISTarget(long mmsi,
2429 std::shared_ptr<AisTargetData> &pTargetData,
2430 std::shared_ptr<AisTargetData> &pStaleTarget,
2431 bool &bnewtarget, int &last_report_ticks,
2432 wxDateTime &now) {
2433 now = wxDateTime::Now();
2434 auto it = AISTargetList.find(mmsi);
2435 if (it == AISTargetList.end()) // not found
2436 {
2437 pTargetData = AisTargetDataMaker::GetInstance().GetTargetData();
2438 bnewtarget = true;
2439 m_n_targets++;
2440 } else {
2441 pTargetData = it->second; // find current entry
2442 pStaleTarget = pTargetData; // save a pointer to stale data
2443 }
2444
2445 // Grab the stale targets's last report time
2446 now.MakeGMT();
2447
2448 if (pStaleTarget)
2449 last_report_ticks = pStaleTarget->PositionReportTicks;
2450 else
2451 last_report_ticks = now.GetTicks();
2452
2453 // Delete the stale AIS Target selectable point
2454 if (pStaleTarget)
2455 pSelectAIS->DeleteSelectablePoint((void *)mmsi, SELTYPE_AISTARGET);
2456}
2457
2458void AisDecoder::getMmsiProperties(
2459 std::shared_ptr<AisTargetData> &pTargetData) {
2460 for (unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
2461 if (pTargetData->MMSI == g_MMSI_Props_Array[i]->MMSI) {
2462 MmsiProperties *props = g_MMSI_Props_Array[i];
2463 pTargetData->b_isFollower = props->m_bFollower;
2464 pTargetData->b_mPropPersistTrack = props->m_bPersistentTrack;
2465 if (TRACKTYPE_NEVER == props->TrackType) {
2466 pTargetData->b_show_track = false;
2467 } else if (TRACKTYPE_ALWAYS == props->TrackType) {
2468 pTargetData->b_show_track = true;
2469 }
2470 break;
2471 }
2472 }
2473}
2474
2475std::shared_ptr<AisTargetData> AisDecoder::ProcessDSx(const wxString &str,
2476 bool b_take_dsc) {
2477 double dsc_lat = 0.;
2478 double dsc_lon = 0.;
2479 double dsc_mins, dsc_degs, dsc_tmp, dsc_addr;
2480 double dse_tmp;
2481 double dse_lat = 0.;
2482 double dse_lon = 0.;
2483 long dsc_fmt, dsc_quadrant, dsc_cat, dsc_nature;
2484
2485 int dsc_mmsi = 0;
2486 int dsc_tx_mmsi = 0;
2487 int dse_mmsi = 0;
2488 double dse_cog = 0.;
2489 double dse_sog = 0.;
2490 wxString dse_shipName = wxEmptyString;
2491 wxString dseSymbol;
2492
2493 int mmsi = 0;
2494
2495 std::shared_ptr<AisTargetData> pTargetData = NULL;
2496
2497 // parse a DSC Position message $CDDSx,.....
2498 // Use a tokenizer to pull out the first 9 fields
2499 wxString string(str);
2500 wxStringTokenizer tkz(string, _T(",*"));
2501
2502 wxString token;
2503 token = tkz.GetNextToken(); // !$CDDS
2504
2505 if (str.Mid(3, 3).IsSameAs(_T("DSC"))) {
2506 m_dsc_last_string = str;
2507
2508 token = tkz.GetNextToken(); // format specifier
2509 token.ToLong(
2510 &dsc_fmt); // (02-area,12-distress,16-allships,20-individual,...)
2511
2512 token = tkz.GetNextToken(); // address i.e. mmsi*10 for received msg, or
2513 // area spec or sender mmsi for (12) and (16)
2514 if (dsc_fmt == 12 || dsc_fmt == 16) {
2515 dsc_mmsi = wxAtoi(token.Mid(0, 9));
2516 } else {
2517 token.ToDouble(&dsc_addr);
2518 dsc_mmsi = 0 - (int)(dsc_addr / 10); // as per NMEA 0183 3.01
2519 }
2520
2521 token = tkz.GetNextToken(); // category
2522 token.ToLong(&dsc_cat); // 12 - Distress (relayed)
2523
2524 token = tkz.GetNextToken(); // nature of distress or telecommand1
2525 if (!token.IsSameAs(wxEmptyString)) { // 00-12 = nature of distress
2526 token.ToLong(&dsc_nature);
2527 } else
2528 dsc_nature = 99;
2529
2530 token = tkz.GetNextToken(); // comm type or telecommand2
2531
2532 token = tkz.GetNextToken(); // position or channel/freq
2533 token.ToDouble(&dsc_tmp);
2534
2535 token = tkz.GetNextToken(); // time or tel. no.
2536 token = tkz.GetNextToken(); // mmsi of ship in distress, relay
2537 if (dsc_fmt == 16 && dsc_cat == 12 && !token.IsSameAs(wxEmptyString)) {
2538 // wxString dmmsi = token.Mid(0,9);
2539 dsc_tx_mmsi = dsc_mmsi; // mmsi of relay issuer
2540 dsc_mmsi = wxAtoi(token.Mid(0, 9));
2541 }
2542 token = tkz.GetNextToken(); // nature of distress, relay
2543 if (dsc_fmt == 16 && dsc_cat == 12) {
2544 if (!token.IsSameAs(wxEmptyString)) { // 00-12 = nature of distress
2545 token.ToLong(&dsc_nature);
2546 } else
2547 dsc_nature = 99;
2548 }
2549 token = tkz.GetNextToken(); // acknowledgement
2550 token = tkz.GetNextToken(); // expansion indicator
2551
2552 dsc_quadrant = (int)(dsc_tmp / 1000000000.0);
2553
2554 if (dsc_quadrant > 3) // Position is "Unspecified", or 9999999999
2555 return NULL;
2556
2557 dsc_lat = (int)(dsc_tmp / 100000.0);
2558 dsc_lon = dsc_tmp - dsc_lat * 100000.0;
2559 dsc_lat = dsc_lat - dsc_quadrant * 10000;
2560 dsc_degs = (int)(dsc_lat / 100.0);
2561 dsc_mins = dsc_lat - dsc_degs * 100.0;
2562 dsc_lat = dsc_degs + dsc_mins / 60.0;
2563
2564 dsc_degs = (int)(dsc_lon / 100.0);
2565 dsc_mins = dsc_lon - dsc_degs * 100.0;
2566 dsc_lon = dsc_degs + dsc_mins / 60.0;
2567 switch (dsc_quadrant) {
2568 case 0:
2569 break; // NE
2570 case 1:
2571 dsc_lon = -dsc_lon;
2572 break; // NW
2573 case 2:
2574 dsc_lat = -dsc_lat;
2575 break; // SE
2576 case 3:
2577 dsc_lon = -dsc_lon;
2578 dsc_lat = -dsc_lat;
2579 break; // SW
2580 default:
2581 break;
2582 }
2583 if (dsc_fmt != 02) mmsi = (int)dsc_mmsi;
2584
2585 } else if (str.Mid(3, 3).IsSameAs(_T("DSE"))) {
2586 token = tkz.GetNextToken(); // total number of sentences
2587 token = tkz.GetNextToken(); // sentence number
2588 token = tkz.GetNextToken(); // query/rely flag
2589 token = tkz.GetNextToken(); // vessel MMSI
2590 dse_mmsi = wxAtoi(token.Mid(
2591 0, 9)); // ITU-R M.493-10 �5.2
2592 // token.ToDouble(&dse_addr);
2593 // 0 - (int)(dse_addr / 10); // as per NMEA 0183 3.01
2594
2595#if 0
2596 token = tkz.GetNextToken(); // code field
2597 token =
2598 tkz.GetNextToken(); // data field - position - 2*4 digits latlon .mins
2599 token.ToDouble(&dse_tmp);
2600 dse_lat = (int)(dse_tmp / 10000.0);
2601 dse_lon = (int)(dse_tmp - dse_lat * 10000.0);
2602 dse_lat = dse_lat / 600000.0;
2603 dse_lon = dse_lon / 600000.0;
2604#endif
2605 // DSE Sentence may contain multiple dse expansion data items
2606 while (tkz.HasMoreTokens()) {
2607 dseSymbol = tkz.GetNextToken(); // dse expansion data symbol
2608 token = tkz.GetNextToken(); // dse expansion data
2609 if (dseSymbol.IsSameAs(_T("00"))) { // Position
2610 token.ToDouble(&dse_tmp);
2611 dse_lat = (int)(dse_tmp / 10000.0);
2612 dse_lon = (int)(dse_tmp - dse_lat * 10000.0);
2613 dse_lat = dse_lat / 600000.0;
2614 dse_lon = dse_lon / 600000.0;
2615 } else if (dseSymbol.IsSameAs(_T("01"))) { // Source & Datum
2616 } else if (dseSymbol.IsSameAs(_T("02"))) { // SOG
2617 token.ToDouble(&dse_tmp);
2618 dse_sog = dse_tmp / 10.0;
2619 } else if (dseSymbol.IsSameAs(_T("03"))) { // COG
2620 token.ToDouble(&dse_tmp);
2621 dse_cog = dse_tmp / 10.0;
2622 } else if (dseSymbol.IsSameAs(_T("04"))) { // Station Information
2623 dse_shipName = DecodeDSEExpansionCharacters(token);
2624 } else if (dseSymbol.IsSameAs(_T("05"))) { // Geographic Information
2625 } else if (dseSymbol.IsSameAs(_T("06"))) { // Persons On Board
2626 }
2627 }
2628 mmsi = abs((int)dse_mmsi);
2629 }
2630
2631 // Get the last report time for this target, if it exists
2632 wxDateTime now = wxDateTime::Now();
2633 now.MakeGMT();
2634 int last_report_ticks = now.GetTicks();
2635
2636 // Search the current AISTargetList for an MMSI match
2637 auto it = AISTargetList.find(mmsi);
2638 std::shared_ptr<AisTargetData> pStaleTarget = NULL;
2639 if (it == AISTargetList.end()) { // not found
2640 } else {
2641 pStaleTarget = it->second; // find current entry
2642 last_report_ticks = pStaleTarget->PositionReportTicks;
2643 }
2644
2645 if (dsc_mmsi) {
2646 // Create a tentative target, but do not post it pending receipt of
2647 // extended data
2648 m_ptentative_dsctarget = AisTargetDataMaker::GetInstance().GetTargetData();
2649
2650 m_ptentative_dsctarget->PositionReportTicks = now.GetTicks();
2651 m_ptentative_dsctarget->StaticReportTicks = now.GetTicks();
2652
2653 m_ptentative_dsctarget->MMSI = mmsi;
2654 m_ptentative_dsctarget->NavStatus =
2655 99; // Undefind. "-" in the AIS target list
2656 m_ptentative_dsctarget->Lat = dsc_lat;
2657 m_ptentative_dsctarget->Lon = dsc_lon;
2658 m_ptentative_dsctarget->b_positionOnceValid = true;
2659 m_ptentative_dsctarget->COG = 0;
2660 m_ptentative_dsctarget->SOG = 0;
2661 m_ptentative_dsctarget->ShipType = dsc_fmt;
2662 m_ptentative_dsctarget->Class = AIS_DSC;
2663 m_ptentative_dsctarget->b_isDSCtarget = true;
2664 m_ptentative_dsctarget->b_nameValid = true;
2665 if (dsc_fmt == 12 || (dsc_fmt == 16 && dsc_cat == 12)) {
2666 snprintf(m_ptentative_dsctarget->ShipName, SHIP_NAME_LEN, "DISTRESS %d",
2667 std::abs(mmsi));
2668 m_ptentative_dsctarget->m_dscNature = int(dsc_nature);
2669 m_ptentative_dsctarget->m_dscTXmmsi = dsc_tx_mmsi;
2670 } else {
2671 snprintf(m_ptentative_dsctarget->ShipName, SHIP_NAME_LEN, "POSITION %d",
2672 std::abs(mmsi));
2673 }
2674
2675 m_ptentative_dsctarget->b_active = true;
2676 m_ptentative_dsctarget->b_lost = false;
2677 m_ptentative_dsctarget->RecentPeriod =
2678 m_ptentative_dsctarget->PositionReportTicks - last_report_ticks;
2679
2680 // Start a timer, looking for an expected DSE extension message
2681 if (!b_take_dsc) m_dsc_timer.Start(1000, wxTIMER_ONE_SHOT);
2682 }
2683
2684 // Got an extension message, or the timer expired and no extension is
2685 // expected
2686 if (dse_mmsi || b_take_dsc) {
2687 if (m_ptentative_dsctarget) {
2688 // stop the timer for sure
2689 m_dsc_timer.Stop();
2690
2691 // Update the extended information
2692 if (dse_mmsi) {
2693 m_ptentative_dsctarget->Lat =
2694 m_ptentative_dsctarget->Lat +
2695 ((m_ptentative_dsctarget->Lat) >= 0 ? dse_lat : -dse_lat);
2696 m_ptentative_dsctarget->Lon =
2697 m_ptentative_dsctarget->Lon +
2698 ((m_ptentative_dsctarget->Lon) >= 0 ? dse_lon : -dse_lon);
2699 if (dse_shipName.length() > 0) {
2700 memset(m_ptentative_dsctarget->ShipName, '\0', SHIP_NAME_LEN);
2701 snprintf(m_ptentative_dsctarget->ShipName, dse_shipName.length(),
2702 "%s", dse_shipName.ToAscii().data());
2703 }
2704 m_ptentative_dsctarget->COG = dse_cog;
2705 m_ptentative_dsctarget->SOG = dse_sog;
2706 }
2707
2708 // Update the most recent report period
2709 m_ptentative_dsctarget->RecentPeriod =
2710 m_ptentative_dsctarget->PositionReportTicks - last_report_ticks;
2711
2712 // And post the target
2713
2714 // Search the current AISTargetList for an MMSI match
2715 auto it = AISTargetList.find(mmsi);
2716 if (it == AISTargetList.end()) { // not found
2717 pTargetData = m_ptentative_dsctarget;
2718 } else {
2719 pTargetData = it->second; // find current entry
2720 std::vector<AISTargetTrackPoint> ptrack =
2721 std::move(pTargetData->m_ptrack);
2722 pTargetData->CloneFrom(
2723 m_ptentative_dsctarget
2724 .get()); // this will make an empty track list
2725
2726 pTargetData->m_ptrack =
2727 std::move(ptrack); // and substitute the old track list
2728
2729 // delete m_ptentative_dsctarget;
2730 }
2731
2732 // Reset for next time
2733 m_ptentative_dsctarget = NULL;
2734
2735 m_pLatestTargetData = pTargetData;
2736
2737 AISTargetList[pTargetData->MMSI] =
2738 pTargetData; // update the hash table entry
2739
2740 long mmsi_long = pTargetData->MMSI;
2741
2742 // Delete any stale Target selectable point
2743 if (pStaleTarget)
2744 pSelectAIS->DeleteSelectablePoint((void *)mmsi_long, SELTYPE_AISTARGET);
2745 // And add the updated target
2746 SelectItem *pSel =
2747 pSelectAIS->AddSelectablePoint(pTargetData->Lat, pTargetData->Lon,
2748 (void *)mmsi_long, SELTYPE_AISTARGET);
2749 pSel->SetUserData(pTargetData->MMSI);
2750
2751 // Calculate CPA info for this target immediately
2752 UpdateOneCPA(pTargetData.get());
2753
2754 // Update this target's track
2755 if (pTargetData->b_show_track) UpdateOneTrack(pTargetData.get());
2756 }
2757 }
2758
2759 return pTargetData;
2760}
2761
2762// DSE Expansion characters, decode table from ITU-R M.825
2763wxString AisDecoder::DecodeDSEExpansionCharacters(wxString dseData) {
2764 wxString result;
2765 char lookupTable[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', ' ',
2766 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K',
2767 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V',
2768 'W', 'X', 'Y', 'Z', '.', ',', '-', '/', ' '};
2769
2770 for (size_t i = 0; i < dseData.length(); i += 2) {
2771 result.append(1, lookupTable[strtol(dseData.Mid(i, 2).data(), NULL, 10)]);
2772 }
2773 return result;
2774}
2775
2776//----------------------------------------------------------------------------
2777// Parse a NMEA VDM/VDO Bitstring
2778//----------------------------------------------------------------------------
2779bool AisDecoder::Parse_VDXBitstring(AisBitstring *bstr,
2780 std::shared_ptr<AisTargetData> ptd) {
2781 bool parse_result = false;
2782 bool b_posn_report = false;
2783
2784 wxDateTime now = wxDateTime::Now();
2785 now.MakeGMT();
2786 int message_ID = bstr->GetInt(1, 6); // Parse on message ID
2787 ptd->MID = message_ID;
2788
2789 // Save for Ais8_001_31 and ais8_367_33 (class AIS_METEO)
2790 int met_mmsi = ptd->MMSI;
2791
2792 // MMSI is always in the same spot in the bitstream
2793 ptd->MMSI = bstr->GetInt(9, 30);
2794
2795 switch (message_ID) {
2796 case 1: // Position Report
2797 case 2:
2798 case 3: {
2799 n_msg1++;
2800
2801 ptd->NavStatus = bstr->GetInt(39, 4);
2802 ptd->SOG = 0.1 * (bstr->GetInt(51, 10));
2803
2804 int lon = bstr->GetInt(62, 28);
2805 if (lon & 0x08000000) // negative?
2806 lon |= 0xf0000000;
2807 double lon_tentative = lon / 600000.;
2808
2809 int lat = bstr->GetInt(90, 27);
2810 if (lat & 0x04000000) // negative?
2811 lat |= 0xf8000000;
2812 double lat_tentative = lat / 600000.;
2813
2814 if ((lon_tentative <= 180.) &&
2815 (lat_tentative <=
2816 90.)) // Ship does not report Lat or Lon "unavailable"
2817 {
2818 ptd->Lon = lon_tentative;
2819 ptd->Lat = lat_tentative;
2820 ptd->b_positionDoubtful = false;
2821 ptd->b_positionOnceValid = true; // Got the position at least once
2822 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
2823 ptd->PositionReportTicks = now.GetTicks();
2824 } else
2825 ptd->b_positionDoubtful = true;
2826
2827 // decode balance of message....
2828 ptd->COG = 0.1 * (bstr->GetInt(117, 12));
2829 ptd->HDG = 1.0 * (bstr->GetInt(129, 9));
2830
2831 ptd->ROTAIS = bstr->GetInt(43, 8);
2832 double rot_dir = 1.0;
2833
2834 if (ptd->ROTAIS == 128)
2835 ptd->ROTAIS = -128; // not available codes as -128
2836 else if ((ptd->ROTAIS & 0x80) == 0x80) {
2837 ptd->ROTAIS = ptd->ROTAIS - 256; // convert to twos complement
2838 rot_dir = -1.0;
2839 }
2840
2841 ptd->ROTIND = wxRound(rot_dir * pow((((double)ptd->ROTAIS) / 4.733),
2842 2)); // Convert to indicated ROT
2843
2844 ptd->m_utc_sec = bstr->GetInt(138, 6);
2845
2846 if ((1 == message_ID) ||
2847 (2 == message_ID)) // decode SOTDMA per 7.6.7.2.2
2848 {
2849 ptd->SyncState = bstr->GetInt(151, 2);
2850 ptd->SlotTO = bstr->GetInt(153, 2);
2851 if ((ptd->SlotTO == 1) && (ptd->SyncState == 0)) // UTCDirect follows
2852 {
2853 ptd->m_utc_hour = bstr->GetInt(155, 5);
2854
2855 ptd->m_utc_min = bstr->GetInt(160, 7);
2856
2857 if ((ptd->m_utc_hour < 24) && (ptd->m_utc_min < 60) &&
2858 (ptd->m_utc_sec < 60)) {
2859 wxDateTime rx_time(ptd->m_utc_hour, ptd->m_utc_min, ptd->m_utc_sec);
2860 rx_ticks = rx_time.GetTicks();
2861 if (!b_firstrx) {
2862 first_rx_ticks = rx_ticks;
2863 b_firstrx = true;
2864 }
2865 }
2866 }
2867 }
2868
2869 // Capture Euro Inland special passing arrangement signal ("stbd-stbd")
2870 ptd->blue_paddle = bstr->GetInt(144, 2);
2871 ptd->b_blue_paddle = (ptd->blue_paddle == 2); // paddle is set
2872
2873 if (!ptd->b_isDSCtarget) ptd->Class = AIS_CLASS_A;
2874
2875 // Check for SART and friends by looking at first two digits of MMSI
2876 int mmsi_start = ptd->MMSI / 10000000;
2877
2878 if (mmsi_start == 97) {
2879 ptd->Class = AIS_SART;
2880 ptd->StaticReportTicks =
2881 now.GetTicks(); // won't get a static report, so fake it here
2882
2883 // On receipt of Msg 3, force any existing SART target out of
2884 // acknowledge mode by adjusting its ack_time to yesterday This will
2885 // cause any previously "Acknowledged" SART to re-alert.
2886
2887 // On reflection, re-alerting seems a little excessive in real life
2888 // use. After all, the target is on-screen, and in the AIS target
2889 // list. So lets just honor the programmed ACK timout value for SART
2890 // targets as well
2891 // ptd->m_ack_time = wxDateTime::Now() - wxTimeSpan::Day();
2892 }
2893
2894 parse_result = true; // so far so good
2895 b_posn_report = true;
2896
2897 break;
2898 }
2899
2900 case 18: {
2901 ptd->NavStatus =
2902 UNDEFINED; // Class B targets have no status. Enforce this...
2903
2904 ptd->SOG = 0.1 * (bstr->GetInt(47, 10));
2905
2906 int lon = bstr->GetInt(58, 28);
2907 if (lon & 0x08000000) // negative?
2908 lon |= 0xf0000000;
2909 double lon_tentative = lon / 600000.;
2910
2911 int lat = bstr->GetInt(86, 27);
2912 if (lat & 0x04000000) // negative?
2913 lat |= 0xf8000000;
2914 double lat_tentative = lat / 600000.;
2915
2916 if ((lon_tentative <= 180.) &&
2917 (lat_tentative <=
2918 90.)) // Ship does not report Lat or Lon "unavailable"
2919 {
2920 ptd->Lon = lon_tentative;
2921 ptd->Lat = lat_tentative;
2922 ptd->b_positionDoubtful = false;
2923 ptd->b_positionOnceValid = true; // Got the position at least once
2924 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
2925 ptd->PositionReportTicks = now.GetTicks();
2926 } else
2927 ptd->b_positionDoubtful = true;
2928
2929 ptd->COG = 0.1 * (bstr->GetInt(113, 12));
2930 ptd->HDG = 1.0 * (bstr->GetInt(125, 9));
2931
2932 ptd->m_utc_sec = bstr->GetInt(134, 6);
2933
2934 if (!ptd->b_isDSCtarget) {
2935 if (!isBuoyMmsi(ptd->MMSI))
2936 ptd->Class = AIS_CLASS_B;
2937 else
2938 ptd->Class = AIS_BUOY;
2939 }
2940 parse_result = true; // so far so good
2941 b_posn_report = true;
2942
2943 break;
2944 }
2945
2946 case 19: { // Class B mes_ID 19 Is same as mes_ID 18 until bit 139
2947 ptd->NavStatus =
2948 UNDEFINED; // Class B targets have no status. Enforce this...
2949 ptd->SOG = 0.1 * (bstr->GetInt(47, 10));
2950 int lon = bstr->GetInt(58, 28);
2951 if (lon & 0x08000000) // negative?
2952 lon |= 0xf0000000;
2953 double lon_tentative = lon / 600000.;
2954
2955 int lat = bstr->GetInt(86, 27);
2956 if (lat & 0x04000000) // negative?
2957 lat |= 0xf8000000;
2958 double lat_tentative = lat / 600000.;
2959
2960 if ((lon_tentative <= 180.) &&
2961 (lat_tentative <=
2962 90.)) // Ship does not report Lat or Lon "unavailable"
2963 {
2964 ptd->Lon = lon_tentative;
2965 ptd->Lat = lat_tentative;
2966 ptd->b_positionDoubtful = false;
2967 ptd->b_positionOnceValid = true; // Got the position at least once
2968 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
2969 ptd->PositionReportTicks = now.GetTicks();
2970 } else
2971 ptd->b_positionDoubtful = true;
2972
2973 ptd->COG = 0.1 * (bstr->GetInt(113, 12));
2974 ptd->HDG = 1.0 * (bstr->GetInt(125, 9));
2975 ptd->m_utc_sec = bstr->GetInt(134, 6);
2976 // From bit 140 and forward data as of mes 5
2977 bstr->GetStr(144, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
2978 ptd->b_nameValid = true;
2979 if (!ptd->b_isDSCtarget) {
2980 ptd->ShipType = (unsigned char)bstr->GetInt(264, 8);
2981 }
2982 ptd->DimA = bstr->GetInt(272, 9);
2983 ptd->DimB = bstr->GetInt(281, 9);
2984 ptd->DimC = bstr->GetInt(290, 6);
2985 ptd->DimD = bstr->GetInt(296, 6);
2986
2987 if (!ptd->b_isDSCtarget) {
2988 // Although outdated, message 19 is used by many "ATON" for net buoys
2989 if (!isBuoyMmsi(ptd->MMSI))
2990 ptd->Class = AIS_CLASS_B;
2991 else
2992 ptd->Class = AIS_BUOY;
2993 }
2994 parse_result = true; // so far so good
2995 b_posn_report = true;
2996
2997 break;
2998 }
2999
3000 case 27: {
3001 // Long-range automatic identification system broadcast message
3002 // This message is used for long-range detection of AIS Class A and Class
3003 // B vessels (typically by satellite).
3004
3005 // Define the constant to do the covertion from the internal encoded
3006 // position in message 27. The position is less accuate : 1/10 minute
3007 // position resolution.
3008 int bitCorrection = 10;
3009 int resolution = 10;
3010
3011 // Default aout of bounce values.
3012 double lon_tentative = 181.;
3013 double lat_tentative = 91.;
3014
3015#ifdef AIS_DEBUG
3016 printf("AIS Message 27 - received:\r\n");
3017 printf("MMSI : %i\r\n", ptd->MMSI);
3018#endif
3019
3020 // It can be both a CLASS A and a CLASS B vessel - We have decided for
3021 // CLASS A
3022 // TODO: Lookup to see if we have seen it as a CLASS B, and adjust.
3023 if (!ptd->b_isDSCtarget) ptd->Class = AIS_CLASS_A;
3024
3025 ptd->NavStatus = bstr->GetInt(39, 4);
3026
3027 int lon = bstr->GetInt(45, 18);
3028 int lat = bstr->GetInt(63, 17);
3029
3030 lat_tentative = lat;
3031 lon_tentative = lon;
3032
3033 // Negative latitude?
3034 if (lat >= (0x4000000 >> bitCorrection)) {
3035 lat_tentative = (0x8000000 >> bitCorrection) - lat;
3036 lat_tentative *= -1;
3037 }
3038
3039 // Negative longitude?
3040 if (lon >= (0x8000000 >> bitCorrection)) {
3041 lon_tentative = (0x10000000 >> bitCorrection) - lon;
3042 lon_tentative *= -1;
3043 }
3044
3045 // Decode the internal position format.
3046 lat_tentative = lat_tentative / resolution / 60.0;
3047 lon_tentative = lon_tentative / resolution / 60.0;
3048
3049#ifdef AIS_DEBUG
3050 printf("Latitude : %f\r\n", lat_tentative);
3051 printf("Longitude : %f\r\n", lon_tentative);
3052#endif
3053
3054 // Get the latency of the position report.
3055 int positionLatency = bstr->GetInt(95, 1);
3056
3057 if ((lon_tentative <= 180.) &&
3058 (lat_tentative <=
3059 90.)) // Ship does not report Lat or Lon "unavailable"
3060 {
3061 ptd->Lon = lon_tentative;
3062 ptd->Lat = lat_tentative;
3063 ptd->b_positionDoubtful = false;
3064 ptd->b_positionOnceValid = true; // Got the position at least once
3065 if (positionLatency == 0) {
3066// The position is less than 5 seconds old.
3067#ifdef AIS_DEBUG
3068 printf("Low latency position report.\r\n");
3069#endif
3070 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3071 ptd->PositionReportTicks = now.GetTicks();
3072 }
3073 } else
3074 ptd->b_positionDoubtful = true;
3075
3076 ptd->SOG = 1.0 * (bstr->GetInt(80, 6));
3077 ptd->COG = 1.0 * (bstr->GetInt(85, 9));
3078
3079 b_posn_report = true;
3080 parse_result = true;
3081 break;
3082 }
3083
3084 case 5: {
3085 n_msg5++;
3086 if (!ptd->b_isDSCtarget) ptd->Class = AIS_CLASS_A;
3087
3088 // Get the AIS Version indicator
3089 // 0 = station compliant with Recommendation ITU-R M.1371-1
3090 // 1 = station compliant with Recommendation ITU-R M.1371-3
3091 // 2-3 = station compliant with future editions
3092 int AIS_version_indicator = bstr->GetInt(39, 2);
3093 if (AIS_version_indicator < 4) {
3094 ptd->IMO = bstr->GetInt(41, 30);
3095
3096 bstr->GetStr(71, 42, &ptd->CallSign[0], 7);
3097 bstr->GetStr(113, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
3098 ptd->b_nameValid = true;
3099 if (!ptd->b_isDSCtarget) {
3100 ptd->ShipType = (unsigned char)bstr->GetInt(233, 8);
3101 }
3102
3103 ptd->DimA = bstr->GetInt(241, 9);
3104 ptd->DimB = bstr->GetInt(250, 9);
3105 ptd->DimC = bstr->GetInt(259, 6);
3106 ptd->DimD = bstr->GetInt(265, 6);
3107
3108 ptd->ETA_Mo = bstr->GetInt(275, 4);
3109 ptd->ETA_Day = bstr->GetInt(279, 5);
3110 ptd->ETA_Hr = bstr->GetInt(284, 5);
3111 ptd->ETA_Min = bstr->GetInt(289, 6);
3112
3113 ptd->Draft = (double)(bstr->GetInt(295, 8)) / 10.0;
3114
3115 bstr->GetStr(303, 120, &ptd->Destination[0], DESTINATION_LEN - 1);
3116
3117 ptd->StaticReportTicks = now.GetTicks();
3118
3119 parse_result = true;
3120 }
3121
3122 break;
3123 }
3124
3125 case 24: { // Static data report
3126 int part_number = bstr->GetInt(39, 2);
3127 if (0 == part_number) {
3128 bstr->GetStr(41, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
3129 ptd->b_nameValid = true;
3130 parse_result = true;
3131 n_msg24++;
3132 } else if (1 == part_number) {
3133 if (!ptd->b_isDSCtarget) {
3134 ptd->ShipType = (unsigned char)bstr->GetInt(41, 8);
3135 }
3136 bstr->GetStr(91, 42, &ptd->CallSign[0], 7);
3137
3138 ptd->DimA = bstr->GetInt(133, 9);
3139 ptd->DimB = bstr->GetInt(142, 9);
3140 ptd->DimC = bstr->GetInt(151, 6);
3141 ptd->DimD = bstr->GetInt(157, 6);
3142 parse_result = true;
3143 }
3144 break;
3145 }
3146 case 4: // base station
3147 {
3148 ptd->Class = AIS_BASE;
3149
3150 ptd->m_utc_hour = bstr->GetInt(62, 5);
3151 ptd->m_utc_min = bstr->GetInt(67, 6);
3152 ptd->m_utc_sec = bstr->GetInt(73, 6);
3153 // (79, 1);
3154 int lon = bstr->GetInt(80, 28);
3155 if (lon & 0x08000000) // negative?
3156 lon |= 0xf0000000;
3157 double lon_tentative = lon / 600000.;
3158
3159 int lat = bstr->GetInt(108, 27);
3160 if (lat & 0x04000000) // negative?
3161 lat |= 0xf8000000;
3162 double lat_tentative = lat / 600000.;
3163
3164 if ((lon_tentative <= 180.) &&
3165 (lat_tentative <=
3166 90.)) // Ship does not report Lat or Lon "unavailable"
3167 {
3168 ptd->Lon = lon_tentative;
3169 ptd->Lat = lat_tentative;
3170 ptd->b_positionDoubtful = false;
3171 ptd->b_positionOnceValid = true; // Got the position at least once
3172 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3173 ptd->PositionReportTicks = now.GetTicks();
3174 } else
3175 ptd->b_positionDoubtful = true;
3176
3177 ptd->COG = -1.;
3178 ptd->HDG = 511;
3179 ptd->SOG = -1.;
3180
3181 parse_result = true;
3182 b_posn_report = true;
3183
3184 break;
3185 }
3186 case 9: // Special Position Report (Standard SAR Aircraft Position Report)
3187 {
3188 ptd->SOG = bstr->GetInt(51, 10);
3189
3190 int lon = bstr->GetInt(62, 28);
3191 if (lon & 0x08000000) // negative?
3192 lon |= 0xf0000000;
3193 double lon_tentative = lon / 600000.;
3194
3195 int lat = bstr->GetInt(90, 27);
3196 if (lat & 0x04000000) // negative?
3197 lat |= 0xf8000000;
3198 double lat_tentative = lat / 600000.;
3199
3200 if ((lon_tentative <= 180.) &&
3201 (lat_tentative <=
3202 90.)) // Ship does not report Lat or Lon "unavailable"
3203 {
3204 ptd->Lon = lon_tentative;
3205 ptd->Lat = lat_tentative;
3206 ptd->b_positionDoubtful = false;
3207 ptd->b_positionOnceValid = true; // Got the position at least once
3208 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3209 ptd->PositionReportTicks = now.GetTicks();
3210 } else
3211 ptd->b_positionDoubtful = true;
3212
3213 // decode balance of message....
3214 ptd->COG = 0.1 * (bstr->GetInt(117, 12));
3215
3216 int alt_tent = bstr->GetInt(39, 12);
3217 ptd->altitude = alt_tent;
3218
3219 ptd->b_SarAircraftPosnReport = true;
3220
3221 parse_result = true;
3222 b_posn_report = true;
3223
3224 break;
3225 }
3226 case 21: // Test Message (Aid to Navigation)
3227 {
3228 ptd->ShipType = (unsigned char)bstr->GetInt(39, 5);
3229 ptd->IMO = 0;
3230 ptd->SOG = 0;
3231 ptd->HDG = 0;
3232 ptd->COG = 0;
3233 ptd->ROTAIS = -128; // i.e. not available
3234 ptd->DimA = bstr->GetInt(220, 9);
3235 ptd->DimB = bstr->GetInt(229, 9);
3236 ptd->DimC = bstr->GetInt(238, 6);
3237 ptd->DimD = bstr->GetInt(244, 6);
3238 ptd->Draft = 0;
3239
3240 ptd->m_utc_sec = bstr->GetInt(254, 6);
3241
3242 int offpos = bstr->GetInt(260, 1); // off position flag
3243 int virt = bstr->GetInt(270, 1); // virtual flag
3244
3245 if (virt)
3246 ptd->NavStatus = ATON_VIRTUAL;
3247 else
3248 ptd->NavStatus = ATON_REAL;
3249 if (ptd->m_utc_sec <= 59 /*&& !virt*/) {
3250 ptd->NavStatus += 1;
3251 if (offpos) ptd->NavStatus += 1;
3252 }
3253
3254 bstr->GetStr(44, 120, &ptd->ShipName[0], SHIP_NAME_LEN);
3255 // short name only, extension wont fit in Ship structure
3256
3257 if (bstr->GetBitCount() > 276) {
3258 int nx = ((bstr->GetBitCount() - 272) / 6) * 6;
3259 bstr->GetStr(273, nx, &ptd->ShipNameExtension[0], 14);
3260 ptd->ShipNameExtension[14] = 0;
3261 } else {
3262 ptd->ShipNameExtension[0] = 0;
3263 }
3264
3265 ptd->b_nameValid = true;
3266
3267 parse_result = true; // so far so good
3268
3269 ptd->Class = AIS_ATON;
3270
3271 int lon = bstr->GetInt(165, 28);
3272
3273 if (lon & 0x08000000) // negative?
3274 lon |= 0xf0000000;
3275 double lon_tentative = lon / 600000.;
3276
3277 int lat = bstr->GetInt(193, 27);
3278
3279 if (lat & 0x04000000) // negative?
3280 lat |= 0xf8000000;
3281 double lat_tentative = lat / 600000.;
3282
3283 if ((lon_tentative <= 180.) &&
3284 (lat_tentative <=
3285 90.)) // Ship does not report Lat or Lon "unavailable"
3286 {
3287 ptd->Lon = lon_tentative;
3288 ptd->Lat = lat_tentative;
3289 ptd->b_positionDoubtful = false;
3290 ptd->b_positionOnceValid = true; // Got the position at least once
3291 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3292 ptd->PositionReportTicks = now.GetTicks();
3293 } else
3294 ptd->b_positionDoubtful = true;
3295
3296 b_posn_report = true;
3297 break;
3298 }
3299 case 8: // Binary Broadcast
3300 {
3301 int dac = bstr->GetInt(41, 10);
3302 int fi = bstr->GetInt(51, 6);
3303
3304 if (dac == 200) // European inland
3305 {
3306 if (fi == 10) // "Inland ship static and voyage related data"
3307 {
3308 ptd->b_isEuroInland = true;
3309
3310 bstr->GetStr(57, 48, &ptd->Euro_VIN[0], 8);
3311 ptd->Euro_Length = ((double)bstr->GetInt(105, 13)) / 10.0;
3312 ptd->Euro_Beam = ((double)bstr->GetInt(118, 10)) / 10.0;
3313 ptd->UN_shiptype = bstr->GetInt(128, 14);
3314 ptd->Euro_Draft = ((double)bstr->GetInt(145, 11)) / 100.0;
3315 parse_result = true;
3316 }
3317 }
3318 if (dac == 1 || dac == 366) // IMO or US
3319 {
3320 if (fi == 22) // Area Notice
3321 {
3322 if (bstr->GetBitCount() >= 111) {
3323 Ais8_001_22 an;
3324 an.link_id = bstr->GetInt(57, 10);
3325 an.notice_type = bstr->GetInt(67, 7);
3326 an.month = bstr->GetInt(74, 4);
3327 an.day = bstr->GetInt(78, 5);
3328 an.hour = bstr->GetInt(83, 5);
3329 an.minute = bstr->GetInt(88, 6);
3330 an.duration_minutes = bstr->GetInt(94, 18);
3331
3332 wxDateTime now = wxDateTime::Now();
3333 now.MakeGMT();
3334
3335 an.start_time.Set(an.day, wxDateTime::Month(an.month - 1),
3336 now.GetYear(), an.hour, an.minute);
3337
3338 // msg is not supposed to be transmitted more than a day before it
3339 // comes into effect, so a start_time less than a day or two away
3340 // might indicate a month rollover
3341 if (an.start_time > now + wxTimeSpan::Hours(48))
3342 an.start_time.Set(an.day, wxDateTime::Month(an.month - 1),
3343 now.GetYear() - 1, an.hour, an.minute);
3344
3345 an.expiry_time =
3346 an.start_time + wxTimeSpan::Minutes(an.duration_minutes);
3347
3348 // msg is not supposed to be transmitted beyond expiration, so
3349 // taking into account a fudge factor for clock issues, assume an
3350 // expiry date in the past indicates incorrect year
3351 if (an.expiry_time < now - wxTimeSpan::Hours(24)) {
3352 an.start_time.Set(an.day, wxDateTime::Month(an.month - 1),
3353 now.GetYear() + 1, an.hour, an.minute);
3354 an.expiry_time =
3355 an.start_time + wxTimeSpan::Minutes(an.duration_minutes);
3356 }
3357
3358 // Default case, the IMO format
3359 // https://www.e-navigation.nl/content/area-notice-0
3360 int subarea_len = 87;
3361 int lon_len = 25;
3362 int lat_len = 24;
3363 float pos_scale = 60000.0;
3364 int prec_size = 3;
3365 if (dac ==
3366 366) { // Deprecated US format
3367 // https://www.e-navigation.nl/content/area-notice-1
3368 subarea_len = 90;
3369 lon_len = 28;
3370 lat_len = 27;
3371 pos_scale = 600000.0;
3372 prec_size = 0; // Not present in the in US format between
3373 // coordinates and radius for some shapes
3374 }
3375
3376 int subarea_count = (bstr->GetBitCount() - 111) / subarea_len;
3377 for (int i = 0; i < subarea_count; ++i) {
3378 int base = 111 + i * subarea_len;
3380 sa.shape = bstr->GetInt(base + 1, 3);
3381 int scale_factor = 1;
3382 if (sa.shape == AIS8_001_22_SHAPE_TEXT) {
3383 char t[15];
3384 t[14] = 0;
3385 bstr->GetStr(base + 4, subarea_len - 3, t, 14);
3386 sa.text = wxString(t, wxConvUTF8);
3387 } else {
3388 int scale_multipliers[4] = {1, 10, 100, 1000};
3389 scale_factor = scale_multipliers[bstr->GetInt(base + 4, 2)];
3390 switch (sa.shape) {
3391 case AIS8_001_22_SHAPE_SECTOR:
3392 sa.left_bound_deg = bstr->GetInt(
3393 base + 6 + lon_len + lat_len + prec_size + 12, 9);
3394 sa.right_bound_deg = bstr->GetInt(
3395 base + 6 + lon_len + lat_len + prec_size + 12 + 9, 9);
3396 case AIS8_001_22_SHAPE_CIRCLE:
3397 sa.radius_m =
3398 bstr->GetInt(base + 6 + lon_len + lat_len + 3, 12) *
3399 scale_factor;
3400 // FALL THROUGH
3401 case AIS8_001_22_SHAPE_RECT:
3402 sa.longitude =
3403 bstr->GetInt(base + 6, lon_len, true) / pos_scale;
3404 sa.latitude =
3405 bstr->GetInt(base + 6 + lon_len, lat_len, true) /
3406 pos_scale;
3407 sa.e_dim_m =
3408 bstr->GetInt(base + 6 + lon_len + lat_len + prec_size,
3409 8) *
3410 scale_factor;
3411 sa.n_dim_m =
3412 bstr->GetInt(
3413 base + 6 + lon_len + lat_len + prec_size + 8, 8) *
3414 scale_factor;
3415 sa.orient_deg = bstr->GetInt(
3416 base + 6 + lon_len + lat_len + prec_size + 8 + 8, 9);
3417 break;
3418 case AIS8_001_22_SHAPE_POLYLINE:
3419 case AIS8_001_22_SHAPE_POLYGON:
3420 for (int i = 0; i < 4; ++i) {
3421 sa.angles[i] = bstr->GetInt(base + 6 + i * 20, 10) * 0.5;
3422 sa.dists_m[i] =
3423 bstr->GetInt(base + 16 + i * 20, 10) * scale_factor;
3424 }
3425 }
3426 }
3427 an.sub_areas.push_back(sa);
3428 }
3429 ptd->area_notices[an.link_id] = an;
3430 parse_result = true;
3431 }
3432 }
3433
3434 // Meteorological and Hydrographic data ref: IMO SN.1/Circ.289
3435 if (fi == 31) {
3436 if (bstr->GetBitCount() >= 360) {
3437 // Ais8_001_31 mmsi can have been changed.
3438 if (met_mmsi != 666) ptd->MMSI = met_mmsi;
3439
3440 // Default out of bounce values.
3441 double lon_tentative = 181.;
3442 double lat_tentative = 91.;
3443
3444 int lon = bstr->GetInt(57, 25);
3445 int lat = bstr->GetInt(82, 24);
3446
3447 if (lon & 0x01000000) // negative?
3448 lon |= 0xFE000000;
3449 lon_tentative = lon / 60000.;
3450
3451 if (lat & 0x00800000) // negative?
3452 lat |= 0xFF000000;
3453 lat_tentative = lat / 60000.;
3454
3455 ptd->Lon = lon_tentative;
3456 ptd->Lat = lat_tentative;
3457
3458 // Try to make unique name for each station based on position
3459 wxString x = ptd->ShipName;
3460 if (x.Find("METEO") == wxNOT_FOUND) {
3461 double id1, id2;
3462 wxString slat = wxString::Format("%0.3f", lat_tentative);
3463 wxString slon = wxString::Format("%0.3f", lon_tentative);
3464 slat.ToDouble(&id1);
3465 slon.ToDouble(&id2);
3466 wxString nameID = "METEO ";
3467 nameID << wxString::Format("%0.3f", abs(id1) + abs(id2)).Right(3);
3468 strncpy(ptd->ShipName, nameID, SHIP_NAME_LEN - 1);
3469 }
3470
3471 ptd->met_data.pos_acc = bstr->GetInt(106, 1);
3472 ptd->met_data.day = bstr->GetInt(107, 5);
3473 ptd->met_data.hour = bstr->GetInt(112, 5);
3474 ptd->met_data.minute = bstr->GetInt(117, 6);
3475 ptd->met_data.wind_kn = bstr->GetInt(123, 7);
3476 ptd->met_data.wind_gust_kn = bstr->GetInt(130, 7);
3477 ptd->met_data.wind_dir = bstr->GetInt(137, 9);
3478 ptd->met_data.wind_gust_dir = bstr->GetInt(146, 9);
3479
3480 int tmp = bstr->GetInt(155, 11);
3481 if (tmp & 0x00000400) // negative?
3482 tmp |= 0xFFFFF800;
3483 ptd->met_data.air_temp = tmp / 10.;
3484 ptd->met_data.rel_humid = bstr->GetInt(166, 7);
3485 int dew = bstr->GetInt(173, 10);
3486 if (dew & 0x00000200) // negative? (bit 9 = 1)
3487 dew |= 0xFFFFFC00;
3488 ptd->met_data.dew_point = dew / 10.;
3489
3490 /*Air pressure, defined as pressure reduced to sea level,
3491 in 1 hPa steps.0 = pressure 799 hPa or less
3492 1 - 401 = 800 - 1200 hPa*/
3493 ptd->met_data.airpress = bstr->GetInt(183, 9) + 799;
3494 ptd->met_data.airpress_tend = bstr->GetInt(192, 2);
3495
3496 int horVis = bstr->GetInt(194, 8);
3497 if (horVis & 0x80u) { // if MSB = 1
3498 horVis &= 0x7F; // We print >x.x
3499 ptd->met_data.hor_vis_GT = true;
3500 } else
3501 ptd->met_data.hor_vis_GT = false;
3502
3503 ptd->met_data.hor_vis = horVis / 10.0;
3504
3505 ptd->met_data.water_lev_dev = (bstr->GetInt(202, 12) / 100.) - 10.;
3506 ptd->met_data.water_lev_trend = bstr->GetInt(214, 2);
3507 ptd->met_data.current = bstr->GetInt(216, 8) / 10.;
3508 ptd->met_data.curr_dir = bstr->GetInt(224, 9);
3509 ptd->met_data.wave_height = bstr->GetInt(277, 8) / 10.;
3510 ptd->met_data.wave_period = bstr->GetInt(285, 6);
3511 ptd->met_data.wave_dir = bstr->GetInt(291, 9);
3512 ptd->met_data.swell_height = bstr->GetInt(300, 8) / 10;
3513 ptd->met_data.swell_per = bstr->GetInt(308, 6);
3514 ptd->met_data.swell_dir = bstr->GetInt(314, 9);
3515 ptd->met_data.seastate = bstr->GetInt(323, 4);
3516
3517 int wt = bstr->GetInt(327, 10);
3518 if (wt & 0x00000200) // negative? (bit 9 = 1)
3519 wt |= 0xFFFFFC00;
3520 ptd->met_data.water_temp = wt / 10.;
3521
3522 ptd->met_data.precipitation = bstr->GetInt(337, 3);
3523 ptd->met_data.salinity = bstr->GetInt(340, 9) / 10.;
3524 ptd->met_data.ice = bstr->GetInt(349, 2);
3525
3526 ptd->Class = AIS_METEO;
3527 ptd->COG = -1.;
3528 ptd->HDG = 511;
3529 ptd->SOG = -1.;
3530 ptd->b_NoTrack = true;
3531 ptd->b_show_track = false;
3532 ptd->b_positionDoubtful = false;
3533 ptd->b_positionOnceValid = true;
3534 b_posn_report = true;
3535 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3536 ptd->PositionReportTicks = now.GetTicks();
3537 ptd->b_nameValid = true;
3538
3539 parse_result = true;
3540 }
3541 }
3542 break;
3543 }
3544
3545 if (dac == 367 && fi == 33) { // ais8_367_33
3546 // US data acording to DAC367_FI33_em_version_release_3-23mar15_0
3547 // We use only the same kind of data as of ais8_001_31 from these
3548 // reports.
3549 const int size = bstr->GetBitCount();
3550 if (size >= 168) {
3551 // Change to meteo mmsi-ID
3552 if (met_mmsi != 666) ptd->MMSI = met_mmsi;
3553 const int startbits = 56;
3554 const int slotsize = 112;
3555 const int slots_count = (size - startbits) / slotsize;
3556 int slotbit;
3557 for (int slot = 0; slot < slots_count; slot++) {
3558 slotbit = slot * slotsize;
3559 int type = bstr->GetInt(slotbit + 57, 4);
3560 ptd->met_data.hour = bstr->GetInt(slotbit + 66, 5);
3561 ptd->met_data.minute = bstr->GetInt(slotbit + 71, 6);
3562 int Site_ID = bstr->GetInt(slotbit + 77, 7);
3563
3564 // Name the station acc to site ID until message type 1
3565 if (!ptd->b_nameValid) {
3566 wxString nameID = "METEO Site: ";
3567 nameID << Site_ID;
3568 strncpy(ptd->ShipName, nameID, SHIP_NAME_LEN - 1);
3569 ptd->b_nameValid = true;
3570 }
3571
3572 if (type == 0) { // Location
3573 int lon = bstr->GetInt(slotbit + 90, 28);
3574 if (lon & 0x08000000) // negative?
3575 lon |= 0xf0000000;
3576 ptd->Lon = lon / 600000.;
3577
3578 int lat = bstr->GetInt(slotbit + 118, 27);
3579 if (lat & 0x04000000) // negative?
3580 lat |= 0xf8000000;
3581 ptd->Lat = lat / 600000.;
3582 ptd->b_positionOnceValid = true;
3583
3584 } else if (type == 1) { // Name
3585 bstr->GetStr(slotbit + 84, 84, &ptd->ShipName[0], SHIP_NAME_LEN);
3586 ptd->b_nameValid = true;
3587
3588 } else if (type == 2) { // Wind
3589 // Description 1 and 2 are real time values.
3590 int descr = bstr->GetInt(slotbit + 116, 3);
3591 if (descr == 1 || descr == 2) {
3592 ptd->met_data.wind_kn = bstr->GetInt(slotbit + 84, 7);
3593 ptd->met_data.wind_gust_kn = bstr->GetInt(slotbit + 91, 7);
3594 ptd->met_data.wind_dir = bstr->GetInt(slotbit + 98, 9);
3595 ptd->met_data.wind_gust_dir = bstr->GetInt(slotbit + 107, 9);
3596 }
3597
3598 } else if (type == 3) { // Water level
3599 // Description 1 and 2 are real time values.
3600 int descr = bstr->GetInt(slotbit + 108, 3);
3601 if (descr == 1 || descr == 2) {
3602 int wltype = bstr->GetInt(slotbit + 84, 1);
3603 int wl = bstr->GetInt(slotbit + 85, 16); // cm
3604 if (wl & 0x00004000) // negative?
3605 wl |= 0xffff0000;
3606
3607 if (wltype == 1) // 0 = deviation from datum; 1 = water depth
3608 ptd->met_data.water_level = wl / 100.; // m
3609 else
3610 ptd->met_data.water_lev_dev = wl / 100.; // m
3611 }
3612 ptd->met_data.water_lev_trend = bstr->GetInt(slotbit + 101, 2);
3613 ptd->met_data.vertical_ref = bstr->GetInt(slotbit + 103, 5);
3614
3615 } else if (type == 6) { // Horizontal Current Profil
3616 int readbearing = bstr->GetInt(slotbit + 84, 9);
3617 int readdistance = bstr->GetInt(slotbit + 93, 9);
3618 ptd->met_data.current = bstr->GetInt(slotbit + 102, 8) / 10.0;
3619 ptd->met_data.curr_dir = bstr->GetInt(slotbit + 110, 9);
3620 int readLevel = bstr->GetInt(slotbit + 119, 9);
3621
3622 } else if (type == 7) { // Sea state
3623 int swell_descr =
3624 bstr->GetInt(slotbit + 111, 3); // Use 1 || 2 real data
3625 if (swell_descr == 1 || swell_descr == 2) {
3626 ptd->met_data.swell_height =
3627 bstr->GetInt(slotbit + 84, 8) / 10.0;
3628 ptd->met_data.swell_per = bstr->GetInt(slotbit + 92, 6);
3629 ptd->met_data.swell_dir = bstr->GetInt(slotbit + 98, 9);
3630 }
3631 ptd->met_data.seastate = bstr->GetInt(slotbit + 107, 4); // Bf
3632 int wt_descr = bstr->GetInt(slotbit + 131, 3);
3633 if (wt_descr == 1 || wt_descr == 2)
3634 ptd->met_data.water_temp =
3635 bstr->GetInt(slotbit + 114, 10) / 10. - 10.;
3636
3637 int wawe_descr = bstr->GetInt(slotbit + 157, 3);
3638 if (wawe_descr == 1 || wawe_descr == 2) { // Only real data
3639 ptd->met_data.wave_height =
3640 bstr->GetInt(slotbit + 134, 8) / 10.0;
3641 ptd->met_data.wave_period = bstr->GetInt(slotbit + 142, 6);
3642 ptd->met_data.wave_dir = bstr->GetInt(slotbit + 148, 9);
3643 }
3644 ptd->met_data.salinity = bstr->GetInt(slotbit + 160, 9 / 10.0);
3645
3646 } else if (type == 8) { // Salinity
3647 ptd->met_data.water_temp =
3648 bstr->GetInt(slotbit + 84, 10) / 10.0 - 10.0;
3649 ptd->met_data.salinity = bstr->GetInt(slotbit + 120, 9) / 10.0;
3650
3651 } else if (type == 9) { // Weather
3652 int tmp = bstr->GetInt(slotbit + 84, 11);
3653 if (tmp & 0x00000400) // negative?
3654 tmp |= 0xFFFFF800;
3655 ptd->met_data.air_temp = tmp / 10.;
3656 int pp, precip = bstr->GetInt(slotbit + 98, 2);
3657 switch (precip) { // Adapt to IMO precipitation
3658 case 0:
3659 pp = 1;
3660 case 1:
3661 pp = 5;
3662 case 2:
3663 pp = 4;
3664 case 3:
3665 pp = 7;
3666 }
3667 ptd->met_data.precipitation = pp;
3668 ptd->met_data.hor_vis = bstr->GetInt(slotbit + 100, 8) / 10.0;
3669 ptd->met_data.dew_point =
3670 bstr->GetInt(slotbit + 108, 10) / 10.0 - 20.0;
3671 ptd->met_data.airpress = bstr->GetInt(slotbit + 121, 9) + 799;
3672 ptd->met_data.airpress_tend = bstr->GetInt(slotbit + 130, 2);
3673 ptd->met_data.salinity = bstr->GetInt(slotbit + 135, 9) / 10.0;
3674
3675 } else if (type == 11) { // Wind V2
3676 // Description 1 and 2 are real time values.
3677 int descr = bstr->GetInt(slotbit + 113, 3);
3678 if (descr == 1 || descr == 2) {
3679 ptd->met_data.wind_kn = bstr->GetInt(slotbit + 84, 7);
3680 ptd->met_data.wind_gust_kn = bstr->GetInt(slotbit + 91, 7);
3681 ptd->met_data.wind_dir = bstr->GetInt(slotbit + 98, 9);
3682 }
3683 }
3684 }
3685
3686 if (ptd->b_positionOnceValid) {
3687 ptd->Class = AIS_METEO;
3688 ptd->COG = -1.;
3689 ptd->HDG = 511;
3690 ptd->SOG = -1.;
3691 ptd->b_NoTrack = true;
3692 ptd->b_show_track = false;
3693 ptd->b_positionDoubtful = false;
3694 b_posn_report = true;
3695 ptd->LastPositionReportTicks = ptd->PositionReportTicks;
3696 ptd->PositionReportTicks = now.GetTicks();
3697 ptd->b_nameValid = true;
3698
3699 parse_result = true;
3700 }
3701 }
3702 break;
3703 }
3704 break;
3705 }
3706 case 14: // Safety Related Broadcast
3707 {
3708 // Always capture the MSG_14 text
3709 char msg_14_text[968];
3710 if (bstr->GetBitCount() > 40) {
3711 int nx = ((bstr->GetBitCount() - 40) / 6) * 6;
3712 int nd = bstr->GetStr(41, nx, msg_14_text, 968);
3713 nd = wxMax(0, nd);
3714 nd = wxMin(nd, 967);
3715 msg_14_text[nd] = 0;
3716 ptd->MSG_14_text = wxString(msg_14_text, wxConvUTF8);
3717 }
3718 parse_result = true; // so far so good
3719
3720 break;
3721 }
3722
3723 case 6: // Addressed Binary Message
3724 {
3725 break;
3726 }
3727 case 7: // Binary Ack
3728 {
3729 break;
3730 }
3731 default: {
3732 break;
3733 }
3734 }
3735
3736 if (b_posn_report) ptd->b_lost = false;
3737
3738 if (true == parse_result) {
3739 // Revalidate the target under some conditions
3740 if (!ptd->b_active && !ptd->b_positionDoubtful && b_posn_report)
3741 ptd->b_active = true;
3742 }
3743
3744 return parse_result;
3745}
3746
3747bool AisDecoder::NMEACheckSumOK(const wxString &str_in) {
3748 unsigned char checksum_value = 0;
3749 int sentence_hex_sum;
3750
3751 wxCharBuffer buf = str_in.ToUTF8();
3752 if (!buf.data()) return false; // cannot decode string
3753
3754 char str_ascii[AIS_MAX_MESSAGE_LEN + 1];
3755 strncpy(str_ascii, buf.data(), AIS_MAX_MESSAGE_LEN);
3756 str_ascii[AIS_MAX_MESSAGE_LEN] = '\0';
3757
3758 int string_length = strlen(str_ascii);
3759
3760 int payload_length = 0;
3761 while ((payload_length < string_length) &&
3762 (str_ascii[payload_length] != '*')) // look for '*'
3763 payload_length++;
3764
3765 if (payload_length == string_length)
3766 return false; // '*' not found at all, no checksum
3767
3768 int index = 1; // Skip over the $ at the begining of the sentence
3769
3770 while (index < payload_length) {
3771 checksum_value ^= str_ascii[index];
3772 index++;
3773 }
3774
3775 if (string_length > 4) {
3776 char scanstr[3];
3777 scanstr[0] = str_ascii[payload_length + 1];
3778 scanstr[1] = str_ascii[payload_length + 2];
3779 scanstr[2] = 0;
3780 sscanf(scanstr, "%2x", &sentence_hex_sum);
3781
3782 if (sentence_hex_sum == checksum_value) return true;
3783 }
3784
3785 return false;
3786}
3787
3788void AisDecoder::UpdateAllCPA(void) {
3789 // Iterate thru all the targets
3790 for (const auto &it : GetTargetList()) {
3791 std::shared_ptr<AisTargetData> td = it.second;
3792
3793 if (NULL != td) UpdateOneCPA(td.get());
3794 }
3795}
3796
3797void AisDecoder::UpdateAllTracks(void) {
3798 // Iterate thru all the targets
3799 for (const auto &it : GetTargetList()) {
3800 std::shared_ptr<AisTargetData> td = it.second;
3801
3802 if (NULL != td) UpdateOneTrack(td.get());
3803 }
3804}
3805
3806int gdup;
3807void AisDecoder::UpdateOneTrack(AisTargetData *ptarget) {
3808 if (!ptarget->b_positionOnceValid) return;
3809 // Reject for unbelievable jumps (corrupted/bad data)
3810 if (ptarget->m_ptrack.size() > 0) {
3811 const AISTargetTrackPoint &LastTrackpoint = ptarget->m_ptrack.back();
3812 if (fabs(LastTrackpoint.m_lat - ptarget->Lat) > .1 ||
3813 fabs(LastTrackpoint.m_lon - ptarget->Lon) > .1) {
3814 // after an unlikely jump in pos, the last trackpoint might also be wrong
3815 // just to be sure we do delete this one as well.
3816 ptarget->m_ptrack.pop_back();
3817 ptarget->b_positionDoubtful = true;
3818 return;
3819 }
3820 }
3821
3822 // Avoid duplicate track points
3823 // Do not add track point if time since last point is < 2 seconds.
3824 if ((ptarget->PositionReportTicks - ptarget->LastPositionReportTicks) > 2) {
3825 // Create the newest point
3826 AISTargetTrackPoint ptrackpoint;
3827 ptrackpoint.m_lat = ptarget->Lat;
3828 ptrackpoint.m_lon = ptarget->Lon;
3829 ptrackpoint.m_time = wxDateTime::Now().GetTicks();
3830
3831 ptarget->m_ptrack.push_back(ptrackpoint);
3832
3833 if (ptarget->b_PersistTrack || ptarget->b_mPropPersistTrack) {
3834 Track *t;
3835 if (0 == m_persistent_tracks.count(ptarget->MMSI)) {
3836 t = new Track();
3837 t->SetName(wxString::Format(
3838 _T("AIS %s (%u) %s %s"), ptarget->GetFullName().c_str(),
3839 ptarget->MMSI, wxDateTime::Now().FormatISODate().c_str(),
3840 wxDateTime::Now().FormatISOTime().c_str()));
3841 g_TrackList.push_back(t);
3842 new_track.Notify(t);
3843 m_persistent_tracks[ptarget->MMSI] = t;
3844 } else {
3845 t = m_persistent_tracks[ptarget->MMSI];
3846 }
3847 TrackPoint *tp = t->GetLastPoint();
3848 vector2D point(ptrackpoint.m_lon, ptrackpoint.m_lat);
3849 TrackPoint *tp1 =
3850 t->AddNewPoint(point, wxDateTime(ptrackpoint.m_time).ToUTC());
3851
3852 if (tp)
3853 pSelect->AddSelectableTrackSegment(tp->m_lat, tp->m_lon, tp1->m_lat,
3854 tp1->m_lon, tp, tp1, t);
3855
3856 // We do not want dependency on the GUI here, do we?
3857 // if( pRouteManagerDialog && pRouteManagerDialog->IsShown() )
3858 // pRouteManagerDialog->UpdateTrkListCtrl();
3859
3860 } else {
3861 // Walk the list, removing any track points that are older than the
3862 // stipulated time
3863 time_t test_time =
3864 wxDateTime::Now().GetTicks() - (time_t)(g_AISShowTracks_Mins * 60);
3865
3866 ptarget->m_ptrack.erase(
3867 std::remove_if(ptarget->m_ptrack.begin(), ptarget->m_ptrack.end(),
3868 [=](const AISTargetTrackPoint &track) {
3869 return track.m_time < test_time;
3870 }),
3871 ptarget->m_ptrack.end());
3872 }
3873 }
3874}
3875
3876void AisDecoder::DeletePersistentTrack(Track *track) {
3877 for (std::map<int, Track *>::iterator iterator = m_persistent_tracks.begin();
3878 iterator != m_persistent_tracks.end(); iterator++) {
3879 if (iterator->second == track) {
3880 int mmsi = iterator->first;
3881 m_persistent_tracks.erase(iterator);
3882 // Last tracks for this target?
3883 if (0 == m_persistent_tracks.count(mmsi)) {
3884 for (unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
3885 if (mmsi == g_MMSI_Props_Array[i]->MMSI) {
3886 MmsiProperties *props = g_MMSI_Props_Array[i];
3887 if (props->m_bPersistentTrack) {
3888 // Ask if mmsi props should be changed.
3889 // Avoid creation of a new track while messaging
3890
3891 std::shared_ptr<AisTargetData> td =
3892 Get_Target_Data_From_MMSI(mmsi);
3893 if (td) {
3894 props->m_bPersistentTrack = false;
3895 td->b_mPropPersistTrack = false;
3896 }
3897 if (!m_callbacks.confirm_stop_track()) {
3898 props->m_bPersistentTrack = true;
3899 }
3900 }
3901 break;
3902 }
3903 }
3904 }
3905 break;
3906 }
3907 }
3908}
3909
3910void AisDecoder::UpdateAllAlarms(void) {
3911 m_bGeneralAlert = false; // no alerts yet
3912
3913 // Iterate thru all the targets
3914 for (const auto &it : GetTargetList()) {
3915 std::shared_ptr<AisTargetData> td = it.second;
3916
3917 if (NULL != td) {
3918 // Maintain General Alert
3919 if (!m_bGeneralAlert) {
3920 // Quick check on basic condition
3921 if ((td->CPA < g_CPAWarn_NM) && (td->TCPA > 0) &&
3922 (td->Class != AIS_ATON) && (td->Class != AIS_BASE))
3923 m_bGeneralAlert = true;
3924
3925 // Some options can suppress general alerts
3926 if (g_bAIS_CPA_Alert_Suppress_Moored && (td->SOG <= g_ShowMoored_Kts))
3927 m_bGeneralAlert = false;
3928
3929 // Skip distant targets if requested
3930 if ((g_bCPAMax) && (td->Range_NM > g_CPAMax_NM))
3931 m_bGeneralAlert = false;
3932
3933 // Skip if TCPA is too long
3934 if ((g_bTCPA_Max) && (td->TCPA > g_TCPA_Max)) m_bGeneralAlert = false;
3935
3936 // SART targets always alert if "Active"
3937 if (td->Class == AIS_SART && td->NavStatus == 14)
3938 m_bGeneralAlert = true;
3939
3940 // DSC Distress targets always alert
3941 if ((td->Class == AIS_DSC) &&
3942 ((td->ShipType == 12) || (td->ShipType == 16)))
3943 m_bGeneralAlert = true;
3944 }
3945
3946 ais_alert_type this_alarm = AIS_NO_ALERT;
3947
3948 // SART targets always alert if "Active"
3949 if (td->Class == AIS_SART && td->NavStatus == 14)
3950 this_alarm = AIS_ALERT_SET;
3951
3952 // DSC Distress targets always alert
3953 if ((td->Class == AIS_DSC) &&
3954 ((td->ShipType == 12) || (td->ShipType == 16)))
3955 this_alarm = AIS_ALERT_SET;
3956
3957 if (g_bCPAWarn && td->b_active && td->b_positionOnceValid &&
3958 (td->Class != AIS_SART) && (td->Class != AIS_DSC)) {
3959 // Skip anchored/moored(interpreted as low speed) targets if
3960 // requested
3961 if ((g_bHideMoored) && (td->SOG <= g_ShowMoored_Kts)) { // dsr
3962 td->n_alert_state = AIS_NO_ALERT;
3963 continue;
3964 }
3965
3966 // No Alert on moored(interpreted as low speed) targets if so
3967 // requested
3968 if (g_bAIS_CPA_Alert_Suppress_Moored &&
3969 (td->SOG <= g_ShowMoored_Kts)) { // dsr
3970 td->n_alert_state = AIS_NO_ALERT;
3971 continue;
3972 }
3973
3974 // Skip distant targets if requested
3975 if (g_bCPAMax) {
3976 if (td->Range_NM > g_CPAMax_NM) {
3977 td->n_alert_state = AIS_NO_ALERT;
3978 continue;
3979 }
3980 }
3981
3982 if ((td->CPA < g_CPAWarn_NM) && (td->TCPA > 0) &&
3983 (td->Class != AIS_ATON) && (td->Class != AIS_BASE) &&
3984 (td->Class != AIS_METEO)) {
3985 if (g_bTCPA_Max) {
3986 if (td->TCPA < g_TCPA_Max) {
3987 if (td->b_isFollower)
3988 this_alarm = AIS_ALERT_NO_DIALOG_SET;
3989 else
3990 this_alarm = AIS_ALERT_SET;
3991 }
3992 } else {
3993 if (td->b_isFollower)
3994 this_alarm = AIS_ALERT_NO_DIALOG_SET;
3995 else
3996 this_alarm = AIS_ALERT_SET;
3997 }
3998 }
3999 }
4000
4001 // Maintain the timer for in_ack flag
4002 // SART and DSC targets always maintain ack timeout
4003
4004 if (g_bAIS_ACK_Timeout || (td->Class == AIS_SART) ||
4005 ((td->Class == AIS_DSC) &&
4006 ((td->ShipType == 12) || (td->ShipType == 16)))) {
4007 if (td->b_in_ack_timeout) {
4008 wxTimeSpan delta = wxDateTime::Now() - td->m_ack_time;
4009 if (delta.GetMinutes() > g_AckTimeout_Mins)
4010 td->b_in_ack_timeout = false;
4011 }
4012 } else {
4013 // Not using ack timeouts.
4014 // If a target has been acknowledged, leave it ack'ed until it goes out
4015 // of AIS_ALARM_SET state
4016 if (td->b_in_ack_timeout) {
4017 if (this_alarm == AIS_NO_ALERT) td->b_in_ack_timeout = false;
4018 }
4019 }
4020
4021 td->n_alert_state = this_alarm;
4022 }
4023 }
4024}
4025
4026void AisDecoder::UpdateOneCPA(AisTargetData *ptarget) {
4027 ptarget->Range_NM = -1.; // Defaults
4028 ptarget->Brg = -1.;
4029
4030 // Compute the current Range/Brg to the target
4031 // This should always be possible even if GPS data is not valid
4032 // because O must always have a position for own-ship. Plugins need
4033 // AIS target range and bearing from own-ship position even if GPS is not
4034 // valid.
4035 double brg, dist;
4036 DistanceBearingMercator(ptarget->Lat, ptarget->Lon, gLat, gLon, &brg, &dist);
4037 ptarget->Range_NM = dist;
4038 ptarget->Brg = brg;
4039
4040 if (dist <= 1e-5) ptarget->Brg = -1.0; // Brg is undefined if Range == 0.
4041
4042 if (!ptarget->b_positionOnceValid || !bGPSValid) {
4043 ptarget->bCPA_Valid = false;
4044 return;
4045 }
4046 // Ais Meteo is not a hard target in danger for collision
4047 if (ptarget->Class == AIS_METEO) {
4048 ptarget->bCPA_Valid = false;
4049 return;
4050 }
4051
4052 // There can be no collision between ownship and itself....
4053 // This can happen if AIVDO messages are received, and there is another
4054 // source of ownship position, like NMEA GLL The two positions are always
4055 // temporally out of sync, and one will always be exactly in front of the
4056 // other one.
4057 if (ptarget->b_OwnShip) {
4058 ptarget->CPA = 100;
4059 ptarget->TCPA = -100;
4060 ptarget->bCPA_Valid = false;
4061 return;
4062 }
4063
4064 double cpa_calc_ownship_cog = gCog;
4065 double cpa_calc_target_cog = ptarget->COG;
4066
4067 // Ownship is not reporting valid SOG, so no way to calculate CPA
4068 if (std::isnan(gSog) || (gSog > 102.2)) {
4069 ptarget->bCPA_Valid = false;
4070 return;
4071 }
4072
4073 // Ownship is maybe anchored and not reporting COG
4074 if (std::isnan(gCog) || gCog == 360.0) {
4075 if (gSog < .01)
4076 cpa_calc_ownship_cog =
4077 0.; // substitute value
4078 // for the case where SOG ~= 0, and COG is unknown.
4079 else {
4080 ptarget->bCPA_Valid = false;
4081 return;
4082 }
4083 }
4084
4085 // Target is maybe anchored and not reporting COG
4086 if (ptarget->COG == 360.0) {
4087 if (ptarget->SOG > 102.2) {
4088 ptarget->bCPA_Valid = false;
4089 return;
4090 } else if (ptarget->SOG < .01)
4091 cpa_calc_target_cog =
4092 0.; // substitute value
4093 // for the case where SOG ~= 0, and COG is unknown.
4094 else {
4095 ptarget->bCPA_Valid = false;
4096 return;
4097 }
4098 }
4099
4100 // Express the SOGs as meters per hour
4101 double v0 = gSog * 1852.;
4102 double v1 = ptarget->SOG * 1852.;
4103
4104 if ((v0 < 1e-6) && (v1 < 1e-6)) {
4105 ptarget->TCPA = 0.;
4106 ptarget->CPA = 0.;
4107
4108 ptarget->bCPA_Valid = false;
4109 } else {
4110 // Calculate the TCPA first
4111
4112 // Working on a Reduced Lat/Lon orthogonal plotting sheet....
4113 // Get easting/northing to target, in meters
4114
4115 double east1 = (ptarget->Lon - gLon) * 60 * 1852;
4116 double north1 = (ptarget->Lat - gLat) * 60 * 1852;
4117
4118 double east = east1 * (cos(gLat * PI / 180.));
4119
4120 double north = north1;
4121
4122 // Convert COGs trigonometry to standard unit circle
4123 double cosa = cos((90. - cpa_calc_ownship_cog) * PI / 180.);
4124 double sina = sin((90. - cpa_calc_ownship_cog) * PI / 180.);
4125 double cosb = cos((90. - cpa_calc_target_cog) * PI / 180.);
4126 double sinb = sin((90. - cpa_calc_target_cog) * PI / 180.);
4127
4128 // These will be useful
4129 double fc = (v0 * cosa) - (v1 * cosb);
4130 double fs = (v0 * sina) - (v1 * sinb);
4131
4132 double d = (fc * fc) + (fs * fs);
4133 double tcpa;
4134
4135 // the tracks are almost parallel
4136 if (fabs(d) < 1e-6)
4137 tcpa = 0.;
4138 else
4139 // Here is the equation for t, which will be in hours
4140 tcpa = ((fc * east) + (fs * north)) / d;
4141
4142 // Convert to minutes
4143 ptarget->TCPA = tcpa * 60.;
4144
4145 // Calculate CPA
4146 // Using TCPA, predict ownship and target positions
4147
4148 double OwnshipLatCPA, OwnshipLonCPA, TargetLatCPA, TargetLonCPA;
4149
4150 ll_gc_ll(gLat, gLon, cpa_calc_ownship_cog, gSog * tcpa, &OwnshipLatCPA,
4151 &OwnshipLonCPA);
4152 ll_gc_ll(ptarget->Lat, ptarget->Lon, cpa_calc_target_cog,
4153 ptarget->SOG * tcpa, &TargetLatCPA, &TargetLonCPA);
4154
4155 // And compute the distance
4156 ptarget->CPA = DistGreatCircle(OwnshipLatCPA, OwnshipLonCPA, TargetLatCPA,
4157 TargetLonCPA);
4158
4159 ptarget->bCPA_Valid = true;
4160
4161 if (ptarget->TCPA < 0) ptarget->bCPA_Valid = false;
4162 }
4163}
4164
4165void AisDecoder::OnTimerDSC(wxTimerEvent &event) {
4166 // Timer expired, no CDDSE message was received, so accept the latest CDDSC
4167 // message
4168 if (m_ptentative_dsctarget) {
4169 ProcessDSx(m_dsc_last_string, true);
4170 }
4171}
4172
4173void AisDecoder::OnTimerAIS(wxTimerEvent &event) {
4174 TimerAIS.Stop();
4175 // Scrub the target hash list
4176 // removing any targets older than stipulated age
4177
4178 wxDateTime now = wxDateTime::Now();
4179 now.MakeGMT();
4180
4181 std::unordered_map<int, std::shared_ptr<AisTargetData>> &current_targets =
4182 GetTargetList();
4183
4184 auto it = current_targets.begin();
4185 std::vector<int> remove_array; // collector for MMSI of targets to be removed
4186
4187 while (it != current_targets.end()) {
4188 if (it->second == NULL) // This should never happen, but I saw it once....
4189 {
4190 current_targets.erase(it);
4191 break; // leave the loop
4192 }
4193 // std::shared_ptr<AisTargetData>
4194 // xtd(std::make_shared<AisTargetData>(*it->second));
4195 std::shared_ptr<AisTargetData> xtd = it->second;
4196
4197 int target_posn_age = now.GetTicks() - xtd->PositionReportTicks;
4198 int target_static_age = now.GetTicks() - xtd->StaticReportTicks;
4199
4200 // Global variables controlling lost target handling
4201 // g_bMarkLost
4202 // g_MarkLost_Mins // Minutes until black "cross out
4203 // g_bRemoveLost
4204 // g_RemoveLost_Mins); // minutes until target is removed from screen and
4205 // internal lists
4206
4207 // g_bInlandEcdis
4208
4209 // Mark lost targets if specified
4210 double removelost_Mins = fmax(g_RemoveLost_Mins, g_MarkLost_Mins);
4211
4212 if (g_bInlandEcdis && (xtd->Class != AIS_ARPA)) {
4213 double iECD_LostTimeOut = 0.0;
4214 // special rules apply for europe inland ecdis timeout settings. overrule
4215 // option settings Won't apply for ARPA targets where the radar has all
4216 // control
4217 if (xtd->Class == AIS_CLASS_B) {
4218 if ((xtd->NavStatus == MOORED) || (xtd->NavStatus == AT_ANCHOR))
4219 iECD_LostTimeOut = 18 * 60;
4220 else
4221 iECD_LostTimeOut = 180;
4222 }
4223 if (xtd->Class == AIS_CLASS_A) {
4224 if ((xtd->NavStatus == MOORED) || (xtd->NavStatus == AT_ANCHOR)) {
4225 if (xtd->SOG < 3.)
4226 iECD_LostTimeOut = 18 * 60;
4227 else
4228 iECD_LostTimeOut = 60;
4229 } else
4230 iECD_LostTimeOut = 60;
4231 }
4232
4233 if ((target_posn_age > iECD_LostTimeOut) &&
4234 (xtd->Class != AIS_GPSG_BUDDY))
4235 xtd->b_active = false;
4236
4237 removelost_Mins = (2 * iECD_LostTimeOut) / 60.;
4238 } else if (g_bMarkLost) {
4239 if ((target_posn_age > g_MarkLost_Mins * 60) &&
4240 (xtd->Class != AIS_GPSG_BUDDY))
4241 xtd->b_active = false;
4242 }
4243
4244 if (xtd->Class == AIS_SART || xtd->Class == AIS_METEO)
4245 removelost_Mins = 18.0;
4246
4247 // Remove lost targets if specified
4248
4249 if (g_bRemoveLost || g_bInlandEcdis) {
4250 bool b_arpalost =
4251 (xtd->Class == AIS_ARPA &&
4252 xtd->b_lost); // A lost ARPA target would be deleted at once
4253 if (((target_posn_age > removelost_Mins * 60) &&
4254 (xtd->Class != AIS_GPSG_BUDDY)) ||
4255 b_arpalost) {
4256 // So mark the target as lost, with unknown position, and make it
4257 // not selectable
4258 xtd->b_lost = true;
4259 xtd->b_positionOnceValid = false;
4260 xtd->COG = 360.0;
4261 xtd->SOG = 103.0;
4262 xtd->HDG = 511.0;
4263 xtd->ROTAIS = -128;
4264
4265 plugin_msg.Notify(xtd, "");
4266
4267 long mmsi_long = xtd->MMSI;
4268 pSelectAIS->DeleteSelectablePoint((void *)mmsi_long, SELTYPE_AISTARGET);
4269
4270 // If we have not seen a static report in 3 times the removal spec,
4271 // then remove the target from all lists
4272 // or a lost ARPA target.
4273 if (target_static_age > removelost_Mins * 60 * 3 || b_arpalost) {
4274 xtd->b_removed = true;
4275 plugin_msg.Notify(xtd, "");
4276 remove_array.push_back(xtd->MMSI); // Add this target to removal list
4277 }
4278 }
4279 }
4280
4281 // Remove any targets specified as to be "ignored", so that they won't
4282 // trigger phantom alerts (e.g. SARTs)
4283 for (unsigned int i = 0; i < g_MMSI_Props_Array.GetCount(); i++) {
4284 MmsiProperties *props = g_MMSI_Props_Array[i];
4285 if (xtd->MMSI == props->MMSI) {
4286 if (props->m_bignore) {
4287 remove_array.push_back(xtd->MMSI); // Add this target to removal list
4288 xtd->b_removed = true;
4289 plugin_msg.Notify(xtd, "");
4290 }
4291 break;
4292 }
4293 }
4294
4295 // Check if the target has recently been set as own MMSI
4296 if (xtd->MMSI == g_OwnShipmmsi) {
4297 remove_array.push_back(xtd->MMSI); // Add this target to removal list
4298 xtd->b_removed = true;
4299 plugin_msg.Notify(xtd, "");
4300 }
4301
4302 ++it;
4303 }
4304
4305 // Remove all the targets collected in remove_array in one pass
4306 for (unsigned int i = 0; i < remove_array.size(); i++) {
4307 auto itd = current_targets.find(remove_array[i]);
4308 if (itd != current_targets.end()) {
4309 std::shared_ptr<AisTargetData> td = itd->second;
4310 current_targets.erase(itd);
4311 // delete td;
4312 }
4313 }
4314
4315 UpdateAllCPA();
4316 UpdateAllAlarms();
4317
4318 // Update the general suppression flag
4319 m_bSuppressed = false;
4320 if (g_bAIS_CPA_Alert_Suppress_Moored || g_bHideMoored ||
4321 (g_bShowScaled && g_bAllowShowScaled))
4322 m_bSuppressed = true;
4323
4324 m_bAIS_Audio_Alert_On = false; // default, may be set on
4325
4326 // Process any Alarms
4327
4328 // If the AIS Alert Dialog is not currently shown....
4329
4330 // Scan all targets, looking for SART, DSC Distress, and CPA incursions
4331 // In the case of multiple targets of the same type, select the shortest
4332 // range or shortest TCPA
4333 std::shared_ptr<AisTargetData> palert_target = NULL;
4334 int audioType = AISAUDIO_NONE;
4335
4336 if (!g_pais_alert_dialog_active) {
4337 pAISMOBRoute = NULL; // Reset the AISMOB auto route.
4338 double tcpa_min = 1e6; // really long
4339 double sart_range = 1e6;
4340 std::shared_ptr<AisTargetData> palert_target_cpa = NULL;
4341 std::shared_ptr<AisTargetData> palert_target_sart = NULL;
4342 std::shared_ptr<AisTargetData> palert_target_dsc = NULL;
4343
4344 for (it = current_targets.begin(); it != current_targets.end(); ++it) {
4345 std::shared_ptr<AisTargetData> td = it->second;
4346 if (td) {
4347 if ((td->Class != AIS_SART) && (td->Class != AIS_DSC)) {
4348 if (g_bAIS_CPA_Alert && td->b_active) {
4349 if ((AIS_ALERT_SET == td->n_alert_state) && !td->b_in_ack_timeout) {
4350 if (td->TCPA < tcpa_min) {
4351 tcpa_min = td->TCPA;
4352 palert_target_cpa = td;
4353 }
4354 }
4355 }
4356 } else if ((td->Class == AIS_DSC) &&
4357 ((td->ShipType == 12) || (td->ShipType == 16))) {
4358 if (td->b_active) {
4359 if ((AIS_ALERT_SET == td->n_alert_state) && !td->b_in_ack_timeout) {
4360 palert_target_dsc = td;
4361 } else { // Reset DCS flag to open for a real AIS for the same
4362 // target
4363 td->b_isDSCtarget = false;
4364 }
4365 }
4366 }
4367
4368 else if (td->Class == AIS_SART) {
4369 if (td->b_active) {
4370 if ((AIS_ALERT_SET == td->n_alert_state) && !td->b_in_ack_timeout) {
4371 if (td->Range_NM < sart_range) {
4372 tcpa_min = sart_range;
4373 palert_target_sart = td;
4374 }
4375 }
4376 }
4377 }
4378 }
4379 }
4380
4381 // Which of multiple targets?
4382 // Give priority to SART targets, then DSC Distress, then CPA incursion
4383 palert_target = palert_target_cpa;
4384 if (palert_target) audioType = AISAUDIO_CPA;
4385
4386 if (palert_target_sart) {
4387 palert_target = palert_target_sart;
4388 audioType = AISAUDIO_SART;
4389 }
4390
4391 if (palert_target_dsc) {
4392 palert_target = palert_target_dsc;
4393 audioType = AISAUDIO_DSC;
4394 }
4395 } else {
4396 // Alert is currently shown, get target from from knowable GUI
4397 palert_target = Get_Target_Data_From_MMSI(m_callbacks.get_target_mmsi());
4398 }
4399 // Show or update the alert
4400 if (palert_target) info_update.Notify(palert_target, "");
4401
4402 TimerAIS.Start(TIMER_AIS_MSEC, wxTIMER_CONTINUOUS);
4403}
4404
4405std::shared_ptr<AisTargetData> AisDecoder::Get_Target_Data_From_MMSI(int mmsi) {
4406 if (AISTargetList.find(mmsi) == AISTargetList.end())
4407 return NULL;
4408 else
4409 return AISTargetList[mmsi];
4410}
4411
4412ArrayOfMmsiProperties g_MMSI_Props_Array;
4413
4414// MmsiProperties Implementation
4415
4416MmsiProperties::MmsiProperties(wxString &spec) {
4417 Init();
4418 wxStringTokenizer tkz(spec, _T(";"));
4419 wxString s;
4420
4421 s = tkz.GetNextToken();
4422 long mmsil;
4423 s.ToLong(&mmsil);
4424 MMSI = (int)mmsil;
4425
4426 s = tkz.GetNextToken();
4427 if (s.Len()) {
4428 if (s.Upper() == _T("ALWAYS"))
4429 TrackType = TRACKTYPE_ALWAYS;
4430 else if (s.Upper() == _T("NEVER"))
4431 TrackType = TRACKTYPE_NEVER;
4432 }
4433
4434 s = tkz.GetNextToken();
4435 if (s.Len()) {
4436 if (s.Upper() == _T("IGNORE")) m_bignore = true;
4437 }
4438
4439 s = tkz.GetNextToken();
4440 if (s.Len()) {
4441 if (s.Upper() == _T("MOB")) m_bMOB = true;
4442 }
4443
4444 s = tkz.GetNextToken();
4445 if (s.Len()) {
4446 if (s.Upper() == _T("VDM")) m_bVDM = true;
4447 }
4448
4449 s = tkz.GetNextToken();
4450 if (s.Len()) {
4451 if (s.Upper() == _T("FOLLOWER")) m_bFollower = true;
4452 }
4453
4454 s = tkz.GetNextToken();
4455 if (s.Len()) {
4456 if (s.Upper() == _T("PERSIST")) m_bPersistentTrack = true;
4457 }
4458
4459 s = tkz.GetNextToken();
4460 if (s.Len()) {
4461 m_ShipName = s.Upper();
4462 }
4463}
4464
4465MmsiProperties::~MmsiProperties() {}
4466
4467void MmsiProperties::Init(void) {
4468 MMSI = -1;
4469 TrackType = TRACKTYPE_DEFAULT;
4470 m_bignore = false;
4471 m_bMOB = false;
4472 m_bVDM = false;
4473 m_bFollower = false;
4474 m_bPersistentTrack = false;
4475 m_ShipName = wxEmptyString;
4476}
4477
4478wxString MmsiProperties::Serialize(void) {
4479 wxString sMMSI;
4480 wxString s;
4481
4482 sMMSI.Printf(_T("%d"), MMSI);
4483 sMMSI << _T(";");
4484
4485 if (TrackType) {
4486 if (TRACKTYPE_ALWAYS == TrackType)
4487 sMMSI << _T("always");
4488 else if (TRACKTYPE_NEVER == TrackType)
4489 sMMSI << _T("never");
4490 }
4491 sMMSI << _T(";");
4492
4493 if (m_bignore) {
4494 sMMSI << _T("ignore");
4495 }
4496 sMMSI << _T(";");
4497
4498 if (m_bMOB) {
4499 sMMSI << _T("MOB");
4500 }
4501 sMMSI << _T(";");
4502
4503 if (m_bVDM) {
4504 sMMSI << _T("VDM");
4505 }
4506 sMMSI << _T(";");
4507
4508 if (m_bFollower) {
4509 sMMSI << _T("Follower");
4510 }
4511 sMMSI << _T(";");
4512
4513 if (m_bPersistentTrack) {
4514 sMMSI << _T("PERSIST");
4515 }
4516 sMMSI << _T(";");
4517
4518 if (m_ShipName == wxEmptyString) {
4519 m_ShipName = GetShipNameFromFile(MMSI);
4520 }
4521 sMMSI << m_ShipName;
4522 return sMMSI;
4523}
4524
4525void AISshipNameCache(AisTargetData *pTargetData,
4526 AIS_Target_Name_Hash *AISTargetNamesC,
4527 AIS_Target_Name_Hash *AISTargetNamesNC, long mmsi) {
4528 if (g_benableAISNameCache) {
4529 wxString ship_name = wxEmptyString;
4530
4531 // Check for valid name data
4532 if (!pTargetData->b_nameValid) {
4533 AIS_Target_Name_Hash::iterator it = AISTargetNamesC->find(mmsi);
4534 if (it != AISTargetNamesC->end()) {
4535 ship_name = (*AISTargetNamesC)[mmsi].Left(20);
4536 strncpy(pTargetData->ShipName, ship_name.mb_str(),
4537 ship_name.length() + 1);
4538 pTargetData->b_nameValid = true;
4539 pTargetData->b_nameFromCache = true;
4540 } else if (!g_bUseOnlyConfirmedAISName) {
4541 it = AISTargetNamesNC->find(mmsi);
4542 if (it != AISTargetNamesNC->end()) {
4543 ship_name = (*AISTargetNamesNC)[mmsi].Left(20);
4544 strncpy(pTargetData->ShipName, ship_name.mb_str(),
4545 ship_name.length() + 1);
4546 pTargetData->b_nameValid = true;
4547 pTargetData->b_nameFromCache = true;
4548 }
4549 }
4550 }
4551 // else there IS a valid name, lets check if it is in one of the hash lists.
4552 else if ((pTargetData->MID == 5) || (pTargetData->MID == 24) ||
4553 (pTargetData->MID == 19) ||
4554 (pTargetData->MID == 123) || // 123: Has got a name from SignalK
4555 (pTargetData->MID == 124)) { // 124: Has got a name from n2k
4556 // This message contains ship static data, so has a name field
4557 pTargetData->b_nameFromCache = false;
4558 ship_name = trimAISField(pTargetData->ShipName);
4559 AIS_Target_Name_Hash::iterator itC = AISTargetNamesC->find(mmsi);
4560 AIS_Target_Name_Hash::iterator itNC = AISTargetNamesNC->find(mmsi);
4561 if (itC !=
4562 AISTargetNamesC->end()) { // There is a confirmed entry for this mmsi
4563 if ((*AISTargetNamesC)[mmsi] ==
4564 ship_name) { // Received name is same as confirmed name
4565 if (itNC != AISTargetNamesNC->end()) { // there is also an entry in
4566 // the NC list, delete it
4567 AISTargetNamesNC->erase(itNC);
4568 }
4569 } else { // There is a confirmed name but that one is different
4570 if (itNC != AISTargetNamesNC->end()) { // there is an entry in the NC
4571 // list check if name is same
4572 if ((*AISTargetNamesNC)[mmsi] ==
4573 ship_name) { // Same name is already in NC list so promote till
4574 // confirmed list
4575 (*AISTargetNamesC)[mmsi] = ship_name;
4576 // And delete from NC list
4577 AISTargetNamesNC->erase(itNC);
4578 } else { // A different name is in the NC list, update with
4579 // received one
4580 (*AISTargetNamesNC)[mmsi] = ship_name;
4581 }
4582 if (g_bUseOnlyConfirmedAISName)
4583 strncpy(pTargetData->ShipName, (*AISTargetNamesC)[mmsi].mb_str(),
4584 (*AISTargetNamesC)[mmsi].Left(20).Length() + 1);
4585 } else { // The C list name is different but no NC entry. Add it.
4586 (*AISTargetNamesNC)[mmsi] = ship_name;
4587 }
4588 }
4589 } else { // No confirmed entry available
4590 if (itNC !=
4591 AISTargetNamesNC->end()) { // there is an entry in the NC list,
4592 if ((*AISTargetNamesNC)[mmsi] ==
4593 ship_name) { // Received name same as already in NC list, promote
4594 // to confirmen
4595 (*AISTargetNamesC)[mmsi] = ship_name;
4596 // And delete from NC list
4597 AISTargetNamesNC->erase(itNC);
4598 } else { // entry in NC list is not same as received one
4599 (*AISTargetNamesNC)[mmsi] = ship_name;
4600 }
4601 } else { // No entry in NC list so add it
4602 (*AISTargetNamesNC)[mmsi] = ship_name;
4603 }
4604 if (g_bUseOnlyConfirmedAISName) { // copy back previous name
4605 pTargetData->ShipName[SHIP_NAME_LEN - 1] = '\0';
4606 strncpy(pTargetData->ShipName, "Unknown ",
4607 SHIP_NAME_LEN - 1);
4608 }
4609 }
4610 }
4611 }
4612}
4613
4614wxString GetShipNameFromFile(int nmmsi) {
4615 wxString name = wxEmptyString;
4616 if (g_benableAISNameCache) {
4617 std::ifstream infile(AISTargetNameFileName.mb_str());
4618 if (infile) {
4619 std::string line;
4620 while (getline(infile, line)) {
4621 wxStringTokenizer tokenizer(wxString::FromUTF8(line.c_str()), _T(","));
4622 if (nmmsi == wxAtoi(tokenizer.GetNextToken())) {
4623 name = tokenizer.GetNextToken().Trim();
4624 break;
4625 } else
4626 tokenizer.GetNextToken();
4627 }
4628 }
4629 infile.close();
4630 }
4631 return name;
4632}
4633
4634// Assign a unique meteo mmsi related to position
4635int AisMeteoNewMmsi(int orig_mmsi, int m_lat, int m_lon, int lon_bits = 0,
4636 int siteID = 0) {
4637 bool found = false;
4638 int new_mmsi = 0;
4639 if ((!lon_bits || lon_bits == 999) && siteID) {
4640 // Check if a ais8_367_33 data report belongs to a present site
4641 // Or SignalK data (lon_bits == 999)
4642 auto &points = AisMeteoPoints::GetInstance().GetPoints();
4643 if (points.size()) {
4644 for (const auto &point : points) {
4645 // Does this station ID exist
4646 if (siteID == point.siteID && orig_mmsi == point.orig_mmsi) {
4647 // Created before. Continue
4648 new_mmsi = point.mmsi;
4649 found = true;
4650 break;
4651 }
4652 }
4653 }
4654 if (!found && !lon_bits) {
4655 // ais8_367_33
4656 return 0;
4657 }
4658 }
4659 double lon_tentative = 181.;
4660 double lat_tentative = 91.;
4661
4662 if (lon_bits == 25) {
4663 if (m_lon & 0x01000000) // negative?
4664 m_lon |= 0xFE000000;
4665 lon_tentative = m_lon / 60000.;
4666
4667 if (m_lat & 0x00800000) // negative?
4668 m_lat |= 0xFF000000;
4669 lat_tentative = m_lat / 60000.;
4670
4671 } else if (lon_bits == 28) {
4672 if (m_lon & 0x08000000) // negative?
4673 m_lon |= 0xf0000000;
4674 lon_tentative = m_lon / 600000.;
4675
4676 if (m_lat & 0x04000000) // negative?
4677 m_lat |= 0xf8000000;
4678 lat_tentative = m_lat / 600000.;
4679 }
4680
4681 // Since buoys can move we use position precision not better
4682 // than 50 m to be able to compare previous messages
4683 wxString slon = wxString::Format("%0.3f", lon_tentative);
4684 wxString slat = wxString::Format("%0.3f", lat_tentative);
4685
4686 // Change mmsi_ID number
4687 // Some countries use one equal mmsi for all meteo stations.
4688 // Others use the same mmsi for a meteo station and a nearby AtoN
4689 // So we create our own fake mmsi to separate them.
4690 // 199 is INMARSAT-A MID, should not occur ever in AIS stream.
4691 // 1992 to 1993 are already used so here we use 1994+
4692 static int nextMeteommsi = 199400000;
4693 auto &points = AisMeteoPoints::GetInstance().GetPoints();
4694
4695 if (lon_bits != 999 && points.size()) { // 999 comes from SignalK
4696 wxString t_lat, t_lon;
4697 for (const auto &point : points) {
4698 // Does this station position exist
4699 if (slat.IsSameAs(point.lat) && slon.IsSameAs(point.lon)) {
4700 // Created before. Continue
4701 new_mmsi = point.mmsi;
4702 found = true;
4703 break;
4704 }
4705 }
4706 }
4707 if (!found) {
4708 // Create a new post
4709 nextMeteommsi++;
4710 points.push_back(
4711 AisMeteoPoint(nextMeteommsi, slat, slon, siteID, orig_mmsi));
4712 new_mmsi = nextMeteommsi;
4713 }
4714 return new_mmsi;
4715}
4716
4717bool isBuoyMmsi(const int msi) {
4718 // IMO standard is not yet(?) implemented for (net)buoys
4719 // This adaption, based on real-world outcomes, is used instead
4720 // Consider any not valid MMSI number for a class B target (message 18 or 19)
4721 // to be a "net buoy"
4722 int mid = msi / 1000000;
4723 if ((mid > 200 && mid < 800) || mid >= 970) {
4724 return false;
4725 } else {
4726 return true;
4727 }
4728 return false;
4729}
Global state for AIS decoder.
int GetInt(int sp, int len, bool signed_flag=false)
sp is starting bit, 1-based
EventVar plugin_msg
A JSON message should be sent.
EventVar new_track
Notified on new track creation.
EventVar info_update
Notified when AIS user dialogs should update.
EventVar touch_state
Notified when gFrame->TouchAISActive() should be invoked.
Add a new point to the list of Meteo stations.
const void Notify()
Notify all listeners, no data supplied.
void LogInputMessage(const wxString &msg, const wxString &stream_name, bool b_filter, bool b_error=false, const wxString error_msg=wxEmptyString)
Logs an input message with context information.
A regular Nmea0183 message.
See: https://github.com/OpenCPN/OpenCPN/issues/2729#issuecomment-1179506343.
void Listen(const std::string &key, wxEvtHandler *listener, wxEventType evt)
Set object to send wxEventType ev to listener on changes in key.
Custom event class for OpenCPN's notification system.
std::shared_ptr< const void > GetSharedPtr() const
Gets the event's payload data.
Represents a waypoint or mark within the navigation system.
Definition route_point.h:68
bool m_bIsolatedMark
Flag indicating if the waypoint is a standalone mark.
A parsed SignalK message over ipv4.
Represents a single point in a track.
Definition track.h:53
Represents a track, which is a series of connected track points.
Definition track.h:79
The JSON document writer.
Definition jsonwriter.h:50
void Write(const wxJSONValue &value, wxString &str)
Write the JSONvalue object to a JSON text.
Raw messages layer, supports sending and recieving navmsg messages.
A generic position and navigation data structure.
Definition ocpn_types.h:74
double kCog
Course over ground in degrees.
Definition ocpn_types.h:92
double kHdt
True heading in degrees.
Definition ocpn_types.h:117
double kHdm
Magnetic heading in degrees.
Definition ocpn_types.h:110
double kLat
Latitude in decimal degrees.
Definition ocpn_types.h:81
double kSog
Speed over ground in knots.
Definition ocpn_types.h:98
double kVar
Magnetic variation in degrees.
Definition ocpn_types.h:104
double kLon
Longitude in decimal degrees.
Definition ocpn_types.h:89