JSBSim Flight Dynamics Model 1.2.2 (22 Mar 2025)
An Open Source Flight Dynamics and Control Software Library in C++
Loading...
Searching...
No Matches
FGInertial.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGInertial.cpp
4 Author: Jon S. Berndt
5 Date started: 09/13/00
6 Purpose: Encapsulates the inertial frame forces (coriolis and centrifugal)
7
8 ------------- Copyright (C) 2000 Jon S. Berndt (jon@jsbsim.org) -------------
9
10 This program is free software; you can redistribute it and/or modify it under
11 the terms of the GNU Lesser General Public License as published by the Free
12 Software Foundation; either version 2 of the License, or (at your option) any
13 later version.
14
15 This program is distributed in the hope that it will be useful, but WITHOUT
16 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
17 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
18 details.
19
20 You should have received a copy of the GNU Lesser General Public License along
21 with this program; if not, write to the Free Software Foundation, Inc., 59
22 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23
24 Further information about the GNU Lesser General Public License can also be
25 found on the world wide web at http://www.gnu.org.
26
27FUNCTIONAL DESCRIPTION
28--------------------------------------------------------------------------------
29
30HISTORY
31--------------------------------------------------------------------------------
3209/13/00 JSB Created
33
34%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
35INCLUDES
36%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
37
38#include "FGInertial.h"
39#include "input_output/FGXMLElement.h"
40#include "GeographicLib/Geodesic.hpp"
41
42using namespace std;
43
44namespace JSBSim {
45
46/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
47CLASS IMPLEMENTATION
48%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
49
50
51FGInertial::FGInertial(FGFDMExec* fgex)
52 : FGModel(fgex)
53{
54 Name = "Earth";
55
56 // Earth defaults
57 double RotationRate = 0.00007292115;
58 GM = 14.0764417572E15; // WGS84 value
59 J2 = 1.08262982E-03; // WGS84 value for J2
60 a = 20925646.32546; // WGS84 semimajor axis length in feet
61 b = 20855486.5951; // WGS84 semiminor axis length in feet
62 gravType = gtWGS84;
63
64 // Lunar defaults
65 /*
66 double RotationRate = 0.0000026617;
67 GM = 1.7314079E14; // Lunar GM
68 J2 = 2.033542482111609E-4; // value for J2
69 a = 5702559.05; // semimajor axis length in feet
70 b = 5695439.63; // semiminor axis length in feet
71 */
72
73 vOmegaPlanet = { 0.0, 0.0, RotationRate };
74 GroundCallback = std::make_unique<FGDefaultGroundCallback>(a, b);
75
76 bind();
77
78 Debug(0);
79}
80
81//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
82
83FGInertial::~FGInertial(void)
84{
85 Debug(1);
86}
87
88//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
89
90bool FGInertial::Load(Element* el)
91{
92 if (!Upload(el, true)) return false;
93
94 Name = el->GetAttributeValue("name");
95
96 if (el->FindElement("semimajor_axis"))
97 a = el->FindElementValueAsNumberConvertTo("semimajor_axis", "FT");
98 else if (el->FindElement("equatorial_radius"))
99 a = el->FindElementValueAsNumberConvertTo("equatorial_radius", "FT");
100 if (el->FindElement("semiminor_axis"))
101 b = el->FindElementValueAsNumberConvertTo("semiminor_axis", "FT");
102 else if (el->FindElement("polar_radius"))
103 b = el->FindElementValueAsNumberConvertTo("polar_radius", "FT");
104 // Trigger GeographicLib exceptions if the equatorial or polar radii are
105 // ill-defined.
106 // This intercepts the exception before being thrown by a destructor.
107 GeographicLib::Geodesic geod(a, 1.-b/a);
108
109 if (el->FindElement("rotation_rate")) {
110 double RotationRate = el->FindElementValueAsNumberConvertTo("rotation_rate", "RAD/SEC");
111 vOmegaPlanet = {0., 0., RotationRate};
112 }
113 if (el->FindElement("GM"))
114 GM = el->FindElementValueAsNumberConvertTo("GM", "FT3/SEC2");
115 if (el->FindElement("J2"))
116 J2 = el->FindElementValueAsNumber("J2"); // Dimensionless
117
118 GroundCallback->SetEllipse(a, b);
119
120 // Messages to warn the user about possible inconsistencies.
121 if (debug_lvl > 0) {
122 if (a != b && J2 == 0.0)
123 cout << "Gravitational constant J2 is null for a non-spherical planet." << endl;
124 if (a == b && J2 != 0.0)
125 cout << "Gravitational constant J2 is non-zero for a spherical planet." << endl;
126 }
127
128 Debug(2);
129
130 return true;
131}
132
133//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
134
135bool FGInertial::Run(bool Holding)
136{
137 // Fast return if we have nothing to do ...
138 if (FGModel::Run(Holding)) return true;
139 if (Holding) return false;
140
141 // Gravitation accel
142 switch (gravType) {
143 case gtStandard:
144 {
145 double radius = in.Position.GetRadius();
146 vGravAccel = -(GetGAccel(radius) / radius) * in.Position;
147 }
148 break;
149 case gtWGS84:
150 vGravAccel = GetGravityJ2(in.Position);
151 break;
152 }
153
154 return false;
155}
156
157//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
158
159FGMatrix33 FGInertial::GetTl2ec(const FGLocation& location) const
160{
161 FGColumnVector3 North, Down, East{-location(eY), location(eX), 0.};
162
163 switch (gravType) {
164 case gtStandard:
165 {
166 Down = location;
167 Down *= -1.0;
168 }
169 break;
170 case gtWGS84:
171 {
172 FGLocation sea_level = location;
173 sea_level.SetPositionGeodetic(location.GetLongitude(),
174 location.GetGeodLatitudeRad(), 0.0);
175 Down = GetGravityJ2(location);
176 Down -= vOmegaPlanet*(vOmegaPlanet*sea_level);}
177 }
178 Down.Normalize();
179 East.Normalize();
180 North = East*Down;
181
182 return FGMatrix33{North(eX), East(eX), Down(eX),
183 North(eY), East(eY), Down(eY),
184 North(eZ), 0.0, Down(eZ)};
185}
186
187//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
188
189double FGInertial::GetGAccel(double r) const
190{
191 return GM/(r*r);
192}
193
194//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
195//
196// Calculate the WGS84 gravitation value in ECEF frame. Pass in the ECEF
197// position via the position parameter. The J2Gravity value returned is in ECEF
198// frame, and therefore may need to be expressed (transformed) in another frame,
199// depending on how it is used. See Stevens and Lewis eqn. 1.4-16.
200
201FGColumnVector3 FGInertial::GetGravityJ2(const FGLocation& position) const
202{
203 FGColumnVector3 J2Gravity;
204
205 // Gravitation accel
206 double r = position.GetRadius();
207 double sinLat = sin(position.GetLatitude());
208
209 double adivr = a/r;
210 double preCommon = 1.5*J2*adivr*adivr;
211 double xy = 1.0 - 5.0*(sinLat*sinLat);
212 double z = 3.0 - 5.0*(sinLat*sinLat);
213 double GMOverr2 = GM/(r*r);
214
215 J2Gravity(1) = -GMOverr2 * ((1.0 + (preCommon * xy)) * position(eX)/r);
216 J2Gravity(2) = -GMOverr2 * ((1.0 + (preCommon * xy)) * position(eY)/r);
217 J2Gravity(3) = -GMOverr2 * ((1.0 + (preCommon * z)) * position(eZ)/r);
218
219 return J2Gravity;
220}
221
222//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
223
224void FGInertial::SetAltitudeAGL(FGLocation& location, double altitudeAGL)
225{
226 FGColumnVector3 vDummy;
227 FGLocation contact;
228 contact.SetEllipse(a, b);
229 GroundCallback->GetAGLevel(location, contact, vDummy, vDummy, vDummy);
230 double groundHeight = contact.GetGeodAltitude();
231 double longitude = location.GetLongitude();
232 double geodLat = location.GetGeodLatitudeRad();
233 location.SetPositionGeodetic(longitude, geodLat,
234 groundHeight + altitudeAGL);
235}
236
237//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
238
239void FGInertial::SetGravityType(int gt)
240{
241 // Messages to warn the user about possible inconsistencies.
242 switch (gt)
243 {
244 case eGravType::gtStandard:
245 if (a != b)
246 cout << "Warning: Standard gravity model has been set for a non-spherical planet" << endl;
247 break;
248 case eGravType::gtWGS84:
249 if (J2 == 0.0)
250 cout << "Warning: WGS84 gravity model has been set without specifying the J2 gravitational constant." << endl;
251 }
252
253 gravType = gt;
254}
255
256//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
257
258void FGInertial::bind(void)
259{
260 PropertyManager->Tie("inertial/sea-level-radius_ft", &in.Position,
261 &FGLocation::GetSeaLevelRadius);
262 PropertyManager->Tie("simulation/gravity-model", this, &FGInertial::GetGravityType,
263 &FGInertial::SetGravityType);
264}
265
266//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
267// The bitmasked value choices are as follows:
268// unset: In this case (the default) JSBSim would only print
269// out the normally expected messages, essentially echoing
270// the config files as they are read. If the environment
271// variable is not set, debug_lvl is set to 1 internally
272// 0: This requests JSBSim not to output any messages
273// whatsoever.
274// 1: This value explicity requests the normal JSBSim
275// startup messages
276// 2: This value asks for a message to be printed out when
277// a class is instantiated
278// 4: When this value is set, a message is displayed when a
279// FGModel object executes its Run() method
280// 8: When this value is set, various runtime state variables
281// are printed out periodically
282// 16: When set various parameters are sanity checked and
283// a message is printed out when they go out of bounds
284
285void FGInertial::Debug(int from)
286{
287 if (debug_lvl <= 0) return;
288
289 if (debug_lvl & 1) { // Standard console startup message output
290 if (from == 0) {} // Constructor
291 if (from == 2) { // Loading
292 cout << endl << " Planet " << Name << endl;
293 cout << " Semi major axis: " << a << endl;
294 cout << " Semi minor axis: " << b << endl;
295 cout << " Rotation rate : " << scientific << vOmegaPlanet(eZ) << endl;
296 cout << " GM : " << GM << endl;
297 cout << " J2 : " << J2 << endl << defaultfloat << endl;
298 }
299 }
300 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
301 if (from == 0) cout << "Instantiated: FGInertial" << endl;
302 if (from == 1) cout << "Destroyed: FGInertial" << endl;
303 }
304 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
305 }
306 if (debug_lvl & 8 ) { // Runtime state variables
307 }
308 if (debug_lvl & 16) { // Sanity checking
309 }
310 if (debug_lvl & 64) {
311 if (from == 0) { // Constructor
312 }
313 }
314}
315}
This class implements a 3 element column vector.
FGColumnVector3 & Normalize(void)
Normalize.
FGLocation holds an arbitrary location in the Earth centered Earth fixed reference frame (ECEF).
Definition FGLocation.h:152
void SetEllipse(double semimajor, double semiminor)
Sets the semimajor and semiminor axis lengths for this planet.
double GetGeodLatitudeRad(void) const
Get the GEODETIC latitude in radians.
Definition FGLocation.h:258
double GetLongitude() const
Get the longitude.
Definition FGLocation.h:234
double GetGeodAltitude(void) const
Gets the geodetic altitude in feet.
Definition FGLocation.h:279
void SetPositionGeodetic(double lon, double lat, double height)
Sets the longitude, latitude and the distance above the reference spheroid.
Handles matrix math operations.
Definition FGMatrix33.h:70