38#include "FGInertial.h"
39#include "input_output/FGXMLElement.h"
40#include "GeographicLib/Geodesic.hpp"
51FGInertial::FGInertial(FGFDMExec* fgex)
57 double RotationRate = 0.00007292115;
58 GM = 14.0764417572E15;
73 vOmegaPlanet = { 0.0, 0.0, RotationRate };
74 GroundCallback = std::make_unique<FGDefaultGroundCallback>(a, b);
83FGInertial::~FGInertial(
void)
90bool FGInertial::Load(Element* el)
92 if (!Upload(el,
true))
return false;
94 Name = el->GetAttributeValue(
"name");
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");
107 GeographicLib::Geodesic geod(a, 1.-b/a);
109 if (el->FindElement(
"rotation_rate")) {
110 double RotationRate = el->FindElementValueAsNumberConvertTo(
"rotation_rate",
"RAD/SEC");
111 vOmegaPlanet = {0., 0., RotationRate};
113 if (el->FindElement(
"GM"))
114 GM = el->FindElementValueAsNumberConvertTo(
"GM",
"FT3/SEC2");
115 if (el->FindElement(
"J2"))
116 J2 = el->FindElementValueAsNumber(
"J2");
118 GroundCallback->SetEllipse(a, b);
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;
135bool FGInertial::Run(
bool Holding)
138 if (FGModel::Run(Holding))
return true;
139 if (Holding)
return false;
145 double radius = in.Position.GetRadius();
146 vGravAccel = -(GetGAccel(radius) / radius) * in.Position;
150 vGravAccel = GetGravityJ2(in.Position);
175 Down = GetGravityJ2(location);
176 Down -= vOmegaPlanet*(vOmegaPlanet*sea_level);}
182 return FGMatrix33{North(eX), East(eX), Down(eX),
183 North(eY), East(eY), Down(eY),
184 North(eZ), 0.0, Down(eZ)};
189double FGInertial::GetGAccel(
double r)
const
201FGColumnVector3 FGInertial::GetGravityJ2(
const FGLocation& position)
const
203 FGColumnVector3 J2Gravity;
206 double r = position.GetRadius();
207 double sinLat = sin(position.GetLatitude());
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);
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);
224void FGInertial::SetAltitudeAGL(
FGLocation& location,
double altitudeAGL)
229 GroundCallback->GetAGLevel(location, contact, vDummy, vDummy, vDummy);
234 groundHeight + altitudeAGL);
239void FGInertial::SetGravityType(
int gt)
244 case eGravType::gtStandard:
246 cout <<
"Warning: Standard gravity model has been set for a non-spherical planet" << endl;
248 case eGravType::gtWGS84:
250 cout <<
"Warning: WGS84 gravity model has been set without specifying the J2 gravitational constant." << endl;
258void FGInertial::bind(
void)
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);
285void FGInertial::Debug(
int from)
287 if (debug_lvl <= 0)
return;
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;
300 if (debug_lvl & 2 ) {
301 if (from == 0) cout <<
"Instantiated: FGInertial" << endl;
302 if (from == 1) cout <<
"Destroyed: FGInertial" << endl;
304 if (debug_lvl & 4 ) {
306 if (debug_lvl & 8 ) {
308 if (debug_lvl & 16) {
310 if (debug_lvl & 64) {
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).
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.
double GetLongitude() const
Get the longitude.
double GetGeodAltitude(void) const
Gets the geodetic altitude in feet.
void SetPositionGeodetic(double lon, double lat, double height)
Sets the longitude, latitude and the distance above the reference spheroid.
Handles matrix math operations.