OpenCPN Partial API docs
Loading...
Searching...
No Matches
autopilot_output.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: Autopilot output support
5 * Author: David Register
6 *
7 ***************************************************************************
8 * Copyright (C) 2025 by David S. Register *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 **************************************************************************/
25#include <cmath>
26#include <memory>
27#include <vector>
28
29#include <wx/wxprec.h>
30
31#include "model/autopilot_output.h"
32#include "model/comm_drv_n2k_serial.h"
34#include "model/comm_n0183_output.h"
35#include "model/comm_vars.h"
36#include "model/config_vars.h"
37#include "model/georef.h"
38#include "model/nmea_ctx_factory.h"
39#include "model/own_ship.h"
40#include "model/routeman.h"
41#include "N2kMsg.h"
42#include "N2kMessages.h"
43
44#include "observable_globvar.h"
45
46#ifdef __ANDROID__
47#include "androidUTIL.h"
48#endif
49
50bool UpdateAutopilotN0183(Routeman &routeman) {
51 NMEA0183 nmea0183 = routeman.GetNMEA0183();
52 RoutePoint *pActivePoint = routeman.GetpActivePoint();
53
54 // Set max WP name length
55 int maxName = 6;
56 if ((g_maxWPNameLength >= 3) && (g_maxWPNameLength <= 32))
57 maxName = g_maxWPNameLength;
58
59 // Avoid a possible not initiated SOG/COG. APs can be confused if in NAV mode
60 // wo valid GPS
61 double r_Sog(0.0), r_Cog(0.0);
62 if (!std::isnan(gSog)) r_Sog = gSog;
63 if (!std::isnan(gCog)) r_Cog = gCog;
64
65 // RMB
66 {
67 nmea0183.TalkerID = "EC";
68 SENTENCE snt;
69 nmea0183.Rmb.IsDataValid = bGPSValid ? NTrue : NFalse;
70 nmea0183.Rmb.CrossTrackError = routeman.GetCurrentXTEToActivePoint();
71 nmea0183.Rmb.DirectionToSteer = routeman.GetXTEDir() < 0 ? Left : Right;
72 nmea0183.Rmb.RangeToDestinationNauticalMiles =
73 routeman.GetCurrentRngToActivePoint();
74 nmea0183.Rmb.BearingToDestinationDegreesTrue =
75 routeman.GetCurrentBrgToActivePoint();
76
77 if (pActivePoint->m_lat < 0.)
78 nmea0183.Rmb.DestinationPosition.Latitude.Set(-pActivePoint->m_lat, "S");
79 else
80 nmea0183.Rmb.DestinationPosition.Latitude.Set(pActivePoint->m_lat, "N");
81
82 if (pActivePoint->m_lon < 0.)
83 nmea0183.Rmb.DestinationPosition.Longitude.Set(-pActivePoint->m_lon, "W");
84 else
85 nmea0183.Rmb.DestinationPosition.Longitude.Set(pActivePoint->m_lon, "E");
86
87 nmea0183.Rmb.DestinationClosingVelocityKnots =
88 r_Sog *
89 cos((r_Cog - routeman.GetCurrentBrgToActivePoint()) * PI / 180.0);
90 nmea0183.Rmb.IsArrivalCircleEntered =
91 routeman.GetArrival() ? NTrue : NFalse;
92 nmea0183.Rmb.FAAModeIndicator = bGPSValid ? "A" : "N";
93 // RMB is close to NMEA0183 length limit
94 // Restrict WP names further if necessary
95 int wp_len = maxName;
96 do {
97 nmea0183.Rmb.To = pActivePoint->GetName().Truncate(wp_len);
98 nmea0183.Rmb.From =
99 routeman.GetpActiveRouteSegmentBeginPoint()->GetName().Truncate(
100 wp_len);
101 nmea0183.Rmb.Write(snt);
102 wp_len -= 1;
103 } while (snt.Sentence.size() > 82 && wp_len > 0);
104
105 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
106 routeman.GetMessageSentEventVar());
107 }
108
109 // RMC
110 {
111 nmea0183.TalkerID = _T("EC");
112
113 SENTENCE snt;
114 nmea0183.Rmc.IsDataValid = NTrue;
115 if (!bGPSValid) nmea0183.Rmc.IsDataValid = NFalse;
116
117 if (gLat < 0.)
118 nmea0183.Rmc.Position.Latitude.Set(-gLat, _T("S"));
119 else
120 nmea0183.Rmc.Position.Latitude.Set(gLat, _T("N"));
121
122 if (gLon < 0.)
123 nmea0183.Rmc.Position.Longitude.Set(-gLon, _T("W"));
124 else
125 nmea0183.Rmc.Position.Longitude.Set(gLon, _T("E"));
126
127 nmea0183.Rmc.SpeedOverGroundKnots = r_Sog;
128 nmea0183.Rmc.TrackMadeGoodDegreesTrue = r_Cog;
129
130 if (!std::isnan(gVar)) {
131 if (gVar < 0.) {
132 nmea0183.Rmc.MagneticVariation = -gVar;
133 nmea0183.Rmc.MagneticVariationDirection = West;
134 } else {
135 nmea0183.Rmc.MagneticVariation = gVar;
136 nmea0183.Rmc.MagneticVariationDirection = East;
137 }
138 } else
139 nmea0183.Rmc.MagneticVariation =
140 361.; // A signal to NMEA converter, gVAR is unknown
141
142 // Send GPS time to autopilot if available else send local system time
143 if (!gRmcTime.IsEmpty() && !gRmcDate.IsEmpty()) {
144 nmea0183.Rmc.UTCTime = gRmcTime;
145 nmea0183.Rmc.Date = gRmcDate;
146 } else {
147 wxDateTime now = wxDateTime::Now();
148 wxDateTime utc = now.ToUTC();
149 wxString time = utc.Format(_T("%H%M%S"));
150 nmea0183.Rmc.UTCTime = time;
151 wxString date = utc.Format(_T("%d%m%y"));
152 nmea0183.Rmc.Date = date;
153 }
154
155 nmea0183.Rmc.FAAModeIndicator = "A";
156 if (!bGPSValid) nmea0183.Rmc.FAAModeIndicator = "N";
157
158 nmea0183.Rmc.Write(snt);
159
160 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
161 routeman.GetMessageSentEventVar());
162 }
163
164 // APB
165 {
166 nmea0183.TalkerID = _T("EC");
167
168 SENTENCE snt;
169
170 nmea0183.Apb.IsLoranBlinkOK =
171 NTrue; // considered as "generic invalid fix" flag
172 if (!bGPSValid) nmea0183.Apb.IsLoranBlinkOK = NFalse;
173
174 nmea0183.Apb.IsLoranCCycleLockOK = NTrue;
175 if (!bGPSValid) nmea0183.Apb.IsLoranCCycleLockOK = NFalse;
176
177 nmea0183.Apb.CrossTrackErrorMagnitude =
178 routeman.GetCurrentXTEToActivePoint();
179
180 if (routeman.GetXTEDir() < 0)
181 nmea0183.Apb.DirectionToSteer = Left;
182 else
183 nmea0183.Apb.DirectionToSteer = Right;
184
185 nmea0183.Apb.CrossTrackUnits = _T("N");
186
187 if (routeman.GetArrival())
188 nmea0183.Apb.IsArrivalCircleEntered = NTrue;
189 else
190 nmea0183.Apb.IsArrivalCircleEntered = NFalse;
191
192 // We never pass the perpendicular, since we declare arrival before
193 // reaching this point
194 nmea0183.Apb.IsPerpendicular = NFalse;
195
196 nmea0183.Apb.To = pActivePoint->GetName().Truncate(maxName);
197
198 double brg1, dist1;
199 DistanceBearingMercator(pActivePoint->m_lat, pActivePoint->m_lon,
200 routeman.GetpActiveRouteSegmentBeginPoint()->m_lat,
201 routeman.GetpActiveRouteSegmentBeginPoint()->m_lon,
202 &brg1, &dist1);
203
204 if (g_bMagneticAPB && !std::isnan(gVar)) {
205 double brg1m =
206 ((brg1 - gVar) >= 0.) ? (brg1 - gVar) : (brg1 - gVar + 360.);
207 double bapm = ((routeman.GetCurrentBrgToActivePoint() - gVar) >= 0.)
208 ? (routeman.GetCurrentBrgToActivePoint() - gVar)
209 : (routeman.GetCurrentBrgToActivePoint() - gVar + 360.);
210
211 nmea0183.Apb.BearingOriginToDestination = brg1m;
212 nmea0183.Apb.BearingOriginToDestinationUnits = _T("M");
213
214 nmea0183.Apb.BearingPresentPositionToDestination = bapm;
215 nmea0183.Apb.BearingPresentPositionToDestinationUnits = _T("M");
216
217 nmea0183.Apb.HeadingToSteer = bapm;
218 nmea0183.Apb.HeadingToSteerUnits = _T("M");
219 } else {
220 nmea0183.Apb.BearingOriginToDestination = brg1;
221 nmea0183.Apb.BearingOriginToDestinationUnits = _T("T");
222
223 nmea0183.Apb.BearingPresentPositionToDestination =
224 routeman.GetCurrentBrgToActivePoint();
225 nmea0183.Apb.BearingPresentPositionToDestinationUnits = _T("T");
226
227 nmea0183.Apb.HeadingToSteer = routeman.GetCurrentBrgToActivePoint();
228 nmea0183.Apb.HeadingToSteerUnits = _T("T");
229 }
230
231 nmea0183.Apb.Write(snt);
232 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
233 routeman.GetMessageSentEventVar());
234 }
235
236 // XTE
237 {
238 nmea0183.TalkerID = _T("EC");
239
240 SENTENCE snt;
241
242 nmea0183.Xte.IsLoranBlinkOK =
243 NTrue; // considered as "generic invalid fix" flag
244 if (!bGPSValid) nmea0183.Xte.IsLoranBlinkOK = NFalse;
245
246 nmea0183.Xte.IsLoranCCycleLockOK = NTrue;
247 if (!bGPSValid) nmea0183.Xte.IsLoranCCycleLockOK = NFalse;
248
249 nmea0183.Xte.CrossTrackErrorDistance =
250 routeman.GetCurrentXTEToActivePoint();
251
252 if (routeman.GetXTEDir() < 0)
253 nmea0183.Xte.DirectionToSteer = Left;
254 else
255 nmea0183.Xte.DirectionToSteer = Right;
256
257 nmea0183.Xte.CrossTrackUnits = _T("N");
258
259 nmea0183.Xte.Write(snt);
260 BroadcastNMEA0183Message(snt.Sentence, routeman.GetNmeaLog(),
261 routeman.GetMessageSentEventVar());
262 }
263
264 return true;
265}
266
267bool UpdateAutopilotN2K(Routeman &routeman) {
268 bool fail_any = false;
269
270 // Get a suitable N2K Driver
271 auto &registry = CommDriverRegistry::GetInstance();
272 const std::vector<DriverPtr> &drivers = registry.GetDrivers();
273
274 AbstractCommDriver *found = nullptr;
275 for (auto key : routeman.GetOutpuDriverArray()) {
276 for (auto &d : drivers) {
277 if (d->Key() == key) {
278 std::unordered_map<std::string, std::string> attributes =
279 GetAttributes(key);
280 auto protocol_it = attributes.find("protocol");
281 if (protocol_it != attributes.end()) {
282 std::string protocol = protocol_it->second;
283
284 if (protocol == "nmea2000") {
285 found = d.get();
286 }
287 }
288 }
289 }
290 }
291 if (!found) return false;
292
293 // N2K serial drivers require maintenance of an enabled PGN TX list
294 auto drv_serial = dynamic_cast<CommDriverN2KSerial *>(found);
295 if (drv_serial) {
296 drv_serial->AddTxPGN(129283);
297 drv_serial->AddTxPGN(129284);
298 drv_serial->AddTxPGN(129285);
299 }
300
301 fail_any |= !SendPGN129285(routeman, found);
302 fail_any |= !SendPGN129284(routeman, found);
303 fail_any |= !SendPGN129283(routeman, found);
304
305 return (fail_any == 0);
306}
307
308bool SendPGN129285(Routeman &routeman, AbstractCommDriver *driver) {
309 bool fail_any = false;
310
311 // Set max WP name length
312 int maxName = 6;
313 if ((g_maxWPNameLength >= 3) && (g_maxWPNameLength <= 32))
314 maxName = g_maxWPNameLength;
315
316 // Create and transmit PGN 129285
317 // The basic PGN129285 head
318 tN2kMsg msg129285;
319 char route_name[] = "Route";
320 SetN2kPGN129285(msg129285, 0, 0, 0, N2kdir_forward, 0, route_name);
321
322 // Append start point of current leg
323 RoutePoint *pLegBeginPoint = routeman.GetpActiveRouteSegmentBeginPoint();
324 wxString start_point_name = pLegBeginPoint->GetName().Truncate(maxName);
325 std::string sname = start_point_name.ToStdString();
326 char *s = (char *)sname.c_str();
327
328 fail_any |= !AppendN2kPGN129285(msg129285, 0, s, pLegBeginPoint->m_lat,
329 pLegBeginPoint->m_lon);
330 // Append destination point of current leg
331 RoutePoint *pActivePoint = routeman.GetpActivePoint();
332 wxString destination_name = pActivePoint->GetName().Truncate(maxName);
333 std::string dname = destination_name.ToStdString();
334 char *d = (char *)dname.c_str();
335 fail_any |= !AppendN2kPGN129285(msg129285, 1, d, pActivePoint->m_lat,
336 pActivePoint->m_lon);
337
338 if (fail_any) return false;
339
340 auto dest_addr = std::make_shared<const NavAddr2000>(driver->iface, 255);
341 std::vector<uint8_t> payload;
342 for (int i = 0; i < msg129285.DataLen; i++)
343 payload.push_back(msg129285.Data[i]);
344 auto PGN129285 =
345 std::make_shared<const Nmea2000Msg>(129285, payload, dest_addr, 6);
346 fail_any |= !driver->SendMessage(PGN129285, dest_addr);
347
348 return (fail_any == 0);
349}
350
351bool SendPGN129284(Routeman &routeman, AbstractCommDriver *driver) {
352 bool fail_any = false;
353 tN2kMsg msg129284;
354 RoutePoint *pActivePoint = routeman.GetpActivePoint();
355
356 // Calculate closing velocity and ETA
357 double vmg = 0.;
358 if (!std::isnan(gCog) && !std::isnan(gSog)) {
359 double brg = routeman.GetCurrentBrgToActivePoint();
360 vmg = gSog * cos((brg - gCog) * PI / 180.);
361 }
362 wxTimeSpan tttg_span;
363 wxDateTime arrival_time = wxDateTime::Now();
364 if (vmg > 0.) {
365 double tttg_sec = (routeman.GetCurrentRngToActivePoint() / gSog) * 3600;
366 tttg_span = wxTimeSpan::Seconds((long)tttg_sec);
367 arrival_time += tttg_span;
368 }
369 double time_days_1979 = arrival_time.GetTicks() / (3600. * 24.);
370
371 // ETA time_seconds, expressed as seconds since midnight
372 // ETA date, expressed as whole days since 1 January 1970
373 double eta_time_days;
374 double eta_time_seconds = modf(time_days_1979, &eta_time_days);
375 int16_t eta_time_days_16 = static_cast<uint16_t>(eta_time_days);
376
377 SetN2kPGN129284(
378 msg129284,
379 0xFF, // SID
380 routeman.GetCurrentRngToActivePoint() * 1852., // DistanceToWaypoint
381 N2khr_true, // tN2kHeadingReference BearingReference
382 false, // PerpendicularCrossed
383 false, // ArrivalCircleEntered
384 N2kdct_RhumbLine, // tN2kDistanceCalculationType CalculationType
385 eta_time_seconds, // double ETATime,
386 eta_time_days_16, // int16_t ETADate,
387 routeman.GetCurrentSegmentCourse() * PI /
388 180., // BearingOriginToDestinationWaypoint,
389 routeman.GetCurrentBrgToActivePoint() * PI /
390 180., // BearingPositionToDestinationWaypoint,
391 0, // uint8_t OriginWaypointNumber,
392 1, // uint8_t DestinationWaypointNumber,
393 pActivePoint->m_lat, // double DestinationLatitude,
394 pActivePoint->m_lon, // double DestinationLongitude,
395 vmg); // double WaypointClosingVelocity);
396
397 auto dest_addr = std::make_shared<const NavAddr2000>(driver->iface, 255);
398 std::vector<uint8_t> payload;
399 for (int i = 0; i < msg129284.DataLen; i++)
400 payload.push_back(msg129284.Data[i]);
401 auto PGN129284 =
402 std::make_shared<const Nmea2000Msg>(129284, payload, dest_addr, 6);
403 fail_any |= !driver->SendMessage(PGN129284, dest_addr);
404
405 return (fail_any == 0);
406}
407
408bool SendPGN129283(Routeman &routeman, AbstractCommDriver *driver) {
409 bool fail_any = false;
410 tN2kMsg msg129283;
411 RoutePoint *pActivePoint = routeman.GetpActivePoint();
412
413 SetN2kPGN129283(msg129283, 0,
414 N2kxtem_Autonomous, // tN2kXTEMode XTEMode,
415 false, // bool NavigationTerminated,
416 routeman.GetCurrentXTEToActivePoint() // double XTE
417 );
418
419 auto dest_addr = std::make_shared<const NavAddr2000>(driver->iface, 255);
420 std::vector<uint8_t> payload;
421 for (int i = 0; i < msg129283.DataLen; i++)
422 payload.push_back(msg129283.Data[i]);
423 auto PGN129283 =
424 std::make_shared<const Nmea2000Msg>(129283, payload, dest_addr, 6);
425 fail_any |= !driver->SendMessage(PGN129283, dest_addr);
426
427 return (fail_any == 0);
428}
Common interface for all drivers.
Definition comm_driver.h:58
const std::string iface
Physical device for 0183, else a unique string.
Definition comm_driver.h:88
Represents a waypoint or mark within the navigation system.
Definition route_point.h:68
Driver registration container, a singleton.
const std::unordered_map< std::string, std::string > GetAttributes(DriverHandle handle)
Query a specific driver for attributes.