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"
94 IncrementThenHolding =
false;
95 TimeStepsUntilHold = -1;
101 AircraftPath =
"aircraft";
102 EnginePath =
"engine";
103 SystemsPath =
"systems";
106 char* num = getenv(
"JSBSIM_DEBUG");
107 if (num) debug_lvl = atoi(num);
113 FDMctr = std::make_shared<unsigned int>();
126 Root = root->GetNode();
128 FGPropertyNode* instanceRoot = Root->GetNode(
"fdm/jsbsim", IdFDM,
true);
129 instance = std::make_shared<FGPropertyManager>(instanceRoot);
132 char* num = getenv(
"JSBSIM_DISPERSE");
134 if (atoi(num) != 0) disperse = 1;
138 std::cerr <<
"Could not process JSBSIM_DISPERSIONS environment variable: Assumed NO dispersions." << endl;
146 catch (
const string& msg) {
147 cerr << endl <<
"Caught error: " << msg << endl;
151 cout << endl <<
"Caught error: " << e.what() << endl;
160 typedef int (
FGFDMExec::*iPMF)(
void)
const;
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);
171 instance->Tie(
"simulation/frame", (
int *)&Frame);
172 instance->Tie(
"simulation/trim-completed", (
int *)&trim_completed);
175 Constructing =
false;
185 }
catch (
const string& msg ) {
186 cout <<
"Caught error: " << msg << endl;
189 if (!FDMctr) (*FDMctr)--;
215bool FGFDMExec::Allocate(
void)
219 Models.resize(eNumStandardModels);
225 Models[eInertial] = std::make_shared<FGInertial>(
this);
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);
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());
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());
265 LoadPlanetConstants();
270 IC = std::make_shared<FGInitialCondition>(
this);
271 IC->bind(instance.get());
282 return static_pointer_cast<FGPropagate>(Models[ePropagate]);
289 return static_pointer_cast<FGInertial>(Models[eInertial]);
296 return static_pointer_cast<FGInput>(Models[eInput]);
301std::shared_ptr<FGAtmosphere> FGFDMExec::GetAtmosphere(
void)
const
303 return static_pointer_cast<FGAtmosphere>(Models[eAtmosphere]);
310 return static_pointer_cast<FGWinds>(Models[eWinds]);
317 return static_pointer_cast<FGFCS>(Models[eSystems]);
324 return static_pointer_cast<FGMassBalance>(Models[eMassBalance]);
331 return static_pointer_cast<FGAuxiliary>(Models[eAuxiliary]);
338 return static_pointer_cast<FGPropulsion>(Models[ePropulsion]);
345 return static_pointer_cast<FGAerodynamics>(Models[eAerodynamics]);
352 return static_pointer_cast<FGGroundReactions>(Models[eGroundReactions]);
359 return static_pointer_cast<FGExternalReactions>(Models[eExternalReactions]);
366 return static_pointer_cast<FGBuoyantForces>(Models[eBuoyantForces]);
373 return static_pointer_cast<FGAircraft>(Models[eAircraft]);
380 return static_pointer_cast<FGAccelerations>(Models[eAccelerations]);
387 return static_pointer_cast<FGOutput>(Models[eOutput]);
392void FGFDMExec::InitializeModels(
void)
394 for (
unsigned int i = 0; i < Models.size(); i++) {
396 if (i == eInput || i == eOutput)
continue;
399 Models[i]->InitModel();
405bool FGFDMExec::DeAllocate(
void)
421 for (
auto &ChildFDM: ChildFDMList) {
422 ChildFDM->AssignState(Propagate);
431 for (
unsigned int i = 0; i < Models.size(); i++) {
433 Models[i]->Run(holding);
436 if (Terminate) success =
false;
443void FGFDMExec::LoadInputs(
unsigned int idx)
447 Propagate->in.vPQRidot = Accelerations->
GetPQRidot();
448 Propagate->in.vUVWidot = Accelerations->
GetUVWidot();
449 Propagate->in.DeltaT = dT;
454 Inertial->in.Position = Propagate->GetLocation();
458 Atmosphere->in.GeodLatitudeDeg = Propagate->GetGeodLatitudeDeg();
459 Atmosphere->in.LongitudeDeg = Propagate->GetLongitudeDeg();
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();
470 Auxiliary->in.Pressure = Atmosphere->
GetPressure();
471 Auxiliary->in.Density = Atmosphere->
GetDensity();
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());
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);
496 Auxiliary->in.TurbPQR = Winds->GetTurbPQR();
502 Propulsion->in.Pressure = Atmosphere->
GetPressure();
506 Propulsion->in.Density = Atmosphere->
GetDensity();
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();
524 Propulsion->in.H_agl = Propagate->GetDistanceAGL();
525 Propulsion->in.PQRi = Propagate->
GetPQRi();
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();
537 case eGroundReactions:
539 GroundReactions->in.Vground = Auxiliary->
GetVground();
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();
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();
558 case eExternalReactions:
562 BuoyantForces->in.Density = Atmosphere->
GetDensity();
563 BuoyantForces->in.Pressure = Atmosphere->
GetPressure();
565 BuoyantForces->in.gravity = Inertial->GetGravity().
Magnitude();
569 MassBalance->in.GasMass = BuoyantForces->
GetGasMass();
571 MassBalance->in.TanksWeight = Propulsion->GetTanksWeight();
572 MassBalance->in.TanksMoment = Propulsion->GetTanksMoment();
573 MassBalance->in.TankInertia = Propulsion->CalculateTankInertias();
574 MassBalance->in.WOW = GroundReactions->GetWOW();
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();
589 Accelerations->in.
J = MassBalance->
GetJ();
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();
604 Accelerations->in.
DeltaT = dT;
605 Accelerations->in.
Mass = MassBalance->GetMass();
606 Accelerations->in.
MultipliersList = GroundReactions->GetMultipliersList();
617void FGFDMExec::LoadPlanetConstants(
void)
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;
630void FGFDMExec::LoadModelConstants(
void)
633 Aerodynamics->in.Wingarea = Aircraft->
GetWingArea();
634 Aerodynamics->in.Wingchord = Aircraft->
Getcbar();
635 Aerodynamics->in.Wingincidence = Aircraft->GetWingIncidence();
636 Aerodynamics->in.Wingspan = Aircraft->
GetWingSpan();
638 Auxiliary->in.Wingchord = Aircraft->
Getcbar();
639 GroundReactions->in.vXYZcg = MassBalance->
GetXYZcg();
650 Models[eInput]->InitModel();
651 Models[eOutput]->InitModel();
654 Propagate->InitializeDerivatives();
658 MassBalance->GetMassPropertiesReport(0);
661 <<
"End of vehicle configuration loading." << endl
662 <<
"-------------------------------------------------------------------------------"
663 <<
reset << std::setprecision(6) << endl;
666 for (
unsigned int n=0; n < Propulsion->
GetNumEngines(); ++n) {
667 if (IC->IsEngineRunning(n)) {
670 }
catch (
const string& str) {
684 Propagate->SetInitialState(FGIC);
693 if (Constructing)
return;
702 Script->ResetEvents();
706 if (!(mode & DONT_EXECUTE_RUN_IC))
717 Propagate->in.vPQRidot = Accelerations->
GetPQRidot();
718 Propagate->in.vUVWidot = Accelerations->
GetUVWidot();
727 vector <string> FDMList;
730 for (
auto &ChildFDM: ChildFDMList)
731 FDMList.push_back(ChildFDM->exec->GetAircraft()->GetAircraftName());
739 const SGPath& initfile)
741 Script = std::make_shared<FGScript>(
this);
742 return Script->LoadScript(GetFullPath(script), deltaT, initfile);
749 SGPath PlanetFileName;
751 if(useAircraftPath && PlanetPath.isRelative()) {
752 PlanetFileName = AircraftPath/PlanetPath.utf8Str();
754 PlanetFileName = PlanetPath;
758 Element* document = XMLFileRead.LoadXMLDocument(PlanetFileName);
763 s <<
"File: " << PlanetFileName <<
" could not be read.";
764 cerr << s.str() << endl;
768 if (document->
GetName() !=
"planet") {
770 s <<
"File: " << PlanetFileName <<
" is not a planet file.";
771 cerr << s.str() << endl;
778 cerr << endl <<
"Planet element has problems in file " << PlanetFileName << endl;
787 bool result = Models[eInertial]->Load(element);
791 LoadPlanetConstants();
797 if (atm_element && atm_element->
HasAttribute(
"model")) {
799 if (model ==
"MSIS") {
801 instance->Unbind(Models[eAtmosphere]);
802 Models[eAtmosphere] = std::make_shared<FGMSIS>(
this);
803 Atmosphere =
static_cast<FGAtmosphere*
>(Models[eAtmosphere].get());
806 LoadInputs(eAtmosphere);
807 Atmosphere->InitModel();
808 result = Atmosphere->Load(atm_element);
810 cerr << endl <<
"Incorrect definition of <atmosphere>." << endl;
824 const SGPath& SystemsPath,
const string& model,
827 FGFDMExec::AircraftPath = GetFullPath(AircraftPath);
828 FGFDMExec::EnginePath = GetFullPath(EnginePath);
829 FGFDMExec::SystemsPath = GetFullPath(SystemsPath);
838 SGPath aircraftCfgFileName;
843 if( AircraftPath.isNull() || EnginePath.isNull() || SystemsPath.isNull()) {
844 cerr <<
"Error: attempted to load aircraft with undefined "
845 <<
"aircraft, engine, and system paths" << endl;
849 FullAircraftPath = AircraftPath;
850 if (addModelToPath) FullAircraftPath.append(model);
851 aircraftCfgFileName = FullAircraftPath/(model +
".xml");
858 int saved_debug_lvl = debug_lvl;
860 Element *document = XMLFileRead.LoadXMLDocument(aircraftCfgFileName);
863 if (IsChild) debug_lvl = 0;
865 ReadPrologue(document);
867 if (IsChild) debug_lvl = saved_debug_lvl;
872 result = ReadFileHeader(element);
874 cerr << endl <<
"Aircraft fileheader element has problems in file " << aircraftCfgFileName << endl;
879 if (IsChild) debug_lvl = 0;
886 cerr << endl <<
"Planet element has problems in file " << aircraftCfgFileName << endl;
894 result = Models[eAircraft]->Load(element);
896 cerr << endl <<
"Aircraft metrics element has problems in file " << aircraftCfgFileName << endl;
900 cerr << endl <<
"No metrics element was found in the aircraft config file." << endl;
907 result = Models[eMassBalance]->Load(element);
909 cerr << endl <<
"Aircraft mass_balance element has problems in file " << aircraftCfgFileName << endl;
913 cerr << endl <<
"No mass_balance element was found in the aircraft config file." << endl;
918 element = document->
FindElement(
"ground_reactions");
920 result = Models[eGroundReactions]->Load(element);
923 <<
"Aircraft ground_reactions element has problems in file "
924 << aircraftCfgFileName << endl;
928 cerr << endl <<
"No ground_reactions element was found in the aircraft config file." << endl;
933 element = document->
FindElement(
"external_reactions");
935 result = Models[eExternalReactions]->Load(element);
937 cerr << endl <<
"Aircraft external_reactions element has problems in file " << aircraftCfgFileName << endl;
945 result = Models[eBuoyantForces]->Load(element);
947 cerr << endl <<
"Aircraft buoyant_forces element has problems in file " << aircraftCfgFileName << endl;
955 result = Propulsion->
Load(element);
957 cerr << endl <<
"Aircraft propulsion element has problems in file " << aircraftCfgFileName << endl;
967 result = Models[eSystems]->Load(element);
969 cerr << endl <<
"Aircraft system element has problems in file " << aircraftCfgFileName << endl;
978 result = Models[eSystems]->Load(element);
980 cerr << endl <<
"Aircraft autopilot element has problems in file " << aircraftCfgFileName << endl;
988 result = Models[eSystems]->Load(element);
990 cerr << endl <<
"Aircraft flight_control element has problems in file " << aircraftCfgFileName << endl;
998 result = Models[eAerodynamics]->Load(element);
1000 cerr << endl <<
"Aircraft aerodynamics element has problems in file " << aircraftCfgFileName << endl;
1004 cerr << endl <<
"No expected aerodynamics element was found in the aircraft config file." << endl;
1010 if (!Input->
Load(element))
1020 if (!Output->
Load(element))
1029 result = ReadChild(element);
1031 cerr << endl <<
"Aircraft child element has problems in file " << aircraftCfgFileName << endl;
1038 LoadModelConstants();
1042 if (IsChild) debug_lvl = saved_debug_lvl;
1046 <<
" JSBSim failed to open the configuration file: " << aircraftCfgFileName
1050 for (
unsigned int i=0; i< Models.size(); i++) LoadInputs(i);
1055 masterPCS.
node = Root;
1064string FGFDMExec::GetPropulsionTankReport()
const
1066 return Propulsion->GetPropulsionTankReport();
1073 auto pcsNew = std::make_unique<struct PropertyCatalogStructure>();
1075 for (
int i=0; i<pcs->
node->nChildren(); i++) {
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);
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);
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+
")");
1101 for (
auto &catalogElm: PropertyCatalog) {
1102 if (catalogElm.find(in) != string::npos) results += catalogElm + end_of_line;
1104 if (results.empty())
return "No matches found"+end_of_line;
1110void FGFDMExec::PrintPropertyCatalog(
void)
1114 << modelName <<
reset << endl << endl;
1115 for (
auto &catalogElm: PropertyCatalog)
1116 cout <<
" " << catalogElm << endl;
1121void FGFDMExec::PrintSimulationConfiguration(
void)
const
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;
1132bool FGFDMExec::ReadFileHeader(Element* el)
1136 if (debug_lvl == 0)
return result;
1139 cout << endl <<
highint <<
fgblue <<
"Reading child model: " << IdFDM <<
reset << endl << endl;
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;
1156bool FGFDMExec::ReadPrologue(Element* el)
1160 if (!el)
return false;
1162 string AircraftName = el->GetAttributeValue(
"name");
1163 Aircraft->SetAircraftName(AircraftName);
1165 if (debug_lvl & 1) cout <<
underon <<
"Reading Aircraft Configuration File"
1168 CFGVersion = el->GetAttributeValue(
"version");
1169 Release = el->GetAttributeValue(
"release");
1172 cout <<
" Version: " <<
highint << CFGVersion
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;
1182 if (Release ==
"ALPHA" && (debug_lvl & 1)) {
1183 cout << endl << endl
1184 <<
highint <<
"This aircraft model is an " <<
fgred << Release
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!!!"
1190 }
else if (Release ==
"BETA" && (debug_lvl & 1)) {
1191 cout << endl << endl
1192 <<
highint <<
"This aircraft model is a " <<
fgred << Release
1194 <<
"This aircraft model probably will not fly as expected." << endl << endl
1195 <<
fgblue <<
highint <<
"Use this model for development purposes ONLY!!!"
1197 }
else if (Release ==
"PRODUCTION" && (debug_lvl & 1)) {
1198 cout << endl << endl
1199 <<
highint <<
"This aircraft model is a " <<
fgblue << Release
1201 }
else if (debug_lvl & 1) {
1202 cout << endl << endl
1203 <<
highint <<
"This aircraft model is an " <<
fgred << Release
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!!!"
1216bool FGFDMExec::ReadChild(Element* el)
1226 auto child = std::make_shared<childData>();
1228 auto pm = std::make_unique<FGPropertyManager>(Root);
1229 child->exec = std::make_unique<FGFDMExec>(pm.get(), FDMctr);
1230 child->exec->SetChild(
true);
1232 string childAircraft = el->GetAttributeValue(
"name");
1233 string sMated = el->GetAttributeValue(
"mated");
1234 if (sMated ==
"false") child->mated =
false;
1235 string sInternal = el->GetAttributeValue(
"internal");
1236 if (sInternal ==
"true") child->internal =
true;
1238 child->exec->SetAircraftPath( AircraftPath );
1239 child->exec->SetEnginePath( EnginePath );
1240 child->exec->SetSystemsPath( SystemsPath );
1241 child->exec->LoadModel(childAircraft);
1243 Element* location = el->FindElement(
"location");
1245 child->Loc = location->FindElementTripletConvertTo(
"IN");
1247 const string s(
" No location was found for this child object!");
1249 << s <<
reset << endl;
1250 throw BaseException(s);
1253 Element* orientation = el->FindElement(
"orient");
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;
1260 ChildFDMList.push_back(child);
1269 Trim = std::make_shared<FGTrim>(
this,tNone);
1278 if( IncrementThenHolding ) {
1280 if (TimeStepsUntilHold == 0) {
1287 IncrementThenHolding =
false;
1288 TimeStepsUntilHold--;
1290 }
else if ( TimeStepsUntilHold > 0 ) {
1292 TimeStepsUntilHold--;
1301 if (Constructing)
return;
1303 if (mode < 0 || mode > JSBSim::tNone)
1304 throw(
"Illegal trimming mode!");
1306 FGTrim trim(
this, (JSBSim::TrimMode)mode);
1307 bool success = trim.DoTrim();
1330void FGFDMExec::SRand(
int sr)
1333 RandomGenerator->seed(RandomSeed);
1355void FGFDMExec::Debug(
int from)
1357 if (debug_lvl <= 0)
return;
1359 if (debug_lvl & 1 && IdFDM == 0) {
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";
1370 if (debug_lvl & 2 ) {
1371 if (from == 0) cout <<
"Instantiated: FGFDMExec" << endl;
1372 if (from == 1) cout <<
"Destroyed: FGFDMExec" << endl;
1374 if (debug_lvl & 4 ) {
1376 cout <<
"================== Frame: " << Frame <<
" Time: "
1377 << sim_time <<
" dt: " << dT << endl;
1380 if (debug_lvl & 8 ) {
1382 if (debug_lvl & 16) {
1384 if (debug_lvl & 64) {
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.
double Getcbar(void) const
Gets the average wing chord.
double GetWingArea(void) const
Gets the wing area.
double GetWingSpan(void) const
Gets the wing span.
const std::string & GetAircraftName(void) const
Gets the aircraft name.
const FGColumnVector3 & GetXYZrp(void) const
Gets the the aero reference point (RP) coordinates.
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.
double GetMixtureCmd(int engine) const
Gets the mixture command.
double GetThrottleCmd(int engine) const
Gets the throttle command.
double GetGearPos(void) const
Gets the gear position (0 up, 1 down), defaults to down.
bool GetPropFeather(int engine) const
Gets the prop feather position.
double GetPropAdvance(int engine) const
Gets the prop pitch position.
double GetMixturePos(int engine) const
Gets the mixture position.
double GetThrottlePos(int engine) const
Gets the throttle position.
Encapsulates the JSBSim simulation executive.
bool GetHoldDown(void) const
Gets the value of the property forces/hold-down.
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.
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.
std::shared_ptr< FGOutput > GetOutput(void) const
Returns the FGOutput pointer.
static const int START_NEW_OUTPUT
Mode flags for ResetToInitialConditions.
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.
double GetDeltaT(void) const
Returns the simulation delta T.
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.
void Unbind(void)
Unbind all tied JSBSim properties.
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
void Setdt(double delta_t)
Sets the integration time step for the simulation executive.
void SuspendIntegration(void)
Suspends the simulation and sets the delta T to zero.
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.
std::shared_ptr< FGInput > GetInput(void) const
Returns the FGInput pointer.
Manages ground reactions modeling.
Models inertial forces (e.g.
void SetTime(double time)
Set the simulation time.
Initializes the simulation run.
FGColumnVector3 GetWindNEDFpsIC(void) const
Gets the initial wind velocity in the NED local frame.
static char normint[6]
normal intensity text
static char fgred[6]
red text
static char fgblue[6]
blue text
static char underon[5]
underlines text
static char fgdef[6]
default text
static char reset[5]
resets text properties
static char underoff[6]
underline off
static char highint[5]
highlights text
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.
Handles simulation output.
void SetStartNewOutput(void)
Reset the output prior to a restart of the simulation.
bool Load(Element *el, const SGPath &dir=SGPath())
Load the output directives and adds a new output instance to the Output Manager list.
Models the EOM and integration/propagation of state.
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.
Models atmospheric disturbances: winds, gusts, turbulence, downbursts, etc.
virtual const FGColumnVector3 & GetTotalWindNED(void) const
Retrieves the total wind components in NED frame.
virtual void SetWindNED(double wN, double wE, double wD)
Sets the wind components in NED frame.
FGPropertyNode_ptr node
The node for the property.
std::string base_string
Name of the property.