1
1
//
2
2
// This file is part of Dire Wolf, an amateur radio packet TNC.
3
3
//
4
- // Copyright (C) 2013, 2014, 2015 John Langner, WB2OSZ
4
+ // Copyright (C) 2013, 2014, 2015, 2020 John Langner, WB2OSZ
5
5
//
6
6
// This program is free software: you can redistribute it and/or modify
7
7
// it under the terms of the GNU General Public License as published by
55
55
56
56
#include <gps.h>
57
57
58
- // Debian bug report: direwolf (1.2-1) FTBFS with libgps22 as part of the gpsd transition (#803605):
59
- // dwgps.c claims to only support GPSD_API_MAJOR_VERSION 5, but also builds successfully with
60
- // GPSD_API_MAJOR_VERSION 6 provided by libgps22 when the attached patch is applied.
61
58
62
- // Also compatible with API 7 & 8 with conditional compilation later.
59
+ // An incompatibility was introduced with version 7
60
+ // and again with 9 and again with 10.
63
61
64
- #if GPSD_API_MAJOR_VERSION < 5 || GPSD_API_MAJOR_VERSION > 8
62
+ #if GPSD_API_MAJOR_VERSION < 5 || GPSD_API_MAJOR_VERSION > 10
65
63
#error libgps API version might be incompatible.
66
64
#endif
67
65
@@ -116,7 +114,7 @@ static void * read_gpsd_thread (void *arg);
116
114
* shared region via dwgps_put_data.
117
115
*
118
116
* The application calls dwgps_read to get the most
119
- 8 recent information.
117
+ * recent information.
120
118
*
121
119
*--------------------------------------------------------------------*/
122
120
@@ -126,7 +124,7 @@ static void * read_gpsd_thread (void *arg);
126
124
* Originally, I wanted to use the shared memory interface to gpsd
127
125
* because it is simpler and more efficient. Just access it when we
128
126
* actually need the data and we don't have a lot of extra unnecessary
129
- * busy work going on.
127
+ * busy work going on constantly polling it when we don't need the information .
130
128
*
131
129
* The current version of gpsd, supplied with Raspian (Wheezy), is 3.6 from back in
132
130
* May 2012, is missing support for the shared memory interface.
@@ -142,13 +140,28 @@ static void * read_gpsd_thread (void *arg);
142
140
* cd gpsd-3.11
143
141
* scons prefix=/usr libdir=lib/arm-linux-gnueabihf shm_export=True python=False
144
142
* sudo scons udev-install
145
- *
143
+ *
146
144
* For now, we will use the socket interface. Maybe get back to this again someday.
147
145
*
148
146
* Update: January 2016.
149
147
*
150
148
* I'm told that the shared memory interface might work in Raspian, Jessie version.
151
149
* Haven't tried it yet.
150
+ *
151
+ * June 2020: This is how to build the most recent.
152
+ *
153
+ * Based on https://www.satsignal.eu/raspberry-pi/UpdatingGPSD.html
154
+ *
155
+ * git clone https://gitlab.com/gpsd/gpsd.git gpsd-gitlab
156
+ * cd gpsd-gitlab
157
+ * scons --config=force
158
+ * scons
159
+ * sudo scons install
160
+ *
161
+ * The problem we have here is that the library is put in /usr/local/lib and direwolf
162
+ * can't find it there. Solution is to define environment variable:
163
+ *
164
+ * export LD_LIBRARY_PATH=/use/local/lib
152
165
*/
153
166
154
167
@@ -232,7 +245,7 @@ int dwgpsd_init (struct misc_config_s *pconfig, int debug)
232
245
*
233
246
*--------------------------------------------------------------------*/
234
247
235
- #define TIMEOUT 30
248
+ #define TIMEOUT 15
236
249
237
250
#if ENABLE_GPSD
238
251
@@ -255,10 +268,19 @@ static void * read_gpsd_thread (void *arg)
255
268
256
269
while (1 ) {
257
270
271
+ // Example code found here:
272
+ // https://lists.nongnu.org/archive/html/gpsd-dev/2017-11/msg00001.html
273
+
258
274
if ( ! gps_waiting (& gpsdata , TIMEOUT * 1000000 )) {
259
275
text_color_set (DW_COLOR_ERROR );
260
- dw_printf ("GPSD: Timeout waiting for GPS data.\n" );
261
- /* Fall thru to read which should get error and bail out. */
276
+ dw_printf ("------------------------------------------\n" );
277
+ dw_printf ("dwgpsd: Timeout waiting for GPS data.\n" );
278
+ dw_printf ("Is GPSD daemon running?\n" );
279
+ dw_printf ("Troubleshooting tip: Try running cgps or xgps.\n" );
280
+ dw_printf ("------------------------------------------\n" );
281
+ info .fix = DWFIX_ERROR ;
282
+ SLEEP_MS (5000 );
283
+ continue ;
262
284
}
263
285
264
286
// https://github.com/wb2osz/direwolf/issues/196
@@ -290,16 +312,47 @@ static void * read_gpsd_thread (void *arg)
290
312
break ; // Jump out of loop and terminate thread.
291
313
}
292
314
315
+ #if GPSD_API_MAJOR_VERSION >= 9
316
+
317
+ // The gps.h revision history says:
318
+ // * mark altitude in gps_fix_t as deprecated and undefined
319
+ // This seems really stupid to me.
320
+ // If it is deprecated and undefined then take it out. Someone trying to use
321
+ // it would get a compile error and know that something needs to be done.
322
+ // Instead we all just go merrily on our way using a field that is [allegedly] undefined.
323
+ // Why not simply add more variables with different definitions of altitude
324
+ // and keep the original variable working as it always did?
325
+ // If it is truly undefined, as the comment would have us believe, numerous
326
+ // people will WASTE VAST AMOUNTS OF TIME pondering why altitude is now broken in
327
+ // their applications.
328
+
329
+ #define stupid_altitude altMSL
330
+ #else
331
+ #define stupid_altitude altitude
332
+ #endif
333
+
334
+ #if GPSD_API_MAJOR_VERSION >= 10
335
+
336
+ // They did it again. Whimsical incompatibilities that cause
337
+ // pain and aggravation for everyone trying to use this library.
338
+ //
339
+ // error: ‘struct gps_data_t’ has no member named ‘status’
340
+ //
341
+ // Yes, I can understand that it is a more logical place but it breaks
342
+ // all existing code that uses this.
343
+ // I'm really getting annoyed about wasting so much time on keeping up with all
344
+ // of these incompatibilities that are completely unnecessary.
345
+
346
+ #define stupid_status fix.status
347
+ #else
348
+ #define stupid_status status
349
+ #endif
350
+
351
+ // Inform user about change in fix status.
352
+
293
353
switch (gpsdata .fix .mode ) {
294
354
default :
295
355
case MODE_NOT_SEEN :
296
- if (info .fix >= DWFIX_2D ) {
297
- text_color_set (DW_COLOR_INFO );
298
- dw_printf ("GPSD: Lost location fix.\n" );
299
- }
300
- info .fix = DWFIX_NOT_SEEN ;
301
- break ;
302
-
303
356
case MODE_NO_FIX :
304
357
if (info .fix >= DWFIX_2D ) {
305
358
text_color_set (DW_COLOR_INFO );
@@ -325,21 +378,23 @@ static void * read_gpsd_thread (void *arg)
325
378
break ;
326
379
}
327
380
328
- /* Data is available. */
329
- // TODO: what is gpsdata.status?
330
-
331
-
332
- if (gpsdata .status >= STATUS_FIX && gpsdata .fix .mode >= MODE_2D ) {
381
+ if (gpsdata .stupid_status >= STATUS_FIX && gpsdata .fix .mode >= MODE_2D ) {
333
382
334
383
info .dlat = isnan (gpsdata .fix .latitude ) ? G_UNKNOWN : gpsdata .fix .latitude ;
335
384
info .dlon = isnan (gpsdata .fix .longitude ) ? G_UNKNOWN : gpsdata .fix .longitude ;
336
385
info .track = isnan (gpsdata .fix .track ) ? G_UNKNOWN : gpsdata .fix .track ;
337
386
info .speed_knots = isnan (gpsdata .fix .speed ) ? G_UNKNOWN : (MPS_TO_KNOTS * gpsdata .fix .speed );
338
387
339
388
if (gpsdata .fix .mode >= MODE_3D ) {
340
- info .altitude = isnan (gpsdata .fix .altitude ) ? G_UNKNOWN : gpsdata .fix .altitude ;
389
+ info .altitude = isnan (gpsdata .fix .stupid_altitude ) ? G_UNKNOWN : gpsdata .fix .stupid_altitude ;
341
390
}
342
391
}
392
+ else {
393
+ // Keep the last known location.
394
+ // Using info.fix, the caller knows if the location is current (DWFIX_[23]D),
395
+ // last known (DWFIX_NONE), or never known (DWFIX_NOT_SEEN).
396
+ info .fix = DWFIX_NO_FIX ;
397
+ }
343
398
344
399
info .timestamp = time (NULL );
345
400
if (s_debug >= 2 ) {
@@ -373,6 +428,7 @@ void dwgpsd_term (void) {
373
428
374
429
#if ENABLE_GPSD
375
430
431
+ gps_stream (& gpsdata , WATCH_DISABLE , NULL );
376
432
gps_close (& gpsdata );
377
433
378
434
#endif
0 commit comments