Coverage Report

Created: 2026-06-04 06:21

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