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
FGStandardAtmosphere.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGStandardAtmosphere.cpp
4 Author: Jon Berndt, Tony Peden
5 Date started: 5/2011
6 Purpose: Models the 1976 U.S. Standard Atmosphere
7 Called by: FGFDMExec
8
9 ------------- Copyright (C) 2011 Jon S. Berndt (jon@jsbsim.org) -------------
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
13 Software Foundation; either version 2 of the License, or (at your option) any
14 later 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
22 with this program; if not, write to the Free Software Foundation, Inc., 59
23 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25 Further information about the GNU Lesser General Public License can also be
26 found on the world wide web at http://www.gnu.org.
27
28FUNCTIONAL DESCRIPTION
29--------------------------------------------------------------------------------
30
31
32HISTORY
33--------------------------------------------------------------------------------
34
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
36COMMENTS, REFERENCES, and NOTES
37%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
38[1] Anderson, John D. "Introduction to Flight, Third Edition", McGraw-Hill,
39 1989, ISBN 0-07-001641-0
40[2] Sonntag, D. "Important New Values of the Physical Constants of 1986,
41 Vapour Pressure Formulations based on the IST-90 and Psychrometer
42 Formulae", Z. Meteorol., 70 (5), pp. 340-344, 1990
43
44%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
45INCLUDES
46%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
47
48#include <iomanip>
49
50#include "FGFDMExec.h"
51#include "FGStandardAtmosphere.h"
52
53using namespace std;
54
55namespace JSBSim {
56
57/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
58CLASS IMPLEMENTATION
59%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
60
62 : FGAtmosphere(fdmex), StdSLpressure(StdDaySLpressure), TemperatureBias(0.0),
63 TemperatureDeltaGradient(0.0), VaporMassFraction(0.0),
64 SaturatedVaporPressure(StdDaySLpressure), StdAtmosTemperatureTable(9),
65 MaxVaporMassFraction(10)
66{
67 Name = "FGStandardAtmosphere";
68
69 // This is the U.S. Standard Atmosphere table for temperature in degrees
70 // Rankine, based on geometric altitude. The table values are often given
71 // in literature relative to geopotential altitude.
72 //
73 // GeoMet Alt Temp GeoPot Alt GeoMet Alt
74 // (ft) (deg R) (km) (km)
75 // -------- -------- ---------- ----------
76 //StdAtmosTemperatureTable << 0.00 << 518.67 // 0.000 0.000
77 // << 36151.80 << 389.97 // 11.000 11.019
78 // << 65823.90 << 389.97 // 20.000 20.063
79 // << 105518.06 << 411.60 // 32.000 32.162
80 // << 155348.07 << 487.20 // 47.000 47.350
81 // << 168676.12 << 487.20 // 51.000 51.413
82 // << 235570.77 << 386.40 // 71.000 71.802
83 // << 282152.08 << 336.50 // 84.852 86.000
84 // << 298556.40 << 336.50; // 91.000 - First layer in high altitude regime
85
86 // GeoPot Alt Temp GeoPot Alt GeoMet Alt
87 // (ft) (deg R) (km) (km)
88 // ----------- -------- ---------- ----------
89 StdAtmosTemperatureTable << 0.0000 << 518.67 // 0.000 0.000
90 << 36089.2388 << 389.97 // 11.000 11.019
91 << 65616.7979 << 389.97 // 20.000 20.063
92 << 104986.8766 << 411.57 // 32.000 32.162
93 << 154199.4751 << 487.17 // 47.000 47.350
94 << 167322.8346 << 487.17 // 51.000 51.413
95 << 232939.6325 << 386.37 // 71.000 71.802
96 << 278385.8268 << 336.5028 // 84.852 86.000
97 << 298556.4304 << 336.5028; // 91.000 - First layer in high altitude regime
98
99
100 // This is the maximum water vapor mass fraction in ppm (parts per million) of
101 // dry air measured in the atmosphere according to the ISA 1976 document.
102 // Values at altitude below 8 km are record high. All other values are 1%
103 // high.
104
105 // Geopot Alt Water Geopot Alt
106 // (ft) (ppm) (km)
107 // ---------- ----- ----------
108 MaxVaporMassFraction << 0.0000 << 35000. // 0.0000 - Record high
109 << 3280.8399 << 31000. // 1.0000
110 << 6561.6798 << 28000. // 2.0000
111 << 13123.3596 << 22000. // 4.0000
112 << 19685.0394 << 8900. // 6.0000
113 << 26246.7192 << 4700. // 8.0000 - Record high
114 << 32808.3990 << 1300. // 10.0000 - 1% high
115 << 39370.0787 << 230. // 12.0000
116 << 45931.7585 << 48. // 14.0000
117 << 52493.4383 << 38.; // 16.0000 - 1% high
118
119 unsigned int numRows = StdAtmosTemperatureTable.GetNumRows();
120
121 // Initialize the standard atmosphere lapse rates.
123 StdLapseRates = LapseRates;
124
125 // Assume the altitude to fade out the gradient at is at the highest
126 // altitude in the table. Above that, other functions are used to
127 // calculate temperature.
128 GradientFadeoutAltitude = StdAtmosTemperatureTable(numRows, 0);
129
130 // Initialize the standard atmosphere pressure break points.
131 PressureBreakpoints.resize(numRows);
132 CalculatePressureBreakpoints(StdSLpressure);
133 StdPressureBreakpoints = PressureBreakpoints;
134
135 StdSLtemperature = StdAtmosTemperatureTable(1, 1);
136 StdSLdensity = StdSLpressure / (Rdry * StdSLtemperature);
137
139 StdSLsoundspeed = sqrt(SHRatio*Rdry*StdSLtemperature);
140
141 bind();
142 Debug(0);
143}
144
145//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
146
151
152//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
153
154bool FGStandardAtmosphere::InitModel(void)
155{
156
157 // Assume the altitude to fade out the gradient at is at the highest
158 // altitude in the table. Above that, other functions are used to
159 // calculate temperature.
160 GradientFadeoutAltitude = StdAtmosTemperatureTable(StdAtmosTemperatureTable.GetNumRows(), 0);
161
162 TemperatureDeltaGradient = 0.0;
163 TemperatureBias = 0.0;
164 LapseRates = StdLapseRates;
165
166 PressureBreakpoints = StdPressureBreakpoints;
167
168 SLpressure = StdSLpressure;
169 SLtemperature = StdSLtemperature;
170 SLdensity = StdSLdensity;
171 SLsoundspeed = StdSLsoundspeed;
172
173 Calculate(0.0);
174
175// PrintStandardAtmosphereTable();
176
177 return true;
178}
179//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
180
182{
183 FGAtmosphere::Calculate(altitude);
184 SaturatedVaporPressure = CalculateVaporPressure(Temperature);
186}
187
188//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
189// Get the actual pressure as modeled at a specified altitude
190// These calculations are from equations 33a and 33b in the U.S. Standard
191// Atmosphere document referenced in the documentation for this code.
192
193double FGStandardAtmosphere::GetPressure(double altitude) const
194{
195 double GeoPotAlt = GeopotentialAltitude(altitude);
196
197 // Iterate through the altitudes to find the current Base Altitude
198 // in the table. That is, if the current altitude (the argument passed in)
199 // is 20000 ft, then the base altitude from the table is 0.0. If the
200 // passed-in altitude is 40000 ft, the base altitude is 36089.2388 ft (and
201 // the index "b" is 2 - the second entry in the table).
202 double BaseAlt = StdAtmosTemperatureTable(1,0);
203 unsigned int numRows = StdAtmosTemperatureTable.GetNumRows();
204 unsigned int b;
205
206 for (b=0; b < numRows-2; ++b) {
207 double testAlt = StdAtmosTemperatureTable(b+2,0);
208 if (GeoPotAlt < testAlt)
209 break;
210 BaseAlt = testAlt;
211 }
212
213 double Tmb = GetTemperature(GeometricAltitude(BaseAlt));
214 double deltaH = GeoPotAlt - BaseAlt;
215 double Lmb = LapseRates[b];
216
217 if (Lmb != 0.0) {
218 double Exp = g0 / (Rdry*Lmb);
219 double factor = Tmb/(Tmb + Lmb*deltaH);
220 return PressureBreakpoints[b]*pow(factor, Exp);
221 } else
222 return PressureBreakpoints[b]*exp(-g0*deltaH/(Rdry*Tmb));
223}
224
225//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
226
228{
229 double p = ConvertToPSF(pressure, unit);
230 SLpressure = ValidatePressure(p, "Sea Level pressure");
233}
234
235//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
236
238{
239 SLsoundspeed = sqrt(SHRatio*Reng*SLtemperature);
241}
242
243//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
244// Get the modeled temperature at a specified altitude, including any bias or gradient
245// effects.
246
247double FGStandardAtmosphere::GetTemperature(double altitude) const
248{
249 double GeoPotAlt = GeopotentialAltitude(altitude);
250
251 double T;
252
253 if (GeoPotAlt >= 0.0) {
254 T = StdAtmosTemperatureTable.GetValue(GeoPotAlt);
255
256 if (GeoPotAlt <= GradientFadeoutAltitude)
257 T -= TemperatureDeltaGradient * GeoPotAlt;
258 }
259 else {
260 // We don't need to add TemperatureDeltaGradient*GeoPotAlt here because
261 // the lapse rate vector already accounts for the temperature gradient.
262 T = StdAtmosTemperatureTable.GetValue(0.0) + GeoPotAlt*LapseRates[0];
263 }
264
265 T += TemperatureBias;
266
267 if (GeoPotAlt <= GradientFadeoutAltitude)
268 T += TemperatureDeltaGradient * GradientFadeoutAltitude;
269
270 return T;
271}
272
273//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
274// Retrieves the standard temperature at a particular altitude.
275
276double FGStandardAtmosphere::GetStdTemperature(double altitude) const
277{
278 double GeoPotAlt = GeopotentialAltitude(altitude);
279
280 if (GeoPotAlt >= 0.0)
281 return StdAtmosTemperatureTable.GetValue(GeoPotAlt);
282 else
283 return StdAtmosTemperatureTable.GetValue(0.0) + GeoPotAlt*LapseRates[0];
284}
285
286//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
287
288double FGStandardAtmosphere::GetStdPressure(double altitude) const
289{
290 double GeoPotAlt = GeopotentialAltitude(altitude);
291
292 // Iterate through the altitudes to find the current Base Altitude
293 // in the table. That is, if the current altitude (the argument passed in)
294 // is 20000 ft, then the base altitude from the table is 0.0. If the
295 // passed-in altitude is 40000 ft, the base altitude is 36089.2388 ft (and
296 // the index "b" is 2 - the second entry in the table).
297 double BaseAlt = StdAtmosTemperatureTable(1,0);
298 unsigned int numRows = StdAtmosTemperatureTable.GetNumRows();
299 unsigned int b;
300
301 for (b=0; b < numRows-2; ++b) {
302 double testAlt = StdAtmosTemperatureTable(b+2,0);
303 if (GeoPotAlt < testAlt)
304 break;
305 BaseAlt = testAlt;
306 }
307
308 double Tmb = GetStdTemperature(GeometricAltitude(BaseAlt));
309 double deltaH = GeoPotAlt - BaseAlt;
310 double Lmb = LapseRates[b];
311
312 if (Lmb != 0.0) {
313 double Exp = g0 / (Rdry*Lmb);
314 double factor = Tmb/(Tmb + Lmb*deltaH);
315 return StdPressureBreakpoints[b]*pow(factor, Exp);
316 } else
317 return StdPressureBreakpoints[b]*exp(-g0*deltaH/(Rdry*Tmb));
318}
319
320//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
321// Get the standard density at a specified altitude
322
323double FGStandardAtmosphere::GetStdDensity(double altitude) const
324{
325 return GetStdPressure(altitude)/(Rdry * GetStdTemperature(altitude));
326}
327
328//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
329
331{
332 double targetTemp = ConvertToRankine(t, unit);
333 double GeoPotAlt = GeopotentialAltitude(h);
334 double bias = targetTemp - GetStdTemperature(h);
335
336 if (GeoPotAlt <= GradientFadeoutAltitude)
337 bias -= TemperatureDeltaGradient * (GradientFadeoutAltitude - GeoPotAlt);
338
339 SetTemperatureBias(eRankine, bias);
341
342 SLtemperature = GetTemperature(0.0);
344}
345
346//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
347
349{
350 // Retrieve the minimum temperature in the standard atmosphere, may not be the
351 // last row in future if for example it's extended and maybe there is some
352 // temperature inversion layer etc. So run through and find the minimum.
353 const double minStdAtmosphereTemp = StdAtmosTemperatureTable.GetMinValue();
354
355 // Minimum known temperature in the universe currently
356 constexpr double minUniverseTemperature = KelvinToRankine(1.0);
357
358 if (unit == eCelsius || unit == eKelvin)
359 t *= 1.80; // If temp delta "t" is given in metric, scale up to English
360
361 TemperatureBias = t;
362 // Confirm the temperature bias isn't going to result in an atmosphere
363 // temperature lower than the lowest known temperature in the universe
364 if (minStdAtmosphereTemp + TemperatureBias < minUniverseTemperature) {
365 double minBias = minUniverseTemperature - minStdAtmosphereTemp;
366 cerr << "The temperature bias " << TemperatureBias << " R is too low. "
367 << "It could result in temperatures below the absolute zero." << endl
368 << "Temperature bias is therefore capped to " << minBias << endl;
369 TemperatureBias = minBias;
370 }
371
373
374 SLtemperature = GetTemperature(0.0);
376}
377
378//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
379// This function calculates a bias based on the supplied temperature for sea
380// level. The bias is applied to the entire temperature profile at all altitudes.
381// Internally, the Rankine scale is used for calculations, so any temperature
382// supplied must be converted to that unit.
383
385{
386 SetTemperature(t, 0.0, unit);
387}
388
389//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
390// Sets a Sea Level temperature delta that is ramped out by 86 km (282,152 ft).
391
393{
394 SetTemperatureGradedDelta(deltemp, 0.0, unit);
395}
396
397//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
398// Sets a temperature delta at the supplied altitude that is ramped out by 86 km.
399// After this calculation is performed, the lapse rates and pressure breakpoints
400// must be recalculated. Since we are calculating a delta here and not an actual
401// temperature, we only need to be concerned about a scale factor and not
402// the actual temperature itself.
403
405{
406 // Retrieve the minimum temperature in the standard atmosphere, may not be the
407 // last row in future if for example it's extended and maybe there is some
408 // temperature inversion layer etc. So run through and find the minimum.
409 const double minStdAtmosphereTemp = StdAtmosTemperatureTable.GetMinValue();
410 const double minDeltaTemperature = minStdAtmosphereTemp - StdSLtemperature;
411
412 if (unit == eCelsius || unit == eKelvin)
413 deltemp *= 1.80; // If temp delta "t" is given in metric, scale up to English
414
415 if (deltemp <= minDeltaTemperature) {
416 cerr << "The temperature delta " << deltemp << " R is too low. "
417 << "It could result in temperatures below the absolute zero." << endl
418 << "Temperature delta is therefore capped to " << minDeltaTemperature << endl;
419 deltemp = minDeltaTemperature;
420 }
421
422 TemperatureDeltaGradient = deltemp/(GradientFadeoutAltitude - GeopotentialAltitude(h));
425
426 SLtemperature = GetTemperature(0.0);
428}
429
430//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
431
433{
434 std::cout << "Altitude (ft) Temp (F) Pressure (psf) Density (sl/ft3)" << std::endl;
435 std::cout << "------------- -------- -------------- ----------------" << std::endl;
436 for (int i=0; i<280000; i+=1000) {
437 Calculate(i);
438 std::cout << std::setw(12) << std::setprecision(2) << i
439 << " " << std::setw(9) << std::setprecision(2) << Temperature - 459.67
440 << " " << std::setw(13) << std::setprecision(4) << Pressure
441 << " " << std::setw(18) << std::setprecision(8) << Density
442 << std::endl;
443 }
444
445 // Re-execute the Run() method to reset the calculated values
446 Run(false);
447}
448
449//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
450// This function calculates (or recalculates) the lapse rate over an altitude range
451// where the "bh" in this case refers to the index of the base height in the
452// StdAtmosTemperatureTable table. This function should be called anytime the
453// temperature table is altered, such as when a gradient is applied across the
454// temperature table for a range of altitudes.
455
457{
458 unsigned int numRows = StdAtmosTemperatureTable.GetNumRows();
459 LapseRates.clear();
460
461 for (unsigned int bh=0; bh < numRows-1; bh++)
462 {
463 double t0 = StdAtmosTemperatureTable(bh+1,1);
464 double t1 = StdAtmosTemperatureTable(bh+2,1);
465 double h0 = StdAtmosTemperatureTable(bh+1,0);
466 double h1 = StdAtmosTemperatureTable(bh+2,0);
467 LapseRates.push_back((t1 - t0) / (h1 - h0) - TemperatureDeltaGradient);
468 }
469}
470
471//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
472
474{
475 PressureBreakpoints[0] = SLpress;
476
477 for (unsigned int b=0; b<PressureBreakpoints.size()-1; b++) {
478 double BaseTemp = StdAtmosTemperatureTable(b+1,1);
479 double BaseAlt = StdAtmosTemperatureTable(b+1,0);
480 double UpperAlt = StdAtmosTemperatureTable(b+2,0);
481 double deltaH = UpperAlt - BaseAlt;
482 double Tmb = BaseTemp
483 + TemperatureBias
484 + (GradientFadeoutAltitude - BaseAlt)*TemperatureDeltaGradient;
485 if (LapseRates[b] != 0.00) {
486 double Lmb = LapseRates[b];
487 double Exp = g0 / (Rdry*Lmb);
488 double factor = Tmb/(Tmb + Lmb*deltaH);
489 PressureBreakpoints[b+1] = PressureBreakpoints[b]*pow(factor, Exp);
490 } else {
491 PressureBreakpoints[b+1] = PressureBreakpoints[b]*exp(-g0*deltaH/(Rdry*Tmb));
492 }
493 }
494}
495
496//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
497
499{
500 TemperatureBias = TemperatureDeltaGradient = 0.0;
503
504 SLtemperature = StdSLtemperature;
506}
507
508//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
509
511{
512 SLpressure = StdSLpressure;
514 CalculatePressureBreakpoints(StdSLpressure);
515}
516
517//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
518
520{
521 StdDensityBreakpoints.clear();
522 for (unsigned int i = 0; i < StdPressureBreakpoints.size(); i++)
523 StdDensityBreakpoints.push_back(StdPressureBreakpoints[i] / (Rdry * StdAtmosTemperatureTable(i + 1, 1)));
524}
525
526//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
527
528double FGStandardAtmosphere::CalculateDensityAltitude(double density, double geometricAlt)
529{
530 // Work out which layer we're dealing with
531 unsigned int b = 0;
532 for (; b < StdDensityBreakpoints.size() - 2; b++) {
533 if (density >= StdDensityBreakpoints[b + 1])
534 break;
535 }
536
537 // Get layer properties
538 double Tmb = StdAtmosTemperatureTable(b + 1, 1);
539 double Hb = StdAtmosTemperatureTable(b + 1, 0);
540 double Lmb = StdLapseRates[b];
541 double pb = StdDensityBreakpoints[b];
542
543 double density_altitude = 0.0;
544
545 // https://en.wikipedia.org/wiki/Barometric_formula for density solved for H
546 if (Lmb != 0.0) {
547 double Exp = -1.0 / (1.0 + g0/(Rdry*Lmb));
548 density_altitude = Hb + (Tmb / Lmb) * (pow(density / pb, Exp) - 1);
549 } else {
550 double Factor = -Rdry*Tmb / g0;
551 density_altitude = Hb + Factor * log(density / pb);
552 }
553
554 return GeometricAltitude(density_altitude);
555}
556
557//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
558
559double FGStandardAtmosphere::CalculatePressureAltitude(double pressure, double geometricAlt)
560{
561 // Work out which layer we're dealing with
562 unsigned int b = 0;
563 for (; b < StdPressureBreakpoints.size() - 2; b++) {
564 if (pressure >= StdPressureBreakpoints[b + 1])
565 break;
566 }
567
568 // Get layer properties
569 double Tmb = StdAtmosTemperatureTable(b + 1, 1);
570 double Hb = StdAtmosTemperatureTable(b + 1, 0);
571 double Lmb = StdLapseRates[b];
572 double Pb = StdPressureBreakpoints[b];
573
574 double pressure_altitude = 0.0;
575
576 if (Lmb != 0.00) {
577 // Equation 33(a) from ISA document solved for H
578 double Exp = -Rdry*Lmb / g0;
579 pressure_altitude = Hb + (Tmb / Lmb) * (pow(pressure / Pb, Exp) - 1);
580 } else {
581 // Equation 33(b) from ISA document solved for H
582 double Factor = -Rdry*Tmb / g0;
583 pressure_altitude = Hb + Factor * log(pressure / Pb);
584 }
585
586 return GeometricAltitude(pressure_altitude);
587}
588
589//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
590
592{
593 double temperature_degC = RankineToCelsius(temperature);
594 return a*exp(b*temperature_degC/(c+temperature_degC));
595}
596
597//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
598
600{
601 if (SaturatedVaporPressure < Pressure) {
602 double VaporPressure = Pressure*VaporMassFraction / (VaporMassFraction+Rdry/Rwater);
603 if (VaporPressure > SaturatedVaporPressure)
604 VaporMassFraction = Rdry * SaturatedVaporPressure / (Rwater * (Pressure - SaturatedVaporPressure));
605 }
606
607 double GeoPotAlt = GeopotentialAltitude(h);
608 double maxFraction = 1E-6*MaxVaporMassFraction.GetValue(GeoPotAlt);
609
610 if ((VaporMassFraction > maxFraction) || (VaporMassFraction < 0.0))
611 VaporMassFraction = maxFraction;
612
613 // Update the gas constant factor
614 Reng = (VaporMassFraction*Rwater + Rdry)/(1.0 + VaporMassFraction);
615}
616
617//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
618
620{
621 double dewPoint_R = ConvertToRankine(dewpoint, unit);
622 constexpr double minDewPoint = -CelsiusToRankine(c) + 1.0;
623
624 if (dewPoint_R <= minDewPoint) {
625 cerr << "The dew point temperature " << dewPoint_R << " is lower than "
626 << minDewPoint << " R." << endl
627 << "Dew point is therefore capped to " << minDewPoint << endl;
628 dewPoint_R = minDewPoint;
629 }
630
631 double VaporPressure = CalculateVaporPressure(dewPoint_R);
632 SetVaporPressure(ePSF, VaporPressure);
633
634 double finalizedDewPoint = GetDewPoint(eRankine);
635 if (finalizedDewPoint < dewPoint_R) {
636 cerr << "Dew point temperature has been capped to " << finalizedDewPoint
637 << endl;
638 }
639}
640
641//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
642
644{
645 double dewpoint_degC;
646 double VaporPressure = Pressure*VaporMassFraction / (VaporMassFraction+Rdry/Rwater);
647
648 if (VaporPressure <= 0.0)
649 dewpoint_degC = -c;
650 else {
651 double x = log(VaporPressure/a);
652 dewpoint_degC = c*x / (b - x);
653 }
654
655 return ConvertFromRankine(CelsiusToRankine(dewpoint_degC), to);
656}
657
658//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
659
661{
662 double altitude = CalculatePressureAltitude(Pressure, 0.0);
663 double VaporPressure = ConvertToPSF(Pa, unit);
664 if (VaporPressure < 0.0) {
665 cerr << "The vapor pressure cannot be negative." << endl
666 << "Vapor pressure is set to 0.0" << endl;
667 VaporPressure = 0.0;
668 } else if (VaporPressure >= Pressure) {
669 cerr << "The vapor pressure " << VaporPressure
670 << " PSF is higher than the ambient pressure." << endl
671 << "Vapor pressure is therefore capped to " << Pressure-1.0 << endl;
672 VaporPressure = Pressure - 1.0;
673 }
674 VaporMassFraction = Rdry * VaporPressure / (Rwater * (Pressure - VaporPressure));
676}
677
678//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
679
681{
682 double VaporPressure = Pressure*VaporMassFraction / (VaporMassFraction+Rdry/Rwater);
683 return ConvertFromPSF(VaporPressure, to);
684}
685
686//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
687
689{
690 return ConvertFromPSF(SaturatedVaporPressure, to);
691}
692
693//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
694
696{
697 double VaporPressure = Pressure*VaporMassFraction / (VaporMassFraction+Rdry/Rwater);
698 return 100.0*VaporPressure/SaturatedVaporPressure;
699}
700
701//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
702
704{
705 if (RH < 0.0) {
706 cerr << "The relative humidity cannot be negative." << endl
707 << "Relative humidity is set to 0%" << endl;
708 RH = 0.0;
709 } else if (RH > 100.0) {
710 cerr << "The relative humidity cannot be higher than 100%." << endl
711 << "Relative humidity is set to 100%" << endl;
712 RH = 100.0;
713 }
714
715 double VaporPressure = 0.01*RH*SaturatedVaporPressure;
716 SetVaporPressure(ePSF, VaporPressure);
717}
718
719//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
720
722{
723 return VaporMassFraction*1E6;
724}
725
726//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
727
729{
730 double altitude = CalculatePressureAltitude(Pressure, 0.0);
731 VaporMassFraction = frac*1E-6;
733
734 if (fabs(VaporMassFraction*1E6-frac)>1E-2) {
735 cerr << "The vapor mass fraction " << frac << " has been capped to "
736 << VaporMassFraction*1E6 << "PPM." << endl;
737 }
738}
739
740//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
741
742void FGStandardAtmosphere::bind(void)
743{
744 typedef double (FGStandardAtmosphere::*PMFi)(int) const;
745 typedef void (FGStandardAtmosphere::*PMF)(int, double);
746 PropertyManager->Tie("atmosphere/delta-T", this, eRankine,
749 PropertyManager->Tie("atmosphere/SL-graded-delta-T", this, eRankine,
752 PropertyManager->Tie("atmosphere/P-sl-psf", this, ePSF,
753 (PMFi)&FGStandardAtmosphere::GetPressureSL,
755 PropertyManager->Tie("atmosphere/dew-point-R", this, eRankine,
758 PropertyManager->Tie("atmosphere/vapor-pressure-psf", this, ePSF,
761 PropertyManager->Tie("atmosphere/saturated-vapor-pressure-psf", this, ePSF,
763 PropertyManager->Tie("atmosphere/RH", this,
766 PropertyManager->Tie("atmosphere/vapor-fraction-ppm", this,
769}
770
771//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
772// The bitmasked value choices are as follows:
773// unset: In this case (the default) JSBSim would only print
774// out the normally expected messages, essentially echoing
775// the config files as they are read. If the environment
776// variable is not set, debug_lvl is set to 1 internally
777// 0: This requests JSBSim not to output any messages
778// whatsoever.
779// 1: This value explicity requests the normal JSBSim
780// startup messages
781// 2: This value asks for a message to be printed out when
782// a class is instantiated
783// 4: When this value is set, a message is displayed when a
784// FGModel object executes its Run() method
785// 8: When this value is set, various runtime state variables
786// are printed out periodically
787// 16: When set various parameters are sanity checked and
788// a message is printed out when they go out of bounds
789
790void FGStandardAtmosphere::Debug(int from)
791{
792 if (debug_lvl <= 0) return;
793
794 if (debug_lvl & 1) { // Standard console startup message output
795 if (from == 0) { // Constructor
796 }
797 }
798 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
799 if (from == 0) std::cout << "Instantiated: FGStandardAtmosphere" << std::endl;
800 if (from == 1) std::cout << "Destroyed: FGStandardAtmosphere" << std::endl;
801 }
802 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
803 }
804 if (debug_lvl & 8 ) { // Runtime state variables
805 }
806 if (debug_lvl & 16) { // Sanity checking
807 }
808 if (debug_lvl & 128) { //
809 }
810 if (debug_lvl & 64) {
811 if (from == 0) { // Constructor
812 }
813 }
814}
815
816} // namespace JSBSim
Models an empty, abstract base atmosphere class.
double ConvertToRankine(double t, eTemperature unit) const
Converts to Rankine from one of several unit systems.
virtual void Calculate(double altitude)
Calculate the atmosphere for the given altitude.
double ConvertFromPSF(double t, ePressure unit=ePSF) const
Converts from PSF (pounds per square foot) to one of several unit systems.
virtual double GetPressure(void) const
Returns the pressure in psf.
eTemperature
Enums for specifying temperature units.
double ConvertFromRankine(double t, eTemperature unit) const
Converts from Rankine to one of several unit systems.
double ValidatePressure(double p, const std::string &msg, bool quiet=false) const
Check that the pressure is within plausible boundaries.
bool Run(bool Holding) override
Runs the atmosphere forces model; called by the Executive.
double ConvertToPSF(double t, ePressure unit=ePSF) const
Converts to PSF (pounds per square foot) from one of several unit systems.
ePressure
Enums for specifying pressure units.
static constexpr double g0
Sea-level acceleration of gravity - ft/s^2.
virtual double GetTemperature() const
Returns the actual, modeled temperature at the current altitude in degrees Rankine.
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
static constexpr double CelsiusToRankine(double celsius)
Converts from degrees Celsius to degrees Rankine.
Definition FGJSBBase.h:192
static constexpr double RankineToCelsius(double rankine)
Converts from degrees Rankine to degrees Celsius.
Definition FGJSBBase.h:199
static constexpr double KelvinToRankine(double kelvin)
Converts from degrees Kelvin to degrees Rankine.
Definition FGJSBBase.h:206
void CalculateStdDensityBreakpoints()
Calculate the atmospheric density breakpoints at the altitudes in the standard temperature table.
double GeometricAltitude(double geopotalt) const
Convert a geopotential altitude to a geometric altitude.
void SetVaporPressure(ePressure unit, double Pv)
Sets the partial pressure of water vapor.
virtual double GetStdPressure(double altitude) const
Returns the standard pressure at the specified altitude.
static constexpr double a
Sonntag constants based on ref [2].
double CalculateDensityAltitude(double density, double geometricAlt) override
Calculates the density altitude given any temperature or pressure bias.
double GetRelativeHumidity(void) const
Returns the relative humidity in percent.
void CalculateLapseRates()
Recalculate the lapse rate vectors when the temperature profile is altered in a way that would change...
virtual double GetTemperatureDeltaGradient(eTemperature to)
Returns the temperature gradient to be applied on top of the standard temperature gradient.
void SetRelativeHumidity(double RH)
Sets the relative humidity.
virtual double GetTemperatureBias(eTemperature to) const
Returns the temperature bias over the sea level value in degrees Rankine.
virtual double GetStdTemperature(double altitude) const
Returns the standard temperature in degrees Rankine at a specified altitude.
void Calculate(double altitude) override
Calculate the atmosphere for the given altitude.
void SetTemperature(double t, double h, eTemperature unit=eFahrenheit) override
Sets the temperature at the supplied altitude, if it is to be different than the standard temperature...
void CalculatePressureBreakpoints(double SLpress)
Calculate (or recalculate) the atmospheric pressure breakpoints at the altitudes in the standard temp...
virtual void SetTemperatureGradedDelta(double t, double h, eTemperature unit=eFahrenheit)
Sets the temperature delta value at the supplied altitude/elevation above sea level,...
void CalculateSLDensity(void)
Calculate the SL density.
virtual void ResetSLPressure()
Resets the sea level to the Standard sea level pressure, and recalculates dependent parameters so tha...
double GetVaporPressure(ePressure to) const
Returns the partial pressure of water vapor.
double GeopotentialAltitude(double geometalt) const
Convert a geometric altitude to a geopotential altitude.
virtual void PrintStandardAtmosphereTable()
Prints the U.S. Standard Atmosphere table.
virtual double GetStdDensity(double altitude) const
Returns the standard density at a specified altitude.
FGStandardAtmosphere(FGFDMExec *)
Constructor.
virtual void SetTemperatureBias(eTemperature unit, double t)
Sets the temperature bias to be added to the standard temperature at all altitudes.
double GetSaturatedVaporPressure(ePressure to) const
Returns the saturated pressure of water vapor.
double StdSLtemperature
Standard sea level conditions.
virtual ~FGStandardAtmosphere()
Destructor.
void SetPressureSL(ePressure unit, double pressure) override
Sets the sea level pressure for modeling an off-standard pressure profile.
void SetTemperatureSL(double t, eTemperature unit=eFahrenheit) override
Sets the Sea Level temperature, if it is to be different than the standard.
double CalculateVaporPressure(double temperature)
Calculate the pressure of water vapor with the Magnus formula.
void CalculateSLSoundSpeedAndDensity(void)
Calculate the SL density and sound speed.
virtual void SetSLTemperatureGradedDelta(eTemperature unit, double t)
Sets a Sea Level temperature delta that is ramped out by 86 km.
void SetVaporMassFractionPPM(double frac)
Sets the vapor mass per million of dry air mass units.
void SetDewPoint(eTemperature unit, double dewpoint)
Sets the dew point.
void ValidateVaporMassFraction(double geometricAlt)
Validate the value of the vapor mass fraction.
virtual void ResetSLTemperature()
This function resets the model to apply no bias or delta gradient to the temperature.
double GetVaporMassFractionPPM(void) const
Returns the vapor mass per million of dry air mass units (ppm).
double CalculatePressureAltitude(double pressure, double geometricAlt) override
Calculates the pressure altitude given any temperature or pressure bias.
double GetDewPoint(eTemperature to) const
Returns the dew point.
double GetValue(void) const
Get the current table value.
Definition FGTable.cpp:465