51#include "FGPropulsion.h"
52#include "models/FGMassBalance.h"
53#include "models/propulsion/FGRocket.h"
54#include "models/propulsion/FGTurbine.h"
55#include "models/propulsion/FGPiston.h"
56#include "models/propulsion/FGElectric.h"
57#include "models/propulsion/FGTurboProp.h"
58#include "models/propulsion/FGTank.h"
59#include "input_output/FGModelLoader.h"
60#include "models/propulsion/FGBrushLessDCMotor.h"
67extern short debug_lvl;
75 Name =
"FGPropulsion";
95bool FGPropulsion::InitModel(
void)
99 if (!FGModel::InitModel())
return false;
101 vForces.InitMatrix();
102 vMoments.InitMatrix();
104 for (
auto& tank: Tanks) tank->ResetToIC();
105 TotalFuelQuantity = 0.0;
106 TotalOxidizerQuantity = 0.0;
107 refuel = dump =
false;
109 for (
auto& engine: Engines) engine->ResetToIC();
119 if (Holding)
return false;
123 vForces.InitMatrix();
124 vMoments.InitMatrix();
126 for (
auto& engine: Engines) {
128 ConsumeFuel(engine.get());
129 vForces += engine->GetBodyForces();
130 vMoments += engine->GetMoments();
133 TotalFuelQuantity = 0.0;
134 TotalOxidizerQuantity = 0.0;
135 for (
auto& tank: Tanks) {
136 tank->Calculate( in.TotalDeltaT, in.TAT_c);
137 switch (tank->GetType()) {
139 TotalFuelQuantity += tank->GetContents();
141 case FGTank::ttOXIDIZER:
142 TotalOxidizerQuantity += tank->GetContents();
149 if (refuel) DoRefuel( in.TotalDeltaT );
150 if (dump) DumpFuel( in.TotalDeltaT );
167void FGPropulsion::ConsumeFuel(
FGEngine* engine)
169 if (FuelFreeze)
return;
170 if (FDMExec->GetTrimStatus())
return;
172 unsigned int TanksWithFuel=0, CurrentFuelTankPriority=1;
173 unsigned int TanksWithOxidizer=0, CurrentOxidizerTankPriority=1;
174 vector <int> FeedListFuel, FeedListOxi;
176 bool hasOxTanks =
false;
184 size_t numTanks = Tanks.size();
187 while ((TanksWithFuel == 0) && (CurrentFuelTankPriority <= numTanks)) {
188 for (
unsigned int i=0; i<engine->GetNumSourceTanks(); i++) {
189 unsigned int TankId = engine->GetSourceTank(i);
190 const auto& Tank = Tanks[TankId];
191 unsigned int TankPriority = Tank->GetPriority();
192 if (TankPriority != 0) {
193 switch(Tank->GetType()) {
195 if ((Tank->GetContents() > Tank->GetUnusable()) && Tank->GetSelected() && (TankPriority == CurrentFuelTankPriority)) {
198 FeedListFuel.push_back(TankId);
201 case FGTank::ttOXIDIZER:
207 if (TanksWithFuel == 0) CurrentFuelTankPriority++;
210 bool FuelStarved = Starved;
214 if (engine->GetType() == FGEngine::etRocket) {
215 while ((TanksWithOxidizer == 0) && (CurrentOxidizerTankPriority <= numTanks)) {
216 for (
unsigned int i=0; i<engine->GetNumSourceTanks(); i++) {
217 unsigned int TankId = engine->GetSourceTank(i);
218 const auto& Tank = Tanks[TankId];
219 unsigned int TankPriority = Tank->GetPriority();
220 if (TankPriority != 0) {
221 switch(Tank->GetType()) {
225 case FGTank::ttOXIDIZER:
227 if (Tank->GetContents() > Tank->GetUnusable() && Tank->GetSelected() && TankPriority == CurrentOxidizerTankPriority) {
229 if (TanksWithFuel > 0) Starved =
false;
230 FeedListOxi.push_back(TankId);
236 if (TanksWithOxidizer == 0) CurrentOxidizerTankPriority++;
240 bool OxiStarved = Starved;
242 engine->SetStarved(FuelStarved || (hasOxTanks && OxiStarved));
246 if (FuelStarved || (hasOxTanks && OxiStarved))
return;
249 double FuelNeededPerTank = FuelToBurn / TanksWithFuel;
250 for (
const auto& feed: FeedListFuel)
251 Tanks[feed]->Drain(FuelNeededPerTank);
253 if (engine->GetType() == FGEngine::etRocket) {
254 double OxidizerToBurn = engine->CalcOxidizerNeed();
255 double OxidizerNeededPerTank = 0;
256 if (TanksWithOxidizer > 0) OxidizerNeededPerTank = OxidizerToBurn / TanksWithOxidizer;
257 for (
const auto& feed: FeedListOxi)
258 Tanks[feed]->Drain(OxidizerNeededPerTank);
267 double currentThrust = 0, lastThrust = -1;
268 int steady_count = 0, j = 0;
270 bool TrimMode = FDMExec->GetTrimStatus();
273 vForces.InitMatrix();
274 vMoments.InitMatrix();
277 FDMExec->SetTrimStatus(
true);
280 in.TotalDeltaT = 0.5;
282 for (
auto& engine: Engines) {
286 while (!steady && j < 6000) {
288 lastThrust = currentThrust;
289 currentThrust = engine->GetThrust();
290 if (fabs(lastThrust-currentThrust) < 0.0001) {
292 if (steady_count > 120) {
300 vForces += engine->GetBodyForces();
301 vMoments += engine->GetMoments();
304 FDMExec->SetTrimStatus(TrimMode);
305 in.TotalDeltaT = TimeStep;
320 throw(
string(
"Tried to initialize a non-existent engine!"));
323 in.ThrottleCmd[n] = in.ThrottlePos[n] = 1;
324 in.MixtureCmd[n] = in.MixturePos[n] = 1;
332 in.ThrottleCmd[i] = in.ThrottlePos[i] = 1;
333 in.MixtureCmd[i] = in.MixturePos[i] = 1;
348 ReadingEngine =
false;
349 double FuelDensity = 6.0;
360 unsigned int numTanks = 0;
362 while (tank_element) {
363 Tanks.push_back(make_shared<FGTank>(FDMExec, tank_element, numTanks));
364 const auto& tank = Tanks.back();
365 if (tank->GetType() == FGTank::ttFUEL)
366 FuelDensity = tank->GetDensity();
367 else if (tank->GetType() != FGTank::ttOXIDIZER) {
368 cerr <<
"Unknown tank type specified." << endl;
375 ReadingEngine =
true;
377 unsigned int numEngines = 0;
379 while (engine_element) {
380 if (!ModelLoader.Open(engine_element))
return false;
385 if (!thruster_element || !ModelLoader.Open(thruster_element))
386 throw(
"No thruster definition supplied with engine definition.");
388 if (engine_element->
FindElement(
"piston_engine")) {
390 Engines.push_back(make_shared<FGPiston>(FDMExec, element, numEngines, in));
391 }
else if (engine_element->
FindElement(
"turbine_engine")) {
393 Engines.push_back(make_shared<FGTurbine>(FDMExec, element, numEngines, in));
394 }
else if (engine_element->
FindElement(
"turboprop_engine")) {
396 Engines.push_back(make_shared<FGTurboProp>(FDMExec, element, numEngines, in));
397 }
else if (engine_element->
FindElement(
"rocket_engine")) {
399 Engines.push_back(make_shared<FGRocket>(FDMExec, element, numEngines, in));
400 }
else if (engine_element->
FindElement(
"electric_engine")) {
402 Engines.push_back(make_shared<FGElectric>(FDMExec, element, numEngines, in));
403 }
else if (engine_element->
FindElement(
"brushless_dc_motor")) {
405 Engines.push_back(make_shared<FGBrushLessDCMotor>(FDMExec, element, numEngines, in));
408 cerr << engine_element->
ReadFrom() <<
" Unknown engine type" << endl;
411 }
catch (std::string& str) {
412 cerr << endl <<
fgred << str <<
reset << endl;
421 if (numEngines) bind();
423 CalculateTankInertias();
430 for (
auto& engine: Engines)
431 engine->SetFuelDensity(FuelDensity);
433 PostLoad(el, FDMExec);
440SGPath FGPropulsion::FindFullPathName(
const SGPath& path)
const
442 SGPath name = FGModel::FindFullPathName(path);
443 if (!ReadingEngine && !name.isNull())
return name;
449 const array<string, 2> dir_names = {
"Engines",
"engine"};
452 const array<string, 4> dir_names = {
"Engines",
"engines",
"Engine",
"engine"};
455 for(
const string& dir_name: dir_names) {
457 if (!name.isNull())
return name;
465string FGPropulsion::GetPropulsionStrings(
const string& delimiter)
const
469 string PropulsionStrings;
470 bool firstime =
true;
473 for (
auto& engine: Engines) {
474 if (firstime) firstime =
false;
475 else PropulsionStrings += delimiter;
477 PropulsionStrings += engine->GetEngineLabels(delimiter);
479 for (
auto& tank: Tanks) {
480 if (tank->GetType() == FGTank::ttFUEL) buf << delimiter <<
"Fuel Tank " << i++;
481 else if (tank->GetType() == FGTank::ttOXIDIZER) buf << delimiter <<
"Oxidizer Tank " << i++;
483 const string& name = tank->GetName();
484 if (!name.empty()) buf <<
" (" << name <<
")";
487 PropulsionStrings += buf.str();
489 return PropulsionStrings;
494string FGPropulsion::GetPropulsionValues(
const string& delimiter)
const
496 string PropulsionValues;
497 bool firstime =
true;
500 for (
const auto& engine: Engines) {
501 if (firstime) firstime =
false;
502 else PropulsionValues += delimiter;
504 PropulsionValues += engine->GetEngineValues(delimiter);
506 for (
const auto& tank: Tanks) {
508 buf << tank->GetContents();
511 PropulsionValues += buf.str();
513 return PropulsionValues;
518string FGPropulsion::GetPropulsionTankReport()
521 stringstream outstream;
524 CalculateTankInertias();
526 for (
const auto& tank: Tanks) {
528 const string& tankname = tank->GetName();
529 if (!tankname.empty()) tankdesc = tankname +
" (";
530 if (tank->GetType() == FGTank::ttFUEL && tank->GetGrainType() != FGTank::gtUNKNOWN) {
531 tankdesc +=
"Solid Fuel";
532 }
else if (tank->GetType() == FGTank::ttFUEL) {
534 }
else if (tank->GetType() == FGTank::ttOXIDIZER) {
535 tankdesc +=
"Oxidizer";
537 tankdesc +=
"Unknown tank type";
539 if (!tankname.empty()) tankdesc +=
")";
540 outstream <<
highint << left << setw(4) << i++ << setw(30) << tankdesc <<
normint
541 << right << setw(12) << tank->GetContents() << setw(8) << tank->GetXYZ(eX)
542 << setw(8) << tank->GetXYZ(eY) << setw(8) << tank->GetXYZ(eZ)
543 << setw(12) << tank->GetIxx() << setw(12) << tank->GetIyy()
544 << setw(12) << tank->GetIzz() << endl;
546 return outstream.str();
551const FGColumnVector3& FGPropulsion::GetTanksMoment(
void)
553 vXYZtank_arm.InitMatrix();
554 for (
const auto& tank: Tanks)
555 vXYZtank_arm += tank->GetXYZ() * tank->GetContents();
562double FGPropulsion::GetTanksWeight(
void)
const
566 for (
const auto& tank: Tanks) Tw += tank->GetContents();
573const FGMatrix33& FGPropulsion::CalculateTankInertias(
void)
575 if (Tanks.empty())
return tankJ;
579 for (
const auto& tank: Tanks) {
580 tankJ += FDMExec->
GetMassBalance()->GetPointmassInertia( lbtoslug * tank->GetContents(),
582 tankJ(1,1) += tank->GetIxx();
583 tankJ(2,2) += tank->GetIyy();
584 tankJ(3,3) += tank->GetIzz();
592void FGPropulsion::SetMagnetos(
int setting)
594 if (ActiveEngine < 0) {
595 for (
auto& engine: Engines) {
599 if (engine->GetType() == FGEngine::etPiston)
600 static_pointer_cast<FGPiston>(engine)->SetMagnetos(setting);
603 auto engine = dynamic_pointer_cast<FGPiston>(Engines[ActiveEngine]);
605 engine->SetMagnetos(setting);
611void FGPropulsion::SetStarter(
int setting)
613 if (ActiveEngine < 0) {
614 for (
auto& engine: Engines) {
616 engine->SetStarter(
false);
618 engine->SetStarter(
true);
622 Engines[ActiveEngine]->SetStarter(
false);
624 Engines[ActiveEngine]->SetStarter(
true);
630int FGPropulsion::GetStarter(
void)
const
632 if (ActiveEngine < 0) {
635 for (
auto& engine: Engines)
636 starter &= engine->GetStarter();
638 return starter ? 1 : 0;
640 return Engines[ActiveEngine]->GetStarter() ? 1: 0;
645void FGPropulsion::SetCutoff(
int setting)
647 bool bsetting = setting == 0 ? false :
true;
649 if (ActiveEngine < 0) {
650 for (
auto& engine: Engines) {
651 switch (engine->GetType()) {
652 case FGEngine::etTurbine:
653 static_pointer_cast<FGTurbine>(engine)->SetCutoff(bsetting);
655 case FGEngine::etTurboprop:
656 static_pointer_cast<FGTurboProp>(engine)->SetCutoff(bsetting);
663 auto engine = Engines[ActiveEngine];
664 switch (engine->GetType()) {
665 case FGEngine::etTurbine:
666 static_pointer_cast<FGTurbine>(engine)->SetCutoff(bsetting);
668 case FGEngine::etTurboprop:
669 static_pointer_cast<FGTurboProp>(engine)->SetCutoff(bsetting);
679int FGPropulsion::GetCutoff(
void)
const
681 if (ActiveEngine < 0) {
684 for (
auto& engine: Engines) {
685 switch (engine->GetType()) {
686 case FGEngine::etTurbine:
687 cutoff &= static_pointer_cast<FGTurbine>(engine)->GetCutoff();
689 case FGEngine::etTurboprop:
690 cutoff &= static_pointer_cast<FGTurboProp>(engine)->GetCutoff();
697 return cutoff ? 1 : 0;
699 auto engine = Engines[ActiveEngine];
700 switch (engine->GetType()) {
701 case FGEngine::etTurbine:
702 return static_pointer_cast<FGTurbine>(engine)->GetCutoff() ? 1 : 0;
703 case FGEngine::etTurboprop:
704 return static_pointer_cast<FGTurboProp>(engine)->GetCutoff() ? 1 : 0;
715void FGPropulsion::SetActiveEngine(
int engine)
717 if (engine >= (
int)Engines.size() || engine < 0)
720 ActiveEngine = engine;
725double FGPropulsion::Transfer(
int source,
int target,
double amount)
727 double shortage, overage;
732 shortage = Tanks[source]->Drain(amount);
737 overage = Tanks[target]->Fill(amount - shortage);
744void FGPropulsion::DoRefuel(
double time_slice)
746 double fillrate = RefuelRate / 60.0 * time_slice;
747 int TanksNotFull = 0;
749 for (
const auto& tank: Tanks) {
750 if (tank->GetPctFull() < 99.99) ++TanksNotFull;
755 for (
unsigned int i=0; i<Tanks.size(); i++) {
756 if (Tanks[i]->GetPctFull() < 99.99)
757 Transfer(-1, i, fillrate/TanksNotFull);
764void FGPropulsion::DumpFuel(
double time_slice)
766 int TanksDumping = 0;
768 for (
const auto& tank: Tanks) {
769 if (tank->GetContents() > tank->GetStandpipe()) ++TanksDumping;
772 if (TanksDumping == 0)
return;
774 double dump_rate_per_tank = DumpRate / 60.0 * time_slice / TanksDumping;
776 for (
unsigned int i=0; i<Tanks.size(); i++) {
777 if (Tanks[i]->GetContents() > Tanks[i]->GetStandpipe()) {
778 Transfer(i, -1, dump_rate_per_tank);
785void FGPropulsion::SetFuelFreeze(
bool f)
788 for (
auto& engine: Engines) engine->SetFuelFreeze(f);
793void FGPropulsion::bind(
void)
795 typedef int (FGPropulsion::*iPMF)(void) const;
796 typedef bool (FGPropulsion::*bPMF)(void) const;
797 bool HavePistonEngine =
false;
798 bool HaveTurboEngine =
false;
800 for (
const auto& engine: Engines) {
801 if (!HavePistonEngine && engine->GetType() == FGEngine::etPiston) HavePistonEngine =
true;
802 if (!HaveTurboEngine && engine->GetType() == FGEngine::etTurbine) HaveTurboEngine =
true;
803 if (!HaveTurboEngine && engine->GetType() == FGEngine::etTurboprop) HaveTurboEngine =
true;
807 if (HaveTurboEngine) {
808 PropertyManager->Tie(
"propulsion/starter_cmd",
this, &FGPropulsion::GetStarter, &FGPropulsion::SetStarter);
809 PropertyManager->Tie(
"propulsion/cutoff_cmd",
this, &FGPropulsion::GetCutoff, &FGPropulsion::SetCutoff);
812 if (HavePistonEngine) {
813 PropertyManager->Tie(
"propulsion/starter_cmd",
this, &FGPropulsion::GetStarter, &FGPropulsion::SetStarter);
814 PropertyManager->Tie(
"propulsion/magneto_cmd",
this, (iPMF)
nullptr, &FGPropulsion::SetMagnetos);
817 PropertyManager->Tie(
"propulsion/active_engine",
this, &FGPropulsion::GetActiveEngine,
818 &FGPropulsion::SetActiveEngine);
819 PropertyManager->Tie(
"forces/fbx-prop-lbs",
this, eX, &FGPropulsion::GetForces);
820 PropertyManager->Tie(
"forces/fby-prop-lbs",
this, eY, &FGPropulsion::GetForces);
821 PropertyManager->Tie(
"forces/fbz-prop-lbs",
this, eZ, &FGPropulsion::GetForces);
822 PropertyManager->Tie(
"moments/l-prop-lbsft",
this, eX, &FGPropulsion::GetMoments);
823 PropertyManager->Tie(
"moments/m-prop-lbsft",
this, eY, &FGPropulsion::GetMoments);
824 PropertyManager->Tie(
"moments/n-prop-lbsft",
this, eZ, &FGPropulsion::GetMoments);
825 PropertyManager->Tie(
"propulsion/total-fuel-lbs", &TotalFuelQuantity);
826 PropertyManager->Tie(
"propulsion/total-oxidizer-lbs", &TotalOxidizerQuantity);
827 PropertyManager->Tie(
"propulsion/refuel", &refuel);
828 PropertyManager->Tie(
"propulsion/fuel_dump", &dump);
829 PropertyManager->Tie(
"propulsion/fuel_freeze",
this, (bPMF)
nullptr, &FGPropulsion::SetFuelFreeze);
851void FGPropulsion::Debug(
int from)
853 if (debug_lvl <= 0)
return;
857 cout << endl <<
" Propulsion:" << endl;
860 if (debug_lvl & 2 ) {
861 if (from == 0) cout <<
"Instantiated: FGPropulsion" << endl;
862 if (from == 1) cout <<
"Destroyed: FGPropulsion" << endl;
864 if (debug_lvl & 4 ) {
866 if (debug_lvl & 8 ) {
868 if (debug_lvl & 16) {
870 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
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...
double FindElementValueAsNumberConvertTo(const std::string &el, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
Base class for all engines.
virtual double CalcFuelNeed(void)
The fuel need is calculated based on power levels and flow rate for that power level.
virtual void Calculate(void)=0
Calculates the thrust of the engine, and other engine functions.
Encapsulates the JSBSim simulation executive.
const SGPath & GetEnginePath(void)
Retrieves the engine path.
const SGPath & GetFullAircraftPath(void)
Retrieves the full aircraft path name.
double GetDeltaT(void) const
Returns the simulation delta T.
std::shared_ptr< FGMassBalance > GetMassBalance(void) const
Returns the FGAircraft pointer.
static char normint[6]
normal intensity text
static char fgred[6]
red text
static char reset[5]
resets text properties
static char highint[5]
highlights text
void InitMatrix(void)
Initialize the matrix.
Base class for all scheduled JSBSim models.
virtual bool Run(bool Holding)
Runs the model; called by the Executive.
bool Upload(Element *el, bool preLoad)
Uploads this model in memory.
bool Load(Element *el) override
Loads the propulsion system (engine[s] and tank[s]).
FGPropulsion(FGFDMExec *)
Constructor.
~FGPropulsion() override
Destructor.
bool GetSteadyState(void)
Loops the engines until thrust output steady (used for trimming)
bool Run(bool Holding) override
Executes the propulsion model.
auto GetEngine(unsigned int index) const
Retrieves an engine object pointer from the list of engines.
size_t GetNumEngines(void) const
Retrieves the number of engines defined for the aircraft.
void InitRunning(int n)
Sets up the engines as running.