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
FGFDMExec.cpp
1
2/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3
4 Module: FGFDMExec.cpp
5 Author: Jon S. Berndt
6 Date started: 11/17/98
7 Purpose: Schedules and runs the model routines.
8
9 ------------- Copyright (C) 1999 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
31This class wraps up the simulation scheduling routines.
32
33HISTORY
34--------------------------------------------------------------------------------
3511/17/98 JSB Created
36
37%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
38COMMENTS, REFERENCES, and NOTES
39%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
40
41%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
42INCLUDES
43%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
44
45#include <iomanip>
46
47#include "FGFDMExec.h"
48#include "models/atmosphere/FGStandardAtmosphere.h"
49#include "models/atmosphere/FGMSIS.h"
50#include "models/atmosphere/FGWinds.h"
51#include "models/FGFCS.h"
52#include "models/FGPropulsion.h"
53#include "models/FGMassBalance.h"
54#include "models/FGExternalReactions.h"
55#include "models/FGBuoyantForces.h"
56#include "models/FGAerodynamics.h"
57#include "models/FGInertial.h"
58#include "models/FGAircraft.h"
59#include "models/FGAccelerations.h"
60#include "models/FGAuxiliary.h"
61#include "models/FGInput.h"
62#include "initialization/FGTrim.h"
63#include "initialization/FGLinearization.h"
64#include "input_output/FGScript.h"
65#include "input_output/FGXMLFileRead.h"
66#include "initialization/FGInitialCondition.h"
67
68using namespace std;
69
70namespace JSBSim {
71
72/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
73CLASS IMPLEMENTATION
74%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
75
76//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
77// Constructor
78
79FGFDMExec::FGFDMExec(FGPropertyManager* root, std::shared_ptr<unsigned int> fdmctr)
80 : RandomSeed(0), RandomGenerator(make_shared<RandomNumberGenerator>(RandomSeed)),
81 FDMctr(fdmctr)
82{
83 Frame = 0;
84 disperse = 0;
85
86 RootDir = "";
87
88 modelLoaded = false;
89 IsChild = false;
90 holding = false;
91 Terminate = false;
92 HoldDown = false;
93
94 IncrementThenHolding = false; // increment then hold is off by default
95 TimeStepsUntilHold = -1;
96
97 sim_time = 0.0;
98 dT = 1.0/120.0; // a default timestep size. This is needed for when JSBSim is
99 // run in standalone mode with no initialization file.
100
101 AircraftPath = "aircraft";
102 EnginePath = "engine";
103 SystemsPath = "systems";
104
105 try {
106 char* num = getenv("JSBSIM_DEBUG");
107 if (num) debug_lvl = atoi(num); // set debug level
108 } catch (...) { // if error set to 1
109 debug_lvl = 1;
110 }
111
112 if (!FDMctr) {
113 FDMctr = std::make_shared<unsigned int>(); // Create and initialize the child FDM counter
114 *FDMctr = 0;
115 }
116
117 // Store this FDM's ID
118 IdFDM = *FDMctr; // The main (parent) JSBSim instance is always the "zeroth"
119
120 // Prepare FDMctr for the next child FDM id
121 (*FDMctr)++; // instance. "child" instances are loaded last.
122
123 if (root == nullptr) // Then this is the root FDM
124 Root = new FGPropertyNode();
125 else
126 Root = root->GetNode();
127
128 FGPropertyNode* instanceRoot = Root->GetNode("fdm/jsbsim", IdFDM, true);
129 instance = std::make_shared<FGPropertyManager>(instanceRoot);
130
131 try {
132 char* num = getenv("JSBSIM_DISPERSE");
133 if (num) {
134 if (atoi(num) != 0) disperse = 1; // set dispersions on
135 }
136 } catch (...) { // if error set to false
137 disperse = 0;
138 std::cerr << "Could not process JSBSIM_DISPERSIONS environment variable: Assumed NO dispersions." << endl;
139 }
140
141 Debug(0);
142 // this is to catch errors in binding member functions to the property tree.
143 try {
144 Allocate();
145 }
146 catch (const string& msg) {
147 cerr << endl << "Caught error: " << msg << endl;
148 throw;
149 }
150 catch (const BaseException& e) {
151 cout << endl << "Caught error: " << e.what() << endl;
152 throw;
153 }
154
155 trim_status = false;
156 ta_mode = 99;
157 trim_completed = 0;
158
159 Constructing = true;
160 typedef int (FGFDMExec::*iPMF)(void) const;
161 instance->Tie("simulation/do_simple_trim", this, (iPMF)0, &FGFDMExec::DoTrim);
162 instance->Tie("simulation/do_linearization", this, (iPMF)0, &FGFDMExec::DoLinearization);
163 instance->Tie("simulation/reset", this, (iPMF)0, &FGFDMExec::ResetToInitialConditions);
164 instance->Tie("simulation/disperse", this, &FGFDMExec::GetDisperse);
165 instance->Tie("simulation/randomseed", this, (iPMF)&FGFDMExec::SRand, &FGFDMExec::SRand);
166 instance->Tie("simulation/terminate", (bool *)&Terminate);
167 instance->Tie("simulation/pause", (bool *)&holding);
168 instance->Tie("simulation/sim-time-sec", this, &FGFDMExec::GetSimTime);
169 instance->Tie("simulation/dt", this, &FGFDMExec::GetDeltaT);
170 instance->Tie("simulation/jsbsim-debug", this, &FGFDMExec::GetDebugLevel, &FGFDMExec::SetDebugLevel);
171 instance->Tie("simulation/frame", (int *)&Frame);
172 instance->Tie("simulation/trim-completed", (int *)&trim_completed);
173 instance->Tie("forces/hold-down", this, &FGFDMExec::GetHoldDown, &FGFDMExec::SetHoldDown);
174
175 Constructing = false;
176}
177
178//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
179
181{
182 try {
183 Unbind();
184 DeAllocate();
185 } catch (const string& msg ) {
186 cout << "Caught error: " << msg << endl;
187 }
188
189 if (!FDMctr) (*FDMctr)--;
190
191 Debug(1);
192}
193
194//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
195
196double FGFDMExec::Setsim_time(double cur_time) {
197 sim_time = cur_time;
198 Inertial->SetTime(sim_time);
199 return sim_time;
200}
201
202//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
203
205 if (!holding && !IntegrationSuspended()) {
206 sim_time += dT;
207 Inertial->SetTime(sim_time);
208 Frame++;
209 }
210 return sim_time;
211}
212
213//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
214
215bool FGFDMExec::Allocate(void)
216{
217 bool result=true;
218
219 Models.resize(eNumStandardModels);
220
221 // First build the inertial model since some other models are relying on
222 // the inertial model and the ground callback to build themselves.
223 // Note that this does not affect the order in which the models will be
224 // executed later.
225 Models[eInertial] = std::make_shared<FGInertial>(this);
226
227 // See the eModels enum specification in the header file. The order of the
228 // enums specifies the order of execution. The Models[] vector is the primary
229 // storage array for the list of models.
230 Models[ePropagate] = std::make_shared<FGPropagate>(this);
231 Models[eInput] = std::make_shared<FGInput>(this);
232 Models[eAtmosphere] = std::make_shared<FGStandardAtmosphere>(this);
233 Models[eWinds] = std::make_shared<FGWinds>(this);
234 Models[eSystems] = std::make_shared<FGFCS>(this);
235 Models[eMassBalance] = std::make_shared<FGMassBalance>(this);
236 Models[eAuxiliary] = std::make_shared<FGAuxiliary>(this);
237 Models[ePropulsion] = std::make_shared<FGPropulsion>(this);
238 Models[eAerodynamics] = std::make_shared<FGAerodynamics> (this);
239 Models[eGroundReactions] = std::make_shared<FGGroundReactions>(this);
240 Models[eExternalReactions] = std::make_shared<FGExternalReactions>(this);
241 Models[eBuoyantForces] = std::make_shared<FGBuoyantForces>(this);
242 Models[eAircraft] = std::make_shared<FGAircraft>(this);
243 Models[eAccelerations] = std::make_shared<FGAccelerations>(this);
244 Models[eOutput] = std::make_shared<FGOutput>(this);
245
246 // Assign the Model shortcuts for internal executive use only.
247 Propagate = static_cast<FGPropagate*>(Models[ePropagate].get());
248 Inertial = static_cast<FGInertial*>(Models[eInertial].get());
249 Input = static_cast<FGInput*>(Models[eInput].get());
250 Atmosphere = static_cast<FGAtmosphere*>(Models[eAtmosphere].get());
251 Winds = static_cast<FGWinds*>(Models[eWinds].get());
252 FCS = static_cast<FGFCS*>(Models[eSystems].get());
253 MassBalance = static_cast<FGMassBalance*>(Models[eMassBalance].get());
254 Auxiliary = static_cast<FGAuxiliary*>(Models[eAuxiliary].get());
255 Propulsion = static_cast<FGPropulsion*>(Models[ePropulsion].get());
256 Aerodynamics = static_cast<FGAerodynamics*>(Models[eAerodynamics].get());
257 GroundReactions = static_cast<FGGroundReactions*>(Models[eGroundReactions].get());
258 ExternalReactions = static_cast<FGExternalReactions*>(Models[eExternalReactions].get());
259 BuoyantForces = static_cast<FGBuoyantForces*>(Models[eBuoyantForces].get());
260 Aircraft = static_cast<FGAircraft*>(Models[eAircraft].get());
261 Accelerations = static_cast<FGAccelerations*>(Models[eAccelerations].get());
262 Output = static_cast<FGOutput*>(Models[eOutput].get());
263
264 // Initialize planet (environment) constants
265 LoadPlanetConstants();
266
267 // Initialize models
268 InitializeModels();
269
270 IC = std::make_shared<FGInitialCondition>(this);
271 IC->bind(instance.get());
272
273 modelLoaded = false;
274
275 return result;
276}
277
278//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
279
280std::shared_ptr<FGPropagate> FGFDMExec::GetPropagate(void) const
281{
282 return static_pointer_cast<FGPropagate>(Models[ePropagate]);
283}
284
285//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
286
287std::shared_ptr<FGInertial> FGFDMExec::GetInertial(void) const
288{
289 return static_pointer_cast<FGInertial>(Models[eInertial]);
290}
291
292//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
293
294std::shared_ptr<FGInput> FGFDMExec::GetInput(void) const
295{
296 return static_pointer_cast<FGInput>(Models[eInput]);
297}
298
299//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
300
301std::shared_ptr<FGAtmosphere> FGFDMExec::GetAtmosphere(void) const
302{
303 return static_pointer_cast<FGAtmosphere>(Models[eAtmosphere]);
304}
305
306//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
307
308std::shared_ptr<FGWinds> FGFDMExec::GetWinds(void) const
309{
310 return static_pointer_cast<FGWinds>(Models[eWinds]);
311}
312
313//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
314
315std::shared_ptr<FGFCS> FGFDMExec::GetFCS(void) const
316{
317 return static_pointer_cast<FGFCS>(Models[eSystems]);
318}
319
320//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
321
322std::shared_ptr<FGMassBalance> FGFDMExec::GetMassBalance(void) const
323{
324 return static_pointer_cast<FGMassBalance>(Models[eMassBalance]);
325}
326
327//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
328
329std::shared_ptr<FGAuxiliary> FGFDMExec::GetAuxiliary(void) const
330{
331 return static_pointer_cast<FGAuxiliary>(Models[eAuxiliary]);
332}
333
334//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
335
336std::shared_ptr<FGPropulsion> FGFDMExec::GetPropulsion(void) const
337{
338 return static_pointer_cast<FGPropulsion>(Models[ePropulsion]);
339}
340
341//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
342
343std::shared_ptr<FGAerodynamics> FGFDMExec::GetAerodynamics(void) const
344{
345 return static_pointer_cast<FGAerodynamics>(Models[eAerodynamics]);
346}
347
348//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
349
350std::shared_ptr<FGGroundReactions> FGFDMExec::GetGroundReactions(void) const
351{
352 return static_pointer_cast<FGGroundReactions>(Models[eGroundReactions]);
353}
354
355//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
356
357std::shared_ptr<FGExternalReactions> FGFDMExec::GetExternalReactions(void) const
358{
359 return static_pointer_cast<FGExternalReactions>(Models[eExternalReactions]);
360}
361
362//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
363
364std::shared_ptr<FGBuoyantForces> FGFDMExec::GetBuoyantForces(void) const
365{
366 return static_pointer_cast<FGBuoyantForces>(Models[eBuoyantForces]);
367}
368
369//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
370
371std::shared_ptr<FGAircraft> FGFDMExec::GetAircraft(void) const
372{
373 return static_pointer_cast<FGAircraft>(Models[eAircraft]);
374}
375
376//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
377
378std::shared_ptr<FGAccelerations> FGFDMExec::GetAccelerations(void) const
379{
380 return static_pointer_cast<FGAccelerations>(Models[eAccelerations]);
381}
382
383//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
384
385std::shared_ptr<FGOutput> FGFDMExec::GetOutput(void) const
386{
387 return static_pointer_cast<FGOutput>(Models[eOutput]);
388}
389
390//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
391
392void FGFDMExec::InitializeModels(void)
393{
394 for (unsigned int i = 0; i < Models.size(); i++) {
395 // The Input/Output models must not be initialized prior to IC loading
396 if (i == eInput || i == eOutput) continue;
397
398 LoadInputs(i);
399 Models[i]->InitModel();
400 }
401}
402
403//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
404
405bool FGFDMExec::DeAllocate(void)
406{
407
408 Models.clear();
409 modelLoaded = false;
410 return modelLoaded;
411}
412
413//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
414
416{
417 bool success=true;
418
419 Debug(2);
420
421 for (auto &ChildFDM: ChildFDMList) {
422 ChildFDM->AssignState(Propagate); // Transfer state to the child FDM
423 ChildFDM->Run();
424 }
425
426 IncrTime();
427
428 // returns true if success, false if complete
429 if (Script && !IntegrationSuspended()) success = Script->RunScript();
430
431 for (unsigned int i = 0; i < Models.size(); i++) {
432 LoadInputs(i);
433 Models[i]->Run(holding);
434 }
435
436 if (Terminate) success = false;
437
438 return success;
439}
440
441//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
442
443void FGFDMExec::LoadInputs(unsigned int idx)
444{
445 switch(idx) {
446 case ePropagate:
447 Propagate->in.vPQRidot = Accelerations->GetPQRidot();
448 Propagate->in.vUVWidot = Accelerations->GetUVWidot();
449 Propagate->in.DeltaT = dT;
450 break;
451 case eInput:
452 break;
453 case eInertial:
454 Inertial->in.Position = Propagate->GetLocation();
455 break;
456 case eAtmosphere:
457 Atmosphere->in.altitudeASL = Propagate->GetAltitudeASL();
458 Atmosphere->in.GeodLatitudeDeg = Propagate->GetGeodLatitudeDeg();
459 Atmosphere->in.LongitudeDeg = Propagate->GetLongitudeDeg();
460 break;
461 case eWinds:
462 Winds->in.AltitudeASL = Propagate->GetAltitudeASL();
463 Winds->in.DistanceAGL = Propagate->GetDistanceAGL();
464 Winds->in.Tl2b = Propagate->GetTl2b();
465 Winds->in.Tw2b = Auxiliary->GetTw2b();
466 Winds->in.V = Auxiliary->GetVt();
467 Winds->in.totalDeltaT = dT * Winds->GetRate();
468 break;
469 case eAuxiliary:
470 Auxiliary->in.Pressure = Atmosphere->GetPressure();
471 Auxiliary->in.Density = Atmosphere->GetDensity();
472 Auxiliary->in.Temperature = Atmosphere->GetTemperature();
473 Auxiliary->in.SoundSpeed = Atmosphere->GetSoundSpeed();
474 Auxiliary->in.KinematicViscosity = Atmosphere->GetKinematicViscosity();
475 Auxiliary->in.DistanceAGL = Propagate->GetDistanceAGL();
476 Auxiliary->in.Mass = MassBalance->GetMass();
477 Auxiliary->in.Tl2b = Propagate->GetTl2b();
478 Auxiliary->in.Tb2l = Propagate->GetTb2l();
479 Auxiliary->in.vPQR = Propagate->GetPQR();
480 Auxiliary->in.vPQRi = Propagate->GetPQRi();
481 Auxiliary->in.vPQRidot = Accelerations->GetPQRidot();
482 Auxiliary->in.vUVW = Propagate->GetUVW();
483 Auxiliary->in.vUVWdot = Accelerations->GetUVWdot();
484 Auxiliary->in.vVel = Propagate->GetVel();
485 Auxiliary->in.vBodyAccel = Accelerations->GetBodyAccel();
486 Auxiliary->in.ToEyePt = MassBalance->StructuralToBody(Aircraft->GetXYZep());
487 Auxiliary->in.VRPBody = MassBalance->StructuralToBody(Aircraft->GetXYZvrp());
488 Auxiliary->in.RPBody = MassBalance->StructuralToBody(Aircraft->GetXYZrp());
489 Auxiliary->in.vFw = Aerodynamics->GetvFw();
490 Auxiliary->in.vLocation = Propagate->GetLocation();
491 Auxiliary->in.CosTht = Propagate->GetCosEuler(eTht);
492 Auxiliary->in.SinTht = Propagate->GetSinEuler(eTht);
493 Auxiliary->in.CosPhi = Propagate->GetCosEuler(ePhi);
494 Auxiliary->in.SinPhi = Propagate->GetSinEuler(ePhi);
495 Auxiliary->in.TotalWindNED = Winds->GetTotalWindNED();
496 Auxiliary->in.TurbPQR = Winds->GetTurbPQR();
497 break;
498 case eSystems:
499 // Dynamic inputs come into the components that FCS manages through properties
500 break;
501 case ePropulsion:
502 Propulsion->in.Pressure = Atmosphere->GetPressure();
503 Propulsion->in.PressureRatio = Atmosphere->GetPressureRatio();
504 Propulsion->in.Temperature = Atmosphere->GetTemperature();
505 Propulsion->in.DensityRatio = Atmosphere->GetDensityRatio();
506 Propulsion->in.Density = Atmosphere->GetDensity();
507 Propulsion->in.Soundspeed = Atmosphere->GetSoundSpeed();
508 Propulsion->in.TotalPressure = Auxiliary->GetTotalPressure();
509 Propulsion->in.Vc = Auxiliary->GetVcalibratedKTS();
510 Propulsion->in.Vt = Auxiliary->GetVt();
511 Propulsion->in.qbar = Auxiliary->Getqbar();
512 Propulsion->in.TAT_c = Auxiliary->GetTAT_C();
513 Propulsion->in.AeroUVW = Auxiliary->GetAeroUVW();
514 Propulsion->in.AeroPQR = Auxiliary->GetAeroPQR();
515 Propulsion->in.alpha = Auxiliary->Getalpha();
516 Propulsion->in.beta = Auxiliary->Getbeta();
517 Propulsion->in.TotalDeltaT = dT * Propulsion->GetRate();
518 Propulsion->in.ThrottlePos = FCS->GetThrottlePos();
519 Propulsion->in.MixturePos = FCS->GetMixturePos();
520 Propulsion->in.ThrottleCmd = FCS->GetThrottleCmd();
521 Propulsion->in.MixtureCmd = FCS->GetMixtureCmd();
522 Propulsion->in.PropAdvance = FCS->GetPropAdvance();
523 Propulsion->in.PropFeather = FCS->GetPropFeather();
524 Propulsion->in.H_agl = Propagate->GetDistanceAGL();
525 Propulsion->in.PQRi = Propagate->GetPQRi();
526
527 break;
528 case eAerodynamics:
529 Aerodynamics->in.Alpha = Auxiliary->Getalpha();
530 Aerodynamics->in.Beta = Auxiliary->Getbeta();
531 Aerodynamics->in.Qbar = Auxiliary->Getqbar();
532 Aerodynamics->in.Vt = Auxiliary->GetVt();
533 Aerodynamics->in.Tb2w = Auxiliary->GetTb2w();
534 Aerodynamics->in.Tw2b = Auxiliary->GetTw2b();
535 Aerodynamics->in.RPBody = MassBalance->StructuralToBody(Aircraft->GetXYZrp());
536 break;
537 case eGroundReactions:
538 // There are no external inputs to this model.
539 GroundReactions->in.Vground = Auxiliary->GetVground();
540 GroundReactions->in.VcalibratedKts = Auxiliary->GetVcalibratedKTS();
541 GroundReactions->in.Temperature = Atmosphere->GetTemperature();
542 GroundReactions->in.TakeoffThrottle = (FCS->GetThrottlePos().size() > 0) ? (FCS->GetThrottlePos(0) > 0.90) : false;
543 GroundReactions->in.BrakePos = FCS->GetBrakePos();
544 GroundReactions->in.FCSGearPos = FCS->GetGearPos();
545 GroundReactions->in.EmptyWeight = MassBalance->GetEmptyWeight();
546 GroundReactions->in.Tb2l = Propagate->GetTb2l();
547 GroundReactions->in.Tec2l = Propagate->GetTec2l();
548 GroundReactions->in.Tec2b = Propagate->GetTec2b();
549 GroundReactions->in.PQR = Propagate->GetPQR();
550 GroundReactions->in.UVW = Propagate->GetUVW();
551 GroundReactions->in.DistanceAGL = Propagate->GetDistanceAGL();
552 GroundReactions->in.DistanceASL = Propagate->GetAltitudeASL();
553 GroundReactions->in.TotalDeltaT = dT * GroundReactions->GetRate();
554 GroundReactions->in.WOW = GroundReactions->GetWOW();
555 GroundReactions->in.Location = Propagate->GetLocation();
556 GroundReactions->in.vXYZcg = MassBalance->GetXYZcg();
557 break;
558 case eExternalReactions:
559 // There are no external inputs to this model.
560 break;
561 case eBuoyantForces:
562 BuoyantForces->in.Density = Atmosphere->GetDensity();
563 BuoyantForces->in.Pressure = Atmosphere->GetPressure();
564 BuoyantForces->in.Temperature = Atmosphere->GetTemperature();
565 BuoyantForces->in.gravity = Inertial->GetGravity().Magnitude();
566 break;
567 case eMassBalance:
568 MassBalance->in.GasInertia = BuoyantForces->GetGasMassInertia();
569 MassBalance->in.GasMass = BuoyantForces->GetGasMass();
570 MassBalance->in.GasMoment = BuoyantForces->GetGasMassMoment();
571 MassBalance->in.TanksWeight = Propulsion->GetTanksWeight();
572 MassBalance->in.TanksMoment = Propulsion->GetTanksMoment();
573 MassBalance->in.TankInertia = Propulsion->CalculateTankInertias();
574 MassBalance->in.WOW = GroundReactions->GetWOW();
575 break;
576 case eAircraft:
577 Aircraft->in.AeroForce = Aerodynamics->GetForces();
578 Aircraft->in.PropForce = Propulsion->GetForces();
579 Aircraft->in.GroundForce = GroundReactions->GetForces();
580 Aircraft->in.ExternalForce = ExternalReactions->GetForces();
581 Aircraft->in.BuoyantForce = BuoyantForces->GetForces();
582 Aircraft->in.AeroMoment = Aerodynamics->GetMoments();
583 Aircraft->in.PropMoment = Propulsion->GetMoments();
584 Aircraft->in.GroundMoment = GroundReactions->GetMoments();
585 Aircraft->in.ExternalMoment = ExternalReactions->GetMoments();
586 Aircraft->in.BuoyantMoment = BuoyantForces->GetMoments();
587 break;
588 case eAccelerations:
589 Accelerations->in.J = MassBalance->GetJ();
590 Accelerations->in.Jinv = MassBalance->GetJinv();
591 Accelerations->in.Ti2b = Propagate->GetTi2b();
592 Accelerations->in.Tb2i = Propagate->GetTb2i();
593 Accelerations->in.Tec2b = Propagate->GetTec2b();
594 Accelerations->in.Tec2i = Propagate->GetTec2i();
595 Accelerations->in.Moment = Aircraft->GetMoments();
596 Accelerations->in.GroundMoment = GroundReactions->GetMoments();
597 Accelerations->in.Force = Aircraft->GetForces();
598 Accelerations->in.GroundForce = GroundReactions->GetForces();
599 Accelerations->in.vGravAccel = Inertial->GetGravity();
600 Accelerations->in.vPQRi = Propagate->GetPQRi();
601 Accelerations->in.vPQR = Propagate->GetPQR();
602 Accelerations->in.vUVW = Propagate->GetUVW();
603 Accelerations->in.vInertialPosition = Propagate->GetInertialPosition();
604 Accelerations->in.DeltaT = dT;
605 Accelerations->in.Mass = MassBalance->GetMass();
606 Accelerations->in.MultipliersList = GroundReactions->GetMultipliersList();
607 Accelerations->in.TerrainVelocity = Propagate->GetTerrainVelocity();
608 Accelerations->in.TerrainAngularVel = Propagate->GetTerrainAngularVelocity();
609 break;
610 default:
611 break;
612 }
613}
614
615//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
616
617void FGFDMExec::LoadPlanetConstants(void)
618{
619 Propagate->in.vOmegaPlanet = Inertial->GetOmegaPlanet();
620 Accelerations->in.vOmegaPlanet = Inertial->GetOmegaPlanet();
621 Propagate->in.SemiMajor = Inertial->GetSemimajor();
622 Propagate->in.SemiMinor = Inertial->GetSemiminor();
623 Propagate->in.GM = Inertial->GetGM();
624 Auxiliary->in.StandardGravity = Inertial->GetStandardGravity();
625 Auxiliary->in.StdDaySLsoundspeed = Atmosphere->StdDaySLsoundspeed;
626}
627
628//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
629
630void FGFDMExec::LoadModelConstants(void)
631{
632 Winds->in.wingspan = Aircraft->GetWingSpan();
633 Aerodynamics->in.Wingarea = Aircraft->GetWingArea();
634 Aerodynamics->in.Wingchord = Aircraft->Getcbar();
635 Aerodynamics->in.Wingincidence = Aircraft->GetWingIncidence();
636 Aerodynamics->in.Wingspan = Aircraft->GetWingSpan();
637 Auxiliary->in.Wingspan = Aircraft->GetWingSpan();
638 Auxiliary->in.Wingchord = Aircraft->Getcbar();
639 GroundReactions->in.vXYZcg = MassBalance->GetXYZcg();
640}
641
642//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
643// This call will cause the sim time to reset to 0.0
644
646{
647 SuspendIntegration(); // saves the integration rate, dt, then sets it to 0.0.
648 Initialize(IC.get());
649
650 Models[eInput]->InitModel();
651 Models[eOutput]->InitModel();
652
653 Run();
654 Propagate->InitializeDerivatives();
655 ResumeIntegration(); // Restores the integration rate to what it was.
656
657 if (debug_lvl > 0) {
658 MassBalance->GetMassPropertiesReport(0);
659
660 cout << endl << fgblue << highint
661 << "End of vehicle configuration loading." << endl
662 << "-------------------------------------------------------------------------------"
663 << reset << std::setprecision(6) << endl;
664 }
665
666 for (unsigned int n=0; n < Propulsion->GetNumEngines(); ++n) {
667 if (IC->IsEngineRunning(n)) {
668 try {
669 Propulsion->InitRunning(n);
670 } catch (const string& str) {
671 cerr << str << endl;
672 return false;
673 }
674 }
675 }
676
677 return true;
678}
679
680//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
681
683{
684 Propagate->SetInitialState(FGIC);
685 Winds->SetWindNED(FGIC->GetWindNEDFpsIC());
686 Run();
687}
688
689//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
690
692{
693 if (Constructing) return;
694
695 // mode flags
696
697 if (mode & START_NEW_OUTPUT) Output->SetStartNewOutput();
698
699 InitializeModels();
700
701 if (Script)
702 Script->ResetEvents();
703 else
704 Setsim_time(0.0);
705
706 if (!(mode & DONT_EXECUTE_RUN_IC))
707 RunIC();
708}
709
710//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
711
713{
714 HoldDown = hd;
715 Accelerations->SetHoldDown(hd);
716 if (hd) {
717 Propagate->in.vPQRidot = Accelerations->GetPQRidot();
718 Propagate->in.vUVWidot = Accelerations->GetUVWidot();
719 }
720 Propagate->SetHoldDown(hd);
721}
722
723//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
724
725vector <string> FGFDMExec::EnumerateFDMs(void)
726{
727 vector <string> FDMList;
728 FDMList.push_back(Aircraft->GetAircraftName());
729
730 for (auto &ChildFDM: ChildFDMList)
731 FDMList.push_back(ChildFDM->exec->GetAircraft()->GetAircraftName());
732
733 return FDMList;
734}
735
736//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
737
738bool FGFDMExec::LoadScript(const SGPath& script, double deltaT,
739 const SGPath& initfile)
740{
741 Script = std::make_shared<FGScript>(this);
742 return Script->LoadScript(GetFullPath(script), deltaT, initfile);
743}
744
745//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
746
747bool FGFDMExec::LoadPlanet(const SGPath& PlanetPath, bool useAircraftPath)
748{
749 SGPath PlanetFileName;
750
751 if(useAircraftPath && PlanetPath.isRelative()) {
752 PlanetFileName = AircraftPath/PlanetPath.utf8Str();
753 } else {
754 PlanetFileName = PlanetPath;
755 }
756
757 FGXMLFileRead XMLFileRead;
758 Element* document = XMLFileRead.LoadXMLDocument(PlanetFileName);
759
760 // Make sure that the document is valid
761 if (!document) {
762 stringstream s;
763 s << "File: " << PlanetFileName << " could not be read.";
764 cerr << s.str() << endl;
765 throw BaseException(s.str());
766 }
767
768 if (document->GetName() != "planet") {
769 stringstream s;
770 s << "File: " << PlanetFileName << " is not a planet file.";
771 cerr << s.str() << endl;
772 throw BaseException(s.str());
773 }
774
775 bool result = LoadPlanet(document);
776
777 if (!result)
778 cerr << endl << "Planet element has problems in file " << PlanetFileName << endl;
779
780 return result;
781}
782
783//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
784
785bool FGFDMExec::LoadPlanet(Element* element)
786{
787 bool result = Models[eInertial]->Load(element);
788
789 if (result) {
790 // Reload the planet constants and re-initialize the models.
791 LoadPlanetConstants();
792 IC->InitializeIC();
793 InitializeModels();
794
795 // Process the atmosphere element. This element is OPTIONAL.
796 Element* atm_element = element->FindElement("atmosphere");
797 if (atm_element && atm_element->HasAttribute("model")) {
798 string model = atm_element->GetAttributeValue("model");
799 if (model == "MSIS") {
800 // Replace the existing atmosphere model
801 instance->Unbind(Models[eAtmosphere]);
802 Models[eAtmosphere] = std::make_shared<FGMSIS>(this);
803 Atmosphere = static_cast<FGAtmosphere*>(Models[eAtmosphere].get());
804
805 // Model initialization sequence
806 LoadInputs(eAtmosphere);
807 Atmosphere->InitModel();
808 result = Atmosphere->Load(atm_element);
809 if (!result) {
810 cerr << endl << "Incorrect definition of <atmosphere>." << endl;
811 return result;
812 }
813 InitializeModels();
814 }
815 }
816 }
817
818 return result;
819}
820
821//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
822
823bool FGFDMExec::LoadModel(const SGPath& AircraftPath, const SGPath& EnginePath,
824 const SGPath& SystemsPath, const string& model,
825 bool addModelToPath)
826{
827 FGFDMExec::AircraftPath = GetFullPath(AircraftPath);
828 FGFDMExec::EnginePath = GetFullPath(EnginePath);
829 FGFDMExec::SystemsPath = GetFullPath(SystemsPath);
830
831 return LoadModel(model, addModelToPath);
832}
833
834//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
835
836bool FGFDMExec::LoadModel(const string& model, bool addModelToPath)
837{
838 SGPath aircraftCfgFileName;
839 bool result = false; // initialize result to false, indicating input file not yet read
840
841 modelName = model; // Set the class modelName attribute
842
843 if( AircraftPath.isNull() || EnginePath.isNull() || SystemsPath.isNull()) {
844 cerr << "Error: attempted to load aircraft with undefined "
845 << "aircraft, engine, and system paths" << endl;
846 return false;
847 }
848
849 FullAircraftPath = AircraftPath;
850 if (addModelToPath) FullAircraftPath.append(model);
851 aircraftCfgFileName = FullAircraftPath/(model + ".xml");
852
853 if (modelLoaded) {
854 DeAllocate();
855 Allocate();
856 }
857
858 int saved_debug_lvl = debug_lvl;
859 FGXMLFileRead XMLFileRead;
860 Element *document = XMLFileRead.LoadXMLDocument(aircraftCfgFileName); // "document" is a class member
861
862 if (document) {
863 if (IsChild) debug_lvl = 0;
864
865 ReadPrologue(document);
866
867 if (IsChild) debug_lvl = saved_debug_lvl;
868
869 // Process the fileheader element in the aircraft config file. This element is OPTIONAL.
870 Element* element = document->FindElement("fileheader");
871 if (element) {
872 result = ReadFileHeader(element);
873 if (!result) {
874 cerr << endl << "Aircraft fileheader element has problems in file " << aircraftCfgFileName << endl;
875 return result;
876 }
877 }
878
879 if (IsChild) debug_lvl = 0;
880
881 // Process the planet element. This element is OPTIONAL.
882 element = document->FindElement("planet");
883 if (element) {
884 result = LoadPlanet(element);
885 if (!result) {
886 cerr << endl << "Planet element has problems in file " << aircraftCfgFileName << endl;
887 return result;
888 }
889 }
890
891 // Process the metrics element. This element is REQUIRED.
892 element = document->FindElement("metrics");
893 if (element) {
894 result = Models[eAircraft]->Load(element);
895 if (!result) {
896 cerr << endl << "Aircraft metrics element has problems in file " << aircraftCfgFileName << endl;
897 return result;
898 }
899 } else {
900 cerr << endl << "No metrics element was found in the aircraft config file." << endl;
901 return false;
902 }
903
904 // Process the mass_balance element. This element is REQUIRED.
905 element = document->FindElement("mass_balance");
906 if (element) {
907 result = Models[eMassBalance]->Load(element);
908 if (!result) {
909 cerr << endl << "Aircraft mass_balance element has problems in file " << aircraftCfgFileName << endl;
910 return result;
911 }
912 } else {
913 cerr << endl << "No mass_balance element was found in the aircraft config file." << endl;
914 return false;
915 }
916
917 // Process the ground_reactions element. This element is REQUIRED.
918 element = document->FindElement("ground_reactions");
919 if (element) {
920 result = Models[eGroundReactions]->Load(element);
921 if (!result) {
922 cerr << endl << element->ReadFrom()
923 << "Aircraft ground_reactions element has problems in file "
924 << aircraftCfgFileName << endl;
925 return result;
926 }
927 } else {
928 cerr << endl << "No ground_reactions element was found in the aircraft config file." << endl;
929 return false;
930 }
931
932 // Process the external_reactions element. This element is OPTIONAL.
933 element = document->FindElement("external_reactions");
934 if (element) {
935 result = Models[eExternalReactions]->Load(element);
936 if (!result) {
937 cerr << endl << "Aircraft external_reactions element has problems in file " << aircraftCfgFileName << endl;
938 return result;
939 }
940 }
941
942 // Process the buoyant_forces element. This element is OPTIONAL.
943 element = document->FindElement("buoyant_forces");
944 if (element) {
945 result = Models[eBuoyantForces]->Load(element);
946 if (!result) {
947 cerr << endl << "Aircraft buoyant_forces element has problems in file " << aircraftCfgFileName << endl;
948 return result;
949 }
950 }
951
952 // Process the propulsion element. This element is OPTIONAL.
953 element = document->FindElement("propulsion");
954 if (element) {
955 result = Propulsion->Load(element);
956 if (!result) {
957 cerr << endl << "Aircraft propulsion element has problems in file " << aircraftCfgFileName << endl;
958 return result;
959 }
960 for (unsigned int i=0; i < Propulsion->GetNumEngines(); i++)
961 FCS->AddThrottle();
962 }
963
964 // Process the system element[s]. This element is OPTIONAL, and there may be more than one.
965 element = document->FindElement("system");
966 while (element) {
967 result = Models[eSystems]->Load(element);
968 if (!result) {
969 cerr << endl << "Aircraft system element has problems in file " << aircraftCfgFileName << endl;
970 return result;
971 }
972 element = document->FindNextElement("system");
973 }
974
975 // Process the autopilot element. This element is OPTIONAL.
976 element = document->FindElement("autopilot");
977 if (element) {
978 result = Models[eSystems]->Load(element);
979 if (!result) {
980 cerr << endl << "Aircraft autopilot element has problems in file " << aircraftCfgFileName << endl;
981 return result;
982 }
983 }
984
985 // Process the flight_control element. This element is OPTIONAL.
986 element = document->FindElement("flight_control");
987 if (element) {
988 result = Models[eSystems]->Load(element);
989 if (!result) {
990 cerr << endl << "Aircraft flight_control element has problems in file " << aircraftCfgFileName << endl;
991 return result;
992 }
993 }
994
995 // Process the aerodynamics element. This element is OPTIONAL, but almost always expected.
996 element = document->FindElement("aerodynamics");
997 if (element) {
998 result = Models[eAerodynamics]->Load(element);
999 if (!result) {
1000 cerr << endl << "Aircraft aerodynamics element has problems in file " << aircraftCfgFileName << endl;
1001 return result;
1002 }
1003 } else {
1004 cerr << endl << "No expected aerodynamics element was found in the aircraft config file." << endl;
1005 }
1006
1007 // Process the input element. This element is OPTIONAL, and there may be more than one.
1008 element = document->FindElement("input");
1009 while (element) {
1010 if (!Input->Load(element))
1011 return false;
1012
1013 element = document->FindNextElement("input");
1014 }
1015
1016 // Process the output element[s]. This element is OPTIONAL, and there may be
1017 // more than one.
1018 element = document->FindElement("output");
1019 while (element) {
1020 if (!Output->Load(element))
1021 return false;
1022
1023 element = document->FindNextElement("output");
1024 }
1025
1026 // Lastly, process the child element. This element is OPTIONAL - and NOT YET SUPPORTED.
1027 element = document->FindElement("child");
1028 if (element) {
1029 result = ReadChild(element);
1030 if (!result) {
1031 cerr << endl << "Aircraft child element has problems in file " << aircraftCfgFileName << endl;
1032 return result;
1033 }
1034 }
1035
1036 // Since all vehicle characteristics have been loaded, place the values in the Inputs
1037 // structure for the FGModel-derived classes.
1038 LoadModelConstants();
1039
1040 modelLoaded = true;
1041
1042 if (IsChild) debug_lvl = saved_debug_lvl;
1043
1044 } else {
1045 cerr << fgred
1046 << " JSBSim failed to open the configuration file: " << aircraftCfgFileName
1047 << fgdef << endl;
1048 }
1049
1050 for (unsigned int i=0; i< Models.size(); i++) LoadInputs(i);
1051
1052 if (result) {
1053 struct PropertyCatalogStructure masterPCS;
1054 masterPCS.base_string = "";
1055 masterPCS.node = Root;
1056 BuildPropertyCatalog(&masterPCS);
1057 }
1058
1059 return result;
1060}
1061
1062//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1063
1064string FGFDMExec::GetPropulsionTankReport() const
1065{
1066 return Propulsion->GetPropulsionTankReport();
1067}
1068
1069//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1070
1072{
1073 auto pcsNew = std::make_unique<struct PropertyCatalogStructure>();
1074
1075 for (int i=0; i<pcs->node->nChildren(); i++) {
1076 string access="";
1077 pcsNew->base_string = pcs->base_string + "/" + pcs->node->getChild(i)->getNameString();
1078 int node_idx = pcs->node->getChild(i)->getIndex();
1079 if (node_idx != 0) {
1080 pcsNew->base_string = CreateIndexedPropertyName(pcsNew->base_string, node_idx);
1081 }
1082 if (pcs->node->getChild(i)->nChildren() == 0) {
1083 if (pcsNew->base_string.substr(0,12) == string("/fdm/jsbsim/")) {
1084 pcsNew->base_string = pcsNew->base_string.erase(0,12);
1085 }
1086 if (pcs->node->getChild(i)->getAttribute(SGPropertyNode::READ)) access="R";
1087 if (pcs->node->getChild(i)->getAttribute(SGPropertyNode::WRITE)) access+="W";
1088 PropertyCatalog.push_back(pcsNew->base_string+" ("+access+")");
1089 } else {
1090 pcsNew->node = (FGPropertyNode*)pcs->node->getChild(i);
1091 BuildPropertyCatalog(pcsNew.get());
1092 }
1093 }
1094}
1095
1096//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1097
1098string FGFDMExec::QueryPropertyCatalog(const string& in, const string& end_of_line)
1099{
1100 string results;
1101 for (auto &catalogElm: PropertyCatalog) {
1102 if (catalogElm.find(in) != string::npos) results += catalogElm + end_of_line;
1103 }
1104 if (results.empty()) return "No matches found"+end_of_line;
1105 return results;
1106}
1107
1108//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1109
1110void FGFDMExec::PrintPropertyCatalog(void)
1111{
1112 cout << endl;
1113 cout << " " << fgblue << highint << underon << "Property Catalog for "
1114 << modelName << reset << endl << endl;
1115 for (auto &catalogElm: PropertyCatalog)
1116 cout << " " << catalogElm << endl;
1117}
1118
1119//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1120
1121void FGFDMExec::PrintSimulationConfiguration(void) const
1122{
1123 cout << endl << "Simulation Configuration" << endl << "------------------------" << endl;
1124 cout << MassBalance->GetName() << endl;
1125 cout << GroundReactions->GetName() << endl;
1126 cout << Aerodynamics->GetName() << endl;
1127 cout << Propulsion->GetName() << endl;
1128}
1129
1130//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1131
1132bool FGFDMExec::ReadFileHeader(Element* el)
1133{
1134 bool result = true; // true for success
1135
1136 if (debug_lvl == 0) return result;
1137
1138 if (IsChild) {
1139 cout << endl <<highint << fgblue << "Reading child model: " << IdFDM << reset << endl << endl;
1140 }
1141
1142 if (el->FindElement("description"))
1143 cout << " Description: " << el->FindElement("description")->GetDataLine() << endl;
1144 if (el->FindElement("author"))
1145 cout << " Model Author: " << el->FindElement("author")->GetDataLine() << endl;
1146 if (el->FindElement("filecreationdate"))
1147 cout << " Creation Date: " << el->FindElement("filecreationdate")->GetDataLine() << endl;
1148 if (el->FindElement("version"))
1149 cout << " Version: " << el->FindElement("version")->GetDataLine() << endl;
1150
1151 return result;
1152}
1153
1154//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1155
1156bool FGFDMExec::ReadPrologue(Element* el) // el for ReadPrologue is the document element
1157{
1158 bool result = true; // true for success
1159
1160 if (!el) return false;
1161
1162 string AircraftName = el->GetAttributeValue("name");
1163 Aircraft->SetAircraftName(AircraftName);
1164
1165 if (debug_lvl & 1) cout << underon << "Reading Aircraft Configuration File"
1166 << underoff << ": " << highint << AircraftName << normint << endl;
1167
1168 CFGVersion = el->GetAttributeValue("version");
1169 Release = el->GetAttributeValue("release");
1170
1171 if (debug_lvl & 1)
1172 cout << " Version: " << highint << CFGVersion
1173 << normint << endl;
1174 if (CFGVersion != needed_cfg_version) {
1175 cerr << endl << fgred << "YOU HAVE AN INCOMPATIBLE CFG FILE FOR THIS AIRCRAFT."
1176 " RESULTS WILL BE UNPREDICTABLE !!" << endl;
1177 cerr << "Current version needed is: " << needed_cfg_version << endl;
1178 cerr << " You have version: " << CFGVersion << endl << fgdef << endl;
1179 return false;
1180 }
1181
1182 if (Release == "ALPHA" && (debug_lvl & 1)) {
1183 cout << endl << endl
1184 << highint << "This aircraft model is an " << fgred << Release
1185 << reset << highint << " release!!!" << endl << endl << reset
1186 << "This aircraft model may not even properly load, and probably"
1187 << " will not fly as expected." << endl << endl
1188 << fgred << highint << "Use this model for development purposes ONLY!!!"
1189 << normint << reset << endl << endl;
1190 } else if (Release == "BETA" && (debug_lvl & 1)) {
1191 cout << endl << endl
1192 << highint << "This aircraft model is a " << fgred << Release
1193 << reset << highint << " release!!!" << endl << endl << reset
1194 << "This aircraft model probably will not fly as expected." << endl << endl
1195 << fgblue << highint << "Use this model for development purposes ONLY!!!"
1196 << normint << reset << endl << endl;
1197 } else if (Release == "PRODUCTION" && (debug_lvl & 1)) {
1198 cout << endl << endl
1199 << highint << "This aircraft model is a " << fgblue << Release
1200 << reset << highint << " release." << endl << endl << reset;
1201 } else if (debug_lvl & 1) {
1202 cout << endl << endl
1203 << highint << "This aircraft model is an " << fgred << Release
1204 << reset << highint << " release!!!" << endl << endl << reset
1205 << "This aircraft model may not even properly load, and probably"
1206 << " will not fly as expected." << endl << endl
1207 << fgred << highint << "Use this model for development purposes ONLY!!!"
1208 << normint << reset << endl << endl;
1209 }
1210
1211 return result;
1212}
1213
1214//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1215
1216bool FGFDMExec::ReadChild(Element* el)
1217{
1218 // Add a new childData object to the child FDM list
1219 // Populate that childData element with a new FDMExec object
1220 // Set the IsChild flag for that FDMExec object
1221 // Get the aircraft name
1222 // set debug level to print out no additional data for child objects
1223 // Load the model given the aircraft name
1224 // reset debug level to prior setting
1225
1226 auto child = std::make_shared<childData>();
1227
1228 auto pm = std::make_unique<FGPropertyManager>(Root);
1229 child->exec = std::make_unique<FGFDMExec>(pm.get(), FDMctr);
1230 child->exec->SetChild(true);
1231
1232 string childAircraft = el->GetAttributeValue("name");
1233 string sMated = el->GetAttributeValue("mated");
1234 if (sMated == "false") child->mated = false; // child objects are mated by default.
1235 string sInternal = el->GetAttributeValue("internal");
1236 if (sInternal == "true") child->internal = true; // child objects are external by default.
1237
1238 child->exec->SetAircraftPath( AircraftPath );
1239 child->exec->SetEnginePath( EnginePath );
1240 child->exec->SetSystemsPath( SystemsPath );
1241 child->exec->LoadModel(childAircraft);
1242
1243 Element* location = el->FindElement("location");
1244 if (location) {
1245 child->Loc = location->FindElementTripletConvertTo("IN");
1246 } else {
1247 const string s(" No location was found for this child object!");
1248 cerr << el->ReadFrom() << endl << highint << fgred
1249 << s << reset << endl;
1250 throw BaseException(s);
1251 }
1252
1253 Element* orientation = el->FindElement("orient");
1254 if (orientation) {
1255 child->Orient = orientation->FindElementTripletConvertTo("RAD");
1256 } else if (debug_lvl > 0) {
1257 cerr << endl << highint << " No orientation was found for this child object! Assuming 0,0,0." << reset << endl;
1258 }
1259
1260 ChildFDMList.push_back(child);
1261
1262 return true;
1263}
1264
1265//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1266
1267std::shared_ptr<FGTrim> FGFDMExec::GetTrim(void)
1268{
1269 Trim = std::make_shared<FGTrim>(this,tNone);
1270 return Trim;
1271}
1272
1273//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1274
1276{
1277 // Only check if increment then hold is on
1278 if( IncrementThenHolding ) {
1279
1280 if (TimeStepsUntilHold == 0) {
1281
1282 // Should hold simulation if TimeStepsUntilHold has reached zero
1283 holding = true;
1284
1285 // Still need to decrement TimeStepsUntilHold as value of -1
1286 // indicates that incremental then hold is turned off
1287 IncrementThenHolding = false;
1288 TimeStepsUntilHold--;
1289
1290 } else if ( TimeStepsUntilHold > 0 ) {
1291 // Keep decrementing until 0 is reached
1292 TimeStepsUntilHold--;
1293 }
1294 }
1295}
1296
1297//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1298
1299void FGFDMExec::DoTrim(int mode)
1300{
1301 if (Constructing) return;
1302
1303 if (mode < 0 || mode > JSBSim::tNone)
1304 throw("Illegal trimming mode!");
1305
1306 FGTrim trim(this, (JSBSim::TrimMode)mode);
1307 bool success = trim.DoTrim();
1308
1309 if (debug_lvl > 0)
1310 trim.Report();
1311
1312 if (!success)
1313 throw TrimFailureException("Trim Failed");
1314
1315 trim_completed = 1;
1316}
1317
1318//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1319
1321{
1322 double dt0 = this->GetDeltaT();
1323 FGLinearization lin(this);
1324 lin.WriteScicoslab();
1325 this->Setdt(dt0);
1326}
1327
1328//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1329
1330void FGFDMExec::SRand(int sr)
1331{
1332 RandomSeed = sr;
1333 RandomGenerator->seed(RandomSeed);
1334}
1335
1336//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1337// The bitmasked value choices are as follows:
1338// unset: In this case (the default) JSBSim would only print
1339// out the normally expected messages, essentially echoing
1340// the config files as they are read. If the environment
1341// variable is not set, debug_lvl is set to 1 internally
1342// 0: This requests JSBSim not to output any messages
1343// whatsoever.
1344// 1: This value explicity requests the normal JSBSim
1345// startup messages
1346// 2: This value asks for a message to be printed out when
1347// a class is instantiated
1348// 4: When this value is set, a message is displayed when a
1349// FGModel object executes its Run() method
1350// 8: When this value is set, various runtime state variables
1351// are printed out periodically
1352// 16: When set various parameters are sanity checked and
1353// a message is printed out when they go out of bounds
1354
1355void FGFDMExec::Debug(int from)
1356{
1357 if (debug_lvl <= 0) return;
1358
1359 if (debug_lvl & 1 && IdFDM == 0) { // Standard console startup message output
1360 if (from == 0) { // Constructor
1361 cout << "\n\n "
1362 << "JSBSim Flight Dynamics Model v" << JSBSim_version << endl;
1363 cout << " [JSBSim-ML v" << needed_cfg_version << "]\n\n";
1364 cout << "JSBSim startup beginning ...\n\n";
1365 if (disperse == 1) cout << "Dispersions are ON." << endl << endl;
1366 } else if (from == 3) {
1367 cout << "\n\nJSBSim startup complete\n\n";
1368 }
1369 }
1370 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
1371 if (from == 0) cout << "Instantiated: FGFDMExec" << endl;
1372 if (from == 1) cout << "Destroyed: FGFDMExec" << endl;
1373 }
1374 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
1375 if (from == 2) {
1376 cout << "================== Frame: " << Frame << " Time: "
1377 << sim_time << " dt: " << dT << endl;
1378 }
1379 }
1380 if (debug_lvl & 8 ) { // Runtime state variables
1381 }
1382 if (debug_lvl & 16) { // Sanity checking
1383 }
1384 if (debug_lvl & 64) {
1385 if (from == 0) { // Constructor
1386 }
1387 }
1388}
1389}
Element * FindElement(const std::string &el="")
Searches for a specified element.
const std::string & GetName(void) const
Retrieves the element name.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
Handles the calculation of accelerations.
const FGColumnVector3 & GetUVWidot(void) const
Retrieves the body axis acceleration in the ECI frame.
const FGColumnVector3 & GetUVWdot(void) const
Retrieves the body axis acceleration.
const FGColumnVector3 & GetBodyAccel(void) const
Retrieves the acceleration resulting from the applied forces.
const FGColumnVector3 & GetPQRidot(void) const
Retrieves the axis angular acceleration vector in the ECI frame.
void SetHoldDown(bool hd)
Sets the property forces/hold-down.
Encapsulates the aerodynamic calculations.
const FGColumnVector3 & GetMoments(void) const
Gets the total aerodynamic moment vector about the CG.
const FGColumnVector3 & GetvFw(void) const
Retrieves the aerodynamic forces in the wind axes.
const FGColumnVector3 & GetForces(void) const
Gets the total aerodynamic force vector.
Encapsulates an Aircraft and its systems.
Definition FGAircraft.h:102
double Getcbar(void) const
Gets the average wing chord.
Definition FGAircraft.h:138
double GetWingArea(void) const
Gets the wing area.
Definition FGAircraft.h:134
double GetWingSpan(void) const
Gets the wing span.
Definition FGAircraft.h:136
const std::string & GetAircraftName(void) const
Gets the aircraft name.
Definition FGAircraft.h:131
const FGColumnVector3 & GetXYZrp(void) const
Gets the the aero reference point (RP) coordinates.
Definition FGAircraft.h:155
Models an empty, abstract base atmosphere class.
virtual double GetPressureRatio(void) const
Returns the ratio of at-altitude pressure over the sea level value.
virtual double GetPressure(void) const
Returns the pressure in psf.
virtual double GetKinematicViscosity(void) const
Returns the kinematic viscosity.
virtual double GetSoundSpeed(void) const
Returns the speed of sound in ft/sec.
virtual double GetDensityRatio(void) const
Returns the ratio of at-altitude density over the sea level value.
virtual double GetDensity(void) const
Returns the density in slugs/ft^3.
virtual double GetTemperature() const
Returns the actual, modeled temperature at the current altitude in degrees Rankine.
Encapsulates various uncategorized scheduled functions.
const FGMatrix33 & GetTb2w(void) const
Calculates and returns the body-to-wind axis transformation matrix.
const FGMatrix33 & GetTw2b(void) const
Calculates and returns the wind-to-body axis transformation matrix.
double GetVground(void) const
Gets the ground speed in feet per second.
double GetVcalibratedKTS(void) const
Returns Calibrated airspeed in knots.
double GetTotalPressure(void) const
Returns the total pressure.
double GetVt(void) const
Gets the magnitude of total vehicle velocity including wind effects in feet per second.
Encapsulates the Buoyant forces calculations.
const FGColumnVector3 & GetMoments(void) const
Gets the total Buoyancy moment vector.
const FGMatrix33 & GetGasMassInertia(void)
Gets the total moments of inertia for the gas mass in the body frame.
double GetGasMass(void) const
Gets the total gas mass.
const FGColumnVector3 & GetGasMassMoment(void)
Gets the total moment from the gas mass.
const FGColumnVector3 & GetForces(void) const
Gets the total Buoyant force vector.
double Magnitude(void) const
Length of the vector.
Manages the external and/or arbitrary forces and moments.
const FGColumnVector3 & GetMoments(void) const
Retrieves the total moment resulting from the forces defined in the external reactions.
const FGColumnVector3 & GetForces(void) const
Retrieves the total forces defined in the external reactions.
Encapsulates the Flight Control System (FCS) functionality.
Definition FGFCS.h:189
double GetMixtureCmd(int engine) const
Gets the mixture command.
Definition FGFCS.h:253
double GetThrottleCmd(int engine) const
Gets the throttle command.
Definition FGFCS.cpp:351
double GetGearPos(void) const
Gets the gear position (0 up, 1 down), defaults to down.
Definition FGFCS.h:342
bool GetPropFeather(int engine) const
Gets the prop feather position.
Definition FGFCS.h:362
double GetPropAdvance(int engine) const
Gets the prop pitch position.
Definition FGFCS.h:355
double GetMixturePos(int engine) const
Gets the mixture position.
Definition FGFCS.h:336
double GetThrottlePos(int engine) const
Gets the throttle position.
Definition FGFCS.cpp:369
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
bool GetHoldDown(void) const
Gets the value of the property forces/hold-down.
Definition FGFDMExec.h:616
void DoLinearization(int)
Executes linearization with state-space output You must trim first to get an accurate state-space mod...
std::shared_ptr< FGWinds > GetWinds(void) const
Returns the FGWinds pointer.
void SetDebugLevel(int level)
Sets the debug level.
Definition FGFDMExec.h:509
std::shared_ptr< FGBuoyantForces > GetBuoyantForces(void) const
Returns the FGBuoyantForces pointer.
bool LoadScript(const SGPath &Script, double deltaT=0.0, const SGPath &initfile=SGPath())
Load a script.
std::shared_ptr< FGAircraft > GetAircraft(void) const
Returns the FGAircraft pointer.
std::shared_ptr< FGFCS > GetFCS(void) const
Returns the FGFCS pointer.
int GetDebugLevel(void) const
Retrieves the current debug level setting.
Definition FGFDMExec.h:601
std::shared_ptr< FGOutput > GetOutput(void) const
Returns the FGOutput pointer.
static const int START_NEW_OUTPUT
Mode flags for ResetToInitialConditions.
Definition FGFDMExec.h:498
std::string QueryPropertyCatalog(const std::string &check, const std::string &end_of_line="\n")
Retrieves property or properties matching the supplied string.
void BuildPropertyCatalog(struct PropertyCatalogStructure *pcs)
Builds a catalog of properties.
FGFDMExec(FGPropertyManager *root=nullptr, std::shared_ptr< unsigned int > fdmctr=nullptr)
Default constructor.
Definition FGFDMExec.cpp:79
double GetDeltaT(void) const
Returns the simulation delta T.
Definition FGFDMExec.h:552
double Setsim_time(double cur_time)
Sets the current sim time.
std::vector< std::string > EnumerateFDMs(void)
Returns a vector of strings representing the names of all loaded models (future)
void DoTrim(int mode)
Executes trimming in the selected mode.
std::shared_ptr< FGPropagate > GetPropagate(void) const
Returns the FGPropagate pointer.
std::shared_ptr< FGAerodynamics > GetAerodynamics(void) const
Returns the FGAerodynamics pointer.
double IncrTime(void)
Increments the simulation time if not in Holding mode.
std::shared_ptr< FGExternalReactions > GetExternalReactions(void) const
Returns the FGExternalReactions pointer.
void CheckIncrementalHold(void)
Checks if required to hold afer increment.
bool LoadPlanet(const SGPath &PlanetPath, bool useAircraftPath=true)
Loads the planet.
bool Run(void)
This function executes each scheduled model in succession.
std::shared_ptr< FGGroundReactions > GetGroundReactions(void) const
Returns the FGGroundReactions pointer.
std::shared_ptr< FGMassBalance > GetMassBalance(void) const
Returns the FGAircraft pointer.
std::shared_ptr< FGTrim > GetTrim(void)
Returns a pointer to the FGTrim object.
std::shared_ptr< FGPropulsion > GetPropulsion(void) const
Returns the FGPropulsion pointer.
bool LoadModel(const SGPath &AircraftPath, const SGPath &EnginePath, const SGPath &SystemsPath, const std::string &model, bool addModelToPath=true)
Loads an aircraft model.
~FGFDMExec()
Default destructor.
void ResetToInitialConditions(int mode)
Resets the initial conditions object and prepares the simulation to run again.
bool IntegrationSuspended(void) const
Returns the simulation suspension state.
Definition FGFDMExec.h:562
void Unbind(void)
Unbind all tied JSBSim properties.
Definition FGFDMExec.h:243
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
Definition FGFDMExec.h:549
void Setdt(double delta_t)
Sets the integration time step for the simulation executive.
Definition FGFDMExec.h:571
void SuspendIntegration(void)
Suspends the simulation and sets the delta T to zero.
Definition FGFDMExec.h:555
std::shared_ptr< FGInertial > GetInertial(void) const
Returns the FGInertial pointer.
void Initialize(const FGInitialCondition *FGIC)
Initializes the simulation with initial conditions.
void SetHoldDown(bool hd)
Sets the property forces/hold-down.
std::shared_ptr< FGAuxiliary > GetAuxiliary(void) const
Returns the FGAuxiliary pointer.
std::shared_ptr< FGAccelerations > GetAccelerations(void) const
Returns the FGAccelerations pointer.
bool RunIC(void)
Initializes the sim from the initial condition object and executes each scheduled model without integ...
void ResumeIntegration(void)
Resumes the simulation by resetting delta T to the correct value.
Definition FGFDMExec.h:558
std::shared_ptr< FGInput > GetInput(void) const
Returns the FGInput pointer.
Manages ground reactions modeling.
Models inertial forces (e.g.
Definition FGInertial.h:65
void SetTime(double time)
Set the simulation time.
Definition FGInertial.h:140
Initializes the simulation run.
FGColumnVector3 GetWindNEDFpsIC(void) const
Gets the initial wind velocity in the NED local frame.
Handles simulation input.
Definition FGInput.h:85
bool Load(Element *el) override
Load the input directives and adds a new input instance to the Input Manager list.
Definition FGInput.cpp:77
static char normint[6]
normal intensity text
Definition FGJSBBase.h:154
static char fgred[6]
red text
Definition FGJSBBase.h:166
static char fgblue[6]
blue text
Definition FGJSBBase.h:162
static char underon[5]
underlines text
Definition FGJSBBase.h:158
static char fgdef[6]
default text
Definition FGJSBBase.h:170
static char reset[5]
resets text properties
Definition FGJSBBase.h:156
static char underoff[6]
underline off
Definition FGJSBBase.h:160
static char highint[5]
highlights text
Definition FGJSBBase.h:150
Class used to create linear models from FGFDMExec instances.
void WriteScicoslab() const
Write Scicoslab source file with the state space model to a file in the current working directory.
Models weight, balance and moment of inertia information.
const FGColumnVector3 & GetXYZcg(void) const
Returns the coordinates of the center of gravity expressed in the structural frame.
FGColumnVector3 StructuralToBody(const FGColumnVector3 &r) const
Conversion from the structural frame to the body frame.
const FGMatrix33 & GetJinv(void) const
Returns the inverse of the inertia matrix expressed in the body frame.
const FGMatrix33 & GetJ(void) const
Returns the inertia matrix expressed in the body frame.
unsigned int GetRate(void)
Get the output rate for the model in frames.
Definition FGModel.h:93
Handles simulation output.
Definition FGOutput.h:126
void SetStartNewOutput(void)
Reset the output prior to a restart of the simulation.
Definition FGOutput.cpp:118
bool Load(Element *el, const SGPath &dir=SGPath())
Load the output directives and adds a new output instance to the Output Manager list.
Definition FGOutput.cpp:239
Models the EOM and integration/propagation of state.
Definition FGPropagate.h:95
const FGMatrix33 & GetTi2b(void) const
Retrieves the ECI-to-body transformation matrix.
const FGColumnVector3 & GetPQR(void) const
Retrieves the body angular rates vector, relative to the ECEF frame.
double GetSinEuler(int idx) const
Retrieves the sine of a vehicle Euler angle component.
const FGMatrix33 & GetTec2i(void) const
Retrieves the ECEF-to-ECI transformation matrix.
const FGMatrix33 & GetTec2l(void) const
Retrieves the ECEF-to-local transformation matrix.
double GetAltitudeASL(void) const
Returns the current altitude above sea level.
const FGMatrix33 & GetTb2i(void) const
Retrieves the body-to-ECI transformation matrix.
const FGColumnVector3 & GetPQRi(void) const
Retrieves the body angular rates vector, relative to the ECI (inertial) frame.
const FGColumnVector3 & GetUVW(void) const
Retrieves the body frame vehicle velocity vector.
const FGMatrix33 & GetTl2b(void) const
Retrieves the local-to-body transformation matrix.
const FGColumnVector3 & GetVel(void) const
Retrieves the velocity vector.
double GetCosEuler(int idx) const
Retrieves the cosine of a vehicle Euler angle component.
void SetHoldDown(bool hd)
Sets the property forces/hold-down.
const FGMatrix33 & GetTec2b(void) const
Retrieves the ECEF-to-body transformation matrix.
const FGMatrix33 & GetTb2l(void) const
Retrieves the body-to-local transformation matrix.
const FGColumnVector3 & GetInertialPosition(void) const
Retrieves the inertial position vector.
Class wrapper for property handling.
Propulsion management class.
bool Load(Element *el) override
Loads the propulsion system (engine[s] and tank[s]).
size_t GetNumEngines(void) const
Retrieves the number of engines defined for the aircraft.
void InitRunning(int n)
Sets up the engines as running.
The trimming routine for JSBSim.
Definition FGTrim.h:126
Models atmospheric disturbances: winds, gusts, turbulence, downbursts, etc.
Definition FGWinds.h:166
virtual const FGColumnVector3 & GetTotalWindNED(void) const
Retrieves the total wind components in NED frame.
Definition FGWinds.h:187
virtual void SetWindNED(double wN, double wE, double wD)
Sets the wind components in NED frame.
Definition FGWinds.h:195
Random number generator.
Definition FGJSBBase.h:74
FGColumnVector3 TerrainVelocity
Terrain velocities with respect to the local frame (expressed in the ECEF frame).
FGColumnVector3 GroundMoment
Moments generated by the ground normal reactions expressed in the body frame. Does not account for fr...
FGColumnVector3 vUVW
Velocities of the body with respect to the local frame (expressed in the body frame).
FGMatrix33 Tb2i
Transformation matrix from the Body to the ECI frame.
FGMatrix33 J
The body inertia matrix expressed in the body frame.
FGColumnVector3 vPQRi
Angular velocities of the body with respect to the ECI frame (expressed in the body frame).
FGColumnVector3 Moment
Total moments applied to the body except friction and gravity (expressed in the body frame)
FGMatrix33 Jinv
The inverse of the inertia matrix J.
FGColumnVector3 vOmegaPlanet
Earth rotating vector (expressed in the ECI frame).
FGColumnVector3 vInertialPosition
Body position (X,Y,Z) measured in the ECI frame.
FGColumnVector3 vPQR
Angular velocities of the body with respect to the local frame (expressed in the body frame).
FGColumnVector3 vGravAccel
Gravity intensity vector (expressed in the ECEF frame).
FGMatrix33 Tec2i
Transformation matrix from the ECEF to the ECI frame.
std::vector< LagrangeMultiplier * > * MultipliersList
List of Lagrange multipliers set by FGLGear for friction forces calculations.
FGMatrix33 Tec2b
Transformation matrix from the ECEF to the Body frame.
FGColumnVector3 GroundForce
Forces generated by the ground normal reactions expressed in the body frame. Does not account for fri...
FGColumnVector3 TerrainAngularVel
Terrain angular velocities with respect to the local frame (expressed in the ECEF frame).
FGColumnVector3 Force
Total forces applied to the body except friction and gravity (expressed in the body frame)
FGMatrix33 Ti2b
Transformation matrix from the ECI to the Body frame.
FGPropertyNode_ptr node
The node for the property.
Definition FGFDMExec.h:515
std::string base_string
Name of the property.
Definition FGFDMExec.h:513