0.3.0
Loading...
Searching...
No Matches
Gravity.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
14
15#pragma once
16
17#include <cstdint>
18#include <Eigen/Dense>
19#include <Eigen/Core>
20
22#include "util/Logger.hpp"
25#include "internal/egm96Coeffs.hpp"
26
27namespace NAV
28{
30enum class GravitationModel : uint8_t
31{
32 None,
33 WGS84,
36 EGM96,
37 COUNT,
38};
39
43const char* to_string(GravitationModel gravitationModel);
44
48bool ComboGravitationModel(const char* label, GravitationModel& gravitationModel);
49
57template<std::floating_point Scalar>
58[[nodiscard]] Eigen::Vector3<Scalar> n_calcGravitation_SomiglianaAltitude(const Scalar& latitude, const Scalar& altitude)
59{
60 // eq 6.16 has a fault in the denominator, it should be a sin^2(latitude)
61 auto g_0 = 9.7803253359 * (1.0 + 1.931853e-3 * std::pow(std::sin(latitude), 2))
62 / std::sqrt(1.0 - InsConst::WGS84::e_squared * std::pow(std::sin(latitude), 2));
63
64 // Altitude compensation (Matlab example from Chapter 6_GNSS_INS_1 - glocal.m)
65 auto k = 1
66 - (2 * altitude / InsConst::WGS84::a)
68 + (std::pow(InsConst::omega_ie * InsConst::WGS84::a, 2))
70 + 3 * std::pow(altitude / InsConst::WGS84::a, 2);
71
72 return { 0.0, 0.0, k * g_0 };
73}
74
82template<std::floating_point Scalar>
83[[nodiscard]] Eigen::Vector3<Scalar> n_calcGravitation_WGS84_Skydel(const Scalar& latitude, const Scalar& altitude)
84{
85 // geocentric latitude determination from geographic latitude
86 auto latitudeGeocentric = std::atan((std::pow(InsConst::WGS84::b, 2.0) / std::pow(InsConst::WGS84::a, 2.0)) * std::tan(latitude));
87 // effective radius determination, i.e. earth radius on WGS84 spheroid plus local altitude --> possible error!! altitude in lla should be added rather than subtracted!
88 auto radiusSpheroid = InsConst::WGS84::a * (1.0 - InsConst::WGS84::f * std::pow(std::sin(latitudeGeocentric), 2.0)) - altitude;
89
90 // Derivation of gravity, i.e. gravitational potential derived after effective radius
91 auto gravitationMagnitude = InsConst::WGS84::MU * std::pow(radiusSpheroid, -2.0)
92 - 3 * InsConst::WGS84::MU * InsConst::WGS84::J2 * std::pow(InsConst::WGS84::a, 2.0) * 0.5 * std::pow(radiusSpheroid, -4.0) * (3 * std::pow(std::sin(latitudeGeocentric), 2.0) - 1)
93 - std::pow(InsConst::omega_ie_Skydel, 2.0) * radiusSpheroid * std::pow(std::cos(latitudeGeocentric), 2.0);
94
95 return { 0, 0, gravitationMagnitude };
96}
97
105template<std::floating_point Scalar>
106[[nodiscard]] Eigen::Vector3<Scalar> n_calcGravitation_WGS84(const Scalar& latitude, const Scalar& altitude)
107{
108 // Geocentric latitude determination from geographic latitude
109 auto latitudeGeocentric = std::atan((std::pow(InsConst::WGS84::b, 2.0) / std::pow(InsConst::WGS84::a, 2.0)) * std::tan(latitude));
110 // Radius of spheroid determination
111 auto radiusSpheroid = InsConst::WGS84::a * (1.0 - InsConst::WGS84::f * std::pow(std::sin(latitudeGeocentric), 2.0)) + altitude;
112
113 // Magnitude of the gravity, i.e. without orientation
114 auto gravitationMagnitude = InsConst::WGS84::MU * std::pow(radiusSpheroid, -2.0)
115 - 3 * InsConst::WGS84::MU * InsConst::WGS84::J2 * std::pow(InsConst::WGS84::a, 2.0) * 0.5 * std::pow(radiusSpheroid, -4.0) * (3 * std::pow(std::sin(latitudeGeocentric), 2.0) - 1)
116 - std::pow(InsConst::omega_ie, 2.0) * radiusSpheroid * std::pow(std::cos(latitudeGeocentric), 2.0);
117
118 return { 0, 0, gravitationMagnitude };
119}
120
128template<typename Derived>
129[[nodiscard]] Eigen::Vector3<typename Derived::Scalar> n_calcGravitation_EGM96(const Eigen::MatrixBase<Derived>& lla_position, size_t ndegree = 10)
130{
133
134 auto e_position = trafo::lla2ecef_WGS84(lla_position);
135
136 // Geocentric latitude determination from Groves (2013) - eq. (2.114)
137 auto latitudeGeocentric = std::atan(e_position(2) / std::sqrt(e_position(0) * e_position(0) + e_position(1) * e_position(1)));
138
139 // Spherical coordinates
140 auto radius = std::sqrt(e_position(0) * e_position(0) + e_position(1) * e_position(1) + e_position(2) * e_position(2));
141 auto elevation = M_PI_2 - latitudeGeocentric; // [rad]
142 auto azimuth = lla_position(1); // [rad]
143
144 // Gravitation vector in local-navigation frame coordinates in [m/s^2]
145 Eigen::Vector3<typename Derived::Scalar> n_gravitation = Eigen::Vector3<typename Derived::Scalar>::Zero();
146
147 typename Derived::Scalar Pnm = 0;
148 typename Derived::Scalar Pnmd = 0;
149
150 auto coeffsRows = egm96Coeffs.size();
151
152 // Associated Legendre Polynomial Coefficients 'P' and their derivatives 'Pd'
153 auto [P, Pd] = associatedLegendre(static_cast<double>(elevation), ndegree);
154
155 for (size_t i = 0; i < coeffsRows; i++) // NOLINT(clang-analyzer-core.UndefinedBinaryOperatorResult) // FIXME: Wrong error message about Eigen (error: The left operand of '*' is a garbage value)
156 {
157 // Retrieving EGM96 coefficients
158 auto n = static_cast<int>(egm96Coeffs.at(i).at(0)); // Degree of the Associated Legendre Polynomial
159 auto m = static_cast<int>(egm96Coeffs.at(i).at(1)); // Order of the Associated Legendre Polynomial
160 auto C = egm96Coeffs.at(i).at(2);
161 auto S = egm96Coeffs.at(i).at(3);
162
163 if (static_cast<size_t>(n) == ndegree + 1)
164 {
165 // Ending of the for-loop once the iterated 'n' becomes larger than the user-defined 'ndegree'
166 i = coeffsRows;
167 }
168 else
169 {
170 // Retrieving the parameters of the associated Legendre Polynomials
171 Pnm = P(n, m);
172 Pnmd = Pd(n, m);
173
174 auto nd = static_cast<double>(n);
175 auto md = static_cast<double>(m);
176
177 // Gravity vector from differentiation of the gravity potential in spherical coordinates (see 'GUT User Guide' eq. 7.4.2)
178 n_gravitation(0) += std::pow((InsConst::WGS84::a / radius), nd) * (C * std::cos(md * azimuth) + S * std::sin(md * azimuth)) * Pnmd;
179 n_gravitation(1) += std::pow((InsConst::WGS84::a / radius), nd) * md * (C * std::sin(md * azimuth) - S * std::cos(md * azimuth)) * Pnm;
180 n_gravitation(2) += (nd + 1.0) * std::pow((InsConst::WGS84::a / radius), nd) * (C * std::cos(md * azimuth) + S * std::sin(md * azimuth)) * Pnm;
181 }
182 }
183
184 return { -InsConst::WGS84::MU / (radius * radius) * n_gravitation(0),
185 (1.0 / std::sin(elevation)) * (-InsConst::WGS84::MU / (radius * radius)) * n_gravitation(1),
186 InsConst::WGS84::MU / (radius * radius) * (1.0 + n_gravitation(2)) };
187}
188
193template<typename Derived>
194[[nodiscard]] Eigen::Vector3<typename Derived::Scalar> n_calcGravitation(const Eigen::MatrixBase<Derived>& lla_position,
195 GravitationModel gravitationModel = GravitationModel::EGM96)
196{
197 const typename Derived::Scalar& latitude = lla_position(0);
198 const typename Derived::Scalar& altitude = lla_position(2);
199
200 if (gravitationModel == GravitationModel::WGS84)
201 {
202 return n_calcGravitation_WGS84(latitude, altitude);
203 }
204 if (gravitationModel == GravitationModel::WGS84_Skydel) // TODO: This function becomes obsolete, once the ImuStream is deactivated due to the 'InstinctDataStream'
205 {
206 return n_calcGravitation_WGS84_Skydel(latitude, altitude);
207 }
208 if (gravitationModel == GravitationModel::Somigliana)
209 {
210 return n_calcGravitation_SomiglianaAltitude(latitude, altitude);
211 }
212 if (gravitationModel == GravitationModel::EGM96)
213 {
214 return n_calcGravitation_EGM96(lla_position);
215 }
216 return Eigen::Vector3<typename Derived::Scalar>::Zero();
217}
218
219} // namespace NAV
Associated Legendre Polynomials for EGM96.
Holds all Constants.
Transformation collection.
@ None
None.
Definition GlobalActions.hpp:21
Eigen::Vector3< typename Derived::Scalar > n_calcGravitation(const Eigen::MatrixBase< Derived > &lla_position, GravitationModel gravitationModel=GravitationModel::EGM96)
Calculates the gravitation (acceleration due to mass attraction of the Earth)
Definition Gravity.hpp:194
GravitationModel
Available Gravitation Models.
Definition Gravity.hpp:31
@ WGS84_Skydel
World Geodetic System 1984 implemented by the Skydel Simulator // FIXME: Remove after Skydel uses the...
@ WGS84
World Geodetic System 1984.
@ EGM96
Earth Gravitational Model 1996.
@ Somigliana
Somigliana gravity model.
Eigen::Vector3< Scalar > n_calcGravitation_WGS84(const Scalar &latitude, const Scalar &altitude)
Calculates the gravitation (acceleration due to mass attraction of the Earth) at the WGS84 reference ...
Definition Gravity.hpp:106
Eigen::Vector3< Scalar > n_calcGravitation_SomiglianaAltitude(const Scalar &latitude, const Scalar &altitude)
Calculates the gravitation (acceleration due to mass attraction of the Earth) at the WGS84 reference ...
Definition Gravity.hpp:58
bool ComboGravitationModel(const char *label, GravitationModel &gravitationModel)
Shows a ComboBox to select the gravitation model.
Eigen::Vector3< typename Derived::Scalar > n_calcGravitation_EGM96(const Eigen::MatrixBase< Derived > &lla_position, size_t ndegree=10)
Calculates the gravitation (acceleration due to mass attraction of the Earth) at the WGS84 reference ...
Definition Gravity.hpp:129
Eigen::Vector3< Scalar > n_calcGravitation_WGS84_Skydel(const Scalar &latitude, const Scalar &altitude)
Calculates the gravitation (acceleration due to mass attraction of the Earth) at the WGS84 reference ...
Definition Gravity.hpp:83
@ COUNT
Amount of items in the enum.
Utility class for logging to console and file.
const char * to_string(gui::widgets::PositionWithFrame::ReferenceFrame refFrame)
Converts the enum to a string.
static constexpr double f
Flattening f = (a-b)/a.
Definition Constants.hpp:52
static constexpr double J2
Dynamic form factor, derived [-].
Definition Constants.hpp:60
static constexpr double b
Semi-minor axis = polar radius.
Definition Constants.hpp:54
static constexpr double MU
Gravitational constant (mass of Earth’s atmosphere included) [m³/s²].
Definition Constants.hpp:58
static constexpr double a
Semi-major axis = equatorial radius.
Definition Constants.hpp:50
static constexpr double e_squared
Square of the first eccentricity of the ellipsoid.
Definition Constants.hpp:56
static constexpr double omega_ie
Nominal mean angular velocity of the Earth in [rad/s].
Definition Constants.hpp:217
static constexpr double omega_ie_Skydel
Nominal mean angular velocity of the Earth in [rad/s]. Value implemented by the Skydel GNSS simulator...
Definition Constants.hpp:219
std::array< std::array< double, 6 >, 65338 > egm96Coeffs
EGM96 coefficients.
std::pair< Eigen::MatrixXd, Eigen::MatrixXd > associatedLegendre(double theta, size_t ndegree=10)
Calculates the associated Legendre Polynomial coefficients necessary for the EGM96.