48#include "FGTurboProp.h"
49#include "FGPropeller.h"
51#include "math/FGFunction.h"
52#include "input_output/FGXMLElement.h"
74 MaxStartingTime = 999999;
79 while(function_element) {
81 if (name ==
"EnginePowerVC")
82 function_element->
SetAttributeValue(
"name",
string(
"propulsion/engine[#]/") + name);
87 FGEngine::Load(exec, el);
88 thrusterType = Thruster->GetType();
90 string property_prefix = CreateIndexedPropertyName(
"propulsion/engine", EngineNumber);
103 BetaRangeThrottleEnd =
Constrain(0.0, BetaRangeThrottleEnd, 0.99999);
110 cerr << el->
ReadFrom() <<
"Note: 'idlefuelflow' is obsolete, "
111 <<
"use the 'CombustionEfficiency_N1' table instead." << endl;
129 while (table_element) {
130 string name = table_element->GetAttributeValue(
"name");
131 if (!EnginePowerVC && name ==
"EnginePowerVC") {
136 table_element->SetAttributeValue(
"name",
string(
"propulsion/engine[#]/") + name);
137 EnginePowerVC = std::make_shared<FGTable>(PropertyManager, table_element,
138 to_string((
int)EngineNumber));
139 table_element->SetAttributeValue(
"name", name);
140 cerr << table_element->ReadFrom()
141 <<
"Note: Using the EnginePowerVC without enclosed <function> tag is deprecated"
143 }
else if (name ==
"EnginePowerRPM_N1") {
144 EnginePowerRPM_N1 = std::make_unique<FGTable>(PropertyManager, table_element);
145 }
else if (name ==
"ITT_N1") {
146 ITT_N1 = std::make_unique<FGTable>(PropertyManager, table_element);
147 }
else if (name ==
"CombustionEfficiency_N1") {
148 CombustionEfficiency_N1 = std::make_unique<FGTable>(PropertyManager, table_element);
150 cerr << el->
ReadFrom() <<
"Unknown table type: " << name
151 <<
" in turboprop definition." << endl;
159 N1_factor = MaxN1 - IdleN1;
160 OilTemp_degK = in.TAT_c + 273.0;
164 if (! CombustionEfficiency_N1) {
165 CombustionEfficiency_N1 = std::make_unique<FGTable>(6);
166 *CombustionEfficiency_N1 << 60.0 << 12.0/52.0;
167 *CombustionEfficiency_N1 << 82.0 << 12.0/30.0;
168 *CombustionEfficiency_N1 << 96.0 << 12.0/16.0;
169 *CombustionEfficiency_N1 << 100.0 << 1.0;
170 *CombustionEfficiency_N1 << 104.0 << 1.5;
171 *CombustionEfficiency_N1 << 110.0 << 6.0;
174 bindmodel(PropertyManager.get());
186 ThrottlePos = in.ThrottlePos[EngineNumber];
190 RPM = Thruster->GetEngineRPM();
191 if (thrusterType == FGThruster::ttPropeller) {
192 ((
FGPropeller*)Thruster)->SetAdvance(in.PropAdvance[EngineNumber]);
193 ((
FGPropeller*)Thruster)->SetFeather(in.PropFeather[EngineNumber]);
196 ((
FGPropeller*)Thruster)->SetReverseCoef(ThrottlePos);
202 if (ThrottlePos < BetaRangeThrottleEnd) {
206 ThrottlePos = (ThrottlePos-BetaRangeThrottleEnd)/(1-BetaRangeThrottleEnd) * ReverseMaxPower;
212 if ((phase == tpTrim) && (in.TotalDeltaT > 0)) {
213 if (Running && !Starved) {
216 OilTemp_degK = 366.0;
221 Eng_ITT_degC = in.TAT_c;
222 Eng_Temperature = in.TAT_c;
223 OilTemp_degK = in.TAT_c+273.15;
227 if (!Running && Starter) {
228 if (phase == tpOff) {
230 if (StartTime < 0) StartTime=0;
233 if (!Running && !Cutoff && (N1 > 15.0)) {
237 if (Cutoff && (phase != tpSpinUp)) phase = tpOff;
238 if (in.TotalDeltaT == 0) phase = tpTrim;
239 if (Starved) phase = tpOff;
240 if (Condition >= 10) {
246 if (Ielu_max_torque > 0.0) {
249 if (thrusterType == FGThruster::ttPropeller) {
251 }
else if (thrusterType == FGThruster::ttRotor) {
252 torque = ((
FGRotor*)(Thruster))->GetTorque();
256 if ( abs(torque) > Ielu_max_torque && ThrottlePos >= OldThrottle ) {
257 ThrottlePos = OldThrottle - 0.1 * in.TotalDeltaT;
258 Ielu_intervent =
true;
259 }
else if ( Ielu_intervent && ThrottlePos >= OldThrottle) {
260 ThrottlePos = OldThrottle + 0.05 * in.TotalDeltaT;
261 Ielu_intervent =
true;
263 Ielu_intervent =
false;
266 Ielu_intervent =
false;
268 OldThrottle = ThrottlePos;
272 case tpOff: HP = Off();
break;
273 case tpRun: HP = Run();
break;
274 case tpSpinUp: HP = SpinUp();
break;
275 case tpStart: HP = Start();
break;
279 LoadThrusterInputs();
281 double power = HP * hptoftlbssec;
282 if (RPM <= 0.1) power = max(power, 0.0);
283 Thruster->Calculate(power);
290double FGTurboProp::Off(
void)
292 Running =
false; EngStarting =
false;
294 FuelFlow_pph = Seek(&FuelFlow_pph, 0, 800.0, 800.0);
297 N1 = ExpSeek(&N1, in.qbar/15.0, Idle_Max_Delay*2.5, Idle_Max_Delay * 5);
299 OilTemp_degK = ExpSeek(&OilTemp_degK,273.15 + in.TAT_c, 400 , 400);
301 Eng_Temperature = ExpSeek(&Eng_Temperature,in.TAT_c,300,400);
302 double ITT_goal = ITT_N1->GetValue(N1,0.1) + ((N1>20) ? 0.0 : (20-N1)/20.0 * Eng_Temperature);
303 Eng_ITT_degC = ExpSeek(&Eng_ITT_degC,ITT_goal,ITT_Delay,ITT_Delay*1.2);
305 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6;
307 if (RPM>5)
return -0.012;
313double FGTurboProp::Run(
void)
317 Running =
true; Starter =
false; EngStarting =
false;
321 N1 = ExpSeek(&N1, IdleN1 + ThrottlePos * N1_factor, Idle_Max_Delay, Idle_Max_Delay * 2.4);
323 EngPower_HP = EnginePowerRPM_N1->GetValue(RPM,N1);
324 EngPower_HP *= EnginePowerVC->GetValue();
325 if (EngPower_HP > MaxPower) EngPower_HP = MaxPower;
327 CombustionEfficiency = CombustionEfficiency_N1->GetValue(N1);
328 FuelFlow_pph = PSFC / CombustionEfficiency * EngPower_HP;
330 Eng_Temperature = ExpSeek(&Eng_Temperature,Eng_ITT_degC,300,400);
331 double ITT_goal = ITT_N1->GetValue((N1-old_N1)*300+N1,1);
332 Eng_ITT_degC = ExpSeek(&Eng_ITT_degC,ITT_goal,ITT_Delay,ITT_Delay*1.2);
334 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6;
337 OilTemp_degK = Seek(&OilTemp_degK, 353.15, 0.4-N1*0.001, 0.04);
339 if (Cutoff) phase = tpOff;
340 if (Starved) phase = tpOff;
347double FGTurboProp::SpinUp(
void)
350 Running =
false; EngStarting =
true;
353 if (!GeneratorPower) {
360 N1 = ExpSeek(&N1, StarterN1, Idle_Max_Delay * 6, Idle_Max_Delay * 2.4);
362 Eng_Temperature = ExpSeek(&Eng_Temperature,in.TAT_c,300,400);
363 double ITT_goal = ITT_N1->GetValue(N1,0.1) + ((N1>20) ? 0.0 : (20-N1)/20.0 * Eng_Temperature);
364 Eng_ITT_degC = ExpSeek(&Eng_ITT_degC,ITT_goal,ITT_Delay,ITT_Delay*1.2);
366 OilTemp_degK = ExpSeek(&OilTemp_degK,273.15 + in.TAT_c, 400 , 400);
368 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6;
370 EngPower_HP = EnginePowerRPM_N1->GetValue(RPM,N1);
371 EngPower_HP *= EnginePowerVC->GetValue();
372 if (EngPower_HP > MaxPower) EngPower_HP = MaxPower;
374 if (StartTime>=0) StartTime+=in.TotalDeltaT;
375 if (StartTime > MaxStartingTime && MaxStartingTime > 0) {
385double FGTurboProp::Start(
void)
387 double EngPower_HP = 0.0;
390 if ((N1 > 15.0) && !Starved) {
394 EngPower_HP = EnginePowerRPM_N1->GetValue(RPM,N1);
395 EngPower_HP *= EnginePowerVC->GetValue();
396 if (EngPower_HP > MaxPower) EngPower_HP = MaxPower;
397 N1 = ExpSeek(&N1, IdleN1*1.1, Idle_Max_Delay*4, Idle_Max_Delay * 2.4);
398 CombustionEfficiency = CombustionEfficiency_N1->GetValue(N1);
399 FuelFlow_pph = PSFC / CombustionEfficiency * EngPower_HP;
400 Eng_Temperature = ExpSeek(&Eng_Temperature,Eng_ITT_degC,300,400);
401 double ITT_goal = ITT_N1->GetValue((N1-old_N1)*300+N1,1);
402 Eng_ITT_degC = ExpSeek(&Eng_ITT_degC,ITT_goal,ITT_Delay,ITT_Delay*1.2);
404 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6;
405 OilTemp_degK = Seek(&OilTemp_degK, 353.15, 0.4-N1*0.001, 0.04);
427 FuelFlowRate = FuelFlow_pph / 3600.0;
428 FuelExpended = FuelFlowRate * in.TotalDeltaT;
429 if (!Starved) FuelUsedLbs += FuelExpended;
435double FGTurboProp::Seek(
double *var,
double target,
double accel,
double decel)
439 v -= in.TotalDeltaT * decel;
440 if (v < target) v = target;
441 }
else if (v < target) {
442 v += in.TotalDeltaT * accel;
443 if (v > target) v = target;
450double FGTurboProp::ExpSeek(
double *var,
double target,
double accel_tau,
double decel_tau)
455 v = (v - target) * exp ( -in.TotalDeltaT / decel_tau) + target;
456 }
else if (v < target) {
457 v = (target - v) * (1 - exp ( -in.TotalDeltaT / accel_tau)) + v;
464void FGTurboProp::SetDefaults(
void)
479 Ielu_intervent=
false;
481 Idle_Max_Delay = 1.0;
483 ThrottlePos = OldThrottle = 0.0;
485 ReverseMaxPower = 0.0;
486 BetaRangeThrottleEnd = 0.0;
487 CombustionEfficiency = 1.0;
493string FGTurboProp::GetEngineLabels(
const string& delimiter)
495 std::ostringstream buf;
497 buf << Name <<
"_N1[" << EngineNumber <<
"]" << delimiter
498 << Name <<
"_PwrAvail[" << EngineNumber <<
"]" << delimiter
499 << Thruster->GetThrusterLabels(EngineNumber, delimiter);
506string FGTurboProp::GetEngineValues(
const string& delimiter)
508 std::ostringstream buf;
510 buf << N1 << delimiter
512 << Thruster->GetThrusterValues(EngineNumber,delimiter);
519int FGTurboProp::InitRunning(
void)
521 double dt = in.TotalDeltaT;
522 in.TotalDeltaT = 0.0;
532void FGTurboProp::bindmodel(FGPropertyManager* PropertyManager)
534 string property_name, base_property_name;
535 base_property_name = CreateIndexedPropertyName(
"propulsion/engine", EngineNumber);
536 property_name = base_property_name +
"/n1";
537 PropertyManager->Tie( property_name.c_str(), &N1);
538 property_name = base_property_name +
"/reverser";
539 PropertyManager->Tie( property_name.c_str(), &Reversed);
540 property_name = base_property_name +
"/power-hp";
541 PropertyManager->Tie( property_name.c_str(), &HP);
542 property_name = base_property_name +
"/itt-c";
543 PropertyManager->Tie( property_name.c_str(), &Eng_ITT_degC);
544 property_name = base_property_name +
"/engtemp-c";
545 PropertyManager->Tie( property_name.c_str(), &Eng_Temperature);
546 property_name = base_property_name +
"/ielu_intervent";
547 PropertyManager->Tie( property_name.c_str(), &Ielu_intervent);
548 property_name = base_property_name +
"/combustion_efficiency";
549 PropertyManager->Tie( property_name.c_str(), &CombustionEfficiency);
571void FGTurboProp::Debug(
int from)
573 if (debug_lvl <= 0)
return;
580 cout <<
"\n ****MUJ MOTOR TURBOPROP****\n";
581 cout <<
"\n Engine Name: " << Name << endl;
582 cout <<
" IdleN1: " << IdleN1 << endl;
583 cout <<
" MaxN1: " << MaxN1 << endl;
588 if (debug_lvl & 2 ) {
589 if (from == 0) cout <<
"Instantiated: FGTurboProp" << endl;
590 if (from == 1) cout <<
"Destroyed: FGTurboProp" << endl;
592 if (debug_lvl & 4 ) {
594 if (debug_lvl & 8 ) {
596 if (debug_lvl & 16) {
598 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.
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.
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.
Encapsulates the JSBSim simulation executive.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
static constexpr double Constrain(double min, double value, double max)
Constrain a value between a minimum and a maximum value.
std::shared_ptr< FGFunction > GetPreFunction(const std::string &name)
Get one of the "pre" function.
FGPropeller models a propeller given the tabular data for Ct (thrust) and Cp (power),...
Models a helicopter rotor.
void Calculate(void)
Calculates the thrust of the engine, and other engine functions.
FGTurboProp(FGFDMExec *Executive, Element *el, int engine_number, struct Inputs &input)
Constructor.
double CalcFuelNeed(void)
The fuel need is calculated based on power levels and flow rate for that power level.