24#include <unordered_map>
25#include <unordered_set>
32#include "Navigation/Transformations/Units.hpp"
38#include <Eigen/src/Core/Matrix.h>
109 LOG_DEBUG(
"Reading ANTEX files started...");
111 auto path = flow::GetProgramRootPath() /
"resources" /
"gnss" /
"antex";
112 if (!std::filesystem::exists(path))
114 LOG_WARN(
"Not reading ANTEX files because path does not exist: {}", path);
117 for (
const auto& entry : std::filesystem::directory_iterator(path))
119 if (entry.path().extension() !=
".atx") {
continue; }
120 LOG_DEBUG(
"Reading {}...", entry.path().string());
122 std::ifstream fs(entry.path());
125 LOG_ERROR(
"Could not read ANTEX file: {}", entry.path().string());
129 auto extHeaderLabel = [](
const std::string& line) {
130 return str::trim_copy(std::string_view(line).substr(60, 20));
134 size_t lineNumber = 0;
136 bool antennaStarted =
false;
138 std::vector<NAV::AntexReader::Antenna::AntennaInfo>::iterator antInfo;
139 std::string antennaType;
144 const double azimuthStart = 0.0;
145 const double azimuthEnd = deg2rad(360.0);
146 double azimuthDelta = 0.0;
147 double zenithStart = 0.0;
148 double zenithEnd = 0.0;
149 double zenithDelta = 0.0;
150#if LOG_LEVEL <= LOG_LEVEL_DATA
151 bool patternLogging =
false;
153 while (std::getline(fs, line) && !fs.eof())
156 auto label = extHeaderLabel(line);
157 if (label ==
"START OF ANTENNA") { antennaStarted =
true; }
158 else if (label ==
"END OF ANTENNA")
160 antennaStarted =
false;
172 else if (antennaStarted)
174 if (label ==
"TYPE / SERIAL NO")
176 antennaType = str::trim_copy(line.substr(0, 20));
177 std::string serialNumber = str::trim_copy(line.substr(20, 20));
178 if (!serialNumber.empty()) { antennaType +=
":" + serialNumber; }
184 else if (label ==
"METH / BY / # / DATE")
187 auto strDate = line.substr(50, 10);
188 auto day = std::stoi(strDate.substr(0, 2));
189 auto strMon = strDate.substr(3, 3);
191 if (strMon ==
"JAN") { mon = 1; }
192 if (strMon ==
"FEB") { mon = 2; }
193 if (strMon ==
"MAR") { mon = 3; }
194 if (strMon ==
"APR") { mon = 4; }
195 if (strMon ==
"MAY") { mon = 5; }
196 if (strMon ==
"JUN") { mon = 6; }
197 if (strMon ==
"JUL") { mon = 7; }
198 if (strMon ==
"AUG") { mon = 8; }
199 if (strMon ==
"SEP") { mon = 9; }
200 if (strMon ==
"OCT") { mon = 10; }
201 if (strMon ==
"NOV") { mon = 11; }
202 if (strMon ==
"DEZ") { mon = 12; }
203 auto year = 2000 + std::stoi(strDate.substr(7, 2));
206 else if (label ==
"DAZI")
216 azimuthDelta = deg2rad(std::stod(line.substr(2, 6)));
218 else if (label ==
"ZEN1 / ZEN2 / DZEN")
230 zenithStart = deg2rad(std::stod(line.substr(2, 6)));
231 zenithEnd = deg2rad(std::stod(line.substr(8, 6)));
232 zenithDelta = deg2rad(std::stod(line.substr(14, 6)));
234 else if (label ==
"VALID FROM")
236 validFrom =
InsTime(std::stoi(line.substr(0, 6)),
237 std::stoi(line.substr(6, 6)),
238 std::stoi(line.substr(12, 6)),
239 std::stoi(line.substr(18, 6)),
240 std::stoi(line.substr(24, 6)),
241 std::stod(line.substr(30, 13)),
244 else if (label ==
"VALID UNTIL")
246 validUntil =
InsTime(std::stoi(line.substr(0, 6)),
247 std::stoi(line.substr(6, 6)),
248 std::stoi(line.substr(12, 6)),
249 std::stoi(line.substr(18, 6)),
250 std::stoi(line.substr(24, 6)),
251 std::stod(line.substr(30, 13)),
254 else if (label ==
"START OF FREQUENCY")
259 LOG_WARN(
" AntexReader: Invalid frequency [{}] in line {} of file '{}'", line.substr(3, 3), lineNumber, entry.path());
263#if LOG_LEVEL <= LOG_LEVEL_DATA
264 patternLogging =
true;
268 return antInfo.from == validFrom && antInfo.until == validUntil;
272 antenna->
antennaInfo.emplace_back(date, validFrom, validUntil);
290 else if (label ==
"END OF FREQUENCY") { frequency =
Freq_None; }
293 if (label ==
"NORTH / EAST / UP")
295 antInfo->freqInformation[frequency].phaseCenterOffsetToARP = Eigen::Vector3d(str::stod(line.substr(0, 10), 0.0),
296 str::stod(line.substr(10, 10), 0.0),
297 str::stod(line.substr(20, 10), 0.0))
299 LOG_DATA(
" Adding antenna '{}' [{}]{} phaseCenterOffsetToARP: {}", antennaType,
Frequency(frequency),
301 antInfo->freqInformation.at(frequency).phaseCenterOffsetToARP.transpose());
303 else if (line.substr(3, 5) ==
"NOAZI")
312 Eigen::Matrix2Xd& pattern = antInfo->freqInformation.at(frequency).patternAzimuthIndependent;
313 pattern = Eigen::Matrix2Xd::Zero(2,
static_cast<int>(std::round(zenithEnd / zenithDelta)) + 1);
314 pattern.row(0).setLinSpaced(zenithStart, zenithEnd);
315 for (
int c = 0; c < pattern.cols(); c++)
317 pattern(1, c) = std::stod(line.substr((
static_cast<size_t>(c) + 1) * 8, 8)) * 1e-3;
319 LOG_DATA(
" Adding antenna '{}' [{}] NOAZI pattern", antennaType,
Frequency(frequency));
321 else if (azimuthDelta > 0.0)
331 Eigen::MatrixXd& pattern = antInfo->freqInformation.at(frequency).pattern;
333 if (pattern.cols() == 0)
335 pattern = Eigen::MatrixXd::Zero(
static_cast<int>(std::round(azimuthEnd / azimuthDelta)) + 2,
336 static_cast<int>(std::round(zenithEnd / zenithDelta)) + 2);
337 pattern.row(0).rightCols(pattern.cols() - 1).setLinSpaced(zenithStart, zenithEnd);
338 pattern.col(0).bottomRows(pattern.rows() - 1).setLinSpaced(azimuthStart, azimuthEnd);
340 double azimuth = deg2rad(std::stod(line.substr(0, 8)));
341 int r =
static_cast<int>(std::round((azimuth - azimuthStart) / azimuthDelta)) + 1;
342 for (
int c = 0; c < pattern.cols() - 1; c++)
344 pattern(r, c + 1) = std::stod(line.substr((
static_cast<size_t>(c) + 1) * 8, 8)) * 1e-3;
347#if LOG_LEVEL <= LOG_LEVEL_DATA
350 LOG_DATA(
" Adding antenna '{}' [{}] azimuth dependent pattern", antennaType,
Frequency(frequency));
351 patternLogging =
false;
359 LOG_DEBUG(
"Reading ANTEX file finished.");
376 [[maybe_unused]]
const std::string& nameId)
const
380 return antFreqInfo->get().phaseCenterOffsetToARP;
395 double elevation, std::optional<double> azimuth,
396 [[maybe_unused]]
const std::string& nameId)
const
398 LOG_DATA(
"{}: getAntennaPhaseCenterVariation({}, {}, {}, {}, {})", nameId, antennaType,
Frequency(freq), insTime.
toYMDHMS(
GPST), elevation, azimuth);
400 if (!antInfo.has_value()) {
return std::nullopt; }
402 double zenith = deg2rad(90.0) - elevation;
404 if (zenith < antInfo->get().zenithStart || zenith > antInfo->get().zenithEnd
405 || (azimuth && (azimuth < antInfo->get().azimuthStart || azimuth > antInfo->get().azimuthEnd)))
407 LOG_DATA(
"{}: The zenith or azimuth provided are outside the pattern in the ANTEX file", nameId);
411 if (!antFreqInfo.has_value()) {
return std::nullopt; }
413 if (!azimuth.has_value())
415 const Eigen::Matrix2Xd& pattern = antFreqInfo->get().patternAzimuthIndependent;
416 Eigen::Index zenithLoc = -1;
417 if (zenith == pattern(0, 0)) { zenithLoc = 1; }
418 else if (zenith == pattern(0, Eigen::last)) { zenithLoc = pattern.cols() - 1; }
421 zenithLoc = std::distance(
422 pattern.row(0).begin(),
423 std::upper_bound(pattern.row(0).begin(),
424 pattern.row(0).end(),
426 if (zenithLoc == 0) { zenithLoc++; }
428 Eigen::Index uLoc = zenithLoc - 1;
429 double a = pattern(0, uLoc);
430 double b = pattern(0, zenithLoc);
431 double t = (zenith - a) / (b - a);
433 LOG_DATA(
"{}: t = {:.3f} [a = {:.1f}°, z = {:.1f}°, b = {:.1f}°]", nameId, t, rad2deg(a), rad2deg(zenith), rad2deg(b));
434 LOG_DATA(
"{}: zenith {:.1f}° at idx {} = {:.5f}", nameId, rad2deg(zenith), zenithLoc, std::lerp(pattern(1, uLoc), pattern(1, zenithLoc), t));
436 return std::lerp(pattern(1, uLoc), pattern(1, zenithLoc), t);
439 const Eigen::MatrixXd& pattern = antFreqInfo->get().pattern;
440 Eigen::Index zenithLoc = -1;
441 Eigen::Index azimuthLoc = -1;
442 if (zenith == pattern(0, 1)) { zenithLoc = 2; }
443 else if (zenith == pattern(0, Eigen::last)) { zenithLoc = pattern.cols() - 1; }
446 zenithLoc = std::distance(
447 pattern.row(0).rightCols(pattern.cols() - 1).begin(),
448 std::upper_bound(pattern.row(0).rightCols(pattern.cols() - 1).begin(),
449 pattern.row(0).rightCols(pattern.cols() - 1).end(),
451 if (zenithLoc != pattern.cols() - 1) { zenithLoc++; }
453 if (*azimuth == pattern(1, 0)) { azimuthLoc = 2; }
454 else if (*azimuth == pattern(Eigen::last, 0)) { azimuthLoc = pattern.rows() - 1; }
457 azimuthLoc = std::distance(
458 pattern.col(0).bottomRows(pattern.rows() - 1).begin(),
459 std::upper_bound(pattern.col(0).bottomRows(pattern.rows() - 1).begin(),
460 pattern.col(0).bottomRows(pattern.rows() - 1).end(),
462 if (azimuthLoc != pattern.rows() - 1) { azimuthLoc++; }
465 Eigen::Index uZenithLoc = zenithLoc - 1;
466 double za = pattern(0, uZenithLoc);
467 double zb = pattern(0, zenithLoc);
468 double zt = (zenith - za) / (zb - za);
469 LOG_DATA(
"{}: zenith: t = {:.3f} [a = {:.1f}°, {:.1f}°, b = {:.1f}°]", nameId, zt, rad2deg(za), rad2deg(zenith), rad2deg(zb));
470 Eigen::Index uAzimuthLoc = azimuthLoc - 1;
471 double aa = pattern(uAzimuthLoc, 0);
472 double ab = pattern(azimuthLoc, 0);
473 double at = (*azimuth - aa) / (ab - aa);
474 LOG_DATA(
"{}: azimuth: t = {:.3f} [a = {:.1f}°, {:.1f}°, b = {:.1f}°]", nameId, at, rad2deg(aa), rad2deg(*azimuth), rad2deg(ab));
476 LOG_DATA(
"{}: bilinearInterpolation(tx = {:.1f}, ty = {:.1f}, c00 = {:.5f}, c10 = {:.5f}, c01 = {:.5f}, c11 = {:.5f})", nameId,
478 pattern(uAzimuthLoc, uZenithLoc), pattern(azimuthLoc, uZenithLoc),
479 pattern(uAzimuthLoc, zenithLoc), pattern(azimuthLoc, zenithLoc));
480 double v = math::bilinearInterpolation(at, zt,
481 pattern(uAzimuthLoc, uZenithLoc), pattern(azimuthLoc, uZenithLoc),
482 pattern(uAzimuthLoc, zenithLoc), pattern(azimuthLoc, zenithLoc));
483 LOG_DATA(
"{}: azimuth {:.1f}°, zenith {:.1f}° at idx ({},{}) = {:.5f}", nameId, rad2deg(*azimuth), rad2deg(zenith), azimuthLoc, zenithLoc, v);
508 mutable std::unordered_set<std::pair<std::string, Frequency_>>
_notFoundFreq;
514 std::optional<std::reference_wrapper<const Antenna::AntennaInfo>>
getAntennaInfo(
const std::string& antennaType,
const InsTime& insTime,
515 [[maybe_unused]]
const std::string& nameId)
const
521 LOG_WARN(
"{}: Antenna type '{}' is not found in the ANTEX files.",
522 nameId, antennaType);
528 const auto& antenna =
_antennas.at(antennaType);
530 auto antInfo = antenna.antennaInfo.cend();
531 if (antenna.antennaInfo.size() == 1)
533 antInfo = antenna.antennaInfo.begin();
538 return (antInfo.from.empty() && antInfo.until.empty())
539 || (antInfo.from.empty() && insTime <= antInfo.until)
540 || (antInfo.until.empty() && antInfo.from <= insTime)
541 || (antInfo.from <= insTime && insTime <= antInfo.until);
544 if (antInfo == antenna.antennaInfo.end())
546 antInfo = antenna.antennaInfo.cend() - 1;
559 const std::string& nameId)
const
575 const std::string& antennaType,
Frequency_ freq,
576 [[maybe_unused]]
const std::string& nameId)
const
582 if (!
_notFoundFreq.contains(std::make_pair(antennaType, freq)))
584 LOG_WARN(
"{}: Antenna type '{}' is does not have frequency [{}] in the ANTEX file."
585 " Please provide a new ANTEX file under 'resources/gnss/antex' or consider not using the frequency, as this can introduce height errors of several centimeter.",
Frequency definition for different satellite systems.
Frequency_
Enumerate for GNSS frequencies.
Definition Frequency.hpp:26
@ Freq_None
None.
Definition Frequency.hpp:27
The class is responsible for all time-related tasks.
Utility class for logging to console and file.
#define LOG_DEBUG
Debug information. Should not be called on functions which receive observations (spamming)
Definition Logger.hpp:67
#define LOG_DATA
All output which occurs repeatedly every time observations are received.
Definition Logger.hpp:29
#define LOG_ERROR
Error occurred, which stops part of the program to work, but not everything.
Definition Logger.hpp:73
#define LOG_WARN
Error occurred, but a fallback option exists and program continues to work normally.
Definition Logger.hpp:71
#define LOG_TRACE
Detailled info to trace the execution of the program. Should not be called on functions which receive...
Definition Logger.hpp:65
Utility functions for std::pair.
Utility functions for working with std::strings.
@ GPST
GPS Time.
Definition TimeSystem.hpp:30
ANTEX file reader.
Definition AntexReader.hpp:46
const std::set< std::string > & antennas() const
Antennas read from the ANTEX files.
Definition AntexReader.hpp:489
AntexReader()=default
Constructor.
std::optional< std::reference_wrapper< const Antenna::AntennaInfo > > getAntennaInfo(const std::string &antennaType, const InsTime &insTime, const std::string &nameId) const
Get the antenna info object.
Definition AntexReader.hpp:514
std::set< std::string > _antennaNames
Ordered names of all antennas.
Definition AntexReader.hpp:502
std::unordered_set< std::string > _notFoundAnt
List of Antennas not found to emit a warning.
Definition AntexReader.hpp:505
static AntexReader & Get()
Get the static Instance of the reader.
Definition AntexReader.hpp:98
std::optional< std::reference_wrapper< const AntennaFreqInfo > > getAntennaFrequencyInfo(const std::string &antennaType, Frequency_ freq, const InsTime &insTime, const std::string &nameId) const
Get the antenna frequency info object.
Definition AntexReader.hpp:557
std::unordered_set< std::pair< std::string, Frequency_ > > _notFoundFreq
List of Frequencies not found to emit a warning.
Definition AntexReader.hpp:508
std::optional< std::reference_wrapper< const AntennaFreqInfo > > getAntennaFrequencyInfo(const Antenna::AntennaInfo &antInfo, const std::string &antennaType, Frequency_ freq, const std::string &nameId) const
Get the antenna frequency info object.
Definition AntexReader.hpp:574
void initialize()
Initialize from ANTEX file.
Definition AntexReader.hpp:105
std::optional< double > getAntennaPhaseCenterVariation(const std::string &antennaType, Frequency_ freq, const InsTime &insTime, double elevation, std::optional< double > azimuth, const std::string &nameId) const
Gets the phase center variation for given elevation and azimuth.
Definition AntexReader.hpp:394
std::unordered_map< std::string, Antenna > _antennas
Antennas read from the ANTEX files.
Definition AntexReader.hpp:499
std::optional< Eigen::Vector3d > getAntennaPhaseCenterOffsetToARP(const std::string &antennaType, Frequency_ freq, const InsTime &insTime, const std::string &nameId) const
Get the Antenna Phase Center Offset To ARP if it is found in the ANTEX file.
Definition AntexReader.hpp:375
void reset()
Reset the temporary variables.
Definition AntexReader.hpp:363
Frequency definition for different satellite systems.
Definition Frequency.hpp:59
static Frequency fromString(const std::string &typeString)
Construct new object from std::string.
The class is responsible for all time-related tasks.
Definition InsTime.hpp:668
constexpr InsTime_YMDHMS toYMDHMS(TimeSystem timesys=UTC, int digits=-1) const
Converts this time object into a different format.
Definition InsTime.hpp:829
constexpr bool empty() const
Checks if the Time object has a value.
Definition InsTime.hpp:1047
void reset()
Resets the InsTime object.
Definition InsTime.hpp:1053
Antenna frequency dependant information.
Definition AntexReader.hpp:50
Eigen::Vector3d phaseCenterOffsetToARP
Eccentricities of the mean antenna phase center relative to the antenna reference point (ARP)....
Definition AntexReader.hpp:52
Eigen::MatrixXd pattern
Definition AntexReader.hpp:60
Eigen::Matrix2Xd patternAzimuthIndependent
Definition AntexReader.hpp:56
Antenna info.
Definition AntexReader.hpp:68
double azimuthStart
Azimuth start of the phase center variation pattern in [rad].
Definition AntexReader.hpp:83
AntennaInfo(const InsTime &date, const InsTime &from, const InsTime &until)
Constructor.
Definition AntexReader.hpp:73
InsTime date
Date of measurement.
Definition AntexReader.hpp:76
double zenithEnd
Zenith end of the phase center variation pattern in [rad].
Definition AntexReader.hpp:81
double zenithDelta
Zenith delta of the phase center variation pattern in [rad].
Definition AntexReader.hpp:82
double azimuthEnd
Azimuth end of the phase center variation pattern in [rad].
Definition AntexReader.hpp:84
double azimuthDelta
Azimuth delta of the phase center variation pattern in [rad].
Definition AntexReader.hpp:85
InsTime until
Valid until.
Definition AntexReader.hpp:78
InsTime from
Valid from.
Definition AntexReader.hpp:77
double zenithStart
Zenith start of the phase center variation pattern in [rad].
Definition AntexReader.hpp:80
std::unordered_map< Frequency_, AntennaFreqInfo > freqInformation
Frequency dependant information.
Definition AntexReader.hpp:88
Antenna information.
Definition AntexReader.hpp:65
std::vector< AntennaInfo > antennaInfo
Antenna Information.
Definition AntexReader.hpp:94
std::string serialNumber
Serial number.
Definition AntexReader.hpp:91