46#include "math/FGFunction.h"
47#include "math/FGRealValue.h"
49#include "FGThruster.h"
50#include "input_output/FGXMLElement.h"
51#include "input_output/string_utilities.h"
62 :
FGEngine(engine_number, input), FDMExec(exec)
66 MilThrust = MaxThrust = 10000.0;
67 TSFC = std::make_unique<FGSimplifiedTSFC>(
this, 0.8);
68 ATSFC = std::make_unique<FGRealValue>(1.7);
71 MaxN1 = MaxN2 = 100.0;
72 Augmented = AugMethod = Injected = 0;
73 BypassRatio = BleedDemand = 0.0;
74 IdleThrustLookup = MilThrustLookup = MaxThrustLookup = InjectionLookup =
nullptr;
75 N1_spinup = 1.0; N2_spinup = 3.0; IgnitionN1 = 5.21; IgnitionN2 = 25.18; N1_start_rate = 1.4; N2_start_rate = 2.0;
76 N1_spindown = 2.0; N2_spindown = 2.0;
78 InjectionTimer = InjWaterNorm = 0.0;
80 disableWindmill =
false;
92 N1 = N2 = InjN1increment = InjN2increment = 0.0;
94 correctedTSFC = TSFC->GetValue();
95 AugmentCmd = InjWaterNorm = 0.0;
96 InletPosition = NozzlePosition = 1.0;
97 Stalled = Seized = Overtemp = Fire = Augmentation = Injection = Reversed =
false;
101 OilTemp_degK = in.TAT_c + 273.0;
114 ThrottlePos = in.ThrottlePos[EngineNumber];
116 if (ThrottlePos > 1.0) {
117 AugmentCmd = ThrottlePos - 1.0;
118 ThrottlePos -= AugmentCmd;
124 if ((phase == tpTrim) && (in.TotalDeltaT > 0)) {
125 if (Running && !Starved) {
127 N1_factor = MaxN1 - IdleN1;
128 N2_factor = MaxN2 - IdleN2;
129 N2 = IdleN2 + ThrottlePos * N2_factor;
130 N1 = IdleN1 + ThrottlePos * N1_factor;
131 OilTemp_degK = 366.0;
140 if (!Running && Cutoff && Starter) {
141 if (phase == tpOff) phase = tpSpinUp;
145 if ((Starter ==
true) || (in.qbar > 30.0)) {
146 if (!Running && !Cutoff && (N2 > 15.0)) phase = tpStart;
149 if (Cutoff && (phase != tpSpinUp)) phase = tpOff;
150 if (in.TotalDeltaT == 0) phase = tpTrim;
151 if (Starved) phase = tpOff;
152 if (Stalled) phase = tpStall;
153 if (Seized) phase = tpSeize;
156 case tpOff: thrust = Off();
break;
157 case tpRun: thrust = Run();
break;
158 case tpSpinUp: thrust = SpinUp();
break;
159 case tpStart: thrust = Start();
break;
160 case tpStall: thrust = Stall();
break;
161 case tpSeize: thrust = Seize();
break;
162 case tpTrim: thrust = Trim();
break;
163 default: thrust = Off();
166 Thruster->Calculate(thrust);
173double FGTurbine::Off(
void)
176 FuelFlow_pph =
Seek(&FuelFlow_pph, 0, 1000.0, 10000.0);
178 if (disableWindmill ==
false) {
180 N1 =
Seek(&N1, in.qbar/10.0, N1/2.0 + 0.1, N1/N1_spindown);
181 N2 =
Seek(&N2, in.qbar/15.0, N2/2.0 + 0.1, N2/N2_spindown);
183 N1 =
Seek(&N1, 0, N1/2.0, N1/N1_spindown);
184 N2 =
Seek(&N2, 0, N2/2.0, N2/N2_spindown);
186 EGT_degC =
Seek(&EGT_degC, in.TAT_c, 11.7, 7.3);
187 OilTemp_degK =
Seek(&OilTemp_degK, in.TAT_c + 273.0, 0.2, 0.2);
188 OilPressure_psi = N2 * 0.62;
189 NozzlePosition =
Seek(&NozzlePosition, 1.0, 0.8, 0.8);
190 EPR =
Seek(&EPR, 1.0, 0.2, 0.2);
191 Augmentation =
false;
197double FGTurbine::Run()
199 double idlethrust, milthrust, thrust;
201 idlethrust = MilThrust * IdleThrustLookup->GetValue();
202 milthrust = (MilThrust - idlethrust) * MilThrustLookup->GetValue();
207 N1_factor = MaxN1 - IdleN1;
208 N2_factor = MaxN2 - IdleN2;
209 if ((Injected == 1) && Injection && (InjWaterNorm > 0)) {
210 N1_factor += InjN1increment;
211 N2_factor += InjN2increment;
213 N2 =
Seek(&N2, IdleN2 + ThrottlePos * N2_factor,
214 N2SpoolUp->GetValue(), N2SpoolDown->GetValue());
215 N1 =
Seek(&N1, IdleN1 + ThrottlePos * N1_factor,
216 N1SpoolUp->GetValue(), N1SpoolDown->GetValue());
217 N2norm = (N2 - IdleN2) / N2_factor;
218 thrust = idlethrust + (milthrust * N2norm * N2norm);
219 EGT_degC = in.TAT_c + 363.1 + ThrottlePos * 357.1;
220 OilPressure_psi = N2 * 0.62;
221 OilTemp_degK =
Seek(&OilTemp_degK, 366.0, 1.2, 0.1);
224 correctedTSFC = TSFC->GetValue();
225 FuelFlow_pph =
Seek(&FuelFlow_pph, thrust * correctedTSFC, 1000.0, 10000.0);
226 if (FuelFlow_pph < IdleFF) FuelFlow_pph = IdleFF;
227 NozzlePosition =
Seek(&NozzlePosition, 1.0 - N2norm, 0.8, 0.8);
228 thrust = thrust * (1.0 - BleedDemand);
229 EPR = 1.0 + thrust/MilThrust;
232 if (AugMethod == 1) {
233 if ((ThrottlePos > 0.99) && (N2 > 97.0)) {Augmentation =
true;}
234 else {Augmentation =
false;}
237 if ((Augmented == 1) && Augmentation && (AugMethod < 2)) {
238 thrust = MaxThrustLookup->GetValue() * MaxThrust;
239 FuelFlow_pph =
Seek(&FuelFlow_pph, thrust * ATSFC->GetValue(), 5000.0, 10000.0);
240 NozzlePosition =
Seek(&NozzlePosition, 1.0, 0.8, 0.8);
243 if (AugMethod == 2) {
244 if (AugmentCmd > 0.0) {
246 double tdiff = (MaxThrust * MaxThrustLookup->GetValue()) - thrust;
247 thrust += (tdiff * std::min(AugmentCmd, 1.0));
248 FuelFlow_pph =
Seek(&FuelFlow_pph, thrust * ATSFC->GetValue(), 5000.0, 10000.0);
249 NozzlePosition =
Seek(&NozzlePosition, 1.0, 0.8, 0.8);
251 Augmentation =
false;
255 if ((Injected == 1) && Injection && (InjWaterNorm > 0.0)) {
256 InjectionTimer += in.TotalDeltaT;
257 if (InjectionTimer < InjectionTime) {
258 thrust = thrust * InjectionLookup->GetValue();
259 InjWaterNorm = 1.0 - (InjectionTimer/InjectionTime);
266 if (Cutoff) phase = tpOff;
267 if (Starved) phase = tpOff;
274double FGTurbine::SpinUp(
void)
278 N2 =
Seek(&N2, IgnitionN2, N2_spinup, N2/2.0);
279 N1 =
Seek(&N1, IgnitionN1, N1_spinup, N1/2.0);
280 EGT_degC =
Seek(&EGT_degC, in.TAT_c, 11.7, 7.3);
281 OilPressure_psi = N2 * 0.62;
282 OilTemp_degK =
Seek(&OilTemp_degK, in.TAT_c + 273.0, 0.2, 0.2);
284 NozzlePosition = 1.0;
285 if (Starter ==
false) phase = tpOff;
291double FGTurbine::Start(
void)
293 if ((N2 > 15.0) && !Starved) {
296 N2 =
Seek(&N2, IdleN2, N2_start_rate, N2/2.0);
297 N1 =
Seek(&N1, IdleN1, N1_start_rate, N1/2.0);
298 EGT_degC =
Seek(&EGT_degC, in.TAT_c + 363.1, 21.3, 7.3);
299 FuelFlow_pph = IdleFF * N2 / IdleN2;
300 OilPressure_psi = N2 * 0.62;
301 if ((Starter ==
false) && (in.qbar < 30.0)) phase = tpOff;
320double FGTurbine::Stall(
void)
322 EGT_degC = in.TAT_c + 903.14;
323 FuelFlow_pph = IdleFF;
324 N1 =
Seek(&N1, in.qbar/10.0, 0, N1/10.0);
325 N2 =
Seek(&N2, in.qbar/15.0, 0, N2/10.0);
326 if (ThrottlePos < 0.01) {
335double FGTurbine::Seize(
void)
338 N1 =
Seek(&N1, in.qbar/20.0, 0, N1/15.0);
339 FuelFlow_pph = Cutoff ? 0.0 : IdleFF;
340 OilPressure_psi = 0.0;
341 OilTemp_degK =
Seek(&OilTemp_degK, in.TAT_c + 273.0, 0, 0.2);
348double FGTurbine::Trim()
350 double idlethrust = MilThrust * IdleThrustLookup->GetValue();
351 double milthrust = (MilThrust - idlethrust) * MilThrustLookup->GetValue();
352 double N2 = IdleN2 + ThrottlePos * N2_factor;
353 double N2norm = (N2 - IdleN2) / N2_factor;
354 double thrust = (idlethrust + (milthrust * N2norm * N2norm))
355 * (1.0 - BleedDemand);
357 if (AugMethod == 1) {
358 if ((ThrottlePos > 0.99) && (N2 > 97.0)) {Augmentation =
true;}
359 else {Augmentation =
false;}
362 if ((Augmented == 1) && Augmentation && (AugMethod < 2)) {
363 thrust = MaxThrust * MaxThrustLookup->GetValue();
366 if (AugMethod == 2) {
367 if (AugmentCmd > 0.0) {
368 double tdiff = (MaxThrust * MaxThrustLookup->GetValue()) - thrust;
369 thrust += (tdiff * std::min(AugmentCmd, 1.0));
373 if ((Injected == 1) && Injection) {
374 thrust = thrust * InjectionLookup->GetValue();
384 FuelFlowRate = FuelFlow_pph / 3600.0;
385 FuelExpended = FuelFlowRate * in.TotalDeltaT;
386 if (!Starved) FuelUsedLbs += FuelExpended;
392double FGTurbine::GetPowerAvailable(
void) {
393 if( ThrottlePos <= 0.77 )
394 return 64.94*ThrottlePos;
396 return 217.38*ThrottlePos - 117.38;
404 v -= in.TotalDeltaT * decel;
405 if (v < target) v = target;
406 }
else if (v < target) {
407 v += in.TotalDeltaT * accel;
408 if (v > target) v = target;
419 while(function_element) {
421 if (name ==
"IdleThrust" || name ==
"MilThrust" || name ==
"AugThrust"
422 || name ==
"Injection" || name ==
"N1SpoolUp" || name ==
"N1SpoolDown"
423 || name ==
"N2SpoolUp" || name ==
"N2SpoolDown")
424 function_element->
SetAttributeValue(
"name",
string(
"propulsion/engine[#]/") + name);
429 FGEngine::Load(exec, el);
481 string property_prefix = CreateIndexedPropertyName(
"propulsion/engine", EngineNumber);
492 TSFC = std::make_unique<FGSimplifiedTSFC>(
this, atof_locale_c(value));
493 }
catch (InvalidNumber&) {
494 TSFC = std::make_unique<FGFunction>(FDMExec, tsfcElement, to_string(EngineNumber));
502 ATSFC = std::make_unique<FGRealValue>(atof_locale_c(value));
503 }
catch (InvalidNumber&) {
504 ATSFC = std::make_unique<FGFunction>(FDMExec, atsfcElement, to_string((
int)EngineNumber));
511 N1SpoolUp = std::make_shared<FGSpoolUp>(
this, BypassRatio, 1.0);
515 N1SpoolDown = std::make_shared<FGSpoolUp>(
this, BypassRatio, 2.4);
519 N2SpoolUp = std::make_shared<FGSpoolUp>(
this, BypassRatio, 1.0);
523 N2SpoolDown = std::make_shared<FGSpoolUp>(
this, BypassRatio, 3.0);
525 N1_factor = MaxN1 - IdleN1;
526 N2_factor = MaxN2 - IdleN2;
527 OilTemp_degK = in.TAT_c + 273.0;
528 IdleFF = pow(MilThrust, 0.2) * 107.0;
536string FGTurbine::GetEngineLabels(
const string& delimiter)
538 std::ostringstream buf;
540 buf << Name <<
"_N1[" << EngineNumber <<
"]" << delimiter
541 << Name <<
"_N2[" << EngineNumber <<
"]" << delimiter
542 << Thruster->GetThrusterLabels(EngineNumber, delimiter);
549string FGTurbine::GetEngineValues(
const string& delimiter)
551 std::ostringstream buf;
553 buf << N1 << delimiter
555 << Thruster->GetThrusterValues(EngineNumber, delimiter);
562void FGTurbine::bindmodel(FGPropertyManager* PropertyManager)
564 string property_name, base_property_name;
565 base_property_name = CreateIndexedPropertyName(
"propulsion/engine", EngineNumber);
566 property_name = base_property_name +
"/n1";
567 PropertyManager->Tie( property_name.c_str(), &N1);
568 property_name = base_property_name +
"/n2";
569 PropertyManager->Tie( property_name.c_str(), &N2);
570 property_name = base_property_name +
"/injection_cmd";
571 PropertyManager->Tie( property_name.c_str(),
this,
572 &FGTurbine::GetInjection, &FGTurbine::SetInjection);
573 property_name = base_property_name +
"/seized";
574 PropertyManager->Tie( property_name.c_str(), &Seized);
575 property_name = base_property_name +
"/stalled";
576 PropertyManager->Tie( property_name.c_str(), &Stalled);
577 property_name = base_property_name +
"/bleed-factor";
578 PropertyManager->Tie( property_name.c_str(),
this, &FGTurbine::GetBleedDemand, &FGTurbine::SetBleedDemand);
579 property_name = base_property_name +
"/MaxN1";
580 PropertyManager->Tie( property_name.c_str(),
this,
581 &FGTurbine::GetMaxN1, &FGTurbine::SetMaxN1);
582 property_name = base_property_name +
"/MaxN2";
583 PropertyManager->Tie( property_name.c_str(),
this,
584 &FGTurbine::GetMaxN2, &FGTurbine::SetMaxN2);
585 property_name = base_property_name +
"/InjectionTimer";
586 PropertyManager->Tie( property_name.c_str(),
this,
587 &FGTurbine::GetInjectionTimer, &FGTurbine::SetInjectionTimer);
588 property_name = base_property_name +
"/InjWaterNorm";
589 PropertyManager->Tie( property_name.c_str(),
this,
590 &FGTurbine::GetInjWaterNorm, &FGTurbine::SetInjWaterNorm);
591 property_name = base_property_name +
"/InjN1increment";
592 PropertyManager->Tie( property_name.c_str(),
this,
593 &FGTurbine::GetInjN1increment, &FGTurbine::SetInjN1increment);
594 property_name = base_property_name +
"/InjN2increment";
595 PropertyManager->Tie( property_name.c_str(),
this,
596 &FGTurbine::GetInjN2increment, &FGTurbine::SetInjN2increment);
597 property_name = base_property_name +
"/atsfc";
598 PropertyManager->Tie(property_name.c_str(), ATSFC.get(), &FGParameter::GetValue);
599 property_name = base_property_name +
"/tsfc";
600 PropertyManager->Tie(property_name.c_str(), &correctedTSFC);
601 auto node = PropertyManager->GetNode(property_name.c_str(),
false);
602 node->setAttribute(SGPropertyNode::WRITE,
false);
607int FGTurbine::InitRunning(
void)
612 N1_factor = MaxN1 - IdleN1;
613 N2_factor = MaxN2 - IdleN2;
614 N2 = IdleN2 + ThrottlePos * N2_factor;
615 N1 = IdleN1 + ThrottlePos * N1_factor;
640void FGTurbine::Debug(
int from)
642 if (debug_lvl <= 0)
return;
649 cout <<
"\n Engine Name: " << Name << endl;
650 cout <<
" MilThrust: " << MilThrust << endl;
651 cout <<
" MaxThrust: " << MaxThrust << endl;
652 cout <<
" BypassRatio: " << BypassRatio << endl;
653 cout <<
" TSFC: " << TSFC->GetValue() << endl;
654 cout <<
" ATSFC: " << ATSFC->GetValue() << endl;
655 cout <<
" IdleN1: " << IdleN1 << endl;
656 cout <<
" IdleN2: " << IdleN2 << endl;
657 cout <<
" MaxN1: " << MaxN1 << endl;
658 cout <<
" MaxN2: " << MaxN2 << endl;
659 cout <<
" Augmented: " << Augmented << endl;
660 cout <<
" AugMethod: " << AugMethod << endl;
661 cout <<
" Injected: " << Injected << endl;
662 cout <<
" MinThrottle: " << MinThrottle << endl;
667 if (debug_lvl & 2 ) {
668 if (from == 0) cout <<
"Instantiated: FGTurbine" << endl;
669 if (from == 1) cout <<
"Destroyed: FGTurbine" << endl;
671 if (debug_lvl & 4 ) {
673 if (debug_lvl & 8 ) {
675 if (debug_lvl & 16) {
677 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
bool SetAttributeValue(const std::string &key, const std::string &value)
Modifies an attribute.
bool FindElementValueAsBoolean(const std::string &el="")
Searches for the named element and returns the data belonging to it as a bool.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string GetDataLine(unsigned int i=0)
Gets a line of data belonging to an element.
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.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
Base class for all engines.
virtual void ResetToIC(void)
Resets the Engine parameters to the initial conditions.
Encapsulates the JSBSim simulation executive.
void SuspendIntegration(void)
Suspends the simulation and sets the delta T to zero.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
void ResumeIntegration(void)
Resumes the simulation by resetting delta T to the correct value.
std::shared_ptr< FGFunction > GetPreFunction(const std::string &name)
Get one of the "pre" function.
double Seek(double *var, double target, double accel, double decel)
A lag filter.
void Calculate(void)
Calculates the thrust of the engine, and other engine functions.
void ResetToIC(void)
Resets the Engine parameters to the initial conditions.
double CalcFuelNeed(void)
The fuel need is calculated based on power levels and flow rate for that power level.
FGTurbine(FGFDMExec *Executive, Element *el, int engine_number, struct Inputs &input)
Constructor.