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_rtcm3.c
Line
Count
Source
1
/*****************************************************************************
2
3
This is a decoder for RTCM-104 3.x, a serial protocol used for
4
broadcasting pseudorange corrections from differential-GPS reference
5
stations.  The applicable specification is RTCM 10403.1: RTCM Paper
6
177-2006-SC104-STD.  This obsolesces the earlier RTCM-104 2.x
7
specifications. The specification document is proprietary; ordering
8
instructions are accessible from <http://www.rtcm.org/>
9
under "Publications".
10
11
Unike the RTCM 2.x protocol, RTCM3.x does not use the strange
12
sliding-bit-window IS-GPS-200 protocol as a transport layer, but is a
13
self-contained byte-oriented packet protocol.  Packet recognition is
14
handled in the GPSD packet-getter state machine; this code is
15
concerned with unpacking the packets into well-behaved C structures,
16
coping with odd field lengths and fields that may overlap byte
17
boundaries.  These report structures live in gps.h.
18
19
Note that the unpacking this module does is probably useful only for
20
RTCM reporting and diagnostic tools.  It is not necessary when
21
passing RTCM corrections to a GPS, which normally should just be
22
passed an entire correction packet for processing by their internal
23
firmware.
24
25
Decodes of the following types have been verified: 1004, 1005, 1006,
26
1008, 1012, 1013, 1029. There is good reason to believe the 1007 code
27
is correct, as it's identical to 1008 up to where it ends.
28
29
The 1033 decode was arrived at by looking at an rtcminspect dump and noting
30
that it carries an information superset of the 1008.  There are additional
31
Receiver and Firmware fields we're not certain to decode without access
32
to an RTCM3 standard at revision 4 or later, but the guess in the code
33
has been observed to correctly analyze a message with a nonempty Receiver
34
field.
35
36
This file is Copyright 2010 by the GPSD project
37
SPDX-License-Identifier: BSD-2-clause
38
39
*****************************************************************************/
40
41
#include "../include/gpsd_config.h"  // must be before all includes
42
43
#include <string.h>
44
45
#include "../include/gpsd.h"
46
#include "../include/bits.h"
47
48
// scaling constants for RTCM3 real number types
49
0
#define GPS_PSEUDORANGE_RESOLUTION      0.02    // DF011
50
0
#define PSEUDORANGE_DIFF_RESOLUTION     0.0005  // DF012, DF042
51
0
#define CARRIER_NOISE_RATIO_UNITS       0.25    // DF015, DF045, DF050
52
0
#define ANTENNA_POSITION_RESOLUTION     0.0001  // DF025, DF026, DF027
53
0
#define GLONASS_PSEUDORANGE_RESOLUTION  0.02    // DF041
54
0
#define ANTENNA_DEGREE_RESOLUTION       25e-6   // DF062
55
#define GPS_EPOCH_TIME_RESOLUTION       0.1     // DF065
56
// DF069, DF070, DF192, DF193, DF194, DF195
57
0
#define PHASE_CORRECTION_RESOLUTION     0.5
58
// DF156, DF157, DF158, DF166, DF167, DF168, DF169, DF196, DF197
59
0
#define TRANSLATION_MM_RESOLUTION       0.001
60
0
#define VALIDITY_RESOLUTION             2.0     // DF152, DF153, DF154, DF155
61
0
#define SCALE_PPM_RESOLUTION            1e-5    // DF162
62
0
#define ROTATION_ARCSEC_RESOLUTION      2e-5    // DF159, DF160, DF161
63
// DF171, DF172, DF176, DF177, DF178, DF179, DF183, DF184, DF185, DF186
64
0
#define PROJ_ORIGIN_RESOLUTION          11e-9
65
0
#define DEG_ARCSEC_RESOLUTION           3600
66
0
#define CM_RESOLUTION                   0.01    // DF198
67
0
#define RES_ARCSEC_RESOLUTION           3e-5    // DF199, DF200
68
69
// Other magic values
70
0
#define GPS_INVALID_PSEUDORANGE         0x80000 // DF012, DF018
71
0
#define GLONASS_INVALID_RANGEINCR       0x2000  // DF047
72
0
#define GLONASS_CHANNEL_BASE            7       // DF040
73
74
// Large case statements make GNU indent very confused
75
// *INDENT-OFF*
76
77
/* good source on message types:
78
 * https://software.rtcm-ntrip.org/export/HEAD/ntrip/trunk/BNC/src/bnchelp.html
79
 * Also look in the BNC source
80
 * and look at the tklib source: http://www.rtklib.com/
81
 */
82
83
0
#define ugrab(width)    (bitcount += width, ubits(buf, \
84
0
                         bitcount - width, width, false))
85
0
#define sgrab(width)    (bitcount += width, sbits(buf,  \
86
0
                         bitcount - width, width, false))
87
88
/* copy strings safely.
89
 * RTCM 3.3 says most strings are 31 chars, or less, of ISO 8859-1
90
 * but 1007 says DF030 is 20 chars, or less, of ASCII
91
 * and 1029 DF140 is 255 bytes, or less, of UTF-8
92
 */
93
static void rtcm3_copy_string_field(char *dst, size_t dstlen,
94
                                    const unsigned char *src, size_t srclen)
95
0
{
96
0
    if (0 == dstlen) {
97
0
        return;
98
0
    }
99
0
    if (srclen >= dstlen) {
100
0
        srclen = dstlen - 1;
101
0
    }
102
0
    (void)memcpy(dst, src, srclen);
103
0
    dst[srclen] = '\0';
104
0
}
105
106
/* decode 1015/1016/1017 header
107
 * they share a common header
108
 * TODO: rtklib has C code for these.
109
 *
110
 * Return: false if decoded
111
 *         true if runt
112
 */
113
static bool rtcm3_101567(const struct gps_context_t *context,
114
                         struct rtcm3_t *rtcm, const unsigned char *buf)
115
0
{
116
0
    int bitcount = 36;  // 8 preamble, 6 zero, 10 length, 12 type
117
118
    // 1015, 1016, and 1017 all use the 1015 struct
119
0
    rtcm->rtcmtypes.rtcm3_1015.header.network_id = ugrab(12);
120
0
    rtcm->rtcmtypes.rtcm3_1015.header.subnetwork_id = ugrab(4);
121
0
    rtcm->rtcmtypes.rtcm3_1015.header.tow = ugrab(23);
122
0
    rtcm->rtcmtypes.rtcm3_1015.header.multimesg = (bool)ugrab(1);
123
0
    rtcm->rtcmtypes.rtcm3_1015.header.master_id = ugrab(12);
124
0
    rtcm->rtcmtypes.rtcm3_1015.header.aux_id = ugrab(12);
125
0
    rtcm->rtcmtypes.rtcm3_1015.header.satcount = ugrab(4);
126
127
0
    GPSD_LOG(LOG_PROG, &context->errout, "RTCM3: rtcm3_10567(%u) "
128
0
             "network_id %u subnetwork_id %u tow %lu multimesg %u "
129
0
             "master_id %u aux_id %u satcount %u",
130
0
             rtcm->type,
131
0
             rtcm->rtcmtypes.rtcm3_1015.header.network_id,
132
0
             rtcm->rtcmtypes.rtcm3_1015.header.subnetwork_id,
133
0
             rtcm->rtcmtypes.rtcm3_1015.header.tow,
134
0
             rtcm->rtcmtypes.rtcm3_1015.header.multimesg,
135
0
             rtcm->rtcmtypes.rtcm3_1015.header.master_id,
136
0
             rtcm->rtcmtypes.rtcm3_1015.header.aux_id,
137
0
             rtcm->rtcmtypes.rtcm3_1015.header.satcount);
138
0
    return false;
139
0
}
140
141
/* decode 4076 header
142
 * IGS State Space Representation (SSR) Format
143
 * www.igs.org
144
 * https://files.igs.org/pub/data/format/igs_ssr_v1.pdf
145
 *
146
 * Return: false if decoded
147
 *         true if runt or undecoded
148
 */
149
static bool rtcm3_4076(const struct gps_context_t *context,
150
                         struct rtcm3_t *rtcm, const unsigned char *buf)
151
0
{
152
0
    int bitcount = 36;         // 8 preamble, 6 zero, 10 length, 12 type
153
154
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_vers = ugrab(3);
155
0
    rtcm->rtcmtypes.rtcm3_4076.igs_num = ugrab(8);
156
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_epoch = ugrab(20);
157
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_update = ugrab(4);
158
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_mmi = ugrab(1);
159
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_iod = ugrab(4);
160
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_provider = ugrab(16);
161
0
    rtcm->rtcmtypes.rtcm3_4076.ssr_solution = ugrab(4);
162
163
0
    GPSD_LOG(LOG_PROG, &context->errout,
164
0
             "RTCM3: rtcm3_4076 ver %u igs_num %u Epoch %u update %u mmi %u "
165
0
             "IOD %u Provider %u Solution %u\n",
166
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_vers,
167
0
             rtcm->rtcmtypes.rtcm3_4076.igs_num,
168
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_epoch,
169
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_update,
170
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_mmi,
171
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_iod,
172
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_provider,
173
0
             rtcm->rtcmtypes.rtcm3_4076.ssr_solution);
174
0
    return true;
175
0
}
176
177
/* decode MSM header
178
 * MSM1 to MSM7 share a common header
179
 * TODO: rtklib has C code for these.
180
 *
181
 * Return: false if decoded
182
 *         true if runt, or error
183
 */
184
static bool rtcm3_decode_msm(const struct gps_context_t *context,
185
                             struct rtcm3_t *rtcm, const unsigned char *buf)
186
0
{
187
0
    int bitcount = 36;  // 8 preamble, 6 zero, 10 length, 12 type
188
0
    unsigned n_sig = 0, n_sat = 0, n_cell = 0;
189
0
    uint64_t sat_mask;
190
0
    uint32_t sig_mask;
191
0
    unsigned i;
192
193
0
    if (22 > rtcm->length) {
194
        // need 169 bits, 21.125 bytes
195
0
        rtcm->length = 0;          // set to zero to prevent JSON decode
196
0
        GPSD_LOG(LOG_WARN, &context->errout,
197
0
                 "RTCM3: rtcm3_decode_msm() type %d runt length %u ",
198
0
                 rtcm->type, rtcm->length);
199
0
        return true;
200
0
    }
201
202
0
    rtcm->rtcmtypes.rtcm3_msm.station_id = ugrab(12);
203
0
    rtcm->rtcmtypes.rtcm3_msm.tow = ugrab(30);
204
0
    rtcm->rtcmtypes.rtcm3_msm.sync = ugrab(1);
205
0
    rtcm->rtcmtypes.rtcm3_msm.IODS = ugrab(3);
206
0
    bitcount += 7;             // skip 7 reserved bits, DF001
207
0
    rtcm->rtcmtypes.rtcm3_msm.steering = ugrab(2);
208
0
    rtcm->rtcmtypes.rtcm3_msm.ext_clk = ugrab(2);
209
0
    rtcm->rtcmtypes.rtcm3_msm.smoothing = ugrab(1);
210
0
    rtcm->rtcmtypes.rtcm3_msm.interval = ugrab(3);
211
    // FIXME: rtcm->rtcmtypes.rtcm3_msm.sat_mask = ugrab(64);
212
    // ugrab(56) is max, can't do 64, so stack it
213
0
    rtcm->rtcmtypes.rtcm3_msm.sat_mask = ugrab(32) << 32;
214
0
    rtcm->rtcmtypes.rtcm3_msm.sat_mask |= ugrab(32);
215
0
    rtcm->rtcmtypes.rtcm3_msm.sig_mask = ugrab(32);
216
217
    // count satellites
218
0
    sat_mask = rtcm->rtcmtypes.rtcm3_msm.sat_mask;
219
0
    while (sat_mask) {
220
0
        n_sat += sat_mask & 1;
221
0
        sat_mask >>= 1;
222
0
    }
223
    // count signals
224
0
    sig_mask = rtcm->rtcmtypes.rtcm3_msm.sig_mask;
225
0
    while (sig_mask) {
226
0
        n_sig += sig_mask & 1;
227
0
        sig_mask >>= 1;
228
0
    }
229
    // determine cells
230
0
    n_cell = n_sat * n_sig;
231
0
    rtcm->rtcmtypes.rtcm3_msm.n_sat = n_sat;
232
0
    rtcm->rtcmtypes.rtcm3_msm.n_sig = n_sig;
233
0
    rtcm->rtcmtypes.rtcm3_msm.n_cell = n_cell;
234
235
0
    if (0 == n_sat ||
236
0
        RTCM3_MAX_SATELLITES < n_sat ||
237
0
        RTCM3_MAX_SATELLITES < n_cell) {
238
0
        GPSD_LOG(LOG_WARN, &context->errout,
239
0
                 "RTCM3: rtcm3_decode_msm(%u) interval %u  sat_mask x%llx "
240
0
                 "sig_mask x%x invalid n_cell %u\n",
241
0
                 rtcm->type,
242
0
                 rtcm->rtcmtypes.rtcm3_msm.interval,
243
0
                 (unsigned long long)rtcm->rtcmtypes.rtcm3_msm.sat_mask,
244
0
                 rtcm->rtcmtypes.rtcm3_msm.sig_mask,
245
0
                 n_cell);
246
0
        return false;
247
0
    }
248
249
    // cell_mask is variable length!  ugrab() width max is 56
250
0
    if (56 >= n_cell) {
251
0
        rtcm->rtcmtypes.rtcm3_msm.cell_mask = ugrab(n_cell);
252
0
    } else {
253
        // 57 to 64, breaks ugrab(), workaround it...
254
0
        rtcm->rtcmtypes.rtcm3_msm.cell_mask = ugrab(56);
255
0
        rtcm->rtcmtypes.rtcm3_msm.cell_mask <<= n_cell - 56;
256
0
        rtcm->rtcmtypes.rtcm3_msm.cell_mask |= ugrab(n_cell - 56);
257
0
    }
258
259
    // Decode Satellite Data
260
261
    // Decode DF397 (MSM 4-7)
262
0
    if (4 == rtcm->rtcmtypes.rtcm3_msm.msm ||
263
0
        5 == rtcm->rtcmtypes.rtcm3_msm.msm ||
264
0
        6 == rtcm->rtcmtypes.rtcm3_msm.msm ||
265
0
        7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
266
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_sat; i++) {
267
0
            rtcm->rtcmtypes.rtcm3_msm.sat[i].rr_ms = ugrab(8);
268
0
        }
269
0
    }
270
271
    // Decode Extended Info (MSM 5+7)
272
0
    if (5 == rtcm->rtcmtypes.rtcm3_msm.msm ||
273
0
        7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
274
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_sat; i++) {
275
0
            rtcm->rtcmtypes.rtcm3_msm.sat[i].ext_info = ugrab(4);
276
0
        }
277
0
    }
278
279
    // Decode DF398 (MSM 1-7)
280
0
    for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_sat; i++) {
281
0
        rtcm->rtcmtypes.rtcm3_msm.sat[i].rr_m1 = ugrab(10);
282
0
    };
283
284
    // Decode DF399 (MSM 5+7)
285
0
    if (5 == rtcm->rtcmtypes.rtcm3_msm.msm ||
286
0
        7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
287
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_sat; i++) {
288
0
            rtcm->rtcmtypes.rtcm3_msm.sat[i].rates_rphr = ugrab(14);
289
0
        }
290
0
    }
291
292
    // Decode Signal Data
293
294
    // Decode DF400 (MSM 1,3,4,5) resp. DF405 (MSM 6+7)
295
0
    if (1 == rtcm->rtcmtypes.rtcm3_msm.msm ||
296
0
        3 == rtcm->rtcmtypes.rtcm3_msm.msm ||
297
0
        4 == rtcm->rtcmtypes.rtcm3_msm.msm ||
298
0
        5 == rtcm->rtcmtypes.rtcm3_msm.msm) {
299
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
300
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].pseudo_r = sgrab(15);
301
0
        }
302
0
    } else if (6 == rtcm->rtcmtypes.rtcm3_msm.msm ||
303
0
               7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
304
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
305
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].pseudo_r = sgrab(20);
306
0
        }
307
0
    }
308
309
    // Decode DF401 (MSM 2,3,4,5) resp. DF406 (MSM 6+7)
310
0
    if (2 == rtcm->rtcmtypes.rtcm3_msm.msm ||
311
0
        3 == rtcm->rtcmtypes.rtcm3_msm.msm ||
312
0
        4 == rtcm->rtcmtypes.rtcm3_msm.msm ||
313
0
        5 == rtcm->rtcmtypes.rtcm3_msm.msm) {
314
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
315
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].phase_r = sgrab(22);
316
0
        }
317
0
    } else if (6 == rtcm->rtcmtypes.rtcm3_msm.msm ||
318
0
               7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
319
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
320
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].phase_r = sgrab(24);
321
0
        }
322
0
    }
323
324
    // Decode DF402 (MSM 2,3,4,5) resp. DF407 (MSM 6+7)
325
0
    if (2 == rtcm->rtcmtypes.rtcm3_msm.msm ||
326
0
        3 == rtcm->rtcmtypes.rtcm3_msm.msm ||
327
0
        4 == rtcm->rtcmtypes.rtcm3_msm.msm ||
328
0
        5 == rtcm->rtcmtypes.rtcm3_msm.msm) {
329
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
330
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].lti = ugrab(4);
331
0
        }
332
0
    } else if (6 == rtcm->rtcmtypes.rtcm3_msm.msm ||
333
0
               7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
334
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
335
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].lti = ugrab(10);
336
0
        }
337
0
    }
338
339
    // Decode DF420 (MSM 2-7)
340
0
    if (2 == rtcm->rtcmtypes.rtcm3_msm.msm ||
341
0
        3 == rtcm->rtcmtypes.rtcm3_msm.msm ||
342
0
        4 == rtcm->rtcmtypes.rtcm3_msm.msm ||
343
0
        5 == rtcm->rtcmtypes.rtcm3_msm.msm ||
344
0
        6 == rtcm->rtcmtypes.rtcm3_msm.msm ||
345
0
        7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
346
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
347
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].half_amb = ugrab(1);
348
0
        }
349
0
    }
350
351
    // Decode DF403 (MSM 4+5) resp. DF408 (MSM 6+7)
352
0
    if (4 == rtcm->rtcmtypes.rtcm3_msm.msm ||
353
0
        5 == rtcm->rtcmtypes.rtcm3_msm.msm) {
354
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
355
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].cnr = ugrab(6);
356
0
        }
357
0
    } else if (6 == rtcm->rtcmtypes.rtcm3_msm.msm ||
358
0
               7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
359
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
360
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].cnr = ugrab(10);
361
0
        }
362
0
    }
363
364
    // Decode DF404 (MSM 5+7)
365
0
    if (5 == rtcm->rtcmtypes.rtcm3_msm.msm ||
366
0
        7 == rtcm->rtcmtypes.rtcm3_msm.msm) {
367
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_msm.n_cell; i++) {
368
0
            rtcm->rtcmtypes.rtcm3_msm.sig[i].cnr = sgrab(15);
369
0
        }
370
0
    }
371
372
    // tow is %llu for 32-bit compatibility
373
0
    GPSD_LOG(LOG_PROG, &context->errout, "RTCM3: rtcm3_decode_msm(%u) "
374
0
             "gnssid %u MSM%u id %u tow %llu sync %u IODS %u "
375
0
             "steering %u ext_clk %u smoothing %u interval %u "
376
0
             "sat_mask x%llx sig_mask x%lx cell_mask %llx\n",
377
0
             rtcm->type,
378
0
             rtcm->rtcmtypes.rtcm3_msm.gnssid,
379
0
             rtcm->rtcmtypes.rtcm3_msm.msm,
380
0
             rtcm->rtcmtypes.rtcm3_msm.station_id,
381
0
             (unsigned long long)rtcm->rtcmtypes.rtcm3_msm.tow,
382
0
             rtcm->rtcmtypes.rtcm3_msm.sync,
383
0
             rtcm->rtcmtypes.rtcm3_msm.IODS,
384
0
             rtcm->rtcmtypes.rtcm3_msm.steering,
385
0
             rtcm->rtcmtypes.rtcm3_msm.ext_clk,
386
0
             rtcm->rtcmtypes.rtcm3_msm.smoothing,
387
0
             rtcm->rtcmtypes.rtcm3_msm.interval,
388
0
             (long long unsigned)rtcm->rtcmtypes.rtcm3_msm.sat_mask,
389
0
             (long unsigned)rtcm->rtcmtypes.rtcm3_msm.sig_mask,
390
0
             (long long unsigned)rtcm->rtcmtypes.rtcm3_msm.cell_mask);
391
0
    return false;
392
0
}
393
394
/* break out the raw bits into the scaled report-structure fields
395
 *
396
 * Return: void
397
 */
398
void rtcm3_unpack(const struct gps_context_t *context,
399
                  struct rtcm3_t *rtcm, const unsigned char *buf)
400
0
{
401
0
    unsigned n, n2, n3, n4;
402
0
    int bitcount = 0;
403
0
    unsigned i;
404
0
    long temp;
405
0
    bool unknown = true;               // we don't know how to decode
406
0
    const char *msg_name = "Unknown";  // we know the name
407
0
    unsigned preamble, mbz;            // preamble 0xd3, and must be zero
408
0
    unsigned bad_len = 0;
409
410
0
#define GPS_PSEUDORANGE(fld, len) \
411
0
    {temp = ugrab(len);                         \
412
0
    if (temp == GPS_INVALID_PSEUDORANGE) {      \
413
0
        fld.pseudorange = 0;                    \
414
0
    } else {                                    \
415
0
        fld.pseudorange = temp * GPS_PSEUDORANGE_RESOLUTION;} \
416
0
    }
417
0
#define RANGEDIFF(fld, len) \
418
0
    temp = (long)sgrab(len);                    \
419
0
    if (temp == GPS_INVALID_PSEUDORANGE) {      \
420
0
        fld.rangediff = 0;                      \
421
0
    } else {                                    \
422
0
        fld.rangediff = temp * PSEUDORANGE_DIFF_RESOLUTION; \
423
0
    }
424
425
0
    memset(rtcm, 0, sizeof(struct rtcm3_t));
426
427
    // check preamble and zero bits
428
0
    preamble = ugrab(8);
429
0
    mbz = ugrab(6);
430
0
    if (0xD3 != preamble ||
431
0
        0 != mbz) {
432
        // The mbz may eventually used for RTCM version.
433
0
        GPSD_LOG(LOG_WARN, &context->errout,
434
0
                 "RTCM3: invalid preamble x%2x or mbz x%x\n",
435
0
                 preamble, mbz);
436
0
        return;
437
0
    }
438
439
0
    rtcm->length = ugrab(10);
440
0
    if (2 > rtcm->length) {
441
        // ignore zero payload messages, they do not evan have type
442
        // need 2 bytes just to read 10 bit type.
443
0
        GPSD_LOG(LOG_PROG, &context->errout,
444
0
                 "RTCM3: bad payload length %u bitcount %d\n",
445
0
                 rtcm->length, bitcount);
446
0
        return;
447
0
    }
448
0
    rtcm->type = ugrab(12);
449
450
0
    GPSD_LOG(LOG_IO, &context->errout,
451
0
             "RTCM3: type %d payload length %u bitcount %d\n",
452
0
             rtcm->type, rtcm->length, bitcount);
453
454
    // RTCM3 message type numbers start at 1001
455
0
    switch (rtcm->type) {
456
0
    case 1001:
457
0
        msg_name = "GPS Basic RTK, L1 Only";
458
0
        rtcm->rtcmtypes.rtcm3_1001.header.station_id = ugrab(12);
459
0
        rtcm->rtcmtypes.rtcm3_1001.header.tow = ugrab(30);
460
0
        rtcm->rtcmtypes.rtcm3_1001.header.sync = (bool)ugrab(1);
461
0
        n = ugrab(5);
462
0
        if ((8 + (7 * n)) > rtcm->length) {
463
            // not exactly: 8 + (7.25 * n)
464
0
            bad_len = 8 + (7 * n);
465
0
            break;
466
0
        }
467
0
        rtcm->rtcmtypes.rtcm3_1001.header.satcount = n;
468
0
        rtcm->rtcmtypes.rtcm3_1001.header.smoothing = (bool)ugrab(1);
469
0
        rtcm->rtcmtypes.rtcm3_1001.header.interval = ugrab(3);
470
0
#define R1001 rtcm->rtcmtypes.rtcm3_1001.rtk_data[i]
471
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1001.header.satcount; i++) {
472
0
            R1001.ident = ugrab(6);
473
0
            R1001.L1.indicator = ugrab(1);
474
0
            GPS_PSEUDORANGE(R1001.L1, 24);
475
0
            RANGEDIFF(R1001.L1, 20);
476
0
            R1001.L1.locktime = (unsigned char)sgrab(7);
477
0
        }
478
0
#undef R1001
479
0
        unknown = false;
480
0
        break;
481
482
0
    case 1002:
483
0
        msg_name = "GPS Extended RTK, L1 Only";
484
0
        rtcm->rtcmtypes.rtcm3_1002.header.station_id = ugrab(12);
485
0
        rtcm->rtcmtypes.rtcm3_1002.header.tow = ugrab(30);
486
0
        rtcm->rtcmtypes.rtcm3_1002.header.sync = (bool)ugrab(1);
487
0
        n = ugrab(5);
488
0
        if ((8 + (9 * n)) > rtcm->length) {
489
            // not exactly: 8 + (9.25 * n)
490
0
            bad_len = 8 + (9 * n);
491
0
            break;
492
0
        }
493
0
        rtcm->rtcmtypes.rtcm3_1002.header.satcount = n;
494
0
        rtcm->rtcmtypes.rtcm3_1002.header.smoothing = (bool)ugrab(1);
495
0
        rtcm->rtcmtypes.rtcm3_1002.header.interval = ugrab(3);
496
0
#define R1002 rtcm->rtcmtypes.rtcm3_1002.rtk_data[i]
497
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1002.header.satcount; i++) {
498
0
            R1002.ident = ugrab(6);
499
0
            R1002.L1.indicator = ugrab(1);
500
0
            GPS_PSEUDORANGE(R1002.L1, 24);
501
0
            RANGEDIFF(R1002.L1, 20);
502
0
            R1002.L1.locktime = (unsigned char)sgrab(7);
503
0
            R1002.L1.ambiguity = ugrab(8);
504
0
            R1002.L1.CNR = (ugrab(8)) * CARRIER_NOISE_RATIO_UNITS;
505
0
        }
506
0
#undef R1002
507
0
        unknown = false;
508
0
        break;
509
510
0
    case 1003:
511
0
        msg_name = "GPS Basic RTK, L1 & L2";
512
0
        rtcm->rtcmtypes.rtcm3_1003.header.station_id = ugrab(12);
513
0
        rtcm->rtcmtypes.rtcm3_1003.header.tow = ugrab(30);
514
0
        rtcm->rtcmtypes.rtcm3_1003.header.sync = (bool)ugrab(1);
515
0
        n = ugrab(5);
516
0
        if ((8 + (12 * n)) > rtcm->length) {
517
            // not exactly: 8 + (12.625 * n)
518
0
            bad_len = 8 + (12 * n);
519
0
            break;
520
0
        }
521
0
        rtcm->rtcmtypes.rtcm3_1003.header.satcount = n;
522
0
        rtcm->rtcmtypes.rtcm3_1003.header.smoothing = (bool)ugrab(1);
523
0
        rtcm->rtcmtypes.rtcm3_1003.header.interval = ugrab(3);
524
0
#define R1003 rtcm->rtcmtypes.rtcm3_1003.rtk_data[i]
525
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1003.header.satcount; i++) {
526
0
            R1003.ident = ugrab(6);
527
0
            R1003.L1.indicator = ugrab(1);
528
0
            GPS_PSEUDORANGE(R1003.L1, 24);
529
0
            RANGEDIFF(R1003.L1, 20);
530
0
            R1003.L1.locktime = (unsigned char)sgrab(7);
531
0
            R1003.L2.indicator = ugrab(2);
532
0
            GPS_PSEUDORANGE(R1003.L2, 24);
533
0
            temp = (long)sgrab(20);
534
0
            if (temp == GPS_INVALID_PSEUDORANGE) {
535
0
                R1003.L2.rangediff = 0;
536
0
            } else {
537
0
                R1003.L2.rangediff = temp * PSEUDORANGE_DIFF_RESOLUTION;
538
0
            }
539
0
            R1003.L2.locktime = (unsigned char)sgrab(7);
540
0
        }
541
0
#undef R1003
542
0
        unknown = false;
543
0
        break;
544
545
0
    case 1004:
546
0
        msg_name = "GPS Extended RTK, L1 & L2";
547
0
        rtcm->rtcmtypes.rtcm3_1004.header.station_id = ugrab(12);
548
0
        rtcm->rtcmtypes.rtcm3_1004.header.tow = ugrab(30);
549
0
        rtcm->rtcmtypes.rtcm3_1004.header.sync = (bool)ugrab(1);
550
0
        n = ugrab(5);
551
0
        bad_len = ceil(8.0 + (15.625* n));
552
0
        if (bad_len != rtcm->length) {
553
            // exactly: 8 + (15.625 * n)
554
0
            break;
555
0
        }
556
0
        bad_len = 0;
557
0
        rtcm->rtcmtypes.rtcm3_1004.header.satcount = n;
558
0
        rtcm->rtcmtypes.rtcm3_1004.header.smoothing = (bool)ugrab(1);
559
0
        rtcm->rtcmtypes.rtcm3_1004.header.interval = ugrab(3);
560
0
#define R1004 rtcm->rtcmtypes.rtcm3_1004.rtk_data[i]
561
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1004.header.satcount; i++) {
562
0
            R1004.ident = ugrab(6);
563
0
            R1004.L1.indicator = (bool)ugrab(1);
564
0
            GPS_PSEUDORANGE(R1004.L1, 24);
565
0
            RANGEDIFF(R1004.L1, 20);
566
0
            R1004.L1.locktime = (unsigned char)sgrab(7);
567
0
            R1004.L1.ambiguity = ugrab(8);
568
0
            R1004.L1.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
569
0
            R1004.L2.indicator = ugrab(2);
570
0
            GPS_PSEUDORANGE(R1004.L2, 14);
571
0
            RANGEDIFF(R1004.L2, 20);
572
0
            R1004.L2.locktime = (unsigned char)sgrab(7);
573
0
            R1004.L2.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
574
0
        }
575
0
#undef R1004
576
0
        unknown = false;
577
0
        break;
578
579
0
    case 1005:
580
0
        msg_name = "Stationary Antenna Reference Point, No Height Information";
581
        // 19 bytes
582
0
        if (19 != rtcm->length) {
583
0
            bad_len = 19;
584
0
            break;
585
0
        }
586
0
#define R1005 rtcm->rtcmtypes.rtcm3_1005
587
0
        R1005.station_id = ugrab(12);
588
0
        ugrab(6);               // reserved
589
0
        R1005.system = ugrab(3);
590
0
        R1005.reference_station = (bool)ugrab(1);
591
0
        R1005.ecef_x = sgrab(38) * ANTENNA_POSITION_RESOLUTION;
592
0
        R1005.single_receiver = ugrab(1);
593
0
        ugrab(1);
594
0
        R1005.ecef_y = sgrab(38) * ANTENNA_POSITION_RESOLUTION;
595
0
        ugrab(2);
596
0
        R1005.ecef_z = sgrab(38) * ANTENNA_POSITION_RESOLUTION;
597
0
#undef R1005
598
0
        unknown = false;
599
0
        break;
600
601
0
    case 1006:
602
0
        msg_name = "Stationary Antenna Reference Point, w/ Height Info";
603
        // 21 bytes
604
0
        if (21 != rtcm->length) {
605
0
            bad_len = 21;
606
0
            break;
607
0
        }
608
0
#define R1006 rtcm->rtcmtypes.rtcm3_1006
609
0
        R1006.station_id = ugrab(12);
610
0
        (void)ugrab(6);         // reserved
611
0
        R1006.system = ugrab(3);
612
0
        R1006.reference_station = (bool)ugrab(1);
613
0
        R1006.ecef_x = sgrab(38) * ANTENNA_POSITION_RESOLUTION;
614
0
        R1006.single_receiver = ugrab(1);
615
0
        ugrab(1);
616
0
        R1006.ecef_y = sgrab(38) * ANTENNA_POSITION_RESOLUTION;
617
0
        ugrab(2);
618
0
        R1006.ecef_z = sgrab(38) * ANTENNA_POSITION_RESOLUTION;
619
0
        R1006.height = ugrab(16) * ANTENNA_POSITION_RESOLUTION;
620
0
#undef R1006
621
0
        unknown = false;
622
0
        break;
623
624
0
    case 1007:
625
0
        msg_name = "Antenna Description";
626
        // 5 to 36 bytes
627
0
        rtcm->rtcmtypes.rtcm3_1007.station_id = ugrab(12);
628
0
        n = ugrab(8);
629
0
        if ((5 + n) != rtcm->length) {
630
0
            bad_len = 5 + n;
631
0
            break;
632
0
        }
633
        // 1007 says DF030 is 20 chars, or less, of ASCII
634
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1007.descriptor,
635
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1007.descriptor),
636
0
                                buf + 7, n);
637
0
        bitcount += 8 * n;
638
0
        rtcm->rtcmtypes.rtcm3_1007.setup_id = ugrab(8);
639
0
        unknown = false;
640
0
        break;
641
642
0
    case 1008:
643
0
        msg_name = "Antenna Description & Serial Number";
644
        // 6 to 68 bytes
645
0
        rtcm->rtcmtypes.rtcm3_1008.station_id = ugrab(12);
646
0
        n = ugrab(8);
647
0
        if ((6 + n) > rtcm->length) {
648
0
            bad_len = 6 + n;
649
0
            break;
650
0
        }
651
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1008.descriptor,
652
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1008.descriptor),
653
0
                                buf + 7, n);
654
0
        bitcount += 8 * n;
655
0
        rtcm->rtcmtypes.rtcm3_1008.setup_id = ugrab(8);
656
0
        n2 = ugrab(8);
657
0
        if ((6 + n + n2) != rtcm->length) {
658
0
            bad_len = 6 + n + n2;
659
0
            break;
660
0
        }
661
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1008.serial,
662
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1008.serial),
663
0
                                buf + 9 + n, n2);
664
        // bitcount += 8 * n2;
665
0
        unknown = false;
666
0
        break;
667
668
0
    case 1009:
669
0
        msg_name = "GLONASS Basic RTK, L1 Only";
670
0
        rtcm->rtcmtypes.rtcm3_1009.header.station_id = ugrab(12);
671
0
        rtcm->rtcmtypes.rtcm3_1009.header.tow = ugrab(27);
672
0
        rtcm->rtcmtypes.rtcm3_1009.header.sync = (bool)ugrab(1);
673
0
        n = ugrab(5);
674
0
        if ((7 + (8 * n)) > rtcm->length) {
675
            // not exactly: 7.625 + (8 * n)
676
0
            bad_len = 7 + (8 * n);
677
0
            break;
678
0
        }
679
0
        rtcm->rtcmtypes.rtcm3_1009.header.satcount = n;
680
0
        rtcm->rtcmtypes.rtcm3_1009.header.smoothing = (bool)ugrab(1);
681
0
        rtcm->rtcmtypes.rtcm3_1009.header.interval = ugrab(3);
682
0
#define R1009 rtcm->rtcmtypes.rtcm3_1009.rtk_data[i]
683
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1009.header.satcount; i++) {
684
0
            R1009.ident = ugrab(6);
685
0
            R1009.L1.indicator = (bool)ugrab(1);
686
0
            R1009.L1.channel = (short)ugrab(5) - GLONASS_CHANNEL_BASE;
687
0
            R1009.L1.pseudorange = ugrab(25) * GLONASS_PSEUDORANGE_RESOLUTION;
688
0
            RANGEDIFF(R1009.L1, 20);
689
0
            R1009.L1.locktime = (unsigned char)sgrab(7);
690
0
        }
691
0
#undef R1009
692
0
        unknown = false;
693
0
        break;
694
695
0
    case 1010:
696
0
        msg_name = "GLONASS Extended RTK, L1 Only";
697
0
        rtcm->rtcmtypes.rtcm3_1010.header.station_id =
698
0
            (unsigned short)ugrab(12);
699
0
        rtcm->rtcmtypes.rtcm3_1010.header.tow = ugrab(27);
700
0
        rtcm->rtcmtypes.rtcm3_1010.header.sync = (bool)ugrab(1);
701
0
        n = ugrab(5);
702
0
        bad_len = ceil(7.625 + (9.875 * n));
703
0
        if (bad_len != rtcm->length) {
704
            // exactly: 7.625 + (9.875 * n)
705
0
            break;
706
0
        }
707
0
        bad_len = 0;
708
0
        rtcm->rtcmtypes.rtcm3_1010.header.satcount = n;
709
0
        rtcm->rtcmtypes.rtcm3_1010.header.smoothing = (bool)ugrab(1);
710
0
        rtcm->rtcmtypes.rtcm3_1010.header.interval = ugrab(3);
711
0
#define R1010 rtcm->rtcmtypes.rtcm3_1010.rtk_data[i]
712
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1010.header.satcount; i++) {
713
0
            R1010.ident = (unsigned short)ugrab(6);
714
0
            R1010.L1.indicator = (bool)ugrab(1);
715
0
            R1010.L1.channel = (short)ugrab(5) - GLONASS_CHANNEL_BASE;
716
0
            R1010.L1.pseudorange = ugrab(25) * GLONASS_PSEUDORANGE_RESOLUTION;
717
0
            RANGEDIFF(R1010.L1, 20);
718
0
            R1010.L1.locktime = (unsigned char)sgrab(7);
719
0
            R1010.L1.ambiguity = (unsigned char)ugrab(7);
720
0
            R1010.L1.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
721
0
        }
722
0
#undef R1010
723
0
        unknown = false;
724
0
        break;
725
726
0
    case 1011:
727
0
        msg_name = "GLONASS Basic RTK, L1 & L2";
728
0
        rtcm->rtcmtypes.rtcm3_1011.header.station_id =
729
0
            (unsigned short)ugrab(12);
730
0
        rtcm->rtcmtypes.rtcm3_1011.header.tow = ugrab(27);
731
0
        rtcm->rtcmtypes.rtcm3_1011.header.sync = (bool)ugrab(1);
732
0
        n = ugrab(5);
733
0
        bad_len = ceil(7.625 + (13.325 * n));
734
0
        if (bad_len != rtcm->length) {
735
            // exactly: 7.625 + (13.325 * n)
736
0
            break;
737
0
        }
738
0
        bad_len = 0;
739
0
        rtcm->rtcmtypes.rtcm3_1011.header.satcount = n;
740
0
        rtcm->rtcmtypes.rtcm3_1011.header.smoothing = (bool)ugrab(1);
741
0
        rtcm->rtcmtypes.rtcm3_1011.header.interval = ugrab(3);
742
0
#define R1011 rtcm->rtcmtypes.rtcm3_1011.rtk_data[i]
743
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1011.header.satcount; i++) {
744
0
            R1011.ident = (unsigned short)ugrab(6);
745
0
            R1011.L1.indicator = (bool)ugrab(1);
746
0
            R1011.L1.channel = (short)ugrab(5) - GLONASS_CHANNEL_BASE;
747
0
            R1011.L1.pseudorange = ugrab(25) * GLONASS_PSEUDORANGE_RESOLUTION;
748
0
            RANGEDIFF(R1011.L1, 20);
749
0
            R1011.L1.locktime = (unsigned char)sgrab(7);
750
0
            R1011.L1.ambiguity = (unsigned char)ugrab(7);
751
0
            R1011.L1.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
752
0
            R1011.L2.indicator = (bool)ugrab(1);
753
0
            R1011.L2.channel = (short)ugrab(5) - GLONASS_CHANNEL_BASE;
754
0
            R1011.L2.pseudorange = ugrab(25) * GLONASS_PSEUDORANGE_RESOLUTION;
755
0
            RANGEDIFF(R1011.L2, 20);
756
0
            R1011.L2.locktime = (unsigned char)sgrab(7);
757
0
            R1011.L2.ambiguity = (unsigned char)ugrab(7);
758
0
            R1011.L2.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
759
0
        }
760
0
#undef R1011
761
0
        unknown = false;
762
0
        break;
763
764
0
    case 1012:
765
0
        msg_name = "GLONASS Extended RTK, L1 & L2";
766
0
        rtcm->rtcmtypes.rtcm3_1012.header.station_id = ugrab(12);
767
0
        rtcm->rtcmtypes.rtcm3_1012.header.tow = ugrab(27);
768
0
        rtcm->rtcmtypes.rtcm3_1012.header.sync = (bool)ugrab(1);
769
0
        n = ugrab(5);
770
0
        bad_len = ceil(7.625 + (16.25 * n));
771
0
        if (bad_len != rtcm->length) {
772
            // exactly: 7.625 + (16.25 * n)
773
0
            break;
774
0
        }
775
0
        bad_len = 0;
776
0
        rtcm->rtcmtypes.rtcm3_1012.header.satcount = n;
777
0
        rtcm->rtcmtypes.rtcm3_1012.header.smoothing = (bool)ugrab(1);
778
0
        rtcm->rtcmtypes.rtcm3_1012.header.interval = ugrab(3);
779
0
#define R1012 rtcm->rtcmtypes.rtcm3_1012.rtk_data[i]
780
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1012.header.satcount; i++) {
781
0
            unsigned rangeincr;
782
783
0
            R1012.ident = ugrab(6);
784
0
            R1012.L1.indicator = (bool)ugrab(1);
785
0
            R1012.L1.channel = (short)ugrab(5) - GLONASS_CHANNEL_BASE;
786
0
            R1012.L1.pseudorange = ugrab(25) * GLONASS_PSEUDORANGE_RESOLUTION;
787
0
            RANGEDIFF(R1012.L1, 20);
788
0
            R1012.L1.locktime = ugrab(7);
789
0
            R1012.L1.ambiguity = ugrab(7);
790
0
            R1012.L1.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
791
0
            R1012.L2.indicator = (bool)ugrab(2);
792
0
            rangeincr = ugrab(14);
793
0
            if (rangeincr == GLONASS_INVALID_RANGEINCR) {
794
0
                R1012.L2.pseudorange = 0;
795
0
            } else {
796
0
                R1012.L2.pseudorange = (rangeincr *
797
0
                                        GLONASS_PSEUDORANGE_RESOLUTION);
798
0
            }
799
0
            RANGEDIFF(R1012.L2, 20);
800
0
            R1012.L2.locktime = (unsigned char)sgrab(7);
801
0
            R1012.L2.CNR = ugrab(8) * CARRIER_NOISE_RATIO_UNITS;
802
0
        }
803
0
#undef R1012
804
0
        unknown = false;
805
0
        break;
806
807
0
    case 1013:
808
0
        msg_name = "System Parameters";
809
0
        rtcm->rtcmtypes.rtcm3_1013.station_id = ugrab(12);
810
0
        rtcm->rtcmtypes.rtcm3_1013.mjd = ugrab(16);
811
0
        rtcm->rtcmtypes.rtcm3_1013.sod = ugrab(17);
812
0
        n = ugrab(5);
813
0
        bad_len = ceil(8.75 + (3.625 * n));
814
0
        if (bad_len != rtcm->length) {
815
            // exactly: 8.75 + (3.625 * n)
816
0
            break;
817
0
        }
818
0
        bad_len = 0;
819
0
        rtcm->rtcmtypes.rtcm3_1013.ncount = n;
820
0
        rtcm->rtcmtypes.rtcm3_1013.leapsecs = ugrab(8);
821
0
#define R1013 rtcm->rtcmtypes.rtcm3_1013.announcements[i]
822
0
        for (i = 0; i < rtcm->rtcmtypes.rtcm3_1013.ncount; i++) {
823
0
            R1013.id = ugrab(12);
824
0
            R1013.sync = (bool)ugrab(1);
825
0
            R1013.interval = ugrab(16);
826
0
        }
827
0
#undef R1013
828
0
        unknown = false;
829
0
        break;
830
831
0
    case 1014:
832
0
        msg_name = "Network Auxiliary Station Data";
833
        // coordinate difference between one Aux station and master station
834
0
        if (15 != rtcm->length) {
835
0
            bad_len = 15;
836
0
            break;
837
0
        }
838
0
        rtcm->rtcmtypes.rtcm3_1014.network_id = (int)ugrab(8);
839
0
        rtcm->rtcmtypes.rtcm3_1014.subnetwork_id = (int)ugrab(4);
840
0
        rtcm->rtcmtypes.rtcm3_1014.stationcount = (char)ugrab(5);
841
0
        rtcm->rtcmtypes.rtcm3_1014.master_id = (int)ugrab(12);
842
0
        rtcm->rtcmtypes.rtcm3_1014.aux_id = (int)ugrab(12);
843
0
        rtcm->rtcmtypes.rtcm3_1014.d_lat =
844
0
            ugrab(20) * ANTENNA_DEGREE_RESOLUTION;
845
0
        rtcm->rtcmtypes.rtcm3_1014.d_lon =
846
0
            ugrab(21) * ANTENNA_DEGREE_RESOLUTION;
847
0
        rtcm->rtcmtypes.rtcm3_1014.d_alt = ugrab(23) / 1000;
848
0
        unknown = false;
849
0
        break;
850
851
0
    case 1015:
852
        /* RTCM 3.1
853
         * GPS Ionospheric Correction Differences for all satellites
854
         * between the master station and one auxiliary station
855
         * 9 bytes minimum
856
         */
857
0
        if (9 > rtcm->length) {
858
            // need 76 bits, 9.5 bytes
859
0
            bad_len = 9;
860
0
            break;
861
0
        }
862
863
0
        unknown = rtcm3_101567(context, rtcm, buf);
864
0
        msg_name = "GPS Ionospheric Correction Differences";
865
0
        break;
866
867
0
    case 1016:
868
        /* RTCM 3.1
869
         * GPS Geometric Correction Differences for all satellites between
870
         * the master station and one auxiliary station.
871
         * 9 bytes minimum
872
         */
873
0
        if (9 > rtcm->length) {
874
            // need 76 bits, 9.5 bytes
875
0
            bad_len = 9;
876
0
            break;
877
0
        }
878
0
        unknown = rtcm3_101567(context, rtcm, buf);
879
0
        msg_name = "GPS Geometric Correction Differences";
880
0
        break;
881
882
0
    case 1017:
883
        /* RTCM 3.1
884
         * GPS Combined Geometric and Ionospheric Correction Differences
885
         * for all satellites between one Aux station and the master station
886
         * (same content as both types 1015 and 1016 together, but less size)
887
         * 9 bytes minimum
888
         */
889
0
        if (9 > rtcm->length) {
890
            // need 76 bits, 9.5 bytes
891
0
            bad_len = 9;
892
0
            break;
893
0
        }
894
0
        unknown = rtcm3_101567(context, rtcm, buf);
895
0
        unknown = rtcm3_101567(context, rtcm, buf);
896
0
        msg_name = "GPS Combined Geometric and Ionospheric "
897
0
                       "Correction Differences";
898
0
        break;
899
900
0
    case 1018:
901
        /* RTCM 3.1
902
         * Reserved for alternative Ionospheric Correction Difference Message
903
         */
904
0
        msg_name = "Reserved for alternative Ionospheric Correction "
905
0
                       "Differences";
906
0
        break;
907
908
0
    case 1019:
909
        /* RTCM 3.1 - 1020
910
         * GPS Ephemeris
911
         * 62 bytes
912
         */
913
0
        if (61 != rtcm->length) {
914
0
            bad_len = 61;
915
0
            break;
916
0
        }
917
        // TODO: rtklib has C code for this one.
918
0
        msg_name = "GPS Ephemeris";
919
0
        break;
920
921
0
    case 1020:
922
        /* RTCM 3.1 - 1020
923
         * GLONASS Ephemeris
924
         * 45 bytes
925
         */
926
0
        if (45 != rtcm->length) {
927
0
            bad_len = 45;
928
0
            break;
929
0
        }
930
        // TODO: rtklib has C code for this one.
931
0
        msg_name = "GLO Ephemeris";
932
0
        break;
933
934
0
    case 1021:
935
        /* RTCM 3.1
936
         * Helmert / Abridged Molodenski Transformation parameters
937
         */
938
0
        msg_name = "Helmert / Abridged Molodenski Transformation "
939
0
                       "parameters";
940
0
        if (51 > rtcm->length) {
941
            // actually 51.5 + n + m
942
0
            bad_len = 51;
943
0
            break;
944
0
        }
945
        // Set Source-Name
946
0
        n = ugrab(5);
947
0
        if ((sizeof(rtcm->rtcmtypes.rtcm3_1021.src_name) - 1) <= n) {
948
            // paranoia
949
0
            n = sizeof(rtcm->rtcmtypes.rtcm3_1021.src_name) - 1;
950
0
        }
951
0
        for (i = 0; i < n; i++) {
952
            // Grrr.  Not byte aligned
953
0
            rtcm->rtcmtypes.rtcm3_1021.src_name[i] = (char)ugrab(8);
954
0
        }
955
0
        rtcm->rtcmtypes.rtcm3_1021.src_name[n] = '\0';
956
        // Set Target-Name
957
0
        n2 = (unsigned)ugrab(5);
958
0
        if ((sizeof(rtcm->rtcmtypes.rtcm3_1021.tar_name) - 1) <= n2) {
959
            // paranoia
960
0
            n2 = sizeof(rtcm->rtcmtypes.rtcm3_1021.tar_name) - 1;
961
0
        }
962
0
        for (i = 0; i < n2; i++) {
963
0
            rtcm->rtcmtypes.rtcm3_1021.tar_name[i] = (char)ugrab(8);
964
0
        }
965
0
        rtcm->rtcmtypes.rtcm3_1021.tar_name[n2] = '\0';
966
0
        rtcm->rtcmtypes.rtcm3_1021.sys_id_num = ugrab(8);
967
0
#define R1021 rtcm->rtcmtypes.rtcm3_1021.ut_tr_msg_id[i]
968
0
        for (i = 0; i < RTCM3_DF148_SIZE; i++) {
969
0
            R1021 = (bool)ugrab(1);
970
0
        }
971
0
#undef R1021
972
0
        rtcm->rtcmtypes.rtcm3_1021.plate_number = ugrab(5);
973
0
        rtcm->rtcmtypes.rtcm3_1021.computation_id = ugrab(4);
974
0
        rtcm->rtcmtypes.rtcm3_1021.height_id = ugrab(2);
975
0
        rtcm->rtcmtypes.rtcm3_1021.lat_origin = sgrab(19) *
976
0
            VALIDITY_RESOLUTION / DEG_ARCSEC_RESOLUTION;
977
0
        rtcm->rtcmtypes.rtcm3_1021.lon_origin = sgrab(20) *
978
0
            VALIDITY_RESOLUTION / DEG_ARCSEC_RESOLUTION;
979
0
        rtcm->rtcmtypes.rtcm3_1021.lat_extension = sgrab(14) *
980
0
            VALIDITY_RESOLUTION / DEG_ARCSEC_RESOLUTION;
981
0
        rtcm->rtcmtypes.rtcm3_1021.lon_extension = sgrab(14) *
982
0
            VALIDITY_RESOLUTION / DEG_ARCSEC_RESOLUTION;
983
0
        rtcm->rtcmtypes.rtcm3_1021.x_trans = sgrab(23) *
984
0
            TRANSLATION_MM_RESOLUTION;
985
0
        rtcm->rtcmtypes.rtcm3_1021.y_trans = sgrab(23) *
986
0
            TRANSLATION_MM_RESOLUTION;
987
0
        rtcm->rtcmtypes.rtcm3_1021.z_trans = sgrab(23) *
988
0
            TRANSLATION_MM_RESOLUTION;
989
0
        rtcm->rtcmtypes.rtcm3_1021.x_rot = sgrab(32) *
990
0
            ROTATION_ARCSEC_RESOLUTION / DEG_ARCSEC_RESOLUTION;
991
0
        rtcm->rtcmtypes.rtcm3_1021.y_rot = sgrab(32) *
992
0
            ROTATION_ARCSEC_RESOLUTION / DEG_ARCSEC_RESOLUTION;
993
0
        rtcm->rtcmtypes.rtcm3_1021.z_rot = sgrab(32) *
994
0
            ROTATION_ARCSEC_RESOLUTION / DEG_ARCSEC_RESOLUTION;
995
0
        rtcm->rtcmtypes.rtcm3_1021.ds = sgrab(25) * SCALE_PPM_RESOLUTION;
996
0
        rtcm->rtcmtypes.rtcm3_1021.add_as = sgrab(24) *
997
0
            TRANSLATION_MM_RESOLUTION;
998
0
        rtcm->rtcmtypes.rtcm3_1021.add_bs = sgrab(25) *
999
0
            TRANSLATION_MM_RESOLUTION;
1000
0
        rtcm->rtcmtypes.rtcm3_1021.add_at = sgrab(24) *
1001
0
            TRANSLATION_MM_RESOLUTION;
1002
0
        rtcm->rtcmtypes.rtcm3_1021.add_bt = sgrab(25) *
1003
0
            TRANSLATION_MM_RESOLUTION;
1004
0
        rtcm->rtcmtypes.rtcm3_1021.quality_hori = ugrab(3);
1005
0
        rtcm->rtcmtypes.rtcm3_1021.quality_vert = ugrab(3);
1006
1007
0
        unknown = false;
1008
0
        break;
1009
1010
0
    case 1022:
1011
        /* RTCM 3.1
1012
         * Molodenski-Badekas transformation parameters
1013
         */
1014
0
        msg_name = "Molodenski-Badekas transformation parameters";
1015
0
        if (64 > rtcm->length) {
1016
            // actually 64,625 + n + m
1017
0
            bad_len = 64;
1018
0
            break;
1019
0
        }
1020
0
        break;
1021
1022
0
    case 1023:
1023
        /* RTCM 3.1
1024
         * Residuals Ellipsoidal Grid Representation
1025
         */
1026
0
        msg_name = "Residuals Ellipsoidal Grid Representation";
1027
0
        if (72 > rtcm->length) {
1028
            // actually 72,25
1029
0
            bad_len = 72;
1030
0
            break;
1031
0
        }
1032
0
        rtcm->rtcmtypes.rtcm3_1023.sys_id_num = ugrab(8);
1033
0
        rtcm->rtcmtypes.rtcm3_1023.shift_id_hori = (bool)ugrab(1);
1034
0
        rtcm->rtcmtypes.rtcm3_1023.shift_id_vert = (bool)ugrab(1);
1035
0
        rtcm->rtcmtypes.rtcm3_1023.lat_origin = sgrab(21) *
1036
0
            PHASE_CORRECTION_RESOLUTION / DEG_ARCSEC_RESOLUTION;
1037
0
        rtcm->rtcmtypes.rtcm3_1023.lon_origin = sgrab(22) *
1038
0
            PHASE_CORRECTION_RESOLUTION / DEG_ARCSEC_RESOLUTION;
1039
0
        rtcm->rtcmtypes.rtcm3_1023.lat_extension = ugrab(12) *
1040
0
            PHASE_CORRECTION_RESOLUTION / DEG_ARCSEC_RESOLUTION;
1041
0
        rtcm->rtcmtypes.rtcm3_1023.lon_extension = ugrab(12) *
1042
0
            PHASE_CORRECTION_RESOLUTION / DEG_ARCSEC_RESOLUTION;
1043
0
        rtcm->rtcmtypes.rtcm3_1023.lat_mean = sgrab(8) *
1044
0
            TRANSLATION_MM_RESOLUTION;
1045
0
        rtcm->rtcmtypes.rtcm3_1023.lon_mean = sgrab(8) *
1046
0
            TRANSLATION_MM_RESOLUTION;
1047
0
        rtcm->rtcmtypes.rtcm3_1023.hgt_mean = sgrab(15) * CM_RESOLUTION;
1048
0
#define R1023 rtcm->rtcmtypes.rtcm3_1023.residuals[i]
1049
0
        for (i = 0; i < RTCM3_GRID_SIZE; i++) {
1050
0
            R1023.lat_res = sgrab(9) * RES_ARCSEC_RESOLUTION;
1051
0
            R1023.lon_res = sgrab(9) * RES_ARCSEC_RESOLUTION;
1052
0
            R1023.hgt_res = sgrab(9) * TRANSLATION_MM_RESOLUTION;
1053
0
        }
1054
0
#undef R1023
1055
0
        rtcm->rtcmtypes.rtcm3_1023.interp_meth_id_hori = ugrab(2);
1056
0
        rtcm->rtcmtypes.rtcm3_1023.interp_meth_id_vert = ugrab(2);
1057
0
        rtcm->rtcmtypes.rtcm3_1023.grd_qual_id_hori = ugrab(3);
1058
0
        rtcm->rtcmtypes.rtcm3_1023.grd_qual_id_vert = ugrab(3);
1059
0
        rtcm->rtcmtypes.rtcm3_1023.mjd = ugrab(16);
1060
0
        unknown = false;
1061
0
        break;
1062
1063
0
    case 1024:
1064
        /* RTCM 3.1
1065
         * Residuals Plane Grid Representation
1066
         */
1067
0
        msg_name = "Residuals Plane Grid Representation";
1068
0
        if (72 > rtcm->length) {
1069
            // actually 72,75
1070
0
            bad_len = 72;
1071
0
            break;
1072
0
        }
1073
0
        break;
1074
1075
0
    case 1025:
1076
        /* RTCM 3.1
1077
         * Projection Parameters, Projection Types other than LCC2SP
1078
         */
1079
0
        msg_name = "Projection Parameters, Projection Types other "
1080
0
                       "than LCC2SP";
1081
0
        if (24 > rtcm->length) {
1082
            // actually 24.5
1083
0
            bad_len = 24;
1084
0
            break;
1085
0
        }
1086
0
        rtcm->rtcmtypes.rtcm3_1025.sys_id_num = ugrab(8);
1087
0
        rtcm->rtcmtypes.rtcm3_1025.projection_type = ugrab(6);
1088
0
        rtcm->rtcmtypes.rtcm3_1025.lat_origin = sgrab(34) *
1089
0
            PROJ_ORIGIN_RESOLUTION;
1090
0
        rtcm->rtcmtypes.rtcm3_1025.lon_origin = sgrab(35) *
1091
0
            PROJ_ORIGIN_RESOLUTION;
1092
0
        rtcm->rtcmtypes.rtcm3_1025.add_sno = ugrab(30) *
1093
0
            SCALE_PPM_RESOLUTION;
1094
0
        rtcm->rtcmtypes.rtcm3_1025.false_east = ugrab(36) *
1095
0
            TRANSLATION_MM_RESOLUTION;
1096
0
        rtcm->rtcmtypes.rtcm3_1025.false_north = ugrab(35) *
1097
0
            TRANSLATION_MM_RESOLUTION;
1098
0
        unknown = false;
1099
0
        break;
1100
1101
0
    case 1026:
1102
        /* RTCM 3.1
1103
         * Projection Parameters, Projection Type LCC2SP
1104
         * (Lambert Conic Conformal)
1105
         */
1106
0
        msg_name = "Projection Parameters, Projection Type LCC2SP";
1107
0
        if (29 > rtcm->length) {
1108
            // actually 29.25
1109
0
            bad_len = 29;
1110
0
            break;
1111
0
        }
1112
0
        break;
1113
1114
0
    case 1027:
1115
        /* RTCM 3.1
1116
         * Projection Parameters, Projection Type OM (Oblique Mercator)
1117
         */
1118
0
        msg_name = "Projection Parameters, Projection Type OM";
1119
0
        if (32 > rtcm->length) {
1120
            // actually 32.25
1121
0
            bad_len = 32;
1122
0
            break;
1123
0
        }
1124
0
        break;
1125
1126
0
    case 1028:
1127
        /* RTCM 3.1
1128
         * Reserved for global to plate fixed transformation
1129
         */
1130
0
        msg_name = "Reserved, Global to Plate Transformation";
1131
0
        break;
1132
1133
0
    case 1029:
1134
0
        msg_name = "Text in UTF8 format";
1135
        /* 9 bytes minimum
1136
         * (max. 127 multibyte characters and max. 255 bytes)
1137
         */
1138
0
        rtcm->rtcmtypes.rtcm3_1029.station_id = ugrab(12);
1139
0
        rtcm->rtcmtypes.rtcm3_1029.mjd = ugrab(16);
1140
0
        rtcm->rtcmtypes.rtcm3_1029.sod = ugrab(17);
1141
0
        rtcm->rtcmtypes.rtcm3_1029.len = ugrab(7);
1142
0
        n = ugrab(8);
1143
0
        if ((9 + n) != rtcm->length) {
1144
0
            bad_len = 9 + n;
1145
0
            break;
1146
0
        }
1147
0
        rtcm->rtcmtypes.rtcm3_1029.unicode_units = n;
1148
0
        rtcm3_copy_string_field((char *)rtcm->rtcmtypes.rtcm3_1029.text,
1149
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1029.text),
1150
0
                                buf + 12,
1151
0
                                rtcm->rtcmtypes.rtcm3_1029.unicode_units);
1152
0
        unknown = false;
1153
0
        break;
1154
1155
0
    case 1030:
1156
        /* RTCM 3.1
1157
         * GPS Network RTK Residual Message
1158
         */
1159
0
        msg_name = "GPS Network RTK Residual";
1160
0
        if (7 > rtcm->length) {
1161
            // actually 7 + (6.125 * n)
1162
0
            bad_len = 7;
1163
0
            break;
1164
0
        }
1165
0
        break;
1166
1167
0
    case 1031:
1168
        /* RTCM 3.1
1169
         * GLONASS Network RTK Residual Message
1170
         */
1171
0
        msg_name = "GLONASS Network RTK Residual";
1172
0
        if (6 > rtcm->length) {
1173
            // actually 6.625 + (6.125 ( n)
1174
0
            bad_len = 6;
1175
0
            break;
1176
0
        }
1177
0
        break;
1178
1179
0
    case 1032:
1180
        /* RTCM 3.1
1181
         * Physical Reference Station Position message
1182
         */
1183
0
        msg_name = "Physical Reference Station Position";
1184
0
        if (19 > rtcm->length) {
1185
            // actually 19.5
1186
0
            bad_len = 19;
1187
0
            break;
1188
0
        }
1189
0
        break;
1190
1191
0
    case 1033:                  // see note in header
1192
0
        msg_name ="Receiver and Antenna Descriptor";
1193
        /* Type1033 is a combined Message Types 1007 and 1008
1194
         * and hence contains antenna descriptor and serial number
1195
         * as well as receiver descriptor and serial number.
1196
         */
1197
        // TODO: rtklib has C code for this one.
1198
0
        rtcm->rtcmtypes.rtcm3_1033.station_id = ugrab(12);
1199
0
        n = ugrab(8);
1200
0
        if ((28 + n) > rtcm->length) {
1201
0
            bad_len = 28 + n;
1202
0
            break;
1203
0
        }
1204
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1033.descriptor,
1205
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1033.descriptor),
1206
0
                                buf + 7, n);
1207
0
        bitcount += 8 * n;
1208
0
        rtcm->rtcmtypes.rtcm3_1033.setup_id = ugrab(8);
1209
0
        n2 = ugrab(8);
1210
1211
0
        if ((8 + n + n2) > rtcm->length) {
1212
0
            bad_len = 8 + n + n2;
1213
0
            break;
1214
0
        }
1215
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1033.serial,
1216
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1033.serial),
1217
0
                                buf + 9 + n, n2);
1218
0
        bitcount += 8 * n2;
1219
0
        n3 = ugrab(8);
1220
0
        if ((8 + n + n2 + n3) > rtcm->length) {
1221
0
            bad_len = 9 + n + n2 + n3;
1222
0
            break;
1223
0
        }
1224
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1033.receiver,
1225
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1033.receiver),
1226
0
                                buf + 10 + n + n2, n3);
1227
0
        bitcount += 8 * n3;
1228
0
        n4 = ugrab(8);
1229
1230
0
        if ((9 + n + n2 + n3 + n4) > rtcm->length) {
1231
0
            bad_len = 9 + n + n2 + n3 + n4;
1232
0
            break;
1233
0
        }
1234
0
        rtcm3_copy_string_field(rtcm->rtcmtypes.rtcm3_1033.firmware,
1235
0
                                sizeof(rtcm->rtcmtypes.rtcm3_1033.firmware),
1236
0
                                buf + 11 + n + n2 + n3, n4);
1237
        // bitcount += 8 * n4;
1238
        // TODO: next is receiver serial number
1239
0
        unknown = false;
1240
0
        break;
1241
1242
0
    case 1034:
1243
        /* RTCM 3.2
1244
         * GPS Network FKP Gradient Message
1245
         */
1246
0
        msg_name = "GPS Network FKP Gradient";
1247
0
        if (6 > rtcm->length) {
1248
            // not exactly: 6.125 + (8.25 * n)
1249
0
            bad_len = 6;
1250
0
            break;
1251
0
        }
1252
0
        break;
1253
1254
0
    case 1035:
1255
        /* RTCM 3.2
1256
         * GLONASS Network FKP Gradient Message
1257
         */
1258
0
        msg_name = "GLO Network FKP Gradient";
1259
0
        if (5 > rtcm->length) {
1260
            // not exactly: 5.75 + (8.25 * n)
1261
0
            bad_len = 5;
1262
0
            break;
1263
0
        }
1264
0
        break;
1265
1266
0
    case 1037:
1267
        /* RTCM 3.2
1268
         * GLONASS Ionospheric Correction Differences
1269
         */
1270
0
        msg_name = "GLO Ionospheric Correction Differences";
1271
0
        if (9 > rtcm->length) {
1272
            // not exactly: 9.125 + (3,5 * n)
1273
0
            bad_len = 9;
1274
0
            break;
1275
0
        }
1276
0
        break;
1277
1278
0
    case 1038:
1279
        /* RTCM 3.2
1280
         * GLONASS Geometric Correction Differences
1281
         */
1282
0
        msg_name = "GLO Geometric Correction Differences";
1283
0
        if (9 > rtcm->length) {
1284
            // not exactly: 9.125 + (4,5 * n)
1285
0
            bad_len = 9;
1286
0
            break;
1287
0
        }
1288
0
        break;
1289
1290
0
    case 1039:
1291
        /* RTCM 3.2
1292
         * GLONASS Combined Geometric and Ionospheric Correction Differences
1293
         */
1294
0
        msg_name = "GLONASS Combined Geometric and Ionospheric "
1295
0
                   "Correction Differences";
1296
0
        if (9 > rtcm->length) {
1297
            // not exactly: 9.125 + (6.625 * n)
1298
0
            bad_len = 9;
1299
0
            break;
1300
0
        }
1301
0
        break;
1302
1303
0
    case 1042:
1304
        /* RTCM 3.x - 1043
1305
         * BeiDou Ephemeris
1306
         * length 64
1307
         */
1308
0
        msg_name = "BD Ephemeris";
1309
0
        if (64 != rtcm->length) {
1310
0
            bad_len = 64;
1311
0
            break;
1312
0
        }
1313
0
        break;
1314
1315
0
    case 1043:
1316
        /* RTCM 3.x - 1043
1317
         * SBAS Ephemeris
1318
         * length 29
1319
         */
1320
0
        msg_name = "SBAS Ephemeris";
1321
0
        if (29 != rtcm->length) {
1322
0
            bad_len = 29;
1323
0
            break;
1324
0
        }
1325
0
        break;
1326
1327
0
    case 1044:
1328
        /* RTCM 3.x - 1044
1329
         * QZSS ephemeris
1330
         * length 61
1331
         */
1332
        // TODO: rtklib has C code for this one.
1333
0
        msg_name = "QZSS Ephemeris";
1334
0
        if (61 != rtcm->length) {
1335
0
            bad_len = 61;
1336
0
            break;
1337
0
        }
1338
0
        break;
1339
1340
0
    case 1045:
1341
        /* RTCM 3.2 - 1045
1342
         * Galileo F/NAV Ephemeris Data
1343
         * 62 bytes
1344
         */
1345
        // TODO: rtklib has C code for this one.
1346
0
        msg_name = "GAL F/NAV Ephemeris Data";
1347
0
        if (62 != rtcm->length) {
1348
0
            bad_len = 62;
1349
0
            break;
1350
0
        }
1351
0
        break;
1352
1353
0
    case 1046:
1354
        /* RTCM 3.x - 1046
1355
         * Galileo I/NAV Ephemeris Data
1356
         * length 63
1357
         */
1358
        // TODO: rtklib has C code for this one.
1359
0
        msg_name = "GAL I/NAV Ephemeris Data";
1360
0
        if (63 != rtcm->length) {
1361
0
            bad_len = 63;
1362
0
            break;
1363
0
        }
1364
0
        break;
1365
1366
0
    case 1057:
1367
        /* RTCM 3.2
1368
         * SSR GPS Orbit Correction
1369
         */
1370
0
        msg_name = "SSR GPS Orbit Correction";
1371
0
        if (8 > rtcm->length) {
1372
            // not exactly: 8.5 + (6.875 * n)
1373
0
            bad_len = 8;
1374
0
            break;
1375
0
        }
1376
0
        break;
1377
1378
0
    case 1058:
1379
        /* RTCM 3.2
1380
         * SSR GPS Clock Correction
1381
         */
1382
0
        msg_name = "SSR GPS Clock Correction";
1383
0
        if (8 > rtcm->length) {
1384
            // not exactly: 8.375 + (9.5 * n)
1385
0
            bad_len = 8;
1386
0
            break;
1387
0
        }
1388
0
        break;
1389
1390
0
    case 1059:
1391
        /* RTCM 3.2
1392
         * SSR GPS Code Bias
1393
         */
1394
0
        msg_name = "SSR GPS Code Bias";
1395
0
        if (8 > rtcm->length) {
1396
            // not exactly: 8.375 + (1.375 * n) + (2.375 * ??)
1397
0
            bad_len = 8;
1398
0
            break;
1399
0
        }
1400
0
        break;
1401
1402
0
    case 1060:
1403
        /* RTCM 3.2
1404
         * SSR GPS Combined Orbit and Clock Correction
1405
         */
1406
0
        msg_name = "SSR GPS Combined Orbit and Clock Correction";
1407
0
        if (8 > rtcm->length) {
1408
            // not exactly: 8.5 + (25.625 * n)
1409
0
            bad_len = 8;
1410
0
            break;
1411
0
        }
1412
0
        break;
1413
1414
0
    case 1061:
1415
        /* RTCM 3.2
1416
         * SSR GPS URA
1417
         */
1418
0
        msg_name = "SSR GPS URA";
1419
0
        if (8 > rtcm->length) {
1420
            // not exactly: 8.375 + (1.5 * n)
1421
0
            bad_len = 8;
1422
0
            break;
1423
0
        }
1424
0
        break;
1425
1426
0
    case 1062:
1427
        /* RTCM 3.2
1428
         * SSR GPS High Rate Clock Correction
1429
         */
1430
0
        msg_name = "SSR GPS High Rate Clock Correction";
1431
0
        if (8 > rtcm->length) {
1432
            // not exactly: 8.375 + (3.5 * n)
1433
0
            bad_len = 8;
1434
0
            break;
1435
0
        }
1436
0
        break;
1437
1438
0
    case 1063:
1439
        /* RTCM 3.2
1440
         * SSR GLO Orbit Correction
1441
         */
1442
0
        msg_name = "SSR GLO Orbit Correction";
1443
0
        if (8 > rtcm->length) {
1444
            // not exactly: 8.125 + (16.75 * n)
1445
0
            bad_len = 8;
1446
0
            break;
1447
0
        }
1448
0
        break;
1449
1450
0
    case 1064:
1451
        /* RTCM 3.2
1452
         * SSR GLO Clock Correction
1453
         */
1454
0
        msg_name = "SSR GLO Clock Correction";
1455
0
        if (8 > rtcm->length) {
1456
            // not exactly: 8.125 + (9.37 * n)
1457
0
            bad_len = 8;
1458
0
            break;
1459
0
        }
1460
0
        break;
1461
1462
0
    case 1065:
1463
        /* RTCM 3.2
1464
         * SSR GLO Code Correction
1465
         */
1466
0
        msg_name = "SSR GLO ode Correction";
1467
0
        if (8 > rtcm->length) {
1468
            // not exactly: 8 + (1.25 * n) + (2.375 * ??)
1469
0
            bad_len = 8;
1470
0
            break;
1471
0
        }
1472
0
        break;
1473
1474
0
    case 1066:
1475
        /* RTCM 3.2
1476
         * SSR GLO Combined Orbit and Clock Correction
1477
         */
1478
0
        msg_name = "SSR GLO Combined Orbit and Clock Correction";
1479
0
        if (8 > rtcm->length) {
1480
            // not exactly: 8.125 + (25.5 * n)
1481
0
            bad_len = 8;
1482
0
            break;
1483
0
        }
1484
0
        break;
1485
1486
0
    case 1067:
1487
        /* RTCM 3.2
1488
         * SSR GLO URA
1489
         */
1490
0
        msg_name = "SSR GLO URA";
1491
0
        if (8 > rtcm->length) {
1492
            // not exactly: 8 + (1.375 * n)
1493
0
            bad_len = 8;
1494
0
            break;
1495
0
        }
1496
0
        break;
1497
1498
0
    case 1068:
1499
        /* RTCM 3.2
1500
         * SSR GPS High Rate Clock Correction
1501
         */
1502
0
        msg_name = "SSR GLO High Rate Clock Correction";
1503
0
        if (8 > rtcm->length) {
1504
            // not exactly: 8 + (3.375 * n)
1505
0
            bad_len = 8;
1506
0
            break;
1507
0
        }
1508
0
        break;
1509
1510
0
    case 1070:
1511
        /* RTCM 3.x
1512
         * Reserved for MSM
1513
         */
1514
0
        msg_name = "Reserved for MSM";
1515
0
        break;
1516
1517
0
    case 1071:
1518
        // RTCM 3.2
1519
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1520
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 1;
1521
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1522
0
        msg_name = "GPS MSM 1";
1523
0
        break;
1524
1525
0
    case 1072:
1526
        // RTCM 3.2
1527
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1528
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 2;
1529
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1530
0
        msg_name = "GPS MSM 2";
1531
0
        break;
1532
1533
0
    case 1073:
1534
        /* RTCM 3.2
1535
         * GPS Multi Signal Message 3
1536
         */
1537
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1538
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 3;
1539
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1540
0
        msg_name = "GPS MSM 3";
1541
0
        break;
1542
1543
0
    case 1074:
1544
        /* RTCM 3.2
1545
         * GPS Multi Signal Message 4
1546
         */
1547
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1548
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 4;
1549
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1550
0
        msg_name = "GPS MSM 4";
1551
0
        break;
1552
1553
0
    case 1075:
1554
        /* RTCM 3.2
1555
         * GPS Multi Signal Message 5
1556
         */
1557
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1558
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 5;
1559
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1560
0
        msg_name = "GPS MSM 5";
1561
0
        break;
1562
1563
0
    case 1076:
1564
        /* RTCM 3.2
1565
         * GPS Multi Signal Message 6
1566
         */
1567
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1568
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 6;
1569
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1570
0
        msg_name = "GPS MSM 6";
1571
0
        break;
1572
1573
0
    case 1077:
1574
        /* RTCM 3.2 - 1077
1575
         * GPS Multi Signal Message 7
1576
         * Full GPS pseudo-ranges, carrier phases, Doppler and
1577
         * signal strength (high resolution)
1578
         * min length 438
1579
         */
1580
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GPS;
1581
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 7;
1582
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1583
0
        msg_name = "GPS MSM7";
1584
0
        break;
1585
1586
0
    case 1078:
1587
        /* RTCM 3.x
1588
         * Reserved for MSM
1589
         */
1590
0
        msg_name = "Reserved for MSM";
1591
0
        break;
1592
1593
0
    case 1079:
1594
        /* RTCM 3.x
1595
         * Reserved for MSM
1596
         */
1597
0
        msg_name = "Reserved for MSM";
1598
0
        break;
1599
1600
0
    case 1080:
1601
        /* RTCM 3.x
1602
         * Reserved for MSM
1603
         */
1604
0
        msg_name = "Reserved for MSM";
1605
0
        break;
1606
1607
0
    case 1081:
1608
        /* RTCM 3.2
1609
         * GLONASS Multi Signal Message 1
1610
         */
1611
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1612
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 1;
1613
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1614
0
        msg_name = "GLO MSM 1";
1615
0
        break;
1616
1617
0
    case 1082:
1618
        /* RTCM 3.2
1619
         * GLONASS Multi Signal Message 2
1620
         */
1621
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1622
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 2;
1623
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1624
0
        msg_name = "GLO MSM 2";
1625
0
        break;
1626
1627
0
    case 1083:
1628
        /* RTCM 3.2
1629
         * GLONASS Multi Signal Message 4
1630
         */
1631
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1632
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 3;
1633
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1634
0
        msg_name = "GLO MSM 3";
1635
0
        break;
1636
1637
0
    case 1084:
1638
        /* RTCM 3.2
1639
         * GLONASS Multi Signal Message 4
1640
         */
1641
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1642
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 4;
1643
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1644
0
        msg_name = "GLO MSM 4";
1645
0
        break;
1646
1647
0
    case 1085:
1648
        /* RTCM 3.2
1649
         * GLONASS Multi Signal Message 5
1650
         */
1651
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1652
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 5;
1653
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1654
0
        msg_name = "GLO MSM 5";
1655
0
        break;
1656
1657
0
    case 1086:
1658
        /* RTCM 3.2
1659
         * GLONASS Multi Signal Message 6
1660
         */
1661
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1662
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 6;
1663
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1664
0
        msg_name = "GLO MSM 6";
1665
0
        break;
1666
1667
0
    case 1087:
1668
        /* RTCM 3.2 - 1087
1669
         * GLONASS Multi Signal Message 7
1670
         * Full GLONASS pseudo-ranges, carrier phases, Doppler and
1671
         * signal strength (high resolution)
1672
         * length 417 or 427
1673
         */
1674
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GLO;
1675
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 7;
1676
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1677
0
        msg_name = "GLO MSM 7";
1678
0
        break;
1679
1680
0
    case 1088:
1681
        /* RTCM 3.x
1682
         * Reserved for MSM
1683
         */
1684
0
        msg_name = "Reserved for MSM";
1685
0
        break;
1686
1687
0
    case 1089:
1688
        /* RTCM 3.x
1689
         * Reserved for MSM
1690
         */
1691
0
        msg_name = "Reserved for MSM";
1692
0
        break;
1693
1694
0
    case 1090:
1695
        /* RTCM 3.x
1696
         * Reserved for MSM
1697
         */
1698
0
        msg_name = "Reserved for MSM";
1699
0
        break;
1700
1701
0
    case 1091:
1702
        /* RTCM 3.2
1703
         * Galileo Multi Signal Message 1
1704
         */
1705
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1706
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 1;
1707
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1708
0
        msg_name = "GAL MSM 1";
1709
0
        break;
1710
1711
0
    case 1092:
1712
        /* RTCM 3.2
1713
         * Galileo Multi Signal Message 2
1714
         */
1715
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1716
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 2;
1717
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1718
0
        msg_name = "GAL MSM 2";
1719
0
        break;
1720
1721
0
    case 1093:
1722
        /* RTCM 3.2
1723
         * Galileo Multi Signal Message 3
1724
         */
1725
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1726
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 3;
1727
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1728
0
        msg_name = "GAL MSM 3";
1729
0
        break;
1730
1731
0
    case 1094:
1732
        /* RTCM 3.2
1733
         * Galileo Multi Signal Message 4
1734
         */
1735
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1736
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 4;
1737
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1738
0
        msg_name = "GAL MSM 4";
1739
0
        break;
1740
1741
0
    case 1095:
1742
        /* RTCM 3.2
1743
         * Galileo Multi Signal Message 5
1744
         */
1745
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1746
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 5;
1747
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1748
0
        msg_name = "GAL MSM 5";
1749
0
        break;
1750
1751
0
    case 1096:
1752
        /* RTCM 3.2
1753
         * Galileo Multi Signal Message 6
1754
         */
1755
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1756
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 6;
1757
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1758
0
        msg_name = "GAL MSM 6";
1759
0
        break;
1760
1761
0
    case 1097:
1762
        /* RTCM 3.2 - 1097
1763
         * Galileo Multi Signal Message 7
1764
         * Full Galileo pseudo-ranges, carrier phases, Doppler and
1765
         * signal strength (high resolution)
1766
         * length 96
1767
         */
1768
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_GAL;
1769
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 7;
1770
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1771
0
        msg_name = "GAL MSM 7";
1772
0
        break;
1773
1774
0
    case 1098:
1775
        /* RTCM 3.x
1776
         * Reserved for MSM
1777
         */
1778
0
        msg_name = "Reserved for MSM";
1779
0
        break;
1780
1781
0
    case 1099:
1782
        /* RTCM 3.x
1783
         * Reserved for MSM
1784
         */
1785
0
        msg_name = "Reserved for MSM";
1786
0
        break;
1787
1788
0
    case 1100:
1789
        /* RTCM 3.x
1790
         * Reserved for MSM
1791
         */
1792
0
        msg_name = "Reserved for MSM";
1793
0
        break;
1794
1795
0
    case 1101:
1796
        /* RTCM 3.3
1797
         * SBAS Multi Signal Message 1
1798
         */
1799
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1800
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 1;
1801
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1802
0
        msg_name = "SBAS MSM 1";
1803
0
        break;
1804
1805
0
    case 1102:
1806
        /* RTCM 3.3
1807
         * SBAS Multi Signal Message 2
1808
         */
1809
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1810
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 2;
1811
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1812
0
        msg_name = "SBAS MSM 2";
1813
0
        break;
1814
1815
0
    case 1103:
1816
        /* RTCM 3.3
1817
         * SBAS Multi Signal Message 3
1818
         */
1819
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1820
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 3;
1821
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1822
0
        msg_name = "SBAS MSM 3";
1823
0
        break;
1824
1825
0
    case 1104:
1826
        /* RTCM 3.3
1827
         * SBAS Multi Signal Message 4
1828
         */
1829
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1830
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 4;
1831
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1832
0
        msg_name = "SBAS MSM 4";
1833
0
        break;
1834
1835
0
    case 1105:
1836
        /* RTCM 3.3
1837
         * SBAS Multi Signal Message 5
1838
         */
1839
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1840
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 5;
1841
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1842
0
        msg_name = "SBAS MSM 5";
1843
0
        break;
1844
1845
0
    case 1106:
1846
        /* RTCM 3.3
1847
         * SBAS Multi Signal Message 6
1848
         */
1849
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1850
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 6;
1851
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1852
0
        msg_name = "SBAS MSM 6";
1853
0
        break;
1854
1855
0
    case 1107:
1856
        /* RTCM 3.3 - 1107
1857
         * 'Multiple Signal Message
1858
         * Full SBAS pseudo-ranges, carrier phases, Doppler and
1859
         * signal strength (high resolution)
1860
         * length 96
1861
         */
1862
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_SBAS;
1863
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 7;
1864
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1865
0
        msg_name = "SBAS MSM 7";
1866
0
        break;
1867
1868
0
    case 1108:
1869
        /* RTCM 3.x
1870
         * Reserved for MSM
1871
         */
1872
0
        msg_name = "Reserved for MSM";
1873
0
        break;
1874
1875
0
    case 1109:
1876
        /* RTCM 3.x
1877
         * Reserved for MSM
1878
         */
1879
0
        msg_name = "Reserved for MSM";
1880
0
        break;
1881
1882
0
    case 1110:
1883
        /* RTCM 3.x
1884
         * Reserved for MSM
1885
         */
1886
0
        msg_name = "Reserved for MSM";
1887
0
        break;
1888
1889
0
    case 1111:
1890
        /* RTCM 3.3
1891
         * QZSS Multi Signal Message 1
1892
         */
1893
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1894
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 1;
1895
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1896
0
        msg_name = "QZSS MSM 1";
1897
0
        break;
1898
1899
0
    case 1112:
1900
        /* RTCM 3.3
1901
         * QZSS Multi Signal Message 2
1902
         */
1903
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1904
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 2;
1905
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1906
0
        msg_name = "QZSS MSM 2";
1907
0
        break;
1908
1909
0
    case 1113:
1910
        /* RTCM 3.3
1911
         * QZSS Multi Signal Message 3
1912
         */
1913
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1914
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 3;
1915
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1916
0
        msg_name = "QZSS MSM 3";
1917
0
        break;
1918
1919
0
    case 1114:
1920
        /* RTCM 3.3
1921
         * QZSS Multi Signal Message 4
1922
         */
1923
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1924
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 4;
1925
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1926
0
        msg_name = "QZSS MSM 4";
1927
0
        break;
1928
1929
0
    case 1115:
1930
        /* RTCM 3.3
1931
         * QZSS Multi Signal Message 5
1932
         */
1933
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1934
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 5;
1935
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1936
0
        msg_name = "QZSS MSM 5";
1937
0
        break;
1938
1939
0
    case 1116:
1940
        /* RTCM 3.3
1941
         * QZSS Multi Signal Message 6
1942
         */
1943
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1944
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 6;
1945
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1946
0
        msg_name = "QZSS MSM 6";
1947
0
        break;
1948
1949
0
    case 1117:
1950
        /* RTCM 3.3
1951
         * QZSS Multi Signal Message 7
1952
         */
1953
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_QZSS;
1954
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 7;
1955
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1956
0
        msg_name = "QZSS MSM 7";
1957
0
        break;
1958
1959
0
    case 1118:
1960
        /* RTCM 3.x
1961
         * Reserved for MSM
1962
         */
1963
0
        msg_name = "Reserved for MSM";
1964
0
        break;
1965
1966
0
    case 1119:
1967
        /* RTCM 3.x
1968
         * Reserved for MSM
1969
         */
1970
0
        msg_name = "Reserved for MSM";
1971
0
        break;
1972
1973
0
    case 1120:
1974
        /* RTCM 3.x
1975
         * Reserved for MSM
1976
         */
1977
0
        msg_name = "Reserved for MSM";
1978
0
        break;
1979
1980
0
    case 1121:
1981
        /* RTCM 3.2 A.1
1982
         * BD Multi Signal Message 1
1983
         */
1984
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
1985
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 1;
1986
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1987
0
        msg_name = "BD MSM 1";
1988
0
        break;
1989
1990
0
    case 1122:
1991
        /* RTCM 3.2 A.1
1992
         * BD Multi Signal Message 2
1993
         */
1994
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
1995
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 2;
1996
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
1997
0
        msg_name = "BD MSM 2";
1998
0
        break;
1999
2000
0
    case 1123:
2001
        /* RTCM 3.2 A.1
2002
         * BD Multi Signal Message 3
2003
         */
2004
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
2005
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 3;
2006
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
2007
0
        msg_name = "BD MSM 3";
2008
0
        break;
2009
2010
0
    case 1124:
2011
        /* RTCM 3.2 A.1
2012
         * BD Multi Signal Message 4
2013
         */
2014
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
2015
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 4;
2016
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
2017
0
        msg_name = "BD MSM 4";
2018
0
        break;
2019
2020
0
    case 1125:
2021
        /* RTCM 3.2 A.1
2022
         * BeiDou Multi Signal Message 5
2023
         */
2024
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
2025
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 5;
2026
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
2027
0
        msg_name = "BD MSM 5";
2028
0
        break;
2029
2030
0
    case 1126:
2031
        /* RTCM 3.2 A.1
2032
         * BeiDou Multi Signal Message 6
2033
         */
2034
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
2035
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 6;
2036
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
2037
0
        msg_name = "BD MSM 6";
2038
0
        break;
2039
2040
0
    case 1127:
2041
        /* RTCM 3.2 A.1
2042
         * BeiDou Multi Signal Message 7
2043
         */
2044
0
        rtcm->rtcmtypes.rtcm3_msm.gnssid = GNSSID_BD;
2045
0
        rtcm->rtcmtypes.rtcm3_msm.msm = 7;
2046
0
        unknown = rtcm3_decode_msm(context, rtcm, buf);
2047
0
        msg_name = "BD MSM 7";
2048
0
        break;
2049
2050
0
    case 1128:
2051
        /* RTCM 3.x
2052
         * Reserved for MSM
2053
         */
2054
0
        msg_name = "Reserved for MSM";
2055
0
        break;
2056
2057
0
    case 1229:
2058
        /* RTCM 3.x
2059
         * Reserved for MSM
2060
         */
2061
0
        msg_name = "Reserved for MSM";
2062
0
        break;
2063
2064
0
    case 1230:
2065
        /* RTCM 3.2
2066
         * GLONASS L1 and L2, C/A and P, Code-Phase Biases.
2067
         */
2068
0
        msg_name = "GLO L1 and L2 Code-Phase Biases";
2069
0
        unknown = false;
2070
0
        if (4 > rtcm->length) {
2071
            // not exactly: 32 + (16 * n), n <= 4
2072
            // but we see 12 and even 4??
2073
0
            bad_len = 4;
2074
0
            break;
2075
0
        }
2076
0
        rtcm->rtcmtypes.rtcm3_1230.station_id = ugrab(12);
2077
0
        rtcm->rtcmtypes.rtcm3_1230.bias_indicator = ugrab(1);
2078
0
        (void)ugrab(1);         // reserved
2079
0
        rtcm->rtcmtypes.rtcm3_1230.signals_mask = ugrab(3);
2080
        // actual mask order is undocumented...
2081
0
        if (1 & rtcm->rtcmtypes.rtcm3_1230.signals_mask) {
2082
0
            rtcm->rtcmtypes.rtcm3_1230.l1_ca_bias = ugrab(16);
2083
0
        }
2084
0
        if (2 & rtcm->rtcmtypes.rtcm3_1230.signals_mask) {
2085
0
            rtcm->rtcmtypes.rtcm3_1230.l1_p_bias = ugrab(16);
2086
0
        }
2087
0
        if (4 & rtcm->rtcmtypes.rtcm3_1230.signals_mask) {
2088
0
            rtcm->rtcmtypes.rtcm3_1230.l2_ca_bias = ugrab(16);
2089
0
        }
2090
0
        if (8 & rtcm->rtcmtypes.rtcm3_1230.signals_mask) {
2091
0
            rtcm->rtcmtypes.rtcm3_1230.l2_p_bias = ugrab(16);
2092
0
        }
2093
0
        break;
2094
2095
    // Message Types 4001 — 4060 Are Reserved
2096
2097
0
    case 4062:
2098
        /* RTCM 3.3
2099
         * Geely Proprietary
2100
         */
2101
0
        msg_name = "Geely Proprietary";
2102
0
        break;
2103
2104
0
    case 4063:
2105
        /* RTCM 3.3
2106
         * CHC Navigation (CHCNAV) Proprietary
2107
         */
2108
0
        msg_name = "CHC Navigation (CHCNAV) Proprietary";
2109
0
        break;
2110
2111
0
    case 4064:
2112
        /* RTCM 3.3
2113
         * NTLab Proprietary
2114
         */
2115
0
        msg_name = "NTLab Proprietary";
2116
0
        break;
2117
2118
0
    case 4065:
2119
        /* RTCM 3.3
2120
         * Allystar Technology (Shenzhen) Co. Ltd. Proprietary
2121
         *
2122
         * sub-id:
2123
         *    0    Reference stations PVT, requires MSM7.
2124
         *    1    Navigation PVT Solyution
2125
         *    2    Attitude Determination (multi antenna)
2126
         *    0x0a Raw Sensor measurements.
2127
         */
2128
0
        msg_name = "Allystar Technology (Shenzhen) Co. Ltd. Proprietary";
2129
0
        break;
2130
2131
0
    case 4066:
2132
        /* RTCM 3.3
2133
         * Lantmateriet Proprietary
2134
         */
2135
0
        msg_name = "Lantmateriet Proprietary";
2136
0
        break;
2137
2138
0
    case 4067:
2139
        /* RTCM 3.x
2140
         * China Transport telecommunications & Information Center Proprietary
2141
         */
2142
0
        msg_name = "China Transport telecommunications & Information "
2143
0
                       "Center Proprietary";
2144
0
        break;
2145
2146
0
    case 4068:
2147
        /* RTCM 3.3
2148
         * Qianxun Location Networks Co. Ltd Proprietary
2149
         */
2150
0
        msg_name = "Qianxun Location Networks Co. Ltd Proprietary";
2151
0
        break;
2152
2153
0
    case 4069:
2154
        /* RTCM 3.3
2155
         * VERIPOS Ltd Proprietary
2156
         */
2157
0
        msg_name = "VERIPOS Ltd Proprietary";
2158
0
        break;
2159
2160
0
    case 4070:
2161
        /* RTCM 3.3
2162
         * Wuhan MengXin Technology
2163
         */
2164
0
        msg_name = "Wuhan MengXin Technology Proprietary";
2165
0
        break;
2166
2167
0
    case 4071:
2168
        /* RTCM 3.3
2169
         * Wuhan Navigation and LBS
2170
         */
2171
0
        msg_name = "Wuhan Navigation and LBS Proprietary";
2172
0
        break;
2173
2174
0
    case 4072:
2175
        /* RTCM 3.x
2176
         * u-blox Proprietary
2177
         * Mitsubishi Electric Corp Proprietary
2178
         * 4072.0 Reference station PVT (u-blox proprietary)
2179
         * 4072.1 Additional reference station information (u-blox proprietary)
2180
         */
2181
0
        msg_name = "u-blox Proprietary";
2182
0
        break;
2183
2184
0
    case 4073:
2185
        /* RTCM 3.x
2186
         * Unicore Communications Proprietary
2187
         */
2188
0
        msg_name = "Alberding GmbH Proprietary";
2189
0
        break;
2190
2191
0
    case 4075:
2192
        /* RTCM 3.x
2193
         * Alberding GmbH Proprietary
2194
         */
2195
0
        msg_name = "Alberding GmbH Proprietary";
2196
0
        break;
2197
2198
0
    case 4076:
2199
        /* RTCM 3.x
2200
         * International GNSS Service Proprietary, www.igs.org
2201
         */
2202
0
        if (22 > rtcm->length) {
2203
            // need 76 bits, 9.5 bytes
2204
0
            bad_len = 22;
2205
0
            break;
2206
0
        }
2207
2208
0
        msg_name = "International GNSS Service Proprietary";
2209
0
        unknown = rtcm3_4076(context, rtcm, buf);
2210
0
        break;
2211
2212
0
    case 4077:
2213
        /* RTCM 3.x
2214
         * Hemisphere GNSS Proprietary
2215
         */
2216
0
        msg_name = "Hemisphere GNSS Proprietary";
2217
0
        break;
2218
2219
0
    case 4078:
2220
        /* RTCM 3.x
2221
         * ComNav Technology Proprietary
2222
         */
2223
0
        msg_name = "ComNav Technology Proprietary";
2224
0
        break;
2225
2226
0
    case 4079:
2227
        /* RTCM 3.x
2228
         * SubCarrier Systems Corp Proprietary
2229
         */
2230
0
        msg_name = "SubCarrier Systems Corp Proprietary";
2231
0
        break;
2232
2233
0
    case 4080:
2234
        /* RTCM 3.x
2235
         * NavCom Technology, Inc.
2236
         */
2237
0
        msg_name = "NavCom Technology, Inc.";
2238
0
        break;
2239
2240
0
    case 4081:
2241
        /* RTCM 3.x
2242
         * Seoul National Universtiry GNSS Lab Proprietary
2243
         */
2244
0
        msg_name = "Seoul National Universtiry GNSS Lab Proprietery";
2245
0
        break;
2246
2247
0
    case 4082:
2248
        /* RTCM 3.x
2249
         * Cooperative Research Centre for Spatial Information Proprietary
2250
         */
2251
0
        msg_name = "Cooperative Research Centre for Spatial Information "
2252
0
                       "Proprietary";
2253
0
        break;
2254
2255
0
    case 4083:
2256
        /* RTCM 3.x
2257
         * German Aerospace Center Proprietary
2258
         */
2259
0
        msg_name = "German Aerospace Center Proprietary";
2260
0
        break;
2261
2262
0
    case 4084:
2263
        /* RTCM 3.x
2264
         * Geodetics Inc Proprietary
2265
         */
2266
0
        msg_name = "Geodetics Inc Proprietary";
2267
0
        break;
2268
2269
0
    case 4085:
2270
        /* RTCM 3.x
2271
         * European GNSS Supervisory Authority Proprietary
2272
         */
2273
0
        msg_name = "European GNSS Supervisory Authority Proprietary";
2274
0
        break;
2275
2276
0
    case 4086:
2277
        /* RTCM 3.x
2278
         * InPosition GmbH Proprietary
2279
         */
2280
0
        msg_name = "InPosition GmbH Proprietary";
2281
0
        break;
2282
2283
0
    case 4087:
2284
        /* RTCM 3.x
2285
         * Fugro Proprietary
2286
         */
2287
0
        msg_name = "Fugro Proprietary";
2288
0
        break;
2289
2290
0
    case 4088:
2291
        /* RTCM 3.x
2292
         * IfEN GmbH Proprietary
2293
         */
2294
0
        msg_name = "IfEN GmbH Proprietary";
2295
0
        break;
2296
2297
0
    case 4089:
2298
        /* RTCM 3.x
2299
         * Septentrio Satellite Navigation Proprietary
2300
         */
2301
0
        msg_name = "Septentrio Satellite Navigation Proprietary";
2302
0
        break;
2303
2304
0
    case 4090:
2305
        /* RTCM 3.x
2306
         * Geo++ Proprietary
2307
         */
2308
0
        msg_name = "Geo++ Proprietary";
2309
0
        break;
2310
2311
0
    case 4091:
2312
        /* RTCM 3.x
2313
         * Topcon Positioning Systems Proprietary
2314
         */
2315
0
        msg_name = "Topcon Positioning Systems Proprietary";
2316
0
        break;
2317
2318
0
    case 4092:
2319
        /* RTCM 3.x
2320
         * Leica Geosystems Proprietary
2321
         */
2322
0
        msg_name = "Leica Geosystems Proprietary";
2323
0
        break;
2324
2325
0
    case 4093:
2326
        /* RTCM 3.x
2327
         * NovAtel Proprietary
2328
         */
2329
0
        msg_name = "NovAtel Pr.orietary";
2330
0
        break;
2331
2332
0
    case 4094:
2333
        /* RTCM 3.x
2334
         * Trimble Proprietary
2335
         */
2336
0
        msg_name = "Trimble Proprietary";
2337
0
        break;
2338
2339
0
    case 4095:
2340
        /* RTCM 3.x
2341
         * Ashtech/Magellan Proprietary
2342
         */
2343
0
        msg_name = "Ashtech/Magellan Proprietary";
2344
0
        break;
2345
2346
0
    case 63:
2347
        // dunno what this is.  Length 64
2348
0
        FALLTHROUGH
2349
0
    default:
2350
0
        break;
2351
0
    }
2352
0
#undef RANGEDIFF
2353
0
#undef GPS_PSEUDORANGE
2354
0
#undef sgrab
2355
0
#undef ugrab
2356
0
    if (0 != bad_len) {
2357
0
            GPSD_LOG(LOG_WARN, &context->errout,
2358
0
                     "RTCM3: type %d (%s) bad n %u length %u s/b %u\n",
2359
0
                     rtcm->type,  msg_name, n, rtcm->length, bad_len);
2360
0
            rtcm->length = 0;          // set to zero to prevent JSON decode
2361
0
            return;
2362
0
    }
2363
0
    if (unknown) {
2364
        /*
2365
         * Leader bytes, message length, and checksum won't be copied.
2366
         * The first 12 bits of the copied payload will be the type field.
2367
         */
2368
0
        rtcm3_copy_string_field((char *)rtcm->rtcmtypes.data,
2369
0
                                sizeof(rtcm->rtcmtypes.data),
2370
0
                                buf + 3, rtcm->length);
2371
0
        GPSD_LOG(LOG_PROG, &context->errout,
2372
0
                 "RTCM3: %d (%s) Undecoded, length %u\n",
2373
0
                 rtcm->type, msg_name, rtcm->length);
2374
0
    } else {
2375
0
        GPSD_LOG(LOG_PROG, &context->errout,
2376
0
                 "RTCM3: %d (%s) length %u\n",
2377
0
                 rtcm->type, msg_name, rtcm->length);
2378
0
    }
2379
0
}
2380
2381
// *INDENT-ON*
2382
2383
// vim: set expandtab shiftwidth=4