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

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