0.3.0
Loading...
Searching...
No Matches
AntexReader.hpp
Go to the documentation of this file.
1// This file is part of INSTINCT, the INS Toolkit for Integrated
2// Navigation Concepts and Training by the Institute of Navigation of
3// the University of Stuttgart, Germany.
4//
5// This Source Code Form is subject to the terms of the Mozilla Public
6// License, v. 2.0. If a copy of the MPL was not distributed with this
7// file, You can obtain one at https://mozilla.org/MPL/2.0/.
8
13
14#pragma once
15
16#include <algorithm>
17#include <cstddef>
18#include <filesystem>
19#include <fstream>
20#include <mutex>
21#include <set>
22#include <optional>
23#include <string>
24#include <unordered_map>
25#include <unordered_set>
26#include <utility>
27#include <vector>
32#include "Navigation/Transformations/Units.hpp"
34#include "util/Eigen.hpp"
35#include "util/Logger.hpp"
37#include "util/StringUtil.hpp"
38#include <Eigen/src/Core/Matrix.h>
39#include <fmt/core.h>
40
41namespace NAV
42{
43
46{
47 public:
50 {
52 Eigen::Vector3d phaseCenterOffsetToARP = Eigen::Vector3d::Zero();
56 Eigen::Matrix2Xd patternAzimuthIndependent;
60 Eigen::MatrixXd pattern;
61 };
62
64 struct Antenna
65 {
68 {
74 : date(date), from(from), until(until) {}
75
79
80 double zenithStart = 0.0;
81 double zenithEnd = 0.0;
82 double zenithDelta = 0.0;
83 double azimuthStart = 0.0;
84 double azimuthEnd = 0.0;
85 double azimuthDelta = 0.0;
86
88 std::unordered_map<Frequency_, AntennaFreqInfo> freqInformation;
89 };
91 std::string serialNumber;
92
94 std::vector<AntennaInfo> antennaInfo;
95 };
96
98 static AntexReader& Get()
99 {
100 static AntexReader self;
101 return self;
102 }
103
106 {
107 if (!_antennas.empty()) { return; }
108
109 LOG_DEBUG("Reading ANTEX files started...");
110
111 auto path = flow::GetProgramRootPath() / "resources" / "gnss" / "antex";
112 if (!std::filesystem::exists(path))
113 {
114 LOG_WARN("Not reading ANTEX files because path does not exist: {}", path);
115 return;
116 }
117 for (const auto& entry : std::filesystem::directory_iterator(path))
118 {
119 if (entry.path().extension() != ".atx") { continue; }
120 LOG_DEBUG("Reading {}...", entry.path().string());
121
122 std::ifstream fs(entry.path());
123 if (!fs.good())
124 {
125 LOG_ERROR("Could not read ANTEX file: {}", entry.path().string());
126 return;
127 }
128
129 auto extHeaderLabel = [](const std::string& line) {
130 return str::trim_copy(std::string_view(line).substr(60, 20));
131 };
132
133 std::string line;
134 size_t lineNumber = 0;
135
136 bool antennaStarted = false;
137 Antenna* antenna = nullptr;
138 std::vector<NAV::AntexReader::Antenna::AntennaInfo>::iterator antInfo;
139 std::string antennaType;
140 InsTime date;
141 InsTime validFrom;
142 InsTime validUntil;
143 Frequency_ frequency = Freq_None;
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;
152#endif
153 while (std::getline(fs, line) && !fs.eof())
154 {
155 lineNumber++;
156 auto label = extHeaderLabel(line);
157 if (label == "START OF ANTENNA") { antennaStarted = true; }
158 else if (label == "END OF ANTENNA")
159 {
160 antennaStarted = false;
161 antenna = nullptr;
162 antInfo = {};
163 antennaType.clear();
164 date.reset();
165 validFrom.reset();
166 validUntil.reset();
167 azimuthDelta = 0.0;
168 zenithStart = 0.0;
169 zenithEnd = 0.0;
170 zenithDelta = 0.0;
171 }
172 else if (antennaStarted)
173 {
174 if (label == "TYPE / SERIAL NO")
175 {
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; }
179
180 antenna = &_antennas[antennaType];
181 antenna->serialNumber = serialNumber;
182 _antennaNames.insert(antennaType);
183 }
184 else if (label == "METH / BY / # / DATE")
185 {
186 // " COD/ESA 0 29-JAN-17 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);
190 int mon = 0;
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));
204 date = InsTime(year, mon, day, 0, 0, 0.0, GPST);
205 }
206 else if (label == "DAZI")
207 {
208 // Increment of the azimuth: 2X,F6.1,52X
209 // 0 to 360 with increment 'DAZI' (in degrees).
210 // 360 degrees have to be divisible by 'DAZI'.
211 // Common value for 'DAZI': 5.0
212 // For non-azimuth-dependent phase center
213 // variations '0.0' has to be specified.
214 //
215 // 5.0 DAZI
216 azimuthDelta = deg2rad(std::stod(line.substr(2, 6)));
217 }
218 else if (label == "ZEN1 / ZEN2 / DZEN")
219 {
220 // Receiver antenna: 2X,3F6.1,40X
221 // Definition of the grid in zenith angle:
222 // Zenith distance 'ZEN1' to 'ZEN2' with increment 'DZEN' (in degrees).
223 // 'DZEN' has to be > 0.0.
224 // 'ZEN1' and 'ZEN2' always have to be multiples of 'DZEN'.
225 // 'ZEN2' always has to be greater than 'ZEN1'.
226 // Common value for 'DZEN': 5.0
227 // Example: ' 0.0 90.0 5.0'
228 //
229 // 0.0 90.0 5.0 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)));
233 }
234 else if (label == "VALID FROM")
235 {
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)),
242 GPST);
243 }
244 else if (label == "VALID UNTIL")
245 {
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)),
252 GPST);
253 }
254 else if (label == "START OF FREQUENCY")
255 {
256 frequency = Frequency_(Frequency::fromString(line.substr(3, 3)));
257 if (frequency == Freq_None)
258 {
259 LOG_WARN(" AntexReader: Invalid frequency [{}] in line {} of file '{}'", line.substr(3, 3), lineNumber, entry.path());
260 continue;
261 }
262
263#if LOG_LEVEL <= LOG_LEVEL_DATA
264 patternLogging = true;
265#endif
266
267 antInfo = std::ranges::find_if(antenna->antennaInfo, [&](const Antenna::AntennaInfo& antInfo) {
268 return antInfo.from == validFrom && antInfo.until == validUntil;
269 });
270 if (antInfo == antenna->antennaInfo.end())
271 {
272 antenna->antennaInfo.emplace_back(date, validFrom, validUntil);
273 antInfo = antenna->antennaInfo.end() - 1;
274 }
275
276 antInfo->zenithStart = zenithStart;
277 antInfo->zenithEnd = zenithEnd;
278 antInfo->zenithDelta = zenithDelta;
279 antInfo->azimuthStart = azimuthStart;
280 antInfo->azimuthEnd = azimuthEnd;
281 antInfo->azimuthDelta = azimuthDelta;
282
283 if (antInfo->freqInformation.contains(frequency) && antInfo->date > date)
284 {
285 LOG_TRACE(" Antenna '{}' [{}]{} already exists.", antennaType, Frequency(frequency),
286 !validFrom.empty() || !validUntil.empty() ? fmt::format(" (valid [{}] - [{}])", validFrom.toYMDHMS(GPST), validUntil.toYMDHMS(GPST)) : "");
287 frequency = Freq_None;
288 }
289 }
290 else if (label == "END OF FREQUENCY") { frequency = Freq_None; }
291 else if (frequency != Freq_None)
292 {
293 if (label == "NORTH / EAST / UP")
294 {
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))
298 * 1e-3;
299 LOG_DATA(" Adding antenna '{}' [{}]{} phaseCenterOffsetToARP: {}", antennaType, Frequency(frequency),
300 !validFrom.empty() || !validUntil.empty() ? fmt::format(" (valid [{}] - [{}])", validFrom.toYMDHMS(GPST), validUntil.toYMDHMS(GPST)) : "",
301 antInfo->freqInformation.at(frequency).phaseCenterOffsetToARP.transpose());
302 }
303 else if (line.substr(3, 5) == "NOAZI")
304 {
305 // (Values of a non- | The flag 'NOAZI' denotes a non-azimuth- | 3X,A5,mF8.2
306 // azimuth-dependent | dependent pattern that has to be |
307 // pattern) | specified in any case (also if 'DAZI' > |
308 // | 0.0). |
309 // | Phase pattern values in millimeters from |
310 // | 'ZEN1' to 'ZEN2' (with increment 'DZEN').|
311 // | All values on one line. |
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++)
316 {
317 pattern(1, c) = std::stod(line.substr((static_cast<size_t>(c) + 1) * 8, 8)) * 1e-3;
318 }
319 LOG_DATA(" Adding antenna '{}' [{}] NOAZI pattern", antennaType, Frequency(frequency));
320 }
321 else if (azimuthDelta > 0.0)
322 {
323 // (Values of an | The azimuth-dependent pattern has to be | F8.1,mF8.2
324 // azimuth-dependent | specified, if 'DAZI' > 0.0. The first |
325 // pattern) | value in each line denotes the azimuth |
326 // | angle followed by the phase pattern |
327 // | values in millimeters from 'ZEN1' to |
328 // | 'ZEN2' (with increment 'DZEN'). |
329 // | All values of one azimuth angle on one |
330 // | line. |
331 Eigen::MatrixXd& pattern = antInfo->freqInformation.at(frequency).pattern;
332
333 if (pattern.cols() == 0)
334 {
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);
339 }
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++)
343 {
344 pattern(r, c + 1) = std::stod(line.substr((static_cast<size_t>(c) + 1) * 8, 8)) * 1e-3;
345 }
346
347#if LOG_LEVEL <= LOG_LEVEL_DATA
348 if (patternLogging)
349 {
350 LOG_DATA(" Adding antenna '{}' [{}] azimuth dependent pattern", antennaType, Frequency(frequency));
351 patternLogging = false;
352 }
353#endif
354 }
355 }
356 }
357 }
358 }
359 LOG_DEBUG("Reading ANTEX file finished.");
360 }
361
363 void reset()
364 {
365 _notFoundAnt.clear();
366 _notFoundFreq.clear();
367 }
368
375 std::optional<Eigen::Vector3d> getAntennaPhaseCenterOffsetToARP(const std::string& antennaType, Frequency_ freq, const InsTime& insTime,
376 [[maybe_unused]] const std::string& nameId) const
377 {
378 if (auto antFreqInfo = getAntennaFrequencyInfo(antennaType, freq, insTime, nameId))
379 {
380 return antFreqInfo->get().phaseCenterOffsetToARP;
381 }
382
383 return std::nullopt;
384 }
385
394 std::optional<double> getAntennaPhaseCenterVariation(const std::string& antennaType, Frequency_ freq, const InsTime& insTime,
395 double elevation, std::optional<double> azimuth,
396 [[maybe_unused]] const std::string& nameId) const
397 {
398 LOG_DATA("{}: getAntennaPhaseCenterVariation({}, {}, {}, {}, {})", nameId, antennaType, Frequency(freq), insTime.toYMDHMS(GPST), elevation, azimuth);
399 auto antInfo = getAntennaInfo(antennaType, insTime, nameId);
400 if (!antInfo.has_value()) { return std::nullopt; }
401
402 double zenith = deg2rad(90.0) - elevation;
403
404 if (zenith < antInfo->get().zenithStart || zenith > antInfo->get().zenithEnd
405 || (azimuth && (azimuth < antInfo->get().azimuthStart || azimuth > antInfo->get().azimuthEnd)))
406 {
407 LOG_DATA("{}: The zenith or azimuth provided are outside the pattern in the ANTEX file", nameId);
408 return std::nullopt;
409 }
410 auto antFreqInfo = getAntennaFrequencyInfo(antInfo->get(), antennaType, freq, nameId);
411 if (!antFreqInfo.has_value()) { return std::nullopt; }
412
413 if (!azimuth.has_value())
414 {
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; }
419 else
420 {
421 zenithLoc = std::distance(
422 pattern.row(0).begin(),
423 std::upper_bound(pattern.row(0).begin(),
424 pattern.row(0).end(),
425 zenith));
426 if (zenithLoc == 0) { zenithLoc++; }
427 }
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);
432
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));
435
436 return std::lerp(pattern(1, uLoc), pattern(1, zenithLoc), t);
437 }
438
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; }
444 else
445 {
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(),
450 zenith));
451 if (zenithLoc != pattern.cols() - 1) { zenithLoc++; }
452 }
453 if (*azimuth == pattern(1, 0)) { azimuthLoc = 2; }
454 else if (*azimuth == pattern(Eigen::last, 0)) { azimuthLoc = pattern.rows() - 1; }
455 else
456 {
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(),
461 *azimuth));
462 if (azimuthLoc != pattern.rows() - 1) { azimuthLoc++; }
463 }
464
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));
475
476 LOG_DATA("{}: bilinearInterpolation(tx = {:.1f}, ty = {:.1f}, c00 = {:.5f}, c10 = {:.5f}, c01 = {:.5f}, c11 = {:.5f})", nameId,
477 zt, at,
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);
484
485 return v;
486 }
487
489 const std::set<std::string>& antennas() const
490 {
491 return _antennaNames;
492 };
493
494 private:
496 AntexReader() = default;
497
499 std::unordered_map<std::string, Antenna> _antennas;
500
502 std::set<std::string> _antennaNames;
503
505 mutable std::unordered_set<std::string> _notFoundAnt;
506
508 mutable std::unordered_set<std::pair<std::string, Frequency_>> _notFoundFreq;
509
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
516 {
517 if (!_antennas.contains(antennaType))
518 {
519 if (!_notFoundAnt.contains(antennaType))
520 {
521 LOG_WARN("{}: Antenna type '{}' is not found in the ANTEX files.",
522 nameId, antennaType);
523 _notFoundAnt.insert(antennaType);
524 }
525 return std::nullopt;
526 }
527
528 const auto& antenna = _antennas.at(antennaType);
529
530 auto antInfo = antenna.antennaInfo.cend();
531 if (antenna.antennaInfo.size() == 1) // One element only, so take it
532 {
533 antInfo = antenna.antennaInfo.begin();
534 }
535 else // No element is not possible, so more than one here, so search for time
536 {
537 antInfo = std::ranges::find_if(antenna.antennaInfo, [&](const Antenna::AntennaInfo& antInfo) {
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);
542 });
543 }
544 if (antInfo == antenna.antennaInfo.end()) // None matching, so take last
545 {
546 antInfo = antenna.antennaInfo.cend() - 1;
547 }
548
549 return *antInfo;
550 }
551
557 std::optional<std::reference_wrapper<const AntennaFreqInfo>> getAntennaFrequencyInfo(const std::string& antennaType, Frequency_ freq,
558 const InsTime& insTime,
559 const std::string& nameId) const
560 {
561 if (auto antInfo = getAntennaInfo(antennaType, insTime, nameId))
562 {
563 return getAntennaFrequencyInfo(antInfo->get(), antennaType, freq, nameId);
564 }
565
566 return std::nullopt;
567 }
568
574 std::optional<std::reference_wrapper<const AntennaFreqInfo>> getAntennaFrequencyInfo(const Antenna::AntennaInfo& antInfo,
575 const std::string& antennaType, Frequency_ freq,
576 [[maybe_unused]] const std::string& nameId) const
577 {
578 if (antInfo.freqInformation.contains(freq))
579 {
580 return antInfo.freqInformation.at(freq);
581 }
582 if (!_notFoundFreq.contains(std::make_pair(antennaType, freq)))
583 {
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.",
586 nameId, antennaType, Frequency(freq));
587 _notFoundFreq.insert(std::make_pair(antennaType, freq));
588 }
589
590 return std::nullopt;
591 }
592};
593
594} // namespace NAV
Vector space operations.
Save/Load the Nodes.
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.
GNSS Satellite System.
Utility functions for working with std::strings.
Time System defintions.
@ 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