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
FGPropeller.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGPropeller.cpp
4 Author: Jon S. Berndt
5 Date started: 08/24/00
6 Purpose: Encapsulates the propeller object
7
8 ------------- Copyright (C) 2000 Jon S. Berndt (jon@jsbsim.org) -------------
9
10 This program is free software; you can redistribute it and/or modify it under
11 the terms of the GNU Lesser General Public License as published by the Free Software
12 Foundation; either version 2 of the License, or (at your option) any later
13 version.
14
15 This program is distributed in the hope that it will be useful, but WITHOUT
16 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
17 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
18 details.
19
20 You should have received a copy of the GNU Lesser General Public License along with
21 this program; if not, write to the Free Software Foundation, Inc., 59 Temple
22 Place - Suite 330, Boston, MA 02111-1307, USA.
23
24 Further information about the GNU Lesser General Public License can also be found on
25 the world wide web at http://www.gnu.org.
26
27FUNCTIONAL DESCRIPTION
28--------------------------------------------------------------------------------
29
30HISTORY
31--------------------------------------------------------------------------------
3208/24/00 JSB Created
33
34%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
35INCLUDES
36%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
37
38#include <iostream>
39#include <sstream>
40
41#include "FGFDMExec.h"
42#include "FGPropeller.h"
43#include "input_output/FGXMLElement.h"
44
45using namespace std;
46
47namespace JSBSim {
48
49/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
50CLASS IMPLEMENTATION
51%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
52
53FGPropeller::FGPropeller(FGFDMExec* exec, Element* prop_element, int num)
54 : FGThruster(exec, prop_element, num)
55{
56 Element *table_element, *local_element;
57 string name="";
58 auto PropertyManager = exec->GetPropertyManager();
59
60 MaxPitch = MinPitch = P_Factor = Pitch = Advance = MinRPM = MaxRPM = 0.0;
61 Sense = 1; // default clockwise rotation
62 ReversePitch = 0.0;
63 Reversed = false;
64 Feathered = false;
65 Reverse_coef = 0.0;
66 GearRatio = 1.0;
67 CtFactor = CpFactor = 1.0;
68 ConstantSpeed = 0;
69 cThrust = cPower = CtMach = CpMach = 0;
70 Vinduced = 0.0;
71
72 if (prop_element->FindElement("ixx"))
73 Ixx = max(prop_element->FindElementValueAsNumberConvertTo("ixx", "SLUG*FT2"), 1e-06);
74
75 Sense_multiplier = 1.0;
76 if (prop_element->HasAttribute("version")
77 && prop_element->GetAttributeValueAsNumber("version") > 1.0)
78 Sense_multiplier = -1.0;
79
80 if (prop_element->FindElement("diameter"))
81 Diameter = max(prop_element->FindElementValueAsNumberConvertTo("diameter", "FT"), 0.001);
82 if (prop_element->FindElement("numblades"))
83 numBlades = (int)prop_element->FindElementValueAsNumber("numblades");
84 if (prop_element->FindElement("gearratio"))
85 GearRatio = max(prop_element->FindElementValueAsNumber("gearratio"), 0.001);
86 if (prop_element->FindElement("minpitch"))
87 MinPitch = prop_element->FindElementValueAsNumber("minpitch");
88 if (prop_element->FindElement("maxpitch"))
89 MaxPitch = prop_element->FindElementValueAsNumber("maxpitch");
90 if (prop_element->FindElement("minrpm"))
91 MinRPM = prop_element->FindElementValueAsNumber("minrpm");
92 if (prop_element->FindElement("maxrpm")) {
93 MaxRPM = prop_element->FindElementValueAsNumber("maxrpm");
94 ConstantSpeed = 1;
95 }
96 if (prop_element->FindElement("constspeed"))
97 ConstantSpeed = (int)prop_element->FindElementValueAsNumber("constspeed");
98 if (prop_element->FindElement("reversepitch"))
99 ReversePitch = prop_element->FindElementValueAsNumber("reversepitch");
100 while((table_element = prop_element->FindNextElement("table")) != 0) {
101 name = table_element->GetAttributeValue("name");
102 try {
103 if (name == "C_THRUST") {
104 cThrust = new FGTable(PropertyManager, table_element);
105 } else if (name == "C_POWER") {
106 cPower = new FGTable(PropertyManager, table_element);
107 } else if (name == "CT_MACH") {
108 CtMach = new FGTable(PropertyManager, table_element);
109 } else if (name == "CP_MACH") {
110 CpMach = new FGTable(PropertyManager, table_element);
111 } else {
112 cerr << "Unknown table type: " << name << " in propeller definition." << endl;
113 }
114 } catch (std::string& str) {
115 throw("Error loading propeller table:" + name + ". " + str);
116 }
117 }
118 if( (cPower == 0) || (cThrust == 0)){
119 cerr << "Propeller configuration must contain C_THRUST and C_POWER tables!" << endl;
120 }
121
122 local_element = prop_element->GetParent()->FindElement("sense");
123 if (local_element) {
124 double Sense = local_element->GetDataAsNumber();
125 SetSense(Sense >= 0.0 ? 1.0 : -1.0);
126 }
127 local_element = prop_element->GetParent()->FindElement("p_factor");
128 if (local_element) {
129 P_Factor = local_element->GetDataAsNumber();
130 }
131 if (P_Factor < 0) {
132 cerr << "P-Factor value in propeller configuration file must be greater than zero" << endl;
133 }
134 if (prop_element->FindElement("ct_factor"))
135 SetCtFactor( prop_element->FindElementValueAsNumber("ct_factor") );
136 if (prop_element->FindElement("cp_factor"))
137 SetCpFactor( prop_element->FindElementValueAsNumber("cp_factor") );
138
139 Type = ttPropeller;
140 RPM = 0;
141 vTorque.InitMatrix();
142 D4 = Diameter*Diameter*Diameter*Diameter;
143 D5 = D4*Diameter;
144 Pitch = MinPitch;
145
146 string property_name, base_property_name;
147 base_property_name = CreateIndexedPropertyName("propulsion/engine", EngineNum);
148 property_name = base_property_name + "/engine-rpm";
149 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetEngineRPM );
150 property_name = base_property_name + "/advance-ratio";
151 PropertyManager->Tie( property_name.c_str(), &J );
152 property_name = base_property_name + "/blade-angle";
153 PropertyManager->Tie( property_name.c_str(), &Pitch );
154 property_name = base_property_name + "/thrust-coefficient";
155 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetThrustCoefficient );
156 property_name = base_property_name + "/propeller-rpm";
157 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetRPM );
158 property_name = base_property_name + "/helical-tip-Mach";
159 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetHelicalTipMach );
160 property_name = base_property_name + "/constant-speed-mode";
161 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetConstantSpeed,
163 property_name = base_property_name + "/prop-induced-velocity_fps"; // [ft/sec]
164 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetInducedVelocity,
166 property_name = base_property_name + "/propeller-power-ftlbps"; // [ft-lbs/sec]
167 PropertyManager->Tie( property_name.c_str(), &PowerRequired );
168 property_name = base_property_name + "/propeller-torque-ftlb"; // [ft-lbs]
169 PropertyManager->Tie( property_name.c_str(), this, &FGPropeller::GetTorque);
170 property_name = base_property_name + "/propeller-sense";
171 PropertyManager->Tie( property_name.c_str(), &Sense );
172
173 Debug(0);
174}
175
176//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
177
179{
180 delete cThrust;
181 delete cPower;
182 delete CtMach;
183 delete CpMach;
184
185 Debug(1);
186}
187
188//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
189
191{
192 FGThruster::ResetToIC();
193 Vinduced = 0.0;
194}
195
196//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
197//
198// We must be getting the aerodynamic velocity here, NOT the inertial velocity.
199// We need the velocity with respect to the wind.
200//
201// Remembering that Torque * omega = Power, we can derive the torque on the
202// propeller and its acceleration to give a new RPM. The current RPM will be
203// used to calculate thrust.
204//
205// Because RPM could be zero, we need to be creative about what RPM is stated as.
206
207double FGPropeller::Calculate(double EnginePower)
208{
209 FGColumnVector3 vDXYZ = MassBalance->StructuralToBody(vXYZn);
210 const FGMatrix33& mT = Transform();
211 // Local air velocity is obtained from Stevens & Lewis' "Aircraft Control and
212 // Simualtion (3rd edition)" eqn 8.2-1
213 // Variables in.AeroUVW and in.AeroPQR include the wind and turbulence effects
214 // as computed by FGAuxiliary.
215 FGColumnVector3 localAeroVel = mT.Transposed() * (in.AeroUVW + in.AeroPQR*vDXYZ);
216 double omega, PowerAvailable;
217
218 double Vel = localAeroVel(eU);
219 double rho = in.Density;
220 double RPS = RPM/60.0;
221
222 // Calculate helical tip Mach
223 double Area = 0.25*Diameter*Diameter*M_PI;
224 double Vtip = RPS * Diameter * M_PI;
225 HelicalTipMach = sqrt(Vtip*Vtip + Vel*Vel) / in.Soundspeed;
226
227 if (RPS > 0.01) J = Vel / (Diameter * RPS); // Calculate J normally
228 else J = Vel / Diameter;
229
230 PowerAvailable = EnginePower - GetPowerRequired();
231
232 if (MaxPitch == MinPitch) { // Fixed pitch prop
233 ThrustCoeff = cThrust->GetValue(J);
234 } else { // Variable pitch prop
235 ThrustCoeff = cThrust->GetValue(J, Pitch);
236 }
237
238 // Apply optional scaling factor to Ct (default value = 1)
239 ThrustCoeff *= CtFactor;
240
241 // Apply optional Mach effects from CT_MACH table
242 if (CtMach) ThrustCoeff *= CtMach->GetValue(HelicalTipMach);
243
244 Thrust = ThrustCoeff*RPS*RPS*D4*rho;
245
246 // Induced velocity in the propeller disk area. This formula is obtained
247 // from momentum theory - see B. W. McCormick, "Aerodynamics, Aeronautics,
248 // and Flight Mechanics" 1st edition, eqn. 6.15 (propeller analysis chapter).
249 // Since Thrust and Vel can both be negative we need to adjust this formula
250 // To handle sign (direction) separately from magnitude.
251 double Vel2sum = Vel*abs(Vel) + 2.0*Thrust/(rho*Area);
252
253 if( Vel2sum > 0.0)
254 Vinduced = 0.5 * (-Vel + sqrt(Vel2sum));
255 else
256 Vinduced = 0.5 * (-Vel - sqrt(-Vel2sum));
257
258 // P-factor is simulated by a shift of the acting location of the thrust.
259 // The shift is a multiple of the angle between the propeller shaft axis
260 // and the relative wind that goes through the propeller disk.
261 if (P_Factor > 0.0001) {
262 double tangentialVel = localAeroVel.Magnitude(eV, eW);
263
264 if (tangentialVel > 0.0001) {
265 // The angle made locally by the air flow with respect to the propeller
266 // axis is influenced by the induced velocity. This attenuates the
267 // influence of a string cross wind and gives a more realistic behavior.
268 double angle = atan2(tangentialVel, Vel+Vinduced);
269 double factor = Sense * P_Factor * angle / tangentialVel;
270 SetActingLocationY( GetLocationY() + factor * localAeroVel(eW));
271 SetActingLocationZ( GetLocationZ() + factor * localAeroVel(eV));
272 }
273 }
274
275 omega = RPS*2.0*M_PI;
276
277 vFn(eX) = Thrust;
278 vTorque(eX) = -Sense*EnginePower / max(0.01, omega);
279
280 // The Ixx value and rotation speed given below are for rotation about the
281 // natural axis of the engine. The transform takes place in the base class
282 // FGForce::GetBodyForces() function.
283
284 FGColumnVector3 vH(Ixx*omega*Sense*Sense_multiplier, 0.0, 0.0);
285
286 if (omega > 0.01) ExcessTorque = PowerAvailable / omega;
287 else ExcessTorque = PowerAvailable / 1.0;
288
289 RPM = (RPS + ((ExcessTorque / Ixx) / (2.0 * M_PI)) * in.TotalDeltaT) * 60.0;
290
291 if (RPM < 0.0) RPM = 0.0; // Engine won't turn backwards
292
293 // Transform Torque and momentum first, as PQR is used in this
294 // equation and cannot be transformed itself.
295 vMn = in.PQRi*(mT*vH) + mT*vTorque;
296
297 return Thrust; // return thrust in pounds
298}
299
300//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
301
303{
304 double cPReq;
305
306 if (MaxPitch == MinPitch) { // Fixed pitch prop
307 cPReq = cPower->GetValue(J);
308
309 } else { // Variable pitch prop
310
311 if (ConstantSpeed != 0) { // Constant Speed Mode
312
313 // do normal calculation when propeller is neither feathered nor reversed
314 // Note: This method of feathering and reversing was added to support the
315 // turboprop model. It's left here for backward compatiblity, but
316 // now feathering and reversing should be done in Manual Pitch Mode.
317 if (!Feathered) {
318 if (!Reversed) {
319
320 double rpmReq = MinRPM + (MaxRPM - MinRPM) * Advance;
321 double dRPM = rpmReq - RPM;
322 // The pitch of a variable propeller cannot be changed when the RPMs are
323 // too low - the oil pump does not work.
324 if (RPM > 200) Pitch -= dRPM * in.TotalDeltaT;
325 if (Pitch < MinPitch) Pitch = MinPitch;
326 else if (Pitch > MaxPitch) Pitch = MaxPitch;
327
328 } else { // Reversed propeller
329
330 // when reversed calculate propeller pitch depending on throttle lever position
331 // (beta range for taxing full reverse for braking)
332 double PitchReq = MinPitch - ( MinPitch - ReversePitch ) * Reverse_coef;
333 // The pitch of a variable propeller cannot be changed when the RPMs are
334 // too low - the oil pump does not work.
335 if (RPM > 200) Pitch += (PitchReq - Pitch) / 200;
336 if (RPM > MaxRPM) {
337 Pitch += (MaxRPM - RPM) / 50;
338 if (Pitch < ReversePitch) Pitch = ReversePitch;
339 else if (Pitch > MaxPitch) Pitch = MaxPitch;
340 }
341 }
342
343 } else { // Feathered propeller
344 // ToDo: Make feathered and reverse settings done via FGKinemat
345 Pitch += (MaxPitch - Pitch) / 300; // just a guess (about 5 sec to fully feathered)
346 }
347
348 } else { // Manual Pitch Mode, pitch is controlled externally
349
350 }
351
352 cPReq = cPower->GetValue(J, Pitch);
353 }
354
355 // Apply optional scaling factor to Cp (default value = 1)
356 cPReq *= CpFactor;
357
358 // Apply optional Mach effects from CP_MACH table
359 if (CpMach) cPReq *= CpMach->GetValue(HelicalTipMach);
360
361 double RPS = RPM / 60.0;
362 double local_RPS = RPS < 0.01 ? 0.01 : RPS;
363
364 PowerRequired = cPReq*local_RPS*local_RPS*local_RPS*D5*in.Density;
365
366 return PowerRequired;
367}
368
369//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
370
372{
373 // These are moments in lbf per ft : the lever arm along Z generates a moment
374 // along the pitch direction.
375 double p_pitch = Thrust * Sense * (GetActingLocationZ() - GetLocationZ()) / 12.0;
376 // The lever arm along Y generates a moment along the yaw direction.
377 double p_yaw = Thrust * Sense * (GetActingLocationY() - GetLocationY()) / 12.0;
378
379 return FGColumnVector3(0.0, p_pitch, p_yaw);
380}
381
382//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
383
384string FGPropeller::GetThrusterLabels(int id, const string& delimeter)
385{
386 std::ostringstream buf;
387
388 buf << Name << " Torque (engine " << id << ")" << delimeter
389 << Name << " PFactor Pitch (engine " << id << ")" << delimeter
390 << Name << " PFactor Yaw (engine " << id << ")" << delimeter
391 << Name << " Thrust (engine " << id << " in lbs)" << delimeter;
392 if (IsVPitch())
393 buf << Name << " Pitch (engine " << id << ")" << delimeter;
394 buf << Name << " RPM (engine " << id << ")";
395
396 return buf.str();
397}
398
399//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
400
401string FGPropeller::GetThrusterValues(int id, const string& delimeter)
402{
403 std::ostringstream buf;
404
405 FGColumnVector3 vPFactor = GetPFactor();
406 buf << vTorque(eX) << delimeter
407 << vPFactor(ePitch) << delimeter
408 << vPFactor(eYaw) << delimeter
409 << Thrust << delimeter;
410 if (IsVPitch())
411 buf << Pitch << delimeter;
412 buf << RPM;
413
414 return buf.str();
415}
416
417//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
418// The bitmasked value choices are as follows:
419// unset: In this case (the default) JSBSim would only print
420// out the normally expected messages, essentially echoing
421// the config files as they are read. If the environment
422// variable is not set, debug_lvl is set to 1 internally
423// 0: This requests JSBSim not to output any messages
424// whatsoever.
425// 1: This value explicity requests the normal JSBSim
426// startup messages
427// 2: This value asks for a message to be printed out when
428// a class is instantiated
429// 4: When this value is set, a message is displayed when a
430// FGModel object executes its Run() method
431// 8: When this value is set, various runtime state variables
432// are printed out periodically
433// 16: When set various parameters are sanity checked and
434// a message is printed out when they go out of bounds
435
436void FGPropeller::Debug(int from)
437{
438 if (debug_lvl <= 0) return;
439
440 if (debug_lvl & 1) { // Standard console startup message output
441 if (from == 0) { // Constructor
442 cout << "\n Propeller Name: " << Name << endl;
443 cout << " IXX = " << Ixx << endl;
444 cout << " Diameter = " << Diameter << " ft." << endl;
445 cout << " Number of Blades = " << numBlades << endl;
446 cout << " Gear Ratio = " << GearRatio << endl;
447 cout << " Minimum Pitch = " << MinPitch << endl;
448 cout << " Maximum Pitch = " << MaxPitch << endl;
449 cout << " Minimum RPM = " << MinRPM << endl;
450 cout << " Maximum RPM = " << MaxRPM << endl;
451// Tables are being printed elsewhere...
452// cout << " Thrust Coefficient: " << endl;
453// cThrust->Print();
454// cout << " Power Coefficient: " << endl;
455// cPower->Print();
456// cout << " Mach Thrust Coefficient: " << endl;
457// if(CtMach)
458// {
459// CtMach->Print();
460// } else {
461// cout << " NONE" << endl;
462// }
463// cout << " Mach Power Coefficient: " << endl;
464// if(CpMach)
465// {
466// CpMach->Print();
467// } else {
468// cout << " NONE" << endl;
469// }
470 }
471 }
472 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
473 if (from == 0) cout << "Instantiated: FGPropeller" << endl;
474 if (from == 1) cout << "Destroyed: FGPropeller" << endl;
475 }
476 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
477 }
478 if (debug_lvl & 8 ) { // Runtime state variables
479 }
480 if (debug_lvl & 16) { // Sanity checking
481 }
482 if (debug_lvl & 64) {
483 if (from == 0) { // Constructor
484 }
485 }
486}
487}
Element * FindElement(const std::string &el="")
Searches for a specified element.
double GetAttributeValueAsNumber(const std::string &key)
Retrieves an attribute value as a double precision real number.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
Element * GetParent(void)
Returns a pointer to the parent of 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 GetDataAsNumber(void)
Converts the element data to a number.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
This class implements a 3 element column vector.
double Magnitude(void) const
Length of the vector.
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
Handles matrix math operations.
Definition FGMatrix33.h:70
FGMatrix33 Transposed(void) const
Transposed matrix.
Definition FGMatrix33.h:221
void SetInducedVelocity(double Vi)
Set the propeller induced velocity.
~FGPropeller()
Destructor for FGPropeller - deletes the FGTable objects.
double Calculate(double EnginePower)
Calculates and returns the thrust produced by this propeller.
double GetPowerRequired(void)
Retrieves the power required (or "absorbed") by the propeller - i.e.
int GetConstantSpeed(void) const
Returns a non-zero value if the propeller is constant speed.
double GetRPM(void) const
Retrieves the RPMs of the propeller.
std::string GetThrusterValues(int id, const std::string &delimeter)
Generate the values for the thruster standard CSV output.
void ResetToIC(void)
Reset the initial conditions.
void SetCtFactor(double ctf)
Sets coefficient of thrust multiplier.
void SetConstantSpeed(int mode)
Sets propeller into constant speed mode, or manual pitch mode.
FGColumnVector3 GetPFactor(void) const
Retrieves the P-Factor constant.
void SetCpFactor(double cpf)
Sets coefficient of power multiplier.
double GetInducedVelocity(void) const
Get the propeller induced velocity.
void SetSense(double s)
Sets the rotation sense of the propeller.
double GetHelicalTipMach(void) const
Retrieves the Mach number at the propeller tips.
std::string GetThrusterLabels(int id, const std::string &delimeter)
Generate the labels for the thruster standard CSV output.
double GetTorque(void) const
Retrieves the Torque in foot-pounds (Don't you love the English system?)
double GetEngineRPM(void) const
Calculates the RPMs of the engine based on gear ratio.
FGPropeller(FGFDMExec *exec, Element *el, int num=0)
Constructor for FGPropeller.
bool IsVPitch(void) const
Returns true of this propeller is variable pitch.
double GetThrustCoefficient(void) const
Retrieves the thrust coefficient.
Lookup table class.
Definition FGTable.h:234
double GetValue(void) const
Get the current table value.
Definition FGTable.cpp:465
Base class for specific thrusting devices such as propellers, nozzles, etc.
Definition FGThruster.h:77