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
FGTurboProp.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGTurboProp.cpp
4 Author: Jiri "Javky" Javurek
5 based on SimTurbine and Turbine engine from David Culp
6 Date started: 05/14/2004
7 Purpose: This module models a turbo propeller engine.
8
9 ------------- Copyright (C) 2004 (javky@email.cz) ---------
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 Software
13 Foundation; either version 2 of the License, or (at your option) any later
14 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 with
22 this program; if not, write to the Free Software Foundation, Inc., 59 Temple
23 Place - Suite 330, Boston, MA 02111-1307, USA.
24
25 Further information about the GNU Lesser General Public License can also be found on
26 the world wide web at http://www.gnu.org.
27
28FUNCTIONAL DESCRIPTION
29--------------------------------------------------------------------------------
30
31This class descends from the FGEngine class and models a Turbo propeller engine
32based on parameters given in the engine config file for this class
33
34HISTORY
35--------------------------------------------------------------------------------
3605/14/2004 Created
3702/08/2011 T. Kreitler, added rotor support
38
39//JVK (mark)
40
41%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
42INCLUDES
43%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
44
45#include <iostream>
46#include <sstream>
47
48#include "FGTurboProp.h"
49#include "FGPropeller.h"
50#include "FGRotor.h"
51#include "math/FGFunction.h"
52#include "input_output/FGXMLElement.h"
53
54using namespace std;
55
56namespace JSBSim {
57
58/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
59CLASS IMPLEMENTATION
60%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
61
62FGTurboProp::FGTurboProp(FGFDMExec* exec, Element *el, int engine_number, struct Inputs& input)
63 : FGEngine(engine_number, input)
64{
65 SetDefaults();
66 Load(exec, el);
67 Debug(0);
68}
69
70//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
71
72bool FGTurboProp::Load(FGFDMExec* exec, Element *el)
73{
74 MaxStartingTime = 999999; //very big timeout -> infinite
75 Ielu_max_torque=-1;
76
77 Element* function_element = el->FindElement("function");
78
79 while(function_element) {
80 string name = function_element->GetAttributeValue("name");
81 if (name == "EnginePowerVC")
82 function_element->SetAttributeValue("name", string("propulsion/engine[#]/") + name);
83
84 function_element = el->FindNextElement("function");
85 }
86
87 FGEngine::Load(exec, el);
88 thrusterType = Thruster->GetType();
89
90 string property_prefix = CreateIndexedPropertyName("propulsion/engine", EngineNumber);
91
92 EnginePowerVC = GetPreFunction(property_prefix+"/EnginePowerVC");
93
94
95// ToDo: Need to make sure units are properly accounted for below.
96
97 if (el->FindElement("idlen1"))
98 IdleN1 = el->FindElementValueAsNumber("idlen1");
99 if (el->FindElement("maxn1"))
100 MaxN1 = el->FindElementValueAsNumber("maxn1");
101 if (el->FindElement("betarangeend"))
102 BetaRangeThrottleEnd = el->FindElementValueAsNumber("betarangeend")/100.0;
103 BetaRangeThrottleEnd = Constrain(0.0, BetaRangeThrottleEnd, 0.99999);
104 if (el->FindElement("reversemaxpower"))
105 ReverseMaxPower = el->FindElementValueAsNumber("reversemaxpower")/100.0;
106
107 if (el->FindElement("maxpower"))
108 MaxPower = el->FindElementValueAsNumber("maxpower");
109 if (el->FindElement("idlefuelflow")) {
110 cerr << el->ReadFrom() << "Note: 'idlefuelflow' is obsolete, "
111 << "use the 'CombustionEfficiency_N1' table instead." << endl;
112 }
113 if (el->FindElement("psfc"))
114 PSFC = el->FindElementValueAsNumber("psfc");
115 if (el->FindElement("n1idle_max_delay"))
116 Idle_Max_Delay = el->FindElementValueAsNumber("n1idle_max_delay");
117 if (el->FindElement("maxstartingtime"))
118 MaxStartingTime = el->FindElementValueAsNumber("maxstartingtime");
119 if (el->FindElement("startern1"))
120 StarterN1 = el->FindElementValueAsNumber("startern1");
121 if (el->FindElement("ielumaxtorque"))
122 Ielu_max_torque = el->FindElementValueAsNumber("ielumaxtorque");
123 if (el->FindElement("itt_delay"))
124 ITT_Delay = el->FindElementValueAsNumber("itt_delay");
125
126 Element *table_element = el->FindElement("table");
127 auto PropertyManager = exec->GetPropertyManager();
128
129 while (table_element) {
130 string name = table_element->GetAttributeValue("name");
131 if (!EnginePowerVC && name == "EnginePowerVC") {
132 // Get a different name for each engines otherwise FGTable::bind() will
133 // complain that the property 'EnginePowerVC' is already bound. This is a
134 // ugly hack but the functionality is obsolete and will be removed some
135 // time in the future.
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"
142 << endl;
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);
149 } else {
150 cerr << el->ReadFrom() << "Unknown table type: " << name
151 << " in turboprop definition." << endl;
152 }
153 table_element = el->FindNextElement("table");
154 }
155
156 // Pre-calculations and initializations
157
158 delay=1;
159 N1_factor = MaxN1 - IdleN1;
160 OilTemp_degK = in.TAT_c + 273.0;
161
162 // default table based on '9.333 - (N1)/12.0' approximation
163 // gives 430%Fuel at 60%N1
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;
172 }
173
174 bindmodel(PropertyManager.get());
175 return true;
176}
177
178//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
179// The main purpose of Calculate() is to determine what phase the engine should
180// be in, then call the corresponding function.
181
183{
184 RunPreFunctions();
185
186 ThrottlePos = in.ThrottlePos[EngineNumber];
187
188 /* The thruster controls the engine RPM because it encapsulates the gear ratio
189 and other transmission variables */
190 RPM = Thruster->GetEngineRPM();
191 if (thrusterType == FGThruster::ttPropeller) {
192 ((FGPropeller*)Thruster)->SetAdvance(in.PropAdvance[EngineNumber]);
193 ((FGPropeller*)Thruster)->SetFeather(in.PropFeather[EngineNumber]);
194 ((FGPropeller*)Thruster)->SetReverse(Reversed);
195 if (Reversed) {
196 ((FGPropeller*)Thruster)->SetReverseCoef(ThrottlePos);
197 } else {
198 ((FGPropeller*)Thruster)->SetReverseCoef(0.0);
199 }
200
201 if (Reversed) {
202 if (ThrottlePos < BetaRangeThrottleEnd) {
203 ThrottlePos = 0.0; // idle when in Beta-range
204 } else {
205 // when reversed:
206 ThrottlePos = (ThrottlePos-BetaRangeThrottleEnd)/(1-BetaRangeThrottleEnd) * ReverseMaxPower;
207 }
208 }
209 }
210
211 // When trimming is finished check if user wants engine OFF or RUNNING
212 if ((phase == tpTrim) && (in.TotalDeltaT > 0)) {
213 if (Running && !Starved) {
214 phase = tpRun;
215 N1 = IdleN1;
216 OilTemp_degK = 366.0;
217 Cutoff = false;
218 } else {
219 phase = tpOff;
220 Cutoff = true;
221 Eng_ITT_degC = in.TAT_c;
222 Eng_Temperature = in.TAT_c;
223 OilTemp_degK = in.TAT_c+273.15;
224 }
225 }
226
227 if (!Running && Starter) {
228 if (phase == tpOff) {
229 phase = tpSpinUp;
230 if (StartTime < 0) StartTime=0;
231 }
232 }
233 if (!Running && !Cutoff && (N1 > 15.0)) {
234 phase = tpStart;
235 StartTime = -1;
236 }
237 if (Cutoff && (phase != tpSpinUp)) phase = tpOff;
238 if (in.TotalDeltaT == 0) phase = tpTrim;
239 if (Starved) phase = tpOff;
240 if (Condition >= 10) {
241 phase = tpOff;
242 StartTime=-1;
243 }
244
245 // limiter intervention wanted?
246 if (Ielu_max_torque > 0.0) {
247 double torque = 0.0;
248
249 if (thrusterType == FGThruster::ttPropeller) {
250 torque = ((FGPropeller*)(Thruster))->GetTorque();
251 } else if (thrusterType == FGThruster::ttRotor) {
252 torque = ((FGRotor*)(Thruster))->GetTorque();
253 }
254
255 if (Condition < 1) {
256 if ( abs(torque) > Ielu_max_torque && ThrottlePos >= OldThrottle ) {
257 ThrottlePos = OldThrottle - 0.1 * in.TotalDeltaT; //IELU down
258 Ielu_intervent = true;
259 } else if ( Ielu_intervent && ThrottlePos >= OldThrottle) {
260 ThrottlePos = OldThrottle + 0.05 * in.TotalDeltaT; //IELU up
261 Ielu_intervent = true;
262 } else {
263 Ielu_intervent = false;
264 }
265 } else {
266 Ielu_intervent = false;
267 }
268 OldThrottle = ThrottlePos;
269 }
270
271 switch (phase) {
272 case tpOff: HP = Off(); break;
273 case tpRun: HP = Run(); break;
274 case tpSpinUp: HP = SpinUp(); break;
275 case tpStart: HP = Start(); break;
276 default: HP = 0;
277 }
278
279 LoadThrusterInputs();
280 // Filters out negative powers when the propeller is not rotating.
281 double power = HP * hptoftlbssec;
282 if (RPM <= 0.1) power = max(power, 0.0);
283 Thruster->Calculate(power);
284
285 RunPostFunctions();
286}
287
288//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
289
290double FGTurboProp::Off(void)
291{
292 Running = false; EngStarting = false;
293
294 FuelFlow_pph = Seek(&FuelFlow_pph, 0, 800.0, 800.0);
295
296 //allow the air turn with generator
297 N1 = ExpSeek(&N1, in.qbar/15.0, Idle_Max_Delay*2.5, Idle_Max_Delay * 5);
298
299 OilTemp_degK = ExpSeek(&OilTemp_degK,273.15 + in.TAT_c, 400 , 400);
300
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);
304
305 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6; //from MPa to psi
306
307 if (RPM>5) return -0.012; // friction in engine when propeller spining (estimate)
308 return 0.0;
309}
310
311//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
312
313double FGTurboProp::Run(void)
314{
315 double EngPower_HP;
316
317 Running = true; Starter = false; EngStarting = false;
318
319//---
320 double old_N1 = N1;
321 N1 = ExpSeek(&N1, IdleN1 + ThrottlePos * N1_factor, Idle_Max_Delay, Idle_Max_Delay * 2.4);
322
323 EngPower_HP = EnginePowerRPM_N1->GetValue(RPM,N1);
324 EngPower_HP *= EnginePowerVC->GetValue();
325 if (EngPower_HP > MaxPower) EngPower_HP = MaxPower;
326
327 CombustionEfficiency = CombustionEfficiency_N1->GetValue(N1);
328 FuelFlow_pph = PSFC / CombustionEfficiency * EngPower_HP;
329
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);
333
334 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6; //from MPa to psi
335//---
336
337 OilTemp_degK = Seek(&OilTemp_degK, 353.15, 0.4-N1*0.001, 0.04);
338
339 if (Cutoff) phase = tpOff;
340 if (Starved) phase = tpOff;
341
342 return EngPower_HP;
343}
344
345//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
346
347double FGTurboProp::SpinUp(void)
348{
349 double EngPower_HP;
350 Running = false; EngStarting = true;
351 FuelFlow_pph = 0.0;
352
353 if (!GeneratorPower) {
354 EngStarting=false;
355 phase=tpOff;
356 StartTime = -1;
357 return 0.0;
358 }
359
360 N1 = ExpSeek(&N1, StarterN1, Idle_Max_Delay * 6, Idle_Max_Delay * 2.4);
361
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);
365
366 OilTemp_degK = ExpSeek(&OilTemp_degK,273.15 + in.TAT_c, 400 , 400);
367
368 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6; //from MPa to psi
369
370 EngPower_HP = EnginePowerRPM_N1->GetValue(RPM,N1);
371 EngPower_HP *= EnginePowerVC->GetValue();
372 if (EngPower_HP > MaxPower) EngPower_HP = MaxPower;
373
374 if (StartTime>=0) StartTime+=in.TotalDeltaT;
375 if (StartTime > MaxStartingTime && MaxStartingTime > 0) { //start failed due timeout
376 phase = tpOff;
377 StartTime = -1;
378 }
379
380 return EngPower_HP;
381}
382
383//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
384
385double FGTurboProp::Start(void)
386{
387 double EngPower_HP = 0.0;
388
389 EngStarting = false;
390 if ((N1 > 15.0) && !Starved) { // minimum 15% N1 needed for start
391 double old_N1 = N1;
392 Cranking = true; // provided for sound effects signal
393 if (N1 < IdleN1) {
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);
403
404 OilPressure_psi = (N1/100.0*0.25+(0.1-(OilTemp_degK-273.15)*0.1/80.0)*N1/100.0) / 7692.0e-6; //from MPa to psi
405 OilTemp_degK = Seek(&OilTemp_degK, 353.15, 0.4-N1*0.001, 0.04);
406
407 } else {
408 phase = tpRun;
409 Running = true;
410 Starter = false;
411 Cranking = false;
412 FuelFlow_pph = 0;
413 }
414 } else { // no start if N1 < 15% or Starved
415 phase = tpOff;
416 Starter = false;
417 }
418
419 return EngPower_HP;
420}
421
422
423//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
424
426{
427 FuelFlowRate = FuelFlow_pph / 3600.0;
428 FuelExpended = FuelFlowRate * in.TotalDeltaT;
429 if (!Starved) FuelUsedLbs += FuelExpended;
430 return FuelExpended;
431}
432
433//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
434
435double FGTurboProp::Seek(double *var, double target, double accel, double decel)
436{
437 double v = *var;
438 if (v > target) {
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;
444 }
445 return v;
446}
447
448//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
449
450double FGTurboProp::ExpSeek(double *var, double target, double accel_tau, double decel_tau)
451{
452// exponential delay instead of the linear delay used in Seek
453 double v = *var;
454 if (v > target) {
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;
458 }
459 return v;
460}
461
462//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
463
464void FGTurboProp::SetDefaults(void)
465{
466// Name = "Not defined";
467 N1 = 0.0;
468 HP = 0.0;
469 Type = etTurboprop;
470 IdleN1 = 30.0;
471 MaxN1 = 100.0;
472 Reversed = false;
473 Cutoff = true;
474 phase = tpOff;
475 Eng_ITT_degC = 0.0;
476
477 GeneratorPower=true;
478 Condition = 0;
479 Ielu_intervent=false;
480
481 Idle_Max_Delay = 1.0;
482
483 ThrottlePos = OldThrottle = 0.0;
484 ITT_Delay = 0.05;
485 ReverseMaxPower = 0.0;
486 BetaRangeThrottleEnd = 0.0;
487 CombustionEfficiency = 1.0;
488}
489
490//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
491
492
493string FGTurboProp::GetEngineLabels(const string& delimiter)
494{
495 std::ostringstream buf;
496
497 buf << Name << "_N1[" << EngineNumber << "]" << delimiter
498 << Name << "_PwrAvail[" << EngineNumber << "]" << delimiter
499 << Thruster->GetThrusterLabels(EngineNumber, delimiter);
500
501 return buf.str();
502}
503
504//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
505
506string FGTurboProp::GetEngineValues(const string& delimiter)
507{
508 std::ostringstream buf;
509
510 buf << N1 << delimiter
511 << HP << delimiter
512 << Thruster->GetThrusterValues(EngineNumber,delimiter);
513
514 return buf.str();
515}
516
517//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
518
519int FGTurboProp::InitRunning(void)
520{
521 double dt = in.TotalDeltaT;
522 in.TotalDeltaT = 0.0;
523 Cutoff=false;
524 Running=true;
525 Calculate();
526 in.TotalDeltaT = dt;
527 return phase==tpRun;
528}
529
530//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
531
532void FGTurboProp::bindmodel(FGPropertyManager* PropertyManager)
533{
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);
550}
551
552//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
553// The bitmasked value choices are as follows:
554// unset: In this case (the default) JSBSim would only print
555// out the normally expected messages, essentially echoing
556// the config files as they are read. If the environment
557// variable is not set, debug_lvl is set to 1 internally
558// 0: This requests JSBSim not to output any messages
559// whatsoever.
560// 1: This value explicity requests the normal JSBSim
561// startup messages
562// 2: This value asks for a message to be printed out when
563// a class is instantiated
564// 4: When this value is set, a message is displayed when a
565// FGModel object executes its Run() method
566// 8: When this value is set, various runtime state variables
567// are printed out periodically
568// 16: When set various parameters are sanity checked and
569// a message is printed out when they go out of bounds
570
571void FGTurboProp::Debug(int from)
572{
573 if (debug_lvl <= 0) return;
574
575 if (debug_lvl & 1) { // Standard console startup message output
576 if (from == 0) { // Constructor
577
578 }
579 if (from == 2) { // called from Load()
580 cout << "\n ****MUJ MOTOR TURBOPROP****\n";
581 cout << "\n Engine Name: " << Name << endl;
582 cout << " IdleN1: " << IdleN1 << endl;
583 cout << " MaxN1: " << MaxN1 << endl;
584
585 cout << endl;
586 }
587 }
588 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
589 if (from == 0) cout << "Instantiated: FGTurboProp" << endl;
590 if (from == 1) cout << "Destroyed: FGTurboProp" << endl;
591 }
592 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
593 }
594 if (debug_lvl & 8 ) { // Runtime state variables
595 }
596 if (debug_lvl & 16) { // Sanity checking
597 }
598 if (debug_lvl & 64) {
599 if (from == 0) { // Constructor
600 }
601 }
602}
603}
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.
Definition FGEngine.h:104
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Definition FGFDMExec.h:421
static constexpr double Constrain(double min, double value, double max)
Constrain a value between a minimum and a maximum value.
Definition FGJSBBase.h:288
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.
Definition FGRotor.h:235
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.