/* ------------------------------------------------------------------------- * BKG NTRIP Client * ------------------------------------------------------------------------- * * Class: bncComb * * Purpose: Combinations of Orbit/Clock Corrections * * Author: L. Mervart * * Created: 22-Jan-2011 * * Changes: * * -----------------------------------------------------------------------*/ #include #include #include #include #include "bnccomb.h" #include #include "bnccore.h" #include "upload/bncrtnetdecoder.h" #include "bncsettings.h" #include "bncutils.h" #include "bncsp3.h" #include "bncantex.h" #include "t_prn.h" const double sig0_offAC = 1000.0; const double sig0_offACSat = 100.0; const double sigP_offACSat = 0.01; const double sig0_clkSat = 100.0; const double sigObs = 0.05; using namespace std; // Constructor //////////////////////////////////////////////////////////////////////////// bncComb::cmbParam::cmbParam(parType type_, int index_, const QString& ac_, const QString& prn_) { type = type_; index = index_; AC = ac_; prn = prn_; xx = 0.0; eph = 0; if (type == offACgnss) { epoSpec = true; sig0 = sig0_offAC; sigP = sig0; } else if (type == offACSat) { epoSpec = false; sig0 = sig0_offACSat; sigP = sigP_offACSat; } else if (type == clkSat) { epoSpec = true; sig0 = sig0_clkSat; sigP = sig0; } } // Destructor //////////////////////////////////////////////////////////////////////////// bncComb::cmbParam::~cmbParam() { } // Partial //////////////////////////////////////////////////////////////////////////// double bncComb::cmbParam::partial(char sys, const QString& AC_, const QString& prn_) { if (type == offACgnss) { if (AC == AC_ && prn_[0].toLatin1() == sys) { return 1.0; } } else if (type == offACSat) { if (AC == AC_ && prn == prn_) { return 1.0; } } else if (type == clkSat) { if (prn == prn_) { return 1.0; } } return 0.0; } // //////////////////////////////////////////////////////////////////////////// QString bncComb::cmbParam::toString(char sys) const { QString outStr; if (type == offACgnss) { outStr = "AC Offset " + AC + " " + sys + " "; } else if (type == offACSat) { outStr = "Sat Offset " + AC + " " + prn.mid(0,3); } else if (type == clkSat) { outStr = "Clk Corr " + prn.mid(0,3); } return outStr; } // Singleton //////////////////////////////////////////////////////////////////////////// bncComb* bncComb::instance() { static bncComb _bncComb; return &_bncComb; } // Constructor //////////////////////////////////////////////////////////////////////////// bncComb::bncComb() : _ephUser(true) { bncSettings settings; QStringList cmbStreams = settings.value("cmbStreams").toStringList(); _cmbSampl = settings.value("cmbSampl").toInt(); if (_cmbSampl <= 0) { _cmbSampl = 5; } _useGps = (Qt::CheckState(settings.value("cmbGps").toInt()) == Qt::Checked) ? true : false; if (_useGps) { _cmbSysPrn['G'] = t_prn::MAXPRN_GPS; _masterMissingEpochs['G'] = 0; } _useGlo = (Qt::CheckState(settings.value("cmbGlo").toInt()) == Qt::Checked) ? true : false; if (_useGlo) { _cmbSysPrn['R'] = t_prn::MAXPRN_GLONASS; _masterMissingEpochs['R'] = 0; } _useGal = (Qt::CheckState(settings.value("cmbGal").toInt()) == Qt::Checked) ? true : false; if (_useGal) { _cmbSysPrn['E'] = t_prn::MAXPRN_GALILEO; _masterMissingEpochs['E'] = 0; } _useBds = (Qt::CheckState(settings.value("cmbBds").toInt()) == Qt::Checked) ? true : false; if (_useBds) { _cmbSysPrn['C'] = t_prn::MAXPRN_BDS; _masterMissingEpochs['C'] = 0; } _useQzss = (Qt::CheckState(settings.value("cmbQzss").toInt()) == Qt::Checked) ? true : false; if (_useQzss) { _cmbSysPrn['J'] = t_prn::MAXPRN_QZSS; _masterMissingEpochs['J'] = 0; } _useSbas = (Qt::CheckState(settings.value("cmbSbas").toInt()) == Qt::Checked) ? true : false; if (_useSbas) { _cmbSysPrn['S'] = t_prn::MAXPRN_SBAS; _masterMissingEpochs['S'] = 0; } _useIrnss = (Qt::CheckState(settings.value("cmbIrnss").toInt()) == Qt::Checked) ? true : false; if (_useIrnss) { _cmbSysPrn['I'] = t_prn::MAXPRN_IRNSS; _masterMissingEpochs['I'] = 0; } if (cmbStreams.size() >= 1 && !cmbStreams[0].isEmpty()) { QListIterator it(cmbStreams); while (it.hasNext()) { QStringList hlp = it.next().split(" "); cmbAC* newAC = new cmbAC(); newAC->mountPoint = hlp[0]; newAC->name = hlp[1]; newAC->weight = hlp[2].toDouble(); QMapIterator itSys(_cmbSysPrn); // init while (itSys.hasNext()) { itSys.next(); char sys = itSys.key(); if (_masterOrbitAC.isEmpty()) { _masterOrbitAC[sys] = newAC->name; } } _ACs.append(newAC); } } QString ssrFormat; _ssrCorr = 0; QListIterator it(settings.value("uploadMountpointsOut").toStringList()); while (it.hasNext()) { QStringList hlp = it.next().split(","); if (hlp.size() > 7) { ssrFormat = hlp[7]; } } if (ssrFormat == "IGS-SSR") { _ssrCorr = new SsrCorrIgs(); } else if (ssrFormat == "RTCM-SSR") { _ssrCorr = new SsrCorrRtcm(); } else { // default _ssrCorr = new SsrCorrIgs(); } _rtnetDecoder = 0; connect(this, SIGNAL(newMessage(QByteArray,bool)), BNC_CORE, SLOT(slotMessage(const QByteArray,bool))); connect(BNC_CORE, SIGNAL(providerIDChanged(QString)), this, SLOT(slotProviderIDChanged(QString))); connect(BNC_CORE, SIGNAL(newOrbCorrections(QList)), this, SLOT(slotNewOrbCorrections(QList))); connect(BNC_CORE, SIGNAL(newClkCorrections(QList)), this, SLOT(slotNewClkCorrections(QList))); connect(BNC_CORE, SIGNAL(newCodeBiases(QList)), this, SLOT(slotNewCodeBiases(QList))); // Combination Method // ------------------ if (settings.value("cmbMethod").toString() == "Single-Epoch") { _method = singleEpoch; } else { _method = filter; } // Initialize Parameters (model: Clk_Corr = AC_Offset + Sat_Offset + Clk) // ---------------------------------------------------------------------- if (_method == filter) { // SYSTEM QMapIterator itSys(_cmbSysPrn); while (itSys.hasNext()) { itSys.next(); int nextPar = 0; char sys = itSys.key(); unsigned maxPrn = itSys.value(); unsigned flag = 0; if (sys == 'E') { flag = 1; } // AC QListIterator itAc(_ACs); while (itAc.hasNext()) { cmbAC* AC = itAc.next(); _params[sys].push_back(new cmbParam(cmbParam::offACgnss, ++nextPar, AC->name, "")); for (unsigned iGnss = 1; iGnss <= maxPrn; iGnss++) { QString prn = QString("%1%2_%3").arg(sys).arg(iGnss, 2, 10, QChar('0')).arg(flag); _params[sys].push_back(new cmbParam(cmbParam::offACSat, ++nextPar, AC->name, prn)); } } for (unsigned iGnss = 1; iGnss <= maxPrn; iGnss++) { QString prn = QString("%1%2_%3").arg(sys).arg(iGnss, 2, 10, QChar('0')).arg(flag); _params[sys].push_back(new cmbParam(cmbParam::clkSat, ++nextPar, "", prn)); } // Initialize Variance-Covariance Matrix // ------------------------------------- _QQ[sys].ReSize(_params[sys].size()); _QQ[sys] = 0.0; for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; _QQ[sys](iPar,iPar) = pp->sig0 * pp->sig0; } } } // ANTEX File // ---------- _antex = 0; QString antexFileName = settings.value("uploadAntexFile").toString(); if (!antexFileName.isEmpty()) { _antex = new bncAntex(); if (_antex->readFile(antexFileName) != success) { emit newMessage("bncCmb: wrong ANTEX file", true); delete _antex; _antex = 0; } } // Bias SINEX File // --------------- _bsx = 0; QString bsxFileName = settings.value("cmbBsxFile").toString(); if (!bsxFileName.isEmpty()) { _bsx = new bncBiasSnx(); if (_bsx->readFile(bsxFileName) != success) { emit newMessage("bncComb: wrong Bias SINEX file", true); delete _bsx; _bsx = 0; } } if (_bsx) { _ms = 6.0 * 3600 * 1000.0; QTimer::singleShot(_ms, this, SLOT(slotReadBiasSnxFile())); } // Maximal Residuum // ---------------- _MAXRES = settings.value("cmbMaxres").toDouble(); if (_MAXRES <= 0.0) { _MAXRES = 999.0; } } // Destructor //////////////////////////////////////////////////////////////////////////// bncComb::~bncComb() { QListIterator icAC(_ACs); while (icAC.hasNext()) { delete icAC.next(); } delete _rtnetDecoder; if (_ssrCorr) { delete _ssrCorr; } delete _antex; delete _bsx; QMapIterator itSys(_cmbSysPrn); while (itSys.hasNext()) { itSys.next(); char sys = itSys.key(); for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { delete _params[sys][iPar-1]; } QListIterator itTime(_buffer[sys].keys()); while (itTime.hasNext()) { bncTime epoTime = itTime.next(); _buffer[sys].remove(epoTime); } } } void bncComb::slotReadBiasSnxFile() { bncSettings settings; QString bsxFileName = settings.value("cmbBsxFile").toString(); if (bsxFileName.isEmpty()) { emit newMessage("bncComb: no Bias SINEX file specified", true); return; } if (!_bsx) { _bsx = new bncBiasSnx(); } if (_bsx->readFile(bsxFileName) != success) { emit newMessage("bncComb: wrong Bias SINEX file", true); delete _bsx; _bsx = 0; } else { emit newMessage("bncComb: successfully read Bias SINEX file", true); } QTimer::singleShot(_ms, this, SLOT(slotReadBiasSnxFile())); } // Remember orbit corrections //////////////////////////////////////////////////////////////////////////// void bncComb::slotNewOrbCorrections(QList orbCorrections) { QMutexLocker locker(&_mutex); for (int ii = 0; ii < orbCorrections.size(); ii++) { t_orbCorr& orbCorr = orbCorrections[ii]; QString staID(orbCorr._staID.c_str()); char sys = orbCorr._prn.system(); if (!_cmbSysPrn.contains(sys)){ continue; } // Find/Check the AC Name // ---------------------- QString acName; QListIterator icAC(_ACs); while (icAC.hasNext()) { cmbAC* AC = icAC.next(); if (AC->mountPoint == staID) { acName = AC->name; break; } } if (acName.isEmpty()) { continue; } // Store the correction // -------------------- QMap& storage = _orbCorrections[acName]; storage[orbCorr._prn] = orbCorr; } } // Remember Satellite Code Biases //////////////////////////////////////////////////////////////////////////// void bncComb::slotNewCodeBiases(QList satCodeBiases) { QMutexLocker locker(&_mutex); for (int ii = 0; ii < satCodeBiases.size(); ii++) { t_satCodeBias& satCodeBias = satCodeBiases[ii]; QString staID(satCodeBias._staID.c_str()); char sys = satCodeBias._prn.system(); if (!_cmbSysPrn.contains(sys)){ continue; } // Find/Check the AC Name // ---------------------- QString acName; QListIterator icAC(_ACs); while (icAC.hasNext()) { cmbAC* AC = icAC.next(); if (AC->mountPoint == staID) { acName = AC->name; break; } } if (acName.isEmpty()) { continue; } // Store the correction // -------------------- QMap& storage = _satCodeBiases[acName]; storage[satCodeBias._prn] = satCodeBias; } } // Process clock corrections //////////////////////////////////////////////////////////////////////////// void bncComb::slotNewClkCorrections(QList clkCorrections) { QMutexLocker locker(&_mutex); bncTime lastTime; for (int ii = 0; ii < clkCorrections.size(); ii++) { t_clkCorr& clkCorr = clkCorrections[ii]; QString staID(clkCorr._staID.c_str()); QString prn(clkCorr._prn.toInternalString().c_str()); char sys = clkCorr._prn.system(); if (!_cmbSysPrn.contains(sys)){ continue; } // Set the last time // ----------------- if (lastTime.undef() || clkCorr._time > lastTime) { lastTime = clkCorr._time; } // Find/Check the AC Name // ---------------------- QString acName; QListIterator icAC(_ACs); while (icAC.hasNext()) { cmbAC* AC = icAC.next(); if (AC->mountPoint == staID) { acName = AC->name; break; } } if (acName.isEmpty()) { continue; } // Check Modulo Time // ----------------- int sec = int(nint(clkCorr._time.gpssec()*10)); if (sec % (_cmbSampl*10) != 0.0) { continue; } // Check Correction Age // -------------------- if (_resTime.valid() && clkCorr._time <= _resTime) { #ifdef BNC_DEBUG_CMB emit newMessage("bncComb: old correction: " + acName.toLatin1() + " " + prn.mid(0,3).toLatin1(), true); #endif continue; } // Create new correction // --------------------- cmbCorr* newCorr = new cmbCorr(); newCorr->_prn = prn; newCorr->_time = clkCorr._time; newCorr->_iod = clkCorr._iod; newCorr->_acName = acName; newCorr->_clkCorr = clkCorr; // Check orbit correction // ---------------------- if (!_orbCorrections.contains(acName)) { delete newCorr; continue; } else { QMap& storage = _orbCorrections[acName]; if (!storage.contains(clkCorr._prn) || storage[clkCorr._prn]._iod != newCorr->_iod) { delete newCorr; continue; } else { newCorr->_orbCorr = storage[clkCorr._prn]; } } // Check the Ephemeris //-------------------- t_eph* ephLast = _ephUser.ephLast(prn); t_eph* ephPrev = _ephUser.ephPrev(prn); if (ephLast == 0) { #ifdef BNC_DEBUG_CMB emit newMessage("bncComb: eph not found for " + prn.mid(0,3).toLatin1(), true); #endif delete newCorr; continue; } else if (ephLast->checkState() == t_eph::bad || ephLast->checkState() == t_eph::outdated || ephLast->checkState() == t_eph::unhealthy) { #ifdef BNC_DEBUG_CMB emit newMessage("bncComb: ephLast not ok (checkState: " + ephLast->checkStateToString().toLatin1() + ") for " + prn.mid(0,3).toLatin1(), true); #endif delete newCorr; continue; } else { if (ephLast->IOD() == newCorr->_iod) { newCorr->_eph = ephLast; } else if (ephPrev && ephPrev->checkState() == t_eph::ok && ephPrev->IOD() == newCorr->_iod) { newCorr->_eph = ephPrev; switchToLastEph(ephLast, newCorr); } else { #ifdef BNC_DEBUG_CMB emit newMessage("bncComb: eph not found for " + prn.mid(0,3).toLatin1() + QString(" with IOD %1").arg(newCorr->_iod).toLatin1(), true); #endif delete newCorr; continue; } } // Check satellite code biases // ---------------------------- if (_satCodeBiases.contains(acName)) { QMap& storage = _satCodeBiases[acName]; if (storage.contains(clkCorr._prn)) { newCorr->_satCodeBias = storage[clkCorr._prn]; QMap codeBiasesRefSig; for (unsigned ii = 1; ii < cmbRefSig::cIF; ii++) { t_frequency::type frqType = cmbRefSig::toFreq(sys, static_cast(ii)); char frqNum = t_frequency::toString(frqType)[1]; char attrib = cmbRefSig::toAttrib(sys, static_cast(ii)); QString rnxType2ch = QString("%1%2").arg(frqNum).arg(attrib); for (unsigned ii = 0; ii < newCorr->_satCodeBias._bias.size(); ii++) { const t_frqCodeBias& bias = newCorr->_satCodeBias._bias[ii]; if (rnxType2ch.toStdString() == bias._rnxType2ch) { codeBiasesRefSig[frqType] = bias._value; } } } if (codeBiasesRefSig.size() == 2) { map codeCoeff; double channel = double(newCorr->_eph->slotNum()); cmbRefSig::coeff(sys, cmbRefSig::cIF, channel, codeCoeff); map::const_iterator it; for (it = codeCoeff.begin(); it != codeCoeff.end(); it++) { t_frequency::type frqType = it->first; newCorr->_satCodeBiasIF += it->second * codeBiasesRefSig[frqType]; } } newCorr->_satCodeBias._bias.clear(); } } // Store correction into the buffer // -------------------------------- QVector& corrs = _buffer[sys][newCorr->_time].corrs; corrs.push_back(newCorr); } // Process previous Epoch(s) // ------------------------- const double outWait = 1.0 * _cmbSampl; QMapIterator itSys(_cmbSysPrn); while (itSys.hasNext()) { itSys.next(); char sys = itSys.key(); QListIterator itTime(_buffer[sys].keys()); while (itTime.hasNext()) { bncTime epoTime = itTime.next(); if (epoTime < lastTime - outWait) { _resTime = epoTime; processEpoch(sys); } } } } // Change the correction so that it refers to last received ephemeris //////////////////////////////////////////////////////////////////////////// void bncComb::switchToLastEph(t_eph* lastEph, cmbCorr* corr) { if (corr->_eph == lastEph) { return; } ColumnVector oldXC(6); ColumnVector oldVV(3); if (corr->_eph->getCrd(corr->_time, oldXC, oldVV, false) != success) { return; } ColumnVector newXC(6); ColumnVector newVV(3); if (lastEph->getCrd(corr->_time, newXC, newVV, false) != success) { return; } ColumnVector dX = newXC.Rows(1,3) - oldXC.Rows(1,3); ColumnVector dV = newVV - oldVV; double dC = newXC(4) - oldXC(4); ColumnVector dRAO(3); XYZ_to_RSW(newXC.Rows(1,3), newVV, dX, dRAO); ColumnVector dDotRAO(3); XYZ_to_RSW(newXC.Rows(1,3), newVV, dV, dDotRAO); QString msg = "switch corr " + corr->_prn.mid(0,3) + QString(" %1 -> %2 %3").arg(corr->_iod,3).arg(lastEph->IOD(),3).arg(dC*t_CST::c, 8, 'f', 4); emit newMessage(msg.toLatin1(), false); corr->_iod = lastEph->IOD(); corr->_eph = lastEph; corr->_orbCorr._xr += dRAO; corr->_orbCorr._dotXr += dDotRAO; corr->_clkCorr._dClk -= dC; } // Process Epoch //////////////////////////////////////////////////////////////////////////// void bncComb::processEpoch(char sys) { _log.clear(); QTextStream out(&_log, QIODevice::WriteOnly); out << "\n" << "Combination: " << sys << "\n" << "--------------------------------" << "\n"; // Observation Statistics // ---------------------- bool masterPresent = false; QListIterator icAC(_ACs); while (icAC.hasNext()) { cmbAC* AC = icAC.next(); AC->numObs[sys] = 0; QVectorIterator itCorr(corrs(sys)); while (itCorr.hasNext()) { cmbCorr* corr = itCorr.next(); if (corr->_acName == AC->name) { AC->numObs[sys] += 1; if (AC->name == _masterOrbitAC[sys]) { masterPresent = true; } } } out << AC->name.toLatin1().data() << ": " << AC->numObs[sys] << "\n"; } // If Master not present, switch to another one // -------------------------------------------- const unsigned switchMasterAfterGap = 1; if (masterPresent) { _masterMissingEpochs[sys] = 0; } else { ++_masterMissingEpochs[sys]; if (_masterMissingEpochs[sys] < switchMasterAfterGap) { out << "Missing Master, Epoch skipped" << "\n"; _buffer[sys].remove(_resTime); emit newMessage(_log, false); return; } else { _masterMissingEpochs[sys] = 0; QListIterator icAC(_ACs); while (icAC.hasNext()) { cmbAC* AC = icAC.next(); if (AC->numObs[sys] > 0) { out << "Switching Master AC " << _masterOrbitAC[sys].toLatin1().data() << " --> " << AC->name.toLatin1().data() << " " << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << "\n"; _masterOrbitAC[sys] = AC->name; break; } } } } QMap resCorr; // Perform the actual Combination using selected Method // ---------------------------------------------------- t_irc irc; ColumnVector dx; if (_method == filter) { irc = processEpoch_filter(sys, out, resCorr, dx); } else { irc = processEpoch_singleEpoch(sys, out, resCorr, dx); } // Update Parameter Values, Print Results // -------------------------------------- if (irc == success) { for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; pp->xx += dx(iPar); if (pp->type == cmbParam::clkSat) { if (resCorr.find(pp->prn) != resCorr.end()) { // set clock result resCorr[pp->prn]->_dClkResult = pp->xx / t_CST::c; // Add Code Biases from SINEX File if (_bsx) { map codeCoeff; double channel = double(resCorr[pp->prn]->_eph->slotNum()); cmbRefSig::coeff(sys, cmbRefSig::cIF, channel, codeCoeff); t_frequency::type fType1 = cmbRefSig::toFreq(sys, cmbRefSig::c1); t_frequency::type fType2 = cmbRefSig::toFreq(sys, cmbRefSig::c2); _bsx->determineSsrSatCodeBiases(pp->prn.mid(0,3), codeCoeff[fType1], codeCoeff[fType2], resCorr[pp->prn]->_satCodeBias); } } } out << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << " "; out.setRealNumberNotation(QTextStream::FixedNotation); out.setFieldWidth(8); out.setRealNumberPrecision(4); out << pp->toString(sys) << " " << pp->xx << " +- " << sqrt(_QQ[sys](pp->index,pp->index)) << "\n"; out.setFieldWidth(0); } printResults(out, resCorr); dumpResults(resCorr); } // Delete Data, emit Message // ------------------------- _buffer[sys].remove(_resTime); emit newMessage(_log, false); } // Process Epoch - Filter Method //////////////////////////////////////////////////////////////////////////// t_irc bncComb::processEpoch_filter(char sys, QTextStream& out, QMap& resCorr, ColumnVector& dx) { // Prediction Step // --------------- int nPar = _params[sys].size(); ColumnVector x0(nPar); for (int iPar = 1; iPar <= nPar; iPar++) { cmbParam* pp = _params[sys][iPar-1]; if (pp->epoSpec) { pp->xx = 0.0; _QQ[sys].Row(iPar) = 0.0; _QQ[sys].Column(iPar) = 0.0; _QQ[sys](iPar,iPar) = pp->sig0 * pp->sig0; } else { _QQ[sys](iPar,iPar) += pp->sigP * pp->sigP; } x0(iPar) = pp->xx; } // Check Satellite Positions for Outliers // -------------------------------------- if (checkOrbits(sys, out) != success) { return failure; } // Update and outlier detection loop // --------------------------------- SymmetricMatrix QQ_sav = _QQ[sys]; while (true) { Matrix AA; ColumnVector ll; DiagonalMatrix PP; if (createAmat(sys, AA, ll, PP, x0, resCorr) != success) { return failure; } dx.ReSize(nPar); dx = 0.0; kalman(AA, ll, PP, _QQ[sys], dx); ColumnVector vv = ll - AA * dx; int maxResIndex; double maxRes = vv.maximum_absolute_value1(maxResIndex); out.setRealNumberNotation(QTextStream::FixedNotation); out.setRealNumberPrecision(3); out << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << " Maximum Residuum " << maxRes << ' ' << corrs(sys)[maxResIndex-1]->_acName << ' ' << corrs(sys)[maxResIndex-1]->_prn.mid(0,3); if (maxRes > _MAXRES) { for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; if (pp->type == cmbParam::offACSat && pp->AC == corrs(sys)[maxResIndex-1]->_acName && pp->prn == corrs(sys)[maxResIndex-1]->_prn.mid(0,3)) { QQ_sav.Row(iPar) = 0.0; QQ_sav.Column(iPar) = 0.0; QQ_sav(iPar,iPar) = pp->sig0 * pp->sig0; } } out << " Outlier" << "\n"; _QQ[sys] = QQ_sav; corrs(sys).remove(maxResIndex-1); } else { out << " OK" << "\n"; out.setRealNumberNotation(QTextStream::FixedNotation); out.setRealNumberPrecision(4); for (int ii = 0; ii < corrs(sys).size(); ii++) { const cmbCorr* corr = corrs(sys)[ii]; out << _resTime.datestr().c_str() << ' ' << _resTime.timestr().c_str() << " " << corr->_acName << ' ' << corr->_prn.mid(0,3); out.setFieldWidth(10); out << " res = " << vv[ii] << "\n"; out.setFieldWidth(0); } break; } } return success; } // Print results //////////////////////////////////////////////////////////////////////////// void bncComb::printResults(QTextStream& out, const QMap& resCorr) { QMapIterator it(resCorr); while (it.hasNext()) { it.next(); cmbCorr* corr = it.value(); const t_eph* eph = corr->_eph; if (eph) { ColumnVector xc(6); ColumnVector vv(3); if (eph->getCrd(_resTime, xc, vv, false) != success) { continue; } out << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << " "; out.setFieldWidth(3); out << "Full Clock " << corr->_prn.mid(0,3) << " " << corr->_iod << " "; out.setFieldWidth(14); out << (xc(4) + corr->_dClkResult) * t_CST::c << "\n"; out.setFieldWidth(0); } else { out << "bncComb::printResults bug" << "\n"; } out.flush(); } } // Send results to RTNet Decoder and directly to PPP Client //////////////////////////////////////////////////////////////////////////// void bncComb::dumpResults(const QMap& resCorr) { QList orbCorrections; QList clkCorrections; QList satCodeBiasList; unsigned year, month, day, hour, minute; double sec; _resTime.civil_date(year, month, day); _resTime.civil_time(hour, minute, sec); QString outLines = QString().asprintf("* %4d %2d %2d %d %d %12.8f\n", year, month, day, hour, minute, sec); QMapIterator it(resCorr); while (it.hasNext()) { it.next(); cmbCorr* corr = it.value(); // ORBIT t_orbCorr orbCorr(corr->_orbCorr); orbCorr._staID = "INTERNAL"; orbCorrections.push_back(orbCorr); // CLOCK t_clkCorr clkCorr(corr->_clkCorr); clkCorr._staID = "INTERNAL"; clkCorr._dClk = corr->_dClkResult; clkCorr._dotDClk = 0.0; clkCorr._dotDotDClk = 0.0; clkCorrections.push_back(clkCorr); // CODE BIASES t_satCodeBias satCodeBias(corr->_satCodeBias); satCodeBias._staID = "INTERNAL"; satCodeBiasList.push_back(satCodeBias); ColumnVector xc(6); ColumnVector vv(3); corr->_eph->setClkCorr(dynamic_cast(&clkCorr)); corr->_eph->setOrbCorr(dynamic_cast(&orbCorr)); if (corr->_eph->getCrd(_resTime, xc, vv, true) != success) { delete corr; continue; } // Correction Phase Center --> CoM // ------------------------------- ColumnVector dx(3); dx = 0.0; if (_antex) { double Mjd = _resTime.mjd() + _resTime.daysec()/86400.0; if (_antex->satCoMcorrection(corr->_prn, Mjd, xc.Rows(1,3), dx) != success) { dx = 0; _log += "antenna not found " + corr->_prn.mid(0,3).toLatin1() + '\n'; } } outLines += corr->_prn.mid(0,3); QString hlp = QString().asprintf(" APC 3 %15.4f %15.4f %15.4f" " Clk 1 %15.4f" " Vel 3 %15.4f %15.4f %15.4f" " CoM 3 %15.4f %15.4f %15.4f", xc(1), xc(2), xc(3), xc(4) * t_CST::c, vv(1), vv(2), vv(3), xc(1)-dx(1), xc(2)-dx(2), xc(3)-dx(3)); outLines += hlp; hlp.clear(); if (satCodeBias._bias.size()) { hlp = QString().asprintf(" CodeBias %2lu", satCodeBias._bias.size()); outLines += hlp; hlp.clear(); for (unsigned ii = 0; ii < satCodeBias._bias.size(); ii++) { const t_frqCodeBias& frqCodeBias = satCodeBias._bias[ii]; if (!frqCodeBias._rnxType2ch.empty()) { hlp = QString().asprintf(" %s%10.6f", frqCodeBias._rnxType2ch.c_str(), frqCodeBias._value); outLines += hlp; hlp.clear(); } } } outLines += "\n"; delete corr; } outLines += "EOE\n"; // End Of Epoch flag //cout << outLines.toStdString(); if (!_rtnetDecoder) { _rtnetDecoder = new bncRtnetDecoder(); } vector errmsg; _rtnetDecoder->Decode(outLines.toLatin1().data(), outLines.length(), errmsg); // Send new Corrections to PPP etc. // -------------------------------- if (orbCorrections.size() > 0 && clkCorrections.size() > 0) { emit newOrbCorrections(orbCorrections); emit newClkCorrections(clkCorrections); } if (satCodeBiasList.size()) { emit newCodeBiases(satCodeBiasList); } } // Create First Design Matrix and Vector of Measurements //////////////////////////////////////////////////////////////////////////// t_irc bncComb::createAmat(char sys, Matrix& AA, ColumnVector& ll, DiagonalMatrix& PP, const ColumnVector& x0, QMap& resCorr) { unsigned nPar = _params[sys].size(); unsigned nObs = corrs(sys).size(); if (nObs == 0) { return failure; } int maxSat = _cmbSysPrn[sys]; const int nCon = (_method == filter) ? 1 + maxSat : 0; AA.ReSize(nObs+nCon, nPar); AA = 0.0; ll.ReSize(nObs+nCon); ll = 0.0; PP.ReSize(nObs+nCon); PP = 1.0 / (sigObs * sigObs); int iObs = 0; QVectorIterator itCorr(corrs(sys)); while (itCorr.hasNext()) { cmbCorr* corr = itCorr.next(); QString prn = corr->_prn; ++iObs; if (corr->_acName == _masterOrbitAC[sys] && resCorr.find(prn) == resCorr.end()) { resCorr[prn] = new cmbCorr(*corr); } for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; AA(iObs, iPar) = pp->partial(sys, corr->_acName, prn); } ll(iObs) = (corr->_clkCorr._dClk * t_CST::c - corr->_satCodeBiasIF) - DotProduct(AA.Row(iObs), x0); } // Regularization // -------------- if (_method == filter) { const double Ph = 1.e6; PP(nObs+1) = Ph; for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; if ( AA.Column(iPar).maximum_absolute_value() > 0.0 && pp->type == cmbParam::clkSat ) { AA(nObs+1, iPar) = 1.0; } } unsigned flag = 0; if (sys == 'E') { flag = 1; } if (sys == 'R') { return success; } int iCond = 1; // GNSS for (unsigned iGnss = 1; iGnss <= _cmbSysPrn[sys]; iGnss++) { QString prn = QString("%1%2_%3").arg(sys).arg(iGnss, 2, 10, QChar('0')).arg(flag); ++iCond; PP(nObs+iCond) = Ph; for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; if ( pp && AA.Column(iPar).maximum_absolute_value() > 0.0 && pp->type == cmbParam::offACSat && pp->prn == prn) { AA(nObs+iCond, iPar) = 1.0; } } } } return success; } // Process Epoch - Single-Epoch Method //////////////////////////////////////////////////////////////////////////// t_irc bncComb::processEpoch_singleEpoch(char sys, QTextStream& out, QMap& resCorr, ColumnVector& dx) { // Check Satellite Positions for Outliers // -------------------------------------- if (checkOrbits(sys, out) != success) { return failure; } // Outlier Detection Loop // ---------------------- while (true) { // Remove Satellites that are not in Master // ---------------------------------------- QMutableVectorIterator it(corrs(sys)); while (it.hasNext()) { cmbCorr* corr = it.next(); QString prn = corr->_prn; bool foundMaster = false; QVectorIterator itHlp(corrs(sys)); while (itHlp.hasNext()) { cmbCorr* corrHlp = itHlp.next(); QString prnHlp = corrHlp->_prn; QString ACHlp = corrHlp->_acName; if (ACHlp == _masterOrbitAC[sys] && prn == prnHlp) { foundMaster = true; break; } } if (!foundMaster) { delete corr; it.remove(); } } // Count Number of Observations per Satellite and per AC // ----------------------------------------------------- QMap numObsPrn; QMap numObsAC; QVectorIterator itCorr(corrs(sys)); while (itCorr.hasNext()) { cmbCorr* corr = itCorr.next(); QString prn = corr->_prn; QString AC = corr->_acName; if (numObsPrn.find(prn) == numObsPrn.end()) { numObsPrn[prn] = 1; } else { numObsPrn[prn] += 1; } if (numObsAC.find(AC) == numObsAC.end()) { numObsAC[AC] = 1; } else { numObsAC[AC] += 1; } } // Clean-Up the Parameters // ----------------------- for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { delete _params[sys][iPar-1]; } _params[sys].clear(); // Set new Parameters // ------------------ int nextPar = 0; QMapIterator itAC(numObsAC); while (itAC.hasNext()) { itAC.next(); const QString& AC = itAC.key(); int numObs = itAC.value(); if (AC != _masterOrbitAC[sys] && numObs > 0) { _params[sys].push_back(new cmbParam(cmbParam::offACgnss, ++nextPar, AC, "")); } } QMapIterator itPrn(numObsPrn); while (itPrn.hasNext()) { itPrn.next(); const QString& prn = itPrn.key(); int numObs = itPrn.value(); if (numObs > 0) { _params[sys].push_back(new cmbParam(cmbParam::clkSat, ++nextPar, "", prn)); } } int nPar = _params[sys].size(); ColumnVector x0(nPar); x0 = 0.0; // Create First-Design Matrix // -------------------------- Matrix AA; ColumnVector ll; DiagonalMatrix PP; if (createAmat(sys, AA, ll, PP, x0, resCorr) != success) { return failure; } ColumnVector vv; try { Matrix ATP = AA.t() * PP; SymmetricMatrix NN; NN << ATP * AA; ColumnVector bb = ATP * ll; _QQ[sys] = NN.i(); dx = _QQ[sys] * bb; vv = ll - AA * dx; } catch (Exception& exc) { out << exc.what() << "\n"; return failure; } int maxResIndex; double maxRes = vv.maximum_absolute_value1(maxResIndex); out.setRealNumberNotation(QTextStream::FixedNotation); out.setRealNumberPrecision(3); out << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << " Maximum Residuum " << maxRes << ' ' << corrs(sys)[maxResIndex-1]->_acName << ' ' << corrs(sys)[maxResIndex-1]->_prn.mid(0,3); if (maxRes > _MAXRES) { out << " Outlier" << "\n"; delete corrs(sys)[maxResIndex-1]; corrs(sys).remove(maxResIndex-1); } else { out << " OK" << "\n"; out.setRealNumberNotation(QTextStream::FixedNotation); out.setRealNumberPrecision(3); for (int ii = 0; ii < vv.Nrows(); ii++) { const cmbCorr* corr = corrs(sys)[ii]; out << _resTime.datestr().c_str() << ' ' << _resTime.timestr().c_str() << " " << corr->_acName << ' ' << corr->_prn.mid(0,3); out.setFieldWidth(6); out << " res = " << vv[ii] << "\n"; out.setFieldWidth(0); } return success; } } return failure; } // Check Satellite Positions for Outliers //////////////////////////////////////////////////////////////////////////// t_irc bncComb::checkOrbits(char sys, QTextStream& out) { const double MAX_DISPLACEMENT = 0.20; // Switch to last ephemeris (if possible) // -------------------------------------- QMutableVectorIterator im(corrs(sys)); while (im.hasNext()) { cmbCorr* corr = im.next(); QString prn = corr->_prn; t_eph* ephLast = _ephUser.ephLast(prn); t_eph* ephPrev = _ephUser.ephPrev(prn); if (ephLast == 0) { out << "checkOrbit: missing eph (not found) " << corr->_prn.mid(0,3) << "\n"; delete corr; im.remove(); } else if (corr->_eph == 0) { out << "checkOrbit: missing eph (zero) " << corr->_prn.mid(0,3) << "\n"; delete corr; im.remove(); } else { if ( corr->_eph == ephLast || corr->_eph == ephPrev ) { switchToLastEph(ephLast, corr); } else { out << "checkOrbit: missing eph (deleted) " << corr->_prn.mid(0,3) << "\n"; delete corr; im.remove(); } } } while (true) { // Compute Mean Corrections for all Satellites // ------------------------------------------- QMap numCorr; QMap meanRao; QVectorIterator it(corrs(sys)); while (it.hasNext()) { cmbCorr* corr = it.next(); QString prn = corr->_prn; if (meanRao.find(prn) == meanRao.end()) { meanRao[prn].ReSize(4); meanRao[prn].Rows(1,3) = corr->_orbCorr._xr; meanRao[prn](4) = 1; } else { meanRao[prn].Rows(1,3) += corr->_orbCorr._xr; meanRao[prn](4) += 1; } if (numCorr.find(prn) == numCorr.end()) { numCorr[prn] = 1; } else { numCorr[prn] += 1; } } // Compute Differences wrt. Mean, find Maximum // ------------------------------------------- QMap maxDiff; it.toFront(); while (it.hasNext()) { cmbCorr* corr = it.next(); QString prn = corr->_prn; if (meanRao[prn](4) != 0) { meanRao[prn] /= meanRao[prn](4); meanRao[prn](4) = 0; } corr->_diffRao = corr->_orbCorr._xr - meanRao[prn].Rows(1,3); if (maxDiff.find(prn) == maxDiff.end()) { maxDiff[prn] = corr; } else { double normMax = maxDiff[prn]->_diffRao.NormFrobenius(); double norm = corr->_diffRao.NormFrobenius(); if (norm > normMax) { maxDiff[prn] = corr; } } } if (_ACs.size() == 1) { break; } // Remove Outliers // --------------- bool removed = false; QMutableVectorIterator im(corrs(sys)); while (im.hasNext()) { cmbCorr* corr = im.next(); QString prn = corr->_prn; if (numCorr[prn] < 2) { delete corr; im.remove(); } else if (corr == maxDiff[prn]) { double norm = corr->_diffRao.NormFrobenius(); if (norm > MAX_DISPLACEMENT) { out << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << " " << "Orbit Outlier: " << corr->_acName.toLatin1().data() << " " << prn.mid(0,3).toLatin1().data() << " " << corr->_iod << " " << norm << "\n"; delete corr; im.remove(); removed = true; } } } if (!removed) { break; } } return success; } // //////////////////////////////////////////////////////////////////////////// void bncComb::slotProviderIDChanged(QString mountPoint) { QMutexLocker locker(&_mutex); QTextStream out(&_log, QIODevice::WriteOnly); // Find the AC Name // ---------------- QString acName; QListIterator icAC(_ACs); while (icAC.hasNext()) { cmbAC* AC = icAC.next(); if (AC->mountPoint == mountPoint) { acName = AC->name; out << "Provider ID changed: AC " << AC->name.toLatin1().data() << " " << _resTime.datestr().c_str() << " " << _resTime.timestr().c_str() << "\n"; break; } } if (acName.isEmpty()) { return; } QMapIterator itSys(_cmbSysPrn); while (itSys.hasNext()) { itSys.next(); char sys = itSys.key(); // Remove all corrections of the corresponding AC // ---------------------------------------------- QListIterator itTime(_buffer[sys].keys()); while (itTime.hasNext()) { bncTime epoTime = itTime.next(); QVector& corrVec = _buffer[sys][epoTime].corrs; QMutableVectorIterator it(corrVec); while (it.hasNext()) { cmbCorr* corr = it.next(); if (acName == corr->_acName) { delete corr; it.remove(); } } } // Reset Satellite Offsets // ----------------------- if (_method == filter) { for (int iPar = 1; iPar <= _params[sys].size(); iPar++) { cmbParam* pp = _params[sys][iPar-1]; if (pp->AC == acName && pp->type == cmbParam::offACSat) { pp->xx = 0.0; _QQ[sys].Row(iPar) = 0.0; _QQ[sys].Column(iPar) = 0.0; _QQ[sys](iPar,iPar) = pp->sig0 * pp->sig0; } } } } }