source: ntrip/trunk/BNC/bncpppclient.cpp@ 2540

Last change on this file since 2540 was 2505, checked in by mervart, 15 years ago
File size: 13.8 KB
RevLine 
[2035]1// Part of BNC, a utility for retrieving decoding and
2// converting GNSS data streams from NTRIP broadcasters.
3//
4// Copyright (C) 2007
5// German Federal Agency for Cartography and Geodesy (BKG)
6// http://www.bkg.bund.de
7// Czech Technical University Prague, Department of Geodesy
8// http://www.fsv.cvut.cz
9//
10// Email: euref-ip@bkg.bund.de
11//
12// This program is free software; you can redistribute it and/or
13// modify it under the terms of the GNU General Public License
14// as published by the Free Software Foundation, version 2.
15//
16// This program is distributed in the hope that it will be useful,
17// but WITHOUT ANY WARRANTY; without even the implied warranty of
18// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19// GNU General Public License for more details.
20//
21// You should have received a copy of the GNU General Public License
22// along with this program; if not, write to the Free Software
23// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25/* -------------------------------------------------------------------------
26 * BKG NTRIP Client
27 * -------------------------------------------------------------------------
28 *
29 * Class: bncPPPclient
30 *
31 * Purpose: Precise Point Positioning
32 *
33 * Author: L. Mervart
34 *
35 * Created: 21-Nov-2009
36 *
37 * Changes:
38 *
39 * -----------------------------------------------------------------------*/
40
[2249]41#include <newmatio.h>
[2238]42#include <iomanip>
43
[2035]44#include "bncpppclient.h"
[2113]45#include "bncapp.h"
[2043]46#include "bncutils.h"
[2044]47#include "bncconst.h"
[2058]48#include "bncmodel.h"
[2226]49#include "bncsettings.h"
[2035]50
51extern "C" {
52#include "clock_orbit_rtcm.h"
53}
54
55using namespace std;
56
57// Constructor
58////////////////////////////////////////////////////////////////////////////
59bncPPPclient::bncPPPclient(QByteArray staID) {
[2113]60
[2226]61 bncSettings settings;
62
63 if ( Qt::CheckState(settings.value("pppGLONASS").toInt()) == Qt::Checked) {
64 _useGlonass = true;
65 }
66 else {
67 _useGlonass = false;
68 }
69
[2357]70 if (settings.value("pppSPP").toString() == "PPP") {
71 _pppMode = true;
72 }
73 else {
74 _pppMode = false;
75 }
76
[2113]77 connect(((bncApp*)qApp), SIGNAL(newEphGPS(gpsephemeris)),
78 this, SLOT(slotNewEphGPS(gpsephemeris)));
[2224]79 connect(((bncApp*)qApp), SIGNAL(newEphGlonass(glonassephemeris)),
80 this, SLOT(slotNewEphGlonass(glonassephemeris)));
[2113]81 connect(((bncApp*)qApp), SIGNAL(newCorrections(QList<QString>)),
82 this, SLOT(slotNewCorrections(QList<QString>)));
83
[2039]84 _staID = staID;
85 _epoData = 0;
[2113]86 _model = new bncModel(staID);
[2182]87 connect(_model, SIGNAL(newNMEAstr(QByteArray)),
88 this, SIGNAL(newNMEAstr(QByteArray)));
[2035]89}
90
91// Destructor
92////////////////////////////////////////////////////////////////////////////
93bncPPPclient::~bncPPPclient() {
[2058]94 delete _model;
[2039]95 delete _epoData;
[2035]96 QMapIterator<QString, t_eph*> it(_eph);
97 while (it.hasNext()) {
98 it.next();
99 delete it.value();
100 }
101 QMapIterator<QString, t_corr*> ic(_corr);
102 while (ic.hasNext()) {
103 ic.next();
104 delete ic.value();
105 }
[2274]106 QMapIterator<QString, t_bias*> ib(_bias);
107 while (ib.hasNext()) {
108 ib.next();
109 delete ib.value();
110 }
[2035]111}
112
113//
114////////////////////////////////////////////////////////////////////////////
115void bncPPPclient::putNewObs(p_obs pp) {
[2037]116 QMutexLocker locker(&_mutex);
[2051]117
[2037]118 t_obsInternal* obs = &(pp->_o);
[2051]119
[2226]120 if (obs->satSys != 'G' && !_useGlonass) {
[2225]121 return;
122 }
123
[2051]124 t_satData* satData = new t_satData();
125
[2274]126 // Satellite Number
127 // ----------------
128 if (obs->satSys == 'G') {
129 QString prn = QString("G%1").arg(obs->satNum, 2, 10, QChar('0'));
130 satData->prn = prn;
131 }
132 else if (obs->satSys == 'R') {
133 QString prn = QString("R%1").arg(obs->satNum, 2, 10, QChar('0'));
134 satData->prn = prn;
135 }
136
[2505]137 // Check Slips
138 // -----------
139 slipInfo& sInfo = _slips[satData->prn];
140 if ( sInfo.slipCntL1 == obs->slip_cnt_L1 &&
141 sInfo.slipCntL2 == obs->slip_cnt_L2 ) {
142 satData->slipFlag = false;
143 }
144 else {
145 satData->slipFlag = true;
146 }
147 sInfo.slipCntL1 = obs->slip_cnt_L1;
148 sInfo.slipCntL2 = obs->slip_cnt_L2;
149
[2274]150 // Handle Code Biases
151 // ------------------
152 t_bias* bb = 0;
153 if (_bias.contains(satData->prn)) {
154 bb = _bias.value(satData->prn);
155 }
156
[2051]157 // Set Code Observations
158 // ---------------------
159 if (obs->P1) {
[2426]160 satData->P1 = obs->P1 + (bb ? bb->p1 : 0.0);
[2051]161 satData->codeTypeF1 = t_satData::P_CODE;
162 }
163 else if (obs->C1) {
[2426]164 satData->P1 = obs->C1 + (bb ? bb->c1 : 0.0);
[2051]165 satData->codeTypeF1 = t_satData::C_CODE;
166 }
167 else {
168 delete satData;
169 return;
170 }
171
172 if (obs->P2) {
[2426]173 satData->P2 = obs->P2 + (bb ? bb->p2 : 0.0);
[2051]174 satData->codeTypeF2 = t_satData::P_CODE;
175 }
176 else if (obs->C2) {
[2317]177 satData->P2 = obs->C2;
[2051]178 satData->codeTypeF2 = t_satData::C_CODE;
179 }
180 else {
181 delete satData;
182 return;
183 }
184
[2228]185 double f1 = t_CST::freq1;
186 double f2 = t_CST::freq2;
187
[2229]188 if (obs->satSys == 'R') {
189 f1 = 1602000000.0 + 562500.0 * obs->slot;
190 f2 = 1246000000.0 + 437500.0 * obs->slot;
191 }
192
[2228]193 // Ionosphere-Free Combination
194 // ---------------------------
195 double c1 = f1 * f1 / (f1 * f1 - f2 * f2);
196 double c2 = - f2 * f2 / (f1 * f1 - f2 * f2);
197
[2051]198 satData->P3 = c1 * satData->P1 + c2 * satData->P2;
199
200 // Set Phase Observations
201 // ----------------------
202 if (obs->L1 && obs->L2) {
[2228]203 satData->L1 = obs->L1 * t_CST::c / f1;
204 satData->L2 = obs->L2 * t_CST::c / f2;
[2051]205 }
206 else {
207 delete satData;
208 return;
209 }
210 satData->L3 = c1 * satData->L1 + c2 * satData->L2;
211
212 // Add new Satellite to the epoch
213 // ------------------------------
[2123]214 bncTime tt(obs->GPSWeek, obs->GPSWeeks);
[2037]215
[2039]216 if (!_epoData) {
217 _epoData = new t_epoData();
218 _epoData->tt = tt;
[2037]219 }
[2039]220 else if (tt != _epoData->tt) {
[2037]221 processEpoch();
[2039]222 delete _epoData;
223 _epoData = new t_epoData();
224 _epoData->tt = tt;
[2037]225 }
[2039]226
[2231]227 if (obs->satSys == 'G') {
[2274]228 _epoData->satDataGPS[satData->prn] = satData;
[2231]229 }
230 else if (obs->satSys == 'R') {
[2274]231 _epoData->satDataGlo[satData->prn] = satData;
[2231]232 }
[2233]233
[2035]234}
235
236//
237////////////////////////////////////////////////////////////////////////////
238void bncPPPclient::slotNewEphGPS(gpsephemeris gpseph) {
239 QMutexLocker locker(&_mutex);
240
241 QString prn = QString("G%1").arg(gpseph.satellite, 2, 10, QChar('0'));
242
243 if (_eph.contains(prn)) {
[2040]244 t_ephGPS* ee = static_cast<t_ephGPS*>(_eph.value(prn));
245 if ( (ee->GPSweek() < gpseph.GPSweek) ||
246 (ee->GPSweek() == gpseph.GPSweek &&
247 ee->TOC() < gpseph.TOC) ) {
248 ee->set(&gpseph);
249 }
[2035]250 }
251 else {
252 t_ephGPS* ee = new t_ephGPS();
253 ee->set(&gpseph);
254 _eph[prn] = ee;
255 }
256}
257
258//
259////////////////////////////////////////////////////////////////////////////
[2224]260void bncPPPclient::slotNewEphGlonass(glonassephemeris gloeph) {
261 QMutexLocker locker(&_mutex);
262
263 QString prn = QString("R%1").arg(gloeph.almanac_number, 2, 10, QChar('0'));
264
265 if (_eph.contains(prn)) {
[2268]266 int ww = gloeph.GPSWeek;
267 int tow = gloeph.GPSTOW;
268 updatetime(&ww, &tow, gloeph.tb*1000, 0); // Moscow -> GPS
[2224]269 t_ephGlo* ee = static_cast<t_ephGlo*>(_eph.value(prn));
[2268]270 if (ee->GPSweek() < ww ||
271 (ee->GPSweek() == ww && ee->GPSweeks() < tow)) {
[2224]272 ee->set(&gloeph);
273 }
274 }
275 else {
276 t_ephGlo* ee = new t_ephGlo();
277 ee->set(&gloeph);
278 _eph[prn] = ee;
279 }
280}
281
282//
283////////////////////////////////////////////////////////////////////////////
[2035]284void bncPPPclient::slotNewCorrections(QList<QString> corrList) {
285 QMutexLocker locker(&_mutex);
[2069]286
287 if (corrList.size() == 0) {
288 return;
289 }
290
291 // Remove All Corrections
292 // ----------------------
293 QMapIterator<QString, t_corr*> ic(_corr);
294 while (ic.hasNext()) {
295 ic.next();
296 delete ic.value();
297 }
298 _corr.clear();
299
[2035]300 QListIterator<QString> it(corrList);
301 while (it.hasNext()) {
302 QTextStream in(it.next().toAscii());
303 int messageType;
304 int updateInterval;
305 int GPSweek;
306 double GPSweeks;
307 QString prn;
308 in >> messageType >> updateInterval >> GPSweek >> GPSweeks >> prn;
309 if ( messageType == COTYPE_GPSCOMBINED ||
[2207]310 messageType == COTYPE_GLONASSCOMBINED ||
311 messageType == COTYPE_GPSORBIT ||
312 messageType == COTYPE_GPSCLOCK ||
313 messageType == COTYPE_GLONASSORBIT ||
314 messageType == COTYPE_GLONASSCLOCK ) {
315
[2035]316 t_corr* cc = 0;
317 if (_corr.contains(prn)) {
318 cc = _corr.value(prn);
319 }
320 else {
321 cc = new t_corr();
322 _corr[prn] = cc;
323 }
[2207]324
[2035]325 cc->tt.set(GPSweek, GPSweeks);
[2207]326
327 if ( messageType == COTYPE_GPSCOMBINED ||
328 messageType == COTYPE_GLONASSCOMBINED ) {
[2366]329 cc->rao.ReSize(3); cc->rao = 0.0;
330 cc->dotRao.ReSize(3); cc->dotRao = 0.0;
331 cc->dotDotRao.ReSize(3); cc->dotDotRao = 0.0;
332 cc->dClk = 0.0;
333 cc->dotDClk = 0.0;
334 cc->dotDotDClk = 0.0;
335 in >> cc->iod
336 >> cc->dClk >> cc->rao[0] >> cc->rao[1] >> cc->rao[2]
337 >> cc->dotDClk >> cc->dotRao[0] >> cc->dotRao[1] >> cc->dotRao[2]
338 >> cc->dotDotDClk >> cc->dotDotRao[0] >> cc->dotDotRao[1] >> cc->dotDotRao[2];
339 cc->dClk /= t_CST::c;
340 cc->dotDClk /= t_CST::c;
341 cc->dotDotDClk /= t_CST::c;
[2208]342 cc->raoSet = true;
343 cc->dClkSet = true;
[2207]344 }
345 else if ( messageType == COTYPE_GPSORBIT ||
346 messageType == COTYPE_GLONASSORBIT ) {
[2366]347 cc->rao.ReSize(3); cc->rao = 0.0;
348 cc->dotRao.ReSize(3); cc->dotRao = 0.0;
349 cc->dotDotRao.ReSize(3); cc->dotDotRao = 0.0;
350 in >> cc->iod
351 >> cc->rao[0] >> cc->rao[1] >> cc->rao[2]
352 >> cc->dotRao[0] >> cc->dotRao[1] >> cc->dotRao[2]
353 >> cc->dotDotRao[0] >> cc->dotDotRao[1] >> cc->dotDotRao[2];
[2208]354 cc->raoSet = true;
[2207]355 }
356 else if ( messageType == COTYPE_GPSCLOCK ||
357 messageType == COTYPE_GLONASSCLOCK ) {
358 int dummyIOD;
[2366]359 cc->dClk = 0.0;
360 cc->dotDClk = 0.0;
361 cc->dotDotDClk = 0.0;
362 in >> dummyIOD >> cc->dClk >> cc->dotDClk >> cc->dotDotDClk;
363 cc->dClk /= t_CST::c;
364 cc->dotDClk /= t_CST::c;
365 cc->dotDotDClk /= t_CST::c;
[2208]366 cc->dClkSet = true;
[2207]367 }
[2035]368 }
[2317]369 else if ( messageType == BTYPE_GPS ) {
[2274]370
371 t_bias* bb = 0;
372 if (_bias.contains(prn)) {
[2341]373 bb = _bias.value(prn);
[2274]374 }
375 else {
376 bb = new t_bias();
377 _bias[prn] = bb;
378 }
379
380 bb->tt.set(GPSweek, GPSweeks);
381
382 int numBiases;
383 in >> numBiases;
384 for (int ii = 0; ii < numBiases; ++ii) {
385 int bType;
386 double bValue;
387 in >> bType >> bValue;
[2426]388 if (bType == CODETYPEGPS_L1_Z) {
[2429]389 bb->p1 = bValue;
[2274]390 }
[2426]391 else if (bType == CODETYPEGPS_L1_CA) {
[2429]392 bb->c1 = bValue;
[2426]393 }
[2365]394 else if (bType == CODETYPEGPS_L2_Z) {
[2429]395 bb->p2 = bValue;
[2274]396 }
397 }
398 }
[2035]399 }
[2208]400
401 QMutableMapIterator<QString, t_corr*> im(_corr);
402 while (im.hasNext()) {
403 im.next();
404 t_corr* cc = im.value();
405 if (!cc->ready()) {
406 delete cc;
407 im.remove();
408 }
409 }
[2035]410}
411
412// Satellite Position
413////////////////////////////////////////////////////////////////////////////
[2123]414t_irc bncPPPclient::getSatPos(const bncTime& tt, const QString& prn,
[2357]415 ColumnVector& xc, ColumnVector& vv) {
[2035]416
[2067]417 const double MAXAGE = 120.0;
[2041]418
[2035]419 if (_eph.contains(prn)) {
[2040]420 t_eph* ee = _eph.value(prn);
421 ee->position(tt.gpsw(), tt.gpssec(), xc.data(), vv.data());
422
[2357]423 if (_pppMode) {
[2068]424 if (_corr.contains(prn)) {
425 t_corr* cc = _corr.value(prn);
426 if (ee->IOD() == cc->iod && (tt - cc->tt) < MAXAGE) {
[2296]427 applyCorr(tt, cc, xc, vv);
[2068]428 return success;
429 }
[2041]430 }
[2068]431 return failure;
[2040]432 }
[2068]433
[2035]434 return success;
435 }
436
437 return failure;
438}
439
440//
441////////////////////////////////////////////////////////////////////////////
[2296]442void bncPPPclient::applyCorr(const bncTime& tt, const t_corr* cc,
443 ColumnVector& xc, ColumnVector& vv) {
[2043]444 ColumnVector dx(3);
[2041]445
[2367]446 double dt = tt - cc->tt;
447 ColumnVector raoHlp = cc->rao + cc->dotRao * dt + cc->dotDotRao * dt * dt;
[2296]448
449 RSW_to_XYZ(xc.Rows(1,3), vv, raoHlp, dx);
450
[2414]451 xc[0] -= dx[0];
452 xc[1] -= dx[1];
453 xc[2] -= dx[2];
[2417]454 xc[3] += cc->dClk + cc->dotDClk * dt + cc->dotDotDClk * dt * dt;
[2041]455}
456
[2049]457// Correct Time of Transmission
458////////////////////////////////////////////////////////////////////////////
[2240]459t_irc bncPPPclient::cmpToT(t_satData* satData) {
[2049]460
[2051]461 double prange = satData->P3;
[2049]462 if (prange == 0.0) {
463 return failure;
464 }
465
466 double clkSat = 0.0;
467 for (int ii = 1; ii <= 10; ii++) {
468
[2123]469 bncTime ToT = _epoData->tt - prange / t_CST::c - clkSat;
[2049]470
471 ColumnVector xc(4);
472 ColumnVector vv(3);
[2357]473 if (getSatPos(ToT, satData->prn, xc, vv) != success) {
[2049]474 return failure;
475 }
476
477 double clkSatOld = clkSat;
[2050]478 clkSat = xc(4);
[2049]479
480 if ( fabs(clkSat-clkSatOld) * t_CST::c < 1.e-4 ) {
481 satData->xx = xc.Rows(1,3);
482 satData->vv = vv;
[2050]483 satData->clk = clkSat * t_CST::c;
[2049]484 return success;
485 }
486 }
487
488 return failure;
489}
490
[2041]491//
492////////////////////////////////////////////////////////////////////////////
[2035]493void bncPPPclient::processEpoch() {
494
[2053]495 // Data Pre-Processing
496 // -------------------
[2231]497 QMutableMapIterator<QString, t_satData*> iGPS(_epoData->satDataGPS);
498 while (iGPS.hasNext()) {
499 iGPS.next();
500 QString prn = iGPS.key();
501 t_satData* satData = iGPS.value();
[2049]502
[2240]503 if (cmpToT(satData) != success) {
[2049]504 delete satData;
[2231]505 iGPS.remove();
[2049]506 continue;
507 }
[2053]508 }
[2035]509
[2231]510 QMutableMapIterator<QString, t_satData*> iGlo(_epoData->satDataGlo);
511 while (iGlo.hasNext()) {
512 iGlo.next();
513 QString prn = iGlo.key();
514 t_satData* satData = iGlo.value();
515
[2240]516 if (cmpToT(satData) != success) {
[2231]517 delete satData;
518 iGlo.remove();
519 continue;
520 }
521 }
522
[2060]523 // Filter Solution
524 // ---------------
[2143]525 if (_model->update(_epoData) == success) {
[2145]526 emit newPosition(_model->time(), _model->x(), _model->y(), _model->z());
[2060]527 }
[2035]528}
[2049]529
Note: See TracBrowser for help on using the repository browser.