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
FGMSIS.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGMSIS.cpp
4 Author: David Culp
5 (incorporated into C++ JSBSim class hierarchy, see model authors below)
6 Date started: 12/14/03
7 Purpose: Models the MSIS-00 atmosphere
8
9 ------------- Copyright (C) 2003 David P. Culp (davidculp2@comcast.net) ------
10
11 This program is free software; you can redistribute it and/or modify it under
12 the terms of the GNU Lesser General Public License as published by the Free Software
13 Foundation; either version 2 of the License, or (at your option) any later
14 version.
15
16 This program is distributed in the hope that it will be useful, but WITHOUT
17 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
18 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
19 details.
20
21 You should have received a copy of the GNU Lesser General Public License along with
22 this program; if not, write to the Free Software Foundation, Inc., 59 Temple
23 Place - Suite 330, Boston, MA 02111-1307, USA.
24
25 Further information about the GNU Lesser General Public License can also be found on
26 the world wide web at http://www.gnu.org.
27
28FUNCTIONAL DESCRIPTION
29--------------------------------------------------------------------------------
30Models the MSIS-00 atmosphere. Provides temperature and density to FGAtmosphere,
31given day-of-year, time-of-day, altitude, latitude, longitude and local time.
32
33HISTORY
34--------------------------------------------------------------------------------
3512/14/03 DPC Created
3601/11/04 DPC Derived from FGAtmosphere
3703/18/23 BC Refactored
38
39 --------------------------------------------------------------------
40 --------- N R L M S I S E - 0 0 M O D E L 2 0 0 1 ----------
41 --------------------------------------------------------------------
42
43 This file is part of the NRLMSISE-00 C source code package - release
44 20020503
45
46 The NRLMSISE-00 model was developed by Mike Picone, Alan Hedin, and
47 Doug Drob. They also wrote a NRLMSISE-00 distribution package in
48 FORTRAN which is available at
49 http://uap-www.nrl.navy.mil/models_web/msis/msis_home.htm
50
51 Dominik Brodowski implemented and maintains this C version. You can
52 reach him at devel@brodo.de. See the file "DOCUMENTATION" for details,
53 and check http://www.brodo.de/english/pub/nrlmsise/index.html for
54 updated releases of this package.
55*/
56
57/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
58INCLUDES
59%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
60
61#include "FGFDMExec.h"
62#include "FGMSIS.h"
63
64using namespace std;
65
66namespace JSBSim {
67
68/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
69CLASS IMPLEMENTATION
70%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
71
72
74{
75 Name = "MSIS";
76
77 for(unsigned int i=0; i<24; ++i)
78 flags.switches[i] = 1;
79 input.year = 0; // Ignored by NRLMSIS
80 input.f107A = 150.;
81 input.f107 = 150.;
82 input.ap = 4.;
83 input.ap_a = nullptr;
84
85 Debug(0);
86}
87
88//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
89
91{
92 Debug(1);
93}
94
95//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
96
98{
99 FGStandardAtmosphere::InitModel();
100
101 Calculate(0.0);
102
103 return true;
104}
105
106//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
107
108bool FGMSIS::Load(Element* el)
109{
110 if (!Upload(el, true)) return false;
111
112 if (el->FindElement("day"))
113 day_of_year = el->FindElementValueAsNumber("day");
114 if (el->FindElement("utc"))
115 seconds_in_day = el->FindElementValueAsNumber("utc");
116
117 Debug(3);
118
119 return true;
120}
121
122//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
123
124void FGMSIS::Calculate(double altitude)
125{
126 double SLRair = 0.0;
127
128 Compute(0.0, SLpressure, SLtemperature, SLdensity, SLRair);
129 Compute(altitude, Pressure, Temperature, Density, Reng);
130
131 SLsoundspeed = sqrt(SHRatio*SLRair*SLtemperature);
132 Soundspeed = sqrt(SHRatio*Reng*Temperature);
133 PressureAltitude = CalculatePressureAltitude(Pressure, altitude);
134 DensityAltitude = CalculateDensityAltitude(Density, altitude);
135
136 Viscosity = Beta * pow(Temperature, 1.5) / (SutherlandConstant + Temperature);
137 KinematicViscosity = Viscosity / Density;
138 // SaturatedVaporPressure = CalculateVaporPressure(Temperature);
139 // ValidateVaporMassFraction(altitude);
140}
141
142//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
143
144void FGMSIS::Compute(double altitude, double& pressure, double& temperature,
145 double& density, double &Rair) const
146{
147 constexpr double fttokm = fttom / 1000.;
148 constexpr double kgm3_to_slugft3 = kgtoslug / m3toft3;
149 constexpr double gtoslug = kgtoslug / 1000.;
150 // Molecular weight (g/mol)
151 // N2 O2 O He H Ar N OA
152 const double species_mmol[8] {28.0134, 31.9988, 31.9988/2.0, 4.0, 1.0, 39.948,
153 28.0134/2.0, 31.9988/2.0};
154
155 double dn[10] {0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0};
156 double h = altitude*fttokm;
157 double lat = in.GeodLatitudeDeg;
158 double lon = in.LongitudeDeg;
159
160 // Compute epoch
161 double utc_seconds = seconds_in_day + FDMExec->GetSimTime();
162 unsigned int days = utc_seconds / 86400.;
163 utc_seconds -= days * 86400.;
164 double today = day_of_year + days;
165 unsigned int year = today / 365.;
166 today -= year * 365.;
167
168 struct nrlmsise_output output;
169
170 input.doy = today;
171 input.sec = utc_seconds;
172 input.alt = h;
173 input.g_lat = lat;
174 input.g_long = lon;
175 input.lst = utc_seconds/3600 + lon/15; // Local Solar Time (hours)
176 assert(flags.switches[9] != -1); // Make sure that input.ap is used.
177
178 gtd7(&input, &flags, &output);
179
180 temperature = KelvinToRankine(output.t[1]);
181 density = output.d[5] * kgm3_to_slugft3;
182 dn[1] = output.d[2]; // N2
183 dn[2] = output.d[3]; // O2
184 dn[3] = output.d[1]; // O
185 dn[4] = output.d[0]; // He
186 dn[5] = output.d[6]; // H
187 dn[6] = output.d[4]; // Ar
188 dn[7] = output.d[7]; // N
189 // SUBROUTINE GTD7 does NOT include anomalous oxygen so we drop it from
190 // the molar mass computation as well for consistency.
191 dn[8] = 0.0; // OA
192
193 // Compute specific gas constant for air
194 double mmol = 0.0;
195 double qty_mol = 0.0;
196
197 for (unsigned int i=1; i<9; ++i) {
198 mmol += dn[i]*species_mmol[i-1];
199 qty_mol += dn[i];
200 }
201 double mair = mmol * gtoslug / qty_mol;
202 Rair = Rstar / mair;
203
204 pressure = density * Rair * temperature;
205}
206
207//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
208// The bitmasked value choices are as follows:
209// unset: In this case (the default) JSBSim would only print
210// out the normally expected messages, essentially echoing
211// the config files as they are read. If the environment
212// variable is not set, debug_lvl is set to 1 internally
213// 0: This requests JSBSim not to output any messages
214// whatsoever.
215// 1: This value explicity requests the normal JSBSim
216// startup messages
217// 2: This value asks for a message to be printed out when
218// a class is instantiated
219// 4: When this value is set, a message is displayed when a
220// FGModel object executes its Run() method
221// 8: When this value is set, various runtime state variables
222// are printed out periodically
223// 16: When set various parameters are sanity checked and
224// a message is printed out when they go out of bounds
225
226void FGMSIS::Debug(int from)
227{
228 if (debug_lvl <= 0) return;
229
230 if (debug_lvl & 1) { // Standard console startup message output
231 if (from == 0) {} // Constructor
232 if (from == 3) { // Loading
233 cout << " NRLMSIS atmosphere model" << endl;
234 cout << " day: " << day_of_year << endl;
235 cout << " UTC: " << seconds_in_day << endl << endl;
236 }
237 }
238 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
239 if (from == 0) std::cout << "Instantiated: MSIS" << std::endl;
240 if (from == 1) std::cout << "Destroyed: MSIS" << std::endl;
241 }
242 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
243 }
244 if (debug_lvl & 8 ) { // Runtime state variables
245 }
246 if (debug_lvl & 16) { // Sanity checking
247 }
248 if (debug_lvl & 128) { //
249 }
250 if (debug_lvl & 64) {
251 if (from == 0) { // Constructor
252 }
253 }
254}
255
256} // namespace JSBSim
Element * FindElement(const std::string &el="")
Searches for a specified element.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
static constexpr double Rstar
Universal gas constant - ft*lbf/R/mol.
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
Definition FGFDMExec.h:549
static constexpr double KelvinToRankine(double kelvin)
Converts from degrees Kelvin to degrees Rankine.
Definition FGJSBBase.h:206
void Calculate(double altitude) override
Calculate the atmosphere for the given altitude.
Definition FGMSIS.cpp:124
~FGMSIS()
Destructor.
Definition FGMSIS.cpp:90
bool InitModel(void) override
Runs the MSIS-00 atmosphere model; called by the Executive Can pass in a value indicating if the exec...
Definition FGMSIS.cpp:97
FGMSIS(FGFDMExec *)
Constructor.
Definition FGMSIS.cpp:73
bool Upload(Element *el, bool preLoad)
Uploads this model in memory.
Definition FGModel.cpp:110
double CalculateDensityAltitude(double density, double geometricAlt) override
Calculates the density altitude given any temperature or pressure bias.
double CalculatePressureAltitude(double pressure, double geometricAlt) override
Calculates the pressure altitude given any temperature or pressure bias.