// Part of BNC, a utility for retrieving decoding and // converting GNSS data streams from NTRIP broadcasters. // // Copyright (C) 2007 // German Federal Agency for Cartography and Geodesy (BKG) // http://www.bkg.bund.de // Czech Technical University Prague, Department of Geodesy // http://www.fsv.cvut.cz // // Email: euref-ip@bkg.bund.de // // This program is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License // as published by the Free Software Foundation, version 2. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. /* ------------------------------------------------------------------------- * BKG NTRIP Client * ------------------------------------------------------------------------- * * Class: t_pppParlist * * Purpose: List of estimated parameters * * Author: L. Mervart * * Created: 29-Jul-2014 * * Changes: * * -----------------------------------------------------------------------*/ #include #include #include #include #include #include #include "pppParlist.h" #include "pppSatObs.h" #include "pppStation.h" #include "bncutils.h" #include "bncconst.h" #include "pppClient.h" using namespace BNC_PPP; using namespace std; // Constructor //////////////////////////////////////////////////////////////////////////// t_pppParam::t_pppParam(e_type type, const t_prn& prn, t_lc::type tLC, const vector* obsVector) { _type = type; _prn = prn; _tLC = tLC; _x0 = 0.0; _indexOld = -1; _indexNew = -1; _noise = 0.0; _ambInfo = 0; switch (_type) { case crdX: _epoSpec = true; _sigma0 = OPT->_sigCrd[0]; break; case crdY: _epoSpec = true; _sigma0 = OPT->_sigCrd[1]; break; case crdZ: _epoSpec = true; _sigma0 = OPT->_sigCrd[2]; break; case clkR: _epoSpec = true; _sigma0 = 1000.0; break; case amb: _ambInfo = new t_ambInfo(); if (obsVector) { for (unsigned ii = 0; ii < obsVector->size(); ii++) { const t_pppSatObs* obs = obsVector->at(ii); if (obs->prn() == _prn) { double offGG = 0; if (_prn.system() == 'R' && tLC != t_lc::MW) { offGG = PPP_CLIENT->offGG(); } _x0 = floor((obs->obsValue(tLC) - offGG - obs->cmpValue(tLC)) / obs->lambda(tLC) + 0.5); break; } } } _epoSpec = false; _sigma0 = 100.0; break; case offGG: _epoSpec = true; _sigma0 = 1000.0; _x0 = PPP_CLIENT->offGG(); break; case trp: _epoSpec = false; _sigma0 = OPT->_sigTropo; _noise = OPT->_noiseTropo; break; case bias: _epoSpec = true; _sigma0 = 10.0; break; } } // Destructor //////////////////////////////////////////////////////////////////////////// t_pppParam::~t_pppParam() { delete _ambInfo; } // //////////////////////////////////////////////////////////////////////////// double t_pppParam::partial(const bncTime& /* epoTime */, const t_pppSatObs* obs, const t_lc::type& tLC) const { // Special Case - Melbourne-Wuebbena // --------------------------------- if (tLC == t_lc::MW && _type != amb && _type != bias) { return 0.0; } const t_pppStation* sta = PPP_CLIENT->staRover(); ColumnVector rhoV = sta->xyzApr() - obs->xc().Rows(1,3); switch (_type) { case crdX: return (sta->xyzApr()[0] - obs->xc()[0]) / rhoV.norm_Frobenius(); case crdY: return (sta->xyzApr()[1] - obs->xc()[1]) / rhoV.norm_Frobenius(); case crdZ: return (sta->xyzApr()[2] - obs->xc()[2]) / rhoV.norm_Frobenius(); case clkR: return 1.0; case offGG: return (obs->prn().system() == 'R') ? 1.0 : 0.0; case amb: if (obs->prn() == _prn) { if (tLC == _tLC) { return (obs->lambda(tLC)); } else if (tLC == t_lc::lIF && _tLC == t_lc::MW) { return obs->lambda(t_lc::lIF) * obs->lambda(t_lc::MW) / obs->lambda(t_lc::l2); } else { ColumnVector coeff(4); obs->lc(tLC, 0.0, 0.0, 0.0, 0.0, &coeff); if (_tLC == t_lc::l1) { return obs->lambda(t_lc::l1) * coeff(1); } else if (_tLC == t_lc::l2) { return obs->lambda(t_lc::l2) * coeff(2); } } } return 0.0; case trp: return 1.0 / sin(obs->eleSat()); case bias: if (tLC == _tLC && obs->prn().system() == _prn.system()) { return 1.0; } else { return 0.0; } } return 0.0; } // //////////////////////////////////////////////////////////////////////////// string t_pppParam::toString() const { stringstream ss; switch (_type) { case crdX: ss << "CRD_X"; break; case crdY: ss << "CRD_Y"; break; case crdZ: ss << "CRD_Z"; break; case clkR: ss << "CLK "; break; case amb: ss << "AMB " << left << setw(3) << t_lc::toString(_tLC) << right << ' ' << _prn.toString(); break; case offGG: ss << "OGG "; break; case trp: ss << "TRP "; break; case bias: ss << "BIAS " << _prn.system() << ' ' << left << setw(3) << t_lc::toString(_tLC); break; } return ss.str(); } // Constructor //////////////////////////////////////////////////////////////////////////// t_pppParlist::t_pppParlist() { } // Destructor //////////////////////////////////////////////////////////////////////////// t_pppParlist::~t_pppParlist() { for (unsigned ii = 0; ii < _params.size(); ii++) { delete _params[ii]; } } // //////////////////////////////////////////////////////////////////////////// t_irc t_pppParlist::set(const bncTime& epoTime, const vector& ambLCs, const vector& obsVector) { // Remove some Parameters // ---------------------- vector::iterator it = _params.begin(); while (it != _params.end()) { t_pppParam* par = *it; bool remove = false; if (par->epoSpec()) { remove = true; } else if (par->type() == t_pppParam::amb) { if (par->lastObsTime().valid() && (epoTime - par->lastObsTime() > 120.0)) { remove = true; } } else if (par->type() == t_pppParam::amb) { if (par->lastObsTime().valid() && (epoTime - par->lastObsTime() > 3600.0)) { remove = true; } } if (remove) { delete par; it = _params.erase(it); } else { ++it; } } // Check whether parameters have observations // ------------------------------------------ for (unsigned ii = 0; ii < _params.size(); ii++) { t_pppParam* par = _params[ii]; if (par->prn() == 0) { par->setLastObsTime(epoTime); if (par->firstObsTime().undef()) { par->setFirstObsTime(epoTime); } } else { for (unsigned jj = 0; jj < obsVector.size(); jj++) { const t_pppSatObs* satObs = obsVector[jj]; if (satObs->prn() == par->prn()) { par->setLastObsTime(epoTime); if (par->firstObsTime().undef()) { par->setFirstObsTime(epoTime); } break; } } } } // Required Set of Parameters // -------------------------- vector required; // Coordinates // ----------- required.push_back(new t_pppParam(t_pppParam::crdX, t_prn(), t_lc::dummy)); required.push_back(new t_pppParam(t_pppParam::crdY, t_prn(), t_lc::dummy)); required.push_back(new t_pppParam(t_pppParam::crdZ, t_prn(), t_lc::dummy)); // Receiver Clock // -------------- required.push_back(new t_pppParam(t_pppParam::clkR, t_prn(), t_lc::dummy)); // GPS-Glonass Clock Offset // ------------------------ if (OPT->useGlonass()) { required.push_back(new t_pppParam(t_pppParam::offGG, t_prn(), t_lc::dummy)); } // Receiver Biases // --------------- for (unsigned ii = 0; ii < OPT->_estBias.size(); ii++) { const t_pppOptions::t_optBias& optBias = OPT->_estBias[ii]; required.push_back(new t_pppParam(t_pppParam::bias, t_prn(optBias._system, 1), optBias._tLC)); } // Troposphere // ----------- if (OPT->estTropo()) { required.push_back(new t_pppParam(t_pppParam::trp, t_prn(), t_lc::dummy)); } // Ambiguities // ----------- for (unsigned ii = 0; ii < ambLCs.size(); ii++) { const t_lc::type& tLC = ambLCs[ii]; for (unsigned jj = 0; jj < obsVector.size(); jj++) { const t_pppSatObs* satObs = obsVector[jj]; required.push_back(new t_pppParam(t_pppParam::amb, satObs->prn(), tLC, &obsVector)); } } // Check if all required parameters are present // -------------------------------------------- for (unsigned ii = 0; ii < required.size(); ii++) { t_pppParam* parReq = required[ii]; bool found = false; for (unsigned jj = 0; jj < _params.size(); jj++) { t_pppParam* parOld = _params[jj]; if (parOld->isEqual(parReq)) { found = true; break; } } if (found) { delete parReq; } else { _params.push_back(parReq); } } // Set Parameter Indices // --------------------- sort(_params.begin(), _params.end(), t_pppParam::sortFunction); for (unsigned ii = 0; ii < _params.size(); ii++) { t_pppParam* par = _params[ii]; par->setIndex(ii); for (unsigned jj = 0; jj < obsVector.size(); jj++) { const t_pppSatObs* satObs = obsVector[jj]; if (satObs->prn() == par->prn()) { par->setAmbEleSat(satObs->eleSat()); par->stepAmbNumEpo(); } } } return success; } // //////////////////////////////////////////////////////////////////////////// void t_pppParlist::printResult(const bncTime& epoTime, const SymmetricMatrix& QQ, const ColumnVector& xx) const { string epoTimeStr = string(epoTime); LOG << endl; t_pppParam* parX = 0; t_pppParam* parY = 0; t_pppParam* parZ = 0; for (unsigned ii = 0; ii < _params.size(); ii++) { t_pppParam* par = _params[ii]; if (par->type() == t_pppParam::crdX) { parX = par; } else if (par->type() == t_pppParam::crdY) { parY = par; } else if (par->type() == t_pppParam::crdZ) { parZ = par; } else { int ind = par->indexNew(); LOG << epoTimeStr << ' ' << par->toString() << ' ' << setw(10) << setprecision(4) << par->x0() << ' ' << showpos << setw(10) << setprecision(4) << xx[ind] << noshowpos << " +- " << setw(8) << setprecision(4) << sqrt(QQ[ind][ind]); if (par->type() == t_pppParam::amb) { LOG << " el = " << setw(6) << setprecision(2) << par->ambEleSat() * 180.0 / M_PI << " epo = " << setw(4) << par->ambNumEpo(); } LOG << endl; } } if (parX && parY && parZ) { const t_pppStation* sta = PPP_CLIENT->staRover(); ColumnVector xyz(3); xyz[0] = xx[parX->indexNew()]; xyz[1] = xx[parY->indexNew()]; xyz[2] = xx[parZ->indexNew()]; ColumnVector neu(3); xyz2neu(sta->ellApr().data(), xyz.data(), neu.data()); SymmetricMatrix QQxyz = QQ.SymSubMatrix(1,3); SymmetricMatrix QQneu(3); covariXYZ_NEU(QQxyz, sta->ellApr().data(), QQneu); LOG << epoTimeStr << " X = " << setprecision(4) << sta->xyzApr()[0] + xyz[0] << " +- " << setprecision(4) << sqrt(QQxyz[0][0]) << " Y = " << setprecision(4) << sta->xyzApr()[1] + xyz[1] << " +- " << setprecision(4) << sqrt(QQxyz[1][1]) << " Z = " << setprecision(4) << sta->xyzApr()[2] + xyz[2] << " +- " << setprecision(4) << sqrt(QQxyz[2][2]) << " dN = " << setprecision(4) << neu[0] << " +- " << setprecision(4) << sqrt(QQneu[0][0]) << " dE = " << setprecision(4) << neu[1] << " +- " << setprecision(4) << sqrt(QQneu[1][1]) << " dU = " << setprecision(4) << neu[2] << " +- " << setprecision(4) << sqrt(QQneu[2][2]); LOG << endl; } }