Coverage Report

Created: 2026-03-03 06:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/gpsd/gpsd-3.27.6~dev/drivers/driver_italk.c
Line
Count
Source
1
/*
2
 * Driver for the iTalk binary protocol used by FasTrax
3
 *
4
 * Week counters are not limited to 10 bits. It's unknown what
5
 * the firmware is doing to disambiguate them, if anything; it might just
6
 * be adding a fixed offset based on a hidden epoch value, in which case
7
 * unhappy things will occur on the next rollover.
8
 *
9
 * This file is Copyright 2010 by the GPSD project
10
 * SPDX-License-Identifier: BSD-2-clause
11
 *
12
 */
13
14
#include "../include/gpsd_config.h"  // must be before all includes
15
16
#include <math.h>
17
#include <stdbool.h>
18
#include <stdio.h>
19
#include <string.h>
20
#include <unistd.h>
21
22
#include "../include/gpsd.h"
23
#if defined(ITRAX_ENABLE)
24
25
#include "../include/bits.h"
26
#include "../include/driver_italk.h"
27
#include "../include/timespec.h"
28
29
static gps_mask_t italk_parse(struct gps_device_t *, unsigned char *, size_t);
30
static gps_mask_t decode_itk_navfix(struct gps_device_t *, unsigned char *,
31
                                    size_t);
32
static gps_mask_t decode_itk_prnstatus(struct gps_device_t *, unsigned char *,
33
                                       size_t);
34
static gps_mask_t decode_itk_utcionomodel(struct gps_device_t *,
35
                                          unsigned char *, size_t);
36
static gps_mask_t decode_itk_subframe(struct gps_device_t *, unsigned char *,
37
                                      size_t);
38
39
// NAVIGATION_MSG, message id 7
40
static gps_mask_t decode_itk_navfix(struct gps_device_t *session,
41
                                    unsigned char *buf, size_t len)
42
0
{
43
0
    unsigned short flags, pflags;
44
0
    timespec_t ts_tow;
45
0
    uint32_t tow;            // Time of week [ms]
46
0
    char ts_buf[TIMESPEC_LEN];
47
48
0
    gps_mask_t mask = 0;
49
0
    if (296 != len) {
50
0
        GPSD_LOG(LOG_PROG, &session->context->errout,
51
0
                 "ITALK: bad NAV_FIX (len %zu, should be 296)\n",
52
0
                 len);
53
0
        return -1;
54
0
    }
55
56
0
    flags = (unsigned short) getleu16(buf, 7 + 4);
57
    //cflags = (unsigned short) getleu16(buf, 7 + 6);
58
0
    pflags = (unsigned short) getleu16(buf, 7 + 8);
59
60
0
    session->newdata.status = STATUS_UNK;
61
0
    session->newdata.mode = MODE_NO_FIX;
62
0
    mask = ONLINE_SET | MODE_SET | STATUS_SET | CLEAR_IS;
63
64
    // just bail out if this fix is not marked valid
65
0
    if (0 != (pflags & FIX_FLAG_MASK_INVALID)
66
0
        || 0 == (flags & FIXINFO_FLAG_VALID)) {
67
0
        return mask;
68
0
    }
69
70
0
    tow = getleu32(buf, 7 + 84);   // tow in ms
71
0
    MSTOTS(&ts_tow, tow);
72
0
    session->newdata.time = gpsd_gpstime_resolv(session,
73
0
        (unsigned short) getles16(buf, 7 + 82), ts_tow);
74
0
    mask |= TIME_SET | NTPTIME_IS;
75
76
0
    session->newdata.ecef.x = (double)(getles32(buf, 7 + 96) / 100.0);
77
0
    session->newdata.ecef.y = (double)(getles32(buf, 7 + 100) / 100.0);
78
0
    session->newdata.ecef.z = (double)(getles32(buf, 7 + 104) / 100.0);
79
0
    session->newdata.ecef.vx = (double)(getles32(buf, 7 + 186) / 1000.0);
80
0
    session->newdata.ecef.vy = (double)(getles32(buf, 7 + 190) / 1000.0);
81
0
    session->newdata.ecef.vz = (double)(getles32(buf, 7 + 194) / 1000.0);
82
0
    mask |= ECEF_SET | VECEF_SET;
83
    /* this eph does not look right, badly documented.
84
     * let gpsd_error_model() handle it
85
     * session->newdata.eph = (double)(getles32(buf, 7 + 252) / 100.0);
86
     */
87
0
    session->newdata.eps = (double)(getles32(buf, 7 + 254) / 100.0);
88
    // compute epx/epy in gpsd_error_model(), not here
89
0
    mask |= HERR_SET;
90
91
0
#define MAX(a,b) (((a) > (b)) ? (a) : (b))
92
0
    session->gpsdata.satellites_used =
93
0
        (int)MAX(getleu16(buf, 7 + 12), getleu16(buf, 7 + 14));
94
0
    mask |= USED_IS;
95
96
0
    if (flags & FIX_CONV_DOP_VALID) {
97
0
        session->gpsdata.dop.hdop = (double)(getleu16(buf, 7 + 56) / 100.0);
98
0
        session->gpsdata.dop.gdop = (double)(getleu16(buf, 7 + 58) / 100.0);
99
0
        session->gpsdata.dop.pdop = (double)(getleu16(buf, 7 + 60) / 100.0);
100
0
        session->gpsdata.dop.vdop = (double)(getleu16(buf, 7 + 62) / 100.0);
101
0
        session->gpsdata.dop.tdop = (double)(getleu16(buf, 7 + 64) / 100.0);
102
0
        mask |= DOP_SET;
103
0
    }
104
105
0
    if (0 == (pflags & FIX_FLAG_MASK_INVALID) &&
106
0
        0 != (flags & FIXINFO_FLAG_VALID)) {
107
0
        if (pflags & FIX_FLAG_3DFIX) {
108
0
            session->newdata.mode = MODE_3D;
109
0
        } else {
110
0
            session->newdata.mode = MODE_2D;
111
0
        }
112
113
0
        if (pflags & FIX_FLAG_DGPS_CORRECTION) {
114
0
            session->newdata.status = STATUS_DGPS;
115
0
        } else {
116
0
            session->newdata.status = STATUS_GPS;
117
0
        }
118
0
    }
119
120
0
    GPSD_LOG(LOG_DATA, &session->context->errout,
121
0
             "NAV_FIX: time=%s, ecef x:%.2f y:%.2f z:%.2f altHAE=%.2f "
122
0
             "speed=%.2f track=%.2f climb=%.2f mode=%d status=%d gdop=%.2f "
123
0
             "pdop=%.2f hdop=%.2f vdop=%.2f tdop=%.2f\n",
124
0
             timespec_str(&session->newdata.time, ts_buf, sizeof(ts_buf)),
125
0
             session->newdata.ecef.x,
126
0
             session->newdata.ecef.y, session->newdata.ecef.z,
127
0
             session->newdata.altHAE, session->newdata.speed,
128
0
             session->newdata.track, session->newdata.climb,
129
0
             session->newdata.mode, session->newdata.status,
130
0
             session->gpsdata.dop.gdop, session->gpsdata.dop.pdop,
131
0
             session->gpsdata.dop.hdop, session->gpsdata.dop.vdop,
132
0
             session->gpsdata.dop.tdop);
133
0
    return mask;
134
0
}
135
136
static gps_mask_t decode_itk_prnstatus(struct gps_device_t *session,
137
                                       unsigned char *buf, size_t len)
138
0
{
139
0
    gps_mask_t mask = 0;
140
0
    unsigned int i, nsv, nchan, st;
141
0
    uint32_t msec;
142
0
    timespec_t ts_tow;
143
0
    char ts_buf[TIMESPEC_LEN];
144
145
0
    if (62 > len) {
146
0
        GPSD_LOG(LOG_PROG, &session->context->errout,
147
0
                 "ITALK: runt PRN_STATUS (len=%zu)\n", len);
148
0
        return mask;
149
0
    }
150
151
0
    msec = getleu32(buf, 7 + 6);
152
153
0
    MSTOTS(&ts_tow, msec);
154
155
0
    session->gpsdata.skyview_time = gpsd_gpstime_resolv(session,
156
0
        (unsigned short)getleu16(buf, 7 + 4), ts_tow);
157
0
    gpsd_zero_satellites(&session->gpsdata);
158
0
    nchan = (unsigned int)getleu16(buf, 7 + 50);
159
0
    if (nchan > MAX_NR_VISIBLE_PRNS) {
160
0
        nchan = MAX_NR_VISIBLE_PRNS;
161
0
    }
162
0
    for (i = st = nsv = 0; i < nchan; i++) {
163
0
        unsigned int off = 7 + 52 + 10 * i;
164
0
        unsigned short flags;
165
0
        bool used;
166
167
0
        flags = (unsigned short) getleu16(buf, off);
168
0
        used = (bool)(flags & PRN_FLAG_USE_IN_NAV);
169
0
        session->gpsdata.skyview[st].PRN =
170
0
            (short)(getleu16(buf, off + 4) & 0xff);
171
0
        session->gpsdata.skyview[st].elevation =
172
0
            (double)(getles16(buf, off + 6) & 0xff);
173
0
        session->gpsdata.skyview[st].azimuth =
174
0
            (double)(getles16(buf, off + 8) & 0xff);
175
0
        session->gpsdata.skyview[st].ss =
176
0
            (double)(getleu16(buf, off + 2) & 0xff);
177
0
        session->gpsdata.skyview[st].used = used;
178
0
        if (session->gpsdata.skyview[st].PRN > 0) {
179
0
            st++;
180
0
            if (used) {
181
0
                nsv++;
182
0
            }
183
0
        }
184
0
    }
185
0
    session->gpsdata.satellites_visible = (int)st;
186
0
    if (MAXCHANNELS < session->gpsdata.satellites_visible) {
187
0
        GPSD_LOG(LOG_WARN, &session->context->errout,
188
0
                "PRN_STTUS: too many satellites %d\n",
189
0
                 session->gpsdata.satellites_visible);
190
0
        session->gpsdata.satellites_visible = MAXCHANNELS;
191
0
    }
192
0
    session->gpsdata.satellites_used = (int)nsv;
193
0
    mask = USED_IS | SATELLITE_SET;
194
195
0
    GPSD_LOG(LOG_DATA, &session->context->errout,
196
0
             "PRN_STATUS: time=%s visible=%d used=%d "
197
0
             "mask={USED|SATELLITE}\n",
198
0
             timespec_str(&session->newdata.time, ts_buf, sizeof(ts_buf)),
199
0
             session->gpsdata.satellites_visible,
200
0
             session->gpsdata.satellites_used);
201
202
0
    return mask;
203
0
}
204
205
static gps_mask_t decode_itk_utcionomodel(struct gps_device_t *session,
206
                                          unsigned char *buf, size_t len)
207
0
{
208
0
    int leap;
209
0
    unsigned short flags;
210
0
    timespec_t ts_tow;
211
0
    uint32_t tow;            // Time of week [ms]
212
0
    char ts_buf[TIMESPEC_LEN];
213
214
0
    if (64 != len) {
215
0
        GPSD_LOG(LOG_PROG, &session->context->errout,
216
0
                 "ITALK: bad UTC_IONO_MODEL (len %zu, should be 64)\n",
217
0
                 len);
218
0
        return 0;
219
0
    }
220
221
0
    flags = (unsigned short) getleu16(buf, 7);
222
0
    if (0 == (flags & UTC_IONO_MODEL_UTCVALID)) {
223
0
        return 0;
224
0
    }
225
226
0
    leap = (int)getleu16(buf, 7 + 24);
227
0
    if (session->context->leap_seconds < leap) {
228
0
        session->context->leap_seconds = leap;
229
0
    }
230
231
0
    tow = getleu32(buf, 7 + 38);    // in ms
232
0
    MSTOTS(&ts_tow, tow);
233
0
    session->newdata.time = gpsd_gpstime_resolv(session,
234
0
        (unsigned short) getleu16(buf, 7 + 36), ts_tow);
235
0
    GPSD_LOG(LOG_DATA, &session->context->errout,
236
0
             "UTC_IONO_MODEL: time=%s mask={TIME}\n",
237
0
             timespec_str(&session->newdata.time, ts_buf, sizeof(ts_buf)));
238
0
    return TIME_SET | NTPTIME_IS;
239
0
}
240
241
static gps_mask_t decode_itk_subframe(struct gps_device_t *session,
242
                                      unsigned char *buf, size_t len)
243
0
{
244
0
    unsigned short flags, prn, sf;
245
0
    unsigned int i;
246
0
    uint32_t words[10];
247
248
0
    if (64 != len) {
249
0
        GPSD_LOG(LOG_PROG, &session->context->errout,
250
0
                 "ITALK: bad SUBFRAME (len %zu, should be 64)\n", len);
251
0
        return 0;
252
0
    }
253
254
0
    flags = (unsigned short) getleu16(buf, 7 + 4);
255
0
    prn = (unsigned short) getleu16(buf, 7 + 6);
256
0
    sf = (unsigned short) getleu16(buf, 7 + 8);
257
0
    GPSD_LOG(LOG_PROG, &session->context->errout,
258
0
             "iTalk 50B SUBFRAME prn %u sf %u - decode %s %s\n",
259
0
             prn, sf,
260
0
             (flags & SUBFRAME_WORD_FLAG_MASK) ? "error" : "ok",
261
0
             (flags & SUBFRAME_GPS_PREAMBLE_INVERTED) ? "(inverted)" : "");
262
0
    if (flags & SUBFRAME_WORD_FLAG_MASK) {
263
0
        return 0;       // don't try decode an erroneous packet
264
0
    }
265
266
    /*
267
     * Timo says "SUBRAME message contains decoded navigation message subframe
268
     * words with parity checking done but parity bits still present."
269
     */
270
0
    for (i = 0; i < 10; i++) {
271
0
        words[i] = (uint32_t)(getleu32(buf, 7 + 14 + 4 * i) >> 6) & 0xffffff;
272
0
    }
273
274
0
    return gpsd_interpret_subframe(session, GNSSID_GPS, prn, words);
275
0
}
276
277
static gps_mask_t decode_itk_pseudo(struct gps_device_t *session,
278
                                      unsigned char *buf, size_t len)
279
0
{
280
0
    unsigned short flags, n, i;
281
0
    unsigned int tow;             // time of week, in ms
282
0
    timespec_t ts_tow;
283
284
0
    n = (unsigned short) getleu16(buf, 7 + 4);
285
0
    if (1 > n ||
286
0
        MAXCHANNELS < n ) {
287
0
        GPSD_LOG(LOG_INF, &session->context->errout,
288
0
                 "ITALK: bad PSEUDO channel count\n");
289
0
        return 0;
290
0
    }
291
292
0
    if (len != (size_t)((n+1)*36)) {
293
0
        GPSD_LOG(LOG_PROG, &session->context->errout,
294
0
                 "ITALK: bad PSEUDO len %zu\n", len);
295
0
    }
296
297
0
    GPSD_LOG(LOG_PROG, &session->context->errout, "iTalk PSEUDO [%u]\n", n);
298
0
    flags = (unsigned short)getleu16(buf, 7 + 6);
299
0
    if ((flags & 0x3) != 0x3) {
300
0
        return 0; // bail if measurement time not valid.
301
0
    }
302
303
0
    tow = (unsigned int)getleu32(buf, 7 + 38);
304
0
    MSTOTS(&ts_tow, tow);
305
0
    session->newdata.time = gpsd_gpstime_resolv(session,
306
0
        (unsigned short int)getleu16((char *)buf, 7 + 8), ts_tow);
307
308
0
    session->gpsdata.raw.mtime = session->newdata.time;
309
310
    // this is so we can tell which never got set
311
0
    for (i = 0; i < MAXCHANNELS; i++) {
312
0
        session->gpsdata.raw.meas[i].svid = 0;
313
0
    }
314
0
    for (i = 0; i < n; i++){
315
0
        session->gpsdata.skyview[i].PRN =
316
0
            getleu16(buf, 7 + 26 + (i*36)) & 0xff;
317
0
        session->gpsdata.skyview[i].ss =
318
0
            getleu16(buf, 7 + 26 + (i*36 + 2)) & 0x3f;
319
0
        session->gpsdata.raw.meas[i].satstat =
320
0
            getleu32(buf, 7 + 26 + (i*36 + 4));
321
0
        session->gpsdata.raw.meas[i].pseudorange =
322
0
            getled64((char *)buf, 7 + 26 + (i*36 + 8));
323
0
        session->gpsdata.raw.meas[i].doppler =
324
0
            getled64((char *)buf, 7 + 26 + (i*36 + 16));
325
0
        session->gpsdata.raw.meas[i].carrierphase =
326
0
            getleu16(buf, 7 + 26 + (i*36 + 28));
327
328
0
        session->gpsdata.raw.meas[i].codephase = NAN;
329
0
        session->gpsdata.raw.meas[i].deltarange = NAN;
330
0
    }
331
    // return RAW_IS; The above decode does not give reasonable results
332
0
    return 0;         // do not report valid until decode is fixed
333
0
}
334
335
static gps_mask_t italk_parse(struct gps_device_t *session,
336
                              unsigned char *buf, size_t len)
337
0
{
338
0
    unsigned int type;
339
0
    gps_mask_t mask = 0;
340
341
0
    if (0 == len) {
342
0
        return 0;
343
0
    }
344
345
0
    type = (unsigned int) getub(buf, 4);
346
    // we may need to dump the raw packet
347
0
    GPSD_LOG(LOG_RAW, &session->context->errout,
348
0
             "raw italk packet type 0x%02x\n", type);
349
350
0
    session->cycle_end_reliable = true;
351
352
0
    switch (type) {
353
0
    case ITALK_NAV_FIX:
354
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
355
0
                 "iTalk NAV_FIX len %zu\n", len);
356
0
        mask = decode_itk_navfix(session, buf, len) | (CLEAR_IS | REPORT_IS);
357
0
        break;
358
0
    case ITALK_PRN_STATUS:
359
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
360
0
                 "iTalk PRN_STATUS len %zu\n", len);
361
0
        mask = decode_itk_prnstatus(session, buf, len);
362
0
        break;
363
0
    case ITALK_UTC_IONO_MODEL:
364
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
365
0
                 "iTalk UTC_IONO_MODEL len %zu\n", len);
366
0
        mask = decode_itk_utcionomodel(session, buf, len);
367
0
        break;
368
369
0
    case ITALK_ACQ_DATA:
370
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
371
0
                 "iTalk ACQ_DATA len %zu\n", len);
372
0
        break;
373
0
    case ITALK_TRACK:
374
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
375
0
                 "iTalk TRACK len %zu\n", len);
376
0
        break;
377
0
    case ITALK_PSEUDO:
378
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
379
0
                 "iTalk PSEUDO len %zu\n", len);
380
0
        mask = decode_itk_pseudo(session, buf, len);
381
0
        break;
382
0
    case ITALK_RAW_ALMANAC:
383
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
384
0
                 "iTalk RAW_ALMANAC len %zu\n", len);
385
0
        break;
386
0
    case ITALK_RAW_EPHEMERIS:
387
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
388
0
                 "iTalk RAW_EPHEMERIS len %zu\n", len);
389
0
        break;
390
0
    case ITALK_SUBFRAME:
391
0
        mask = decode_itk_subframe(session, buf, len);
392
0
        break;
393
0
    case ITALK_BIT_STREAM:
394
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
395
0
                 "iTalk BIT_STREAM len %zu\n", len);
396
0
        break;
397
398
0
    case ITALK_AGC:
399
0
    case ITALK_SV_HEALTH:
400
0
    case ITALK_PRN_PRED:
401
0
    case ITALK_FREQ_PRED:
402
0
    case ITALK_DBGTRACE:
403
0
    case ITALK_START:
404
0
    case ITALK_STOP:
405
0
    case ITALK_SLEEP:
406
0
    case ITALK_STATUS:
407
0
    case ITALK_ITALK_CONF:
408
0
    case ITALK_SYSINFO:
409
0
    case ITALK_ITALK_TASK_ROUTE:
410
0
    case ITALK_PARAM_CTRL:
411
0
    case ITALK_PARAMS_CHANGED:
412
0
    case ITALK_START_COMPLETED:
413
0
    case ITALK_STOP_COMPLETED:
414
0
    case ITALK_LOG_CMD:
415
0
    case ITALK_SYSTEM_START:
416
0
    case ITALK_STOP_SEARCH:
417
0
    case ITALK_SEARCH:
418
0
    case ITALK_PRED_SEARCH:
419
0
    case ITALK_SEARCH_DONE:
420
0
    case ITALK_TRACK_DROP:
421
0
    case ITALK_TRACK_STATUS:
422
0
    case ITALK_HANDOVER_DATA:
423
0
    case ITALK_CORE_SYNC:
424
0
    case ITALK_WAAS_RAWDATA:
425
0
    case ITALK_ASSISTANCE:
426
0
    case ITALK_PULL_FIX:
427
0
    case ITALK_MEMCTRL:
428
0
    case ITALK_STOP_TASK:
429
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
430
0
                 "iTalk not processing packet: id 0x%02x length %zu\n",
431
0
                 type, len);
432
0
        break;
433
0
    default:
434
0
        GPSD_LOG(LOG_DATA, &session->context->errout,
435
0
                 "iTalk unknown packet: id 0x%02x length %zu\n",
436
0
                 type, len);
437
0
    }
438
439
0
    return mask | ONLINE_SET;
440
0
}
441
442
443
static gps_mask_t italk_parse_input(struct gps_device_t *session)
444
0
{
445
0
    if (ITALK_PACKET == session->lexer.type) {
446
0
        return italk_parse(session, session->lexer.outbuffer,
447
0
                           session->lexer.outbuflen);
448
0
    }
449
0
    if (NMEA_PACKET == session->lexer.type) {
450
0
        return nmea_parse((char *)session->lexer.outbuffer, session);
451
0
    }
452
0
    return 0;
453
0
}
454
455
#ifdef __UNUSED__
456
// send a "ping". it may help us detect an itrax more quickly
457
static void italk_ping(struct gps_device_t *session)
458
{
459
    char *ping = "<?>";
460
    (void)gpsd_write(session, ping, 3);
461
}
462
#endif  // __UNUSED__
463
464
// *INDENT-OFF*
465
const struct gps_type_t driver_italk =
466
{
467
    .type_name      = "iTalk",          // full name of type
468
    .packet_type    = ITALK_PACKET,     // associated lexer packet type
469
    .flags          = DRIVER_STICKY,    // no rollover or other flags
470
    .trigger        = NULL,             // recognize the type
471
    .channels       = 12,               // consumer-grade GPS
472
    .probe_detect   = NULL,             // how to detect at startup time
473
    .get_packet     = packet_get1,      // use generic packet grabber
474
    .parse_packet   = italk_parse_input,// parse message packets
475
    .rtcm_writer    = gpsd_write,       // send RTCM data straight
476
    .init_query     = NULL,             // non-perturbing initial query
477
    .event_hook     = NULL,             // lifetime event handler
478
    .speed_switcher = NULL,             // no speed switcher
479
    .mode_switcher  = NULL,             // no mode switcher
480
    .rate_switcher  = NULL,             // no sample-rate switcher
481
    .min_cycle.tv_sec  = 1,             // not relevant, no rate switch
482
    .min_cycle.tv_nsec = 0,             // not relevant, no rate switch
483
    .control_send   = NULL,             // no control string sender
484
    .time_offset     = NULL,            // no method for NTP fudge factor
485
};
486
// *INDENT-ON*
487
#endif  // defined(ITRAX_ENABLE)
488
// vim: set expandtab shiftwidth=4